利用Python开发微信支付的注意事项


Posted in Python onAugust 19, 2016

前言

微信支付是由微信及财付通联合推出的移动支付创新产品。如今,随着微信支付的全面开放,相关需求也越来越多,很多开发人员进行微信支付开发及商家申请微信支付时,面临着诸多疑惑。

要想开发顺利进行,首先要对业务流程有个清晰的认识。这里以微信公众号支付为例,因此也借用微信支付官方文档中的业务流程图:

利用Python开发微信支付的注意事项

接下来来关注几个开发过程中的关键点,包括:

      1、生成商户订单与调用统一下单 API

      2、微信服务器交互的数据格式

      3、公众号支付下网页内通过 JS-API 调起支付

      4、异步通知商户支付结果(回调) 

一、生成商户订单与调用统一下单 API

这对应业务流程中的第 4 和 第 5 步,商户后台首先为用户生成订单,然后调用微信的【统一下单】接口向微信支付系统提交订单。这里有一个关键点就是签名的生成

简单来讲分为以下几个步骤:

      1、将所有有效参数以“k=v”的形式进行拼接,有效参数是指非空参数,也就是说如果参数为空,则不参与签名;

      2、将所有的“k=v”对用“&”连接,得到“k1=v1&k2=v2&k3=v3”这样的字符串;

      3、将微信支付 API 密钥 拼接在最后,如“k1=v1&k2=v2&k3=v3&key=secret”;

      4、对整体进行 MD5 运算,即得到签名。

这种签名方法有一个高大上的名字叫做 HMAC(Hash-based Message Authentication Code,基于哈希的消息码)。基于此思路,可以实现如下签名方法:

def gen_sign(params, key):
  """
  签名生成函数
 
  :param params: 参数,dict 对象
  :param key: API 密钥
  :return: sign string
  """
 
  param_list = []
  for k in sorted(params.keys()):
    v = params.get(k)
    if not v:
      # 参数的值为空不参与签名
      continue
    param_list.append('{0}={1}'.format(k, v))
  # 在最后拼接 key
  param_list.append('key={}'.format(key))
  # 用 & 连接各 k-v 对,然后对字符串进行 MD5 运算
  return md5('&'.join(param_list).encode('utf8')).hexdigest()

参与签名的参数中有一个随机字符串,在 Python 中有很多方法,当然也可以利用 uuid 库来生成:

def gen_nonce_str():
  """
  生成随机字符串,有效字符a-zA-Z0-9
 
  :return: 随机字符串
  """
 
  return ''.join(str(uuid.uuid4()).split('-'))

 

二、微信服务器交互的数据格式

微信服务器与商户服务器之间采用 XML 格式进行交互,这就涉及到与语言原生数据类型进行转换以方便处理。交互的数据参数都是 key-value 的形式,因此在 Python 中使用字典会更加方便。而要解析 XML,也有一大把第三方库供使用,比如 BeautifulSoup

以下是具体实现:

def trans_xml_to_dict(xml):
  """
  将微信支付交互返回的 XML 格式数据转化为 Python Dict 对象
 
  :param xml: 原始 XML 格式数据
  :return: dict 对象
  """
 
  soup = BeautifulSoup(xml, features='xml')
  xml = soup.find('xml')
  if not xml:
    return {}
 
  # 将 XML 数据转化为 Dict
  data = dict([(item.name, item.text) for item in xml.find_all()])
  return data
 
 
def trans_dict_to_xml(data):
  """
  将 dict 对象转换成微信支付交互所需的 XML 格式数据
 
  :param data: dict 对象
  :return: xml 格式数据
  """
 
  xml = []
  for k in sorted(data.keys()):
    v = data.get(k)
    if k == 'detail' and not v.startswith('<![CDATA['):
      v = '<![CDATA[{}]]>'.format(v)
    xml.append('<{key}>{value}</{key}>'.format(key=k, value=v))
  return '<xml>{}</xml>'.format(''.join(xml))

注意 detail 参数,即商品详情,其值为 JSON 格式,在转换为 XML 数据时应前注意使用 CDATA 标签将其保护起来。

如:

<detail><![CDATA[{"goods_detail": [{"wxpay_goods_id": "10010001", "price": 1, "goods_num": 1, "goods_name": "\\u82f9\\u679c", "goods_id": "10010001"}, {"wxpay_goods_id": "10010002", "price": 1, "goods_num": 1, "goods_name": "\\u9999\\u8549", "goods_id": "10010002"}]}]]></detail>

三、公众号支付下网页内通过 JS-API 调起支付

这一点对应业务流程中的第 7 步。之所以提及它是因为微信官方文档在此给开发者挖了一个坑(至少截至我在写这篇文章时是的),就是在“网页端调起支付API”中关于 JS 的示例代码是采用的 WeixinJSBridge,这在很早以前就是 Deprecated 的“玩意儿”,如今更是已经不可用了。正确的做法是使用 JS-SDK,可以参考微信公众号的 wiki。

使用 JS-SDK 前需要先调用 config,这里也包含一个签名,但注意这个签名与之前微信支付的签名并不相干。其首先需要用微信公众号的 APPID 和 APPKEY 来换取 access_token,然后用该 access_token 调用 JS-SDK 换取 ticket 的接口得到 ticket,最后再使用该 ticket 和用户当前页面的 URI 通过 sha1 运算生成签名。

在此之后,即可调用 wx.chooseWXPay 来调起支付,这里也有一个坑:timestamp。wx.chooseWXPay 中的参数要求 timestamp 是全小写。而微信支付中签名时要求 timestamp 中的“s”是大写。真的是要傻傻分不清了。 

四、异步通知商户支付结果(回调)

最后是关于异步回调,对应业务流程中的第 10 步。在用户支付操作完成后,微信服务器会通过回调的形式告知商户服务器支付结果。回调的地址与【统一下单】中定义的 notify_url 一致。当接收到回调时,首先应验证签名的有效性以保证“来源可靠”,然后可以通过回调中所带的 openid、out_trade_no 等来定位唯一订单。

总结

微信支付还有很多种形式,在业务流程上也不尽相同。不过只要能玩转其中一种,其他的也基本来说能很快实现。另外,支付功能的实现涉及业务流程中的安全性,因此一定要注意理清业务流程,并卡好各个关键结点。以上就是本文的全部内容,希望对大家使用Python开发微信支付能有所帮助。

Python 相关文章推荐
python列表操作实例
Jan 14 Python
Python中使用strip()方法删除字符串中空格的教程
May 20 Python
在Python中处理日期和时间的基本知识点整理汇总
May 22 Python
python判断一个集合是否包含了另外一个集合中所有项的方法
Jun 30 Python
python字符串的方法与操作大全
Jan 30 Python
在numpy矩阵中令小于0的元素改为0的实例
Jan 26 Python
Python:Numpy 求平均向量的实例
Jun 29 Python
python实现opencv+scoket网络实时图传
Mar 20 Python
Python内存映射文件读写方式
Apr 24 Python
全网首秀之Pycharm十大实用技巧(推荐)
Apr 27 Python
如何在scrapy中集成selenium爬取网页的方法
Nov 18 Python
python 爬取腾讯视频评论的实现步骤
Feb 18 Python
Python用模块pytz来转换时区
Aug 19 #Python
教你用python3根据关键词爬取百度百科的内容
Aug 18 #Python
利用Python爬取可用的代理IP
Aug 18 #Python
总结用Pdb库调试Python的方式及常用的命令
Aug 18 #Python
Python实现命令行通讯录实例教程
Aug 18 #Python
Python采用Django开发自己的博客系统
Sep 29 #Python
浅析Python中元祖、列表和字典的区别
Aug 17 #Python
You might like
IIS+fastcgi下PHP运行超时问题的解决办法详解
2013/06/20 PHP
PHP中HTML标签过滤技巧
2014/01/07 PHP
php微信支付之APP支付方法
2015/03/04 PHP
php 指定范围内多个随机数代码实例
2016/07/18 PHP
JS获取复选框的值,并传递到后台的实现方法
2016/05/30 Javascript
微信QQ的二维码登录原理js代码解析
2016/06/23 Javascript
KnockoutJS 3.X API 第四章之事件event绑定
2016/10/10 Javascript
web前端开发upload上传头像js示例代码
2016/10/22 Javascript
AngularJS实现的回到顶部指令功能实例
2017/05/17 Javascript
jQuery实现简单的回到顶部totop功能示例
2017/10/16 jQuery
世界上最短的数字判断js代码
2019/09/09 Javascript
Vuex实现数据增加和删除功能
2019/11/11 Javascript
vue-cli创建的项目中的gitHooks原理解析
2020/02/14 Javascript
node.js中对Event Loop事件循环的理解与应用实例分析
2020/02/14 Javascript
JavaScript ES6 Class类实现原理详解
2020/05/08 Javascript
vue - props 声明数组和对象操作
2020/07/30 Javascript
全面解析Vue中的$nextTick
2020/12/24 Vue.js
Python实现3行代码解简单的一元一次方程
2014/08/18 Python
python+requests+unittest API接口测试实例(详解)
2017/06/10 Python
python整小时 整天时间戳获取算法示例
2019/02/20 Python
Python实现的栈、队列、文件目录遍历操作示例
2019/05/06 Python
python数据化运营的重要意义
2019/11/25 Python
Python不支持 i ++ 语法的原因解析
2020/07/22 Python
浅析HTML5的WebSocket与服务器推送事件
2016/02/19 HTML / CSS
实习期自我鉴定
2013/10/11 职场文书
ktv中秋节活动方案
2014/01/30 职场文书
公司联欢晚会主持词
2014/03/22 职场文书
幼儿发展评估方案
2014/06/11 职场文书
档案工作汇报材料
2014/08/21 职场文书
学校领导班子四风对照检查材料
2014/09/27 职场文书
先进个人推荐材料
2014/12/29 职场文书
行政申诉状范文
2015/05/20 职场文书
python读取并查看npz/npy文件数据以及数据显示方法
2022/04/14 Python
使用 DataAnt 监控 Apache APISIX的原理解析
2022/07/07 Servers
Nginx报404错误的详细解决方法
2022/07/23 Servers
CSS中理解层叠性及权重如何分配
2022/12/24 HTML / CSS