小程序支付
业务流程时序图 官方文档
步骤:
1. Openid
在小程序初次加载的时候就已经获取(详情见 小程序登录)
2. 生成商户订单
1.商品信息由小程序端提供
2.提供支付统一下单接口所需参数
3. 调用支付统一下单API
4. 拿到返回预付单信息并处理
5. 再次签名
案例:
小程序端
test.wxml
<button bind:tap="pay">支付</button>
test.js
Page({ pay:function(){ wx.request({ url: "http://127.0.0.1:8000/pay/", method: "POST", data:{"login_key":wx.getStorageSync("login_key")}, header: { "content-type": "application/json" }, success: function (e) { console.log(e) // 签权调起支付 wx.requestPayment({ \'timeStamp\': e.data.data.timeStamp, \'nonceStr\': e.data.data.nonceStr, \'package\': e.data.data.package, \'signType\': e.data.data.signType, \'paySign\': e.data.data.paySign, \'success\': function (res) { console.log(res,"成功") }, \'fail\': function (res) { console.log("支付失败",res) }, }) } }) }, })
后端 django
wx ├── settings.py # 小程序id,code2Session等配置 ├── wx_login.py # 用于调用code2Session拿到openid等 └── WXBizDataCrypt.py # 获取用户授权信息的解密算法,官方下载
wx/settings.py
AppId="..." 微信分配的小程序id AppSecret="..." code2Session="https://api.weixin.qq.com/sns/jscode2session?appid={}&secret={}&js_code={}&grant_type=authorization_code" pay_mchid =\'...\' 微信配的商户id pay_apikey = \'...\' 商户平台设置的秘钥key
项目/views.py
from rest_framework.views import APIView from rest_framework.response import Response from django.core.cache import cache import hashlib,time import random from app01.wx import settings import requests class Pay(APIView): def post(self,request): param=request.data if param.get("login_key"):
#从redis中拿到小程序端login_key所对应得opendi&session_key值 openid,session_key=cache.get(param.get("login_key")).split("&") self.openid=openid # 获取用户IP # 1.如果是Nginx做的负载就要HTTP_X_FORWARDED_FOR if request.META.get(\'HTTP_X_FORWARDED_FOR\'): self.ip =request.META[\'HTTP_X_FORWARDED_FOR\'] else: # 2.如果没有用Nginx就用REMOTE_ADDR self.ip = request.META[\'REMOTE_ADDR\'] # 调用 生成商户订单 方法 data = self.pay() return Response({"code":200,"msg":"ok","data":data}) else: return Response({"code":200,"msg":"缺少参数"}) # 生成随机字符串 def get_str(self): str_all="1234567890abcdefghjklmasdwery" # 注意 开发活动功能时, 去掉1,i,0,o nonce_str="".join(random.sample(str_all,20)) return nonce_str # 生成订单号 def get_order(self): order_id=str(time.strftime("%Y%m%d%H%M%S")) return order_id # 处理返回预付单方法 def xml_to_dict(self,data): import xml.etree.ElementTree as ET xml_dict={} data_dic=ET.fromstring(data) for item in data_dic: xml_dict[item.tag]=item.text return xml_dict # 获取sign签名方法 def get_sign(self): data_dic = { "nonce_str": self.nonce_str, "out_trade_no": self.out_trade_no, "spbill_create_ip": self.ip, "notify_url": self.notify_url, "openid": self.openid, "body": self.body, "trade_type": "JSAPI", "appid": self.appid, "total_fee": self.total_fee, "mch_id": self.mch_id } sign_str = "&".join([f"{k}={data_dic[k]}" for k in sorted(data_dic)]) sign_str = f"{sign_str}&key={settings.pay_apikey}" md5 = hashlib.md5() md5.update(sign_str.encode("utf-8")) return md5.hexdigest().upper() # 1.生成商户订单 提供 支付统一下单 所需参数 def pay(self): self.appid=settings.AppId # appid 微信分配的小程序ID self.mch_id=settings.pay_mchid # mch_id 微信分配的商户号 self.nonce_str=self.get_str() # 随机字符串 self.body="商品名" # 商品名一般由小程序端传到后端 self.out_trade_no=self.get_order() # 订单号 self.total_fee=1 # 订单总金额 self.spbill_create_ip=self.ip # 用户ip self.notify_url="http://www.baidu.com" # 异步接收微信支付结果通知的回调地址 self.trade_type="JSAPI" # 固定写法 self.sign = self.get_sign() # 获取sign 签名 data=f\'\'\' <xml> <appid>{self.appid}</appid> <body>{ self.body}</body> <mch_id>{self.mch_id}</mch_id> <nonce_str>{self.nonce_str}</nonce_str> <notify_url>{self.notify_url}</notify_url> <openid>{self.openid}</openid> <out_trade_no>{self.out_trade_no}</out_trade_no> <spbill_create_ip>{self.spbill_create_ip}</spbill_create_ip> <total_fee>{self.total_fee}</total_fee> <trade_type>{self.trade_type}</trade_type> <sign>{self.sign}</sign> </xml> \'\'\' # 2.支付统一下单接口 url="https://api.mch.weixin.qq.com/pay/unifiedorder" # 3.返回预付单信息 response=requests.post(url,data.encode("utf-8"),headers={"content-type":"application/xml"}) res_data=self.xml_to_dict(response.content) data=self.two_sign(res_data["prepay_id"]) # prepay_id 预支付订单回话标识 return data # 4.将组合数据再次签名 def two_sign(self,prepay_id): timeStamp=str(int(time.time())) nonceStr=self.get_str() data_dict={ "appId":settings.AppId, "timeStamp":timeStamp, "nonceStr":nonceStr, "package":f"prepay_id={prepay_id}", "signType":"MD5" } sign_str = "&".join([f"{k}={data_dict[k]}" for k in sorted(data_dict)]) sign_str = f"{sign_str}&key={settings.pay_apikey}" md5 = hashlib.md5() md5.update(sign_str.encode("utf-8")) sign=md5.hexdigest().upper() data_dict["paySign"]=sign data_dict.pop("appId") # 5.返回支付参数到小程序端,小程序端获取所需参数向微信服务器发送 调起支付 方法 return data_dict
项目/urls.py
url(r\'^pay/\',views.Pay.as_view())