最近发现某些小程序有了提现功能,原来小程序是不支持提现的,所以当初实现方法是打算让用户去关注公众号,再从公众号提现,当然前提要公众号跟小程序使用同一的unionid来标记唯一用户,既然现在支持小程序直接提现了,那就不再使用关注公众号这种影响用户体验的方法了
小程序提现使用的是企业付款接口(这什么烂文档,这个接口印象中已经出现很久了,因为没有在小程序开发中就没太留意,产生了小程序不能做提现的错觉,虽然也不可能单独出现在小程序文档中,你好歹也提提吧),首先申请商户号,现在使用企业付款接口也需要开通条件
开通商户号并达到开通条件后,就是接口的使用了,无非就是一个http请求罢了,根据之前做小程序支付调用统一下单接口的经验,这里也理所当然的使用python的http请求库requests,这里先附上统一下单接口的代码(请注意使用的库)
#python 3.5.1
from xml.etree.ElementTree import Element, tostring
import requests,hashlib,random
url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'
data = {
'appid': APPID,#申请商户号后提供
'mch_id': MCH_ID,#申请商户号后提供
'nonce_str': get_nonce_str(),
'body': '测试',
'out_trade_no': get_out_trade_no(),
'total_fee': 1,
'spbill_create_ip': get_host(), #服务器ip
'notify_url': 'http://www.weixin.qq.com/wxpay/pay.php',#暂时没用到,微信以http请求方式将支付结果发到这个请求地址上
'trade_type': 'JSAPI',
'openid': openid,
}
# 签名算法
data['sign'] = wx_sign(data)
xml_data = bytes.decode(tostring(dict_to_xml_data(data)))
response = requests.post(url=url, data=xml_data)
----------------------------------------------------------------
# 签名算法
def wx_sign(proto_data):
'''根据数组的key排序,并拼接加密'''
a = ''
for key in (sorted(proto_data.keys())):
if a == '':
a = key + '=' + str(proto_data[key])
else:
a = a + '&' + key + '=' + str(proto_data[key])
a = a + '&key=' + KEY
# md5加密
a = hashlib.md5(a.encode('utf-8')).hexdigest().upper()
return a
def dict_to_xml_data(data):
# dict to xml
elem = Element('xml')
for key, value in data.items():
child = Element(key)
child.text = str(value)
elem.append(child)
return elem
def get_out_trade_no():
'''商户订单号'''
return time.strftime("%Y%m%d%H%M%S", time.localtime()) + str(uuid.uuid1())[:8]
def get_nonce_str():
'''随机字符串'''
return ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(32))
理所当然的企业付款接口的写法,requests.post()有一个cert的参数,就是微信提供的私钥和公钥证书,据说requests还不支持p12的证书
url = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers'
data = {
'mch_appid': APPID,
'mchid': MCH_ID,
'nonce_str': get_nonce_str(),
'partner_trade_no': get_out_trade_no(),
'check_name': 'NO_CHECK',
'amount': 100,
'spbill_create_ip': get_host(),
'desc': u'测试',
'openid': openid,
}
data['sign'] = wx_sign(data)
xml_data = bytes.decode(tostring(dict_to_xml_data(data)))
#logger.debug('xml_data=======' + xml_data)
response = requests.post(url=url, data=xml_data,cert=('/home/pem/apiclient_cert.pem','/home/pem/apiclient_key.pem'))
运行后报【参数错误:描述信息存在非UTF8字符】,指的就是我desc写的内容有问题罗,尝试着把desc内容改成英文字母和数字,运行成功在手机微信的微信支付上可以看到入账信息,但是我这里的入账详情就是要写中文啊
把发送前的xml打印出来看一下发现在xml中desc的内容为
测试
,接着把
xml_data = bytes.decode(tostring(dict_to_xml_data(data)))
改成
xml_data = bytes.decode(tostring(dict_to_xml_data(data),encoding='utf8'))
思路是将body换成utf8编码再发送,报了下面的错误
File "/opt/rh/rh-python35/root/usr/lib64/python3.5/http/client.py", line 1127, in _send_request
body = body.encode('iso-8859-1')
UnicodeEncodeError: 'latin-1' codec can't encode characters in position 375-376: ordinal not in range(256)
意思是说requests库发出请求body需以iso-8859来解码,根据错误信息跳到requests源码,body默认的编码为iso 8895-1,没有找到发送请求时改变编码的方式(如果发现requests能修改的小伙伴希望告知),只有修改response返回的编码方式
# RFC 2616 Section 3.7.1 says that text default has a
# default charset of iso-8859-1.
body = body.encode('iso-8859-1')
关键时刻只能曲线救国了,决定换一个http的请求库,在使用了urllib3发现有一样的问题,原来requests库就是在urllib3上封装的,最后使用urllib成功实现了,使用urllib库的棘手地方在于发送请求时带上证书,实现代码如下
url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'
data = {
'mch_appid': APPID,
'mchid': MCH_ID,
'nonce_str': get_nonce_str(),
'partner_trade_no': get_out_trade_no(),
'check_name': 'NO_CHECK',
'amount': 100,
'spbill_create_ip': get_host(),
'desc': u'测试',
'openid': openid,
}
# 签名算法
data['sign'] = wx_sign(data)
xml_data = tostring(dict_to_xml_data(data), encoding='utf8') #这个地方加了utf8编码,并且返回byte数据
from urllib import request as res
import ssl
context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)
context.load_cert_chain(certfile='/home/pem/apiclient_cert.pem', keyfile='/home/pem/apiclient_key.pem')
response = res.urlopen(url=url, data=xml_data, context=context)