微信小程序支付/微信小程序+node服务 支付爬坑 v2
2021-03-16 本文已影响0人
莫伊剑客
一、前端
小程序登录及支付请求和唤起支付界面
// app.js
const {request} = require('./assets/js/utils')
// app.js
App({
onLaunch() {
// 展示本地存储能力
const logs = wx.getStorageSync('logs') || []
logs.unshift(Date.now())
wx.setStorageSync('logs', logs)
// 登录
wx.login({
success: res => {
console.log(this)
// 发送 res.code 到后台换取 openId, sessionKey, unionId
if (res.code) {
request('/wx/login', {code: res.code}, 'POST').then(result => {
console.log(result)
this.globalData = {...this.globalData, ...result.data.data}
})
}
}
})
},
globalData: {
userInfo: null
}
})
wxml
<!--pages/payment/index.wxml-->
<view class="container">
<block wx:if="{{!hasUserInfo}}">
<button wx:if="{{canIUseGetUserProfile}}" bindtap="getUserProfile"> 获取头像昵称 </button>
<button wx:elif="{{canIUse}}" open-type="getUserInfo" bindgetuserinfo="getUserInfo"> 获取头像昵称 </button>
<view wx:else> 请使用1.4.4及以上版本基础库 </view>
</block>
<button bindtap="payment">点击进行支付</button>
</view>
payment/index.js
// pages/payment/index.js
// 获取应用实例
const app = getApp()
const {request} = require('../../assets/js/utils')
Page({
/**
* 页面的初始数据
*/
data: {
hasUserInfo: false,
canIUse: wx.canIUse('button.open-type.getUserInfo'),
canIUseGetUserProfile: false,
userInfo: ''
},
// 第一种获取用户信息方法
getUserProfile(e) {
// 推荐使用wx.getUserProfile获取用户信息,开发者每次通过该接口获取用户个人信息均需用户确认,开发者妥善保管用户快速填写的头像昵称,避免重复弹窗
wx.getUserProfile({
desc: '展示用户信息', // 声明获取用户个人信息后的用途,后续会展示在弹窗中,请谨慎填写
success: (res) => {
console.log(res)
// 存储用户信息
wx.setStorageSync('userInfo', JSON.stringify(res.userInfo));
this.setData({
userInfo: res.userInfo,
hasUserInfo: true
})
}
})
},
// 第二种获取用户信息方法
getUserInfo(e) {
// 不推荐使用getUserInfo获取用户信息,预计自2021年4月13日起,getUserInfo将不再弹出弹窗,并直接返回匿名的用户个人信息
console.log(e)
// 存储用户信息
wx.setStorageSync('userInfo', JSON.stringify(e.detail.userInfo));
this.setData({
userInfo: e.detail.userInfo,
hasUserInfo: true
})
},
// 支付事件
payment() {
const params = {
openid: app.globalData.openid,
total: 0.01,
orderCode: new Date().toLocaleString().replace(/[^\d]/g, '')
}
request('/wx/pay/unifiedorder', params, "POST").then(res => {
const {nonceStr, timeStamp, signType, paySign} = res.data.data
const pack = res.data.data.package
wx.requestPayment({
nonceStr,
timeStamp,
package: pack,
signType,
paySign,
success(res) {
console.log(res)
},
fail(res) {
console.log(res)
},
complete(res){
console.log(res)
}
})
})
},
pageInit() {
if (wx.getUserProfile) {
this.setData({
canIUseGetUserProfile: true
})
}
// 如果缓存不存在用户信息时暂时获取用户信息按钮
if (this.data.userInfo || wx.getStorageSync('userInfo')) {
const userInfo = this.data.userInfo || JSON.parse(wx.getStorageSync('userInfo'))
this.setData({
hasUserInfo: true,
userInfo
})
}
}
util.js request封装
exports.request = function (requestMapping, data, requestWay, contentType) {
wx.showLoading({
title: '请稍后',
})
return new Promise(function (resolve, reject) {
console.log('请求中。。。。。')
wx.request({
url: baseUrl + requestMapping,
data: data,
header: {
'content-type': contentType || "application/x-www-form-urlencoded" // 默认值
},
timeout: 3000,
method: requestWay,
success(res) {
//console.log(res)
if (res.data.success == false || res.data.statusCode == 404) {
reject(res)
} else {
resolve(res)
}
},
fail: (e) => {
wx.showToast({
title: '连接失败',
icon: 'none'
})
},
complete: () => {
wx.hideLoading()
}
})
})
}
二、node服务端
我这里使用的是eggjs,使用koa,express同理
router.js
'use strict';
/**
* @param {Egg.Application} app - egg application
*/
module.exports = app => {
const { router, controller } = app;
router.post('/wx/login', controller.wx.login);
router.post('/wx/pay/unifiedorder', controller.wx.unifiedorder);
};
Controller
'use strict';
const { Controller } = require('egg');
class WxController extends Controller {
// 登录
async login() {
const { ctx } = this;
ctx.body = await ctx.service.wx.login()
}
// 统一下单
async unifiedorder(){
const { ctx } = this;
ctx.body = await ctx.service.wx.unifiedorder()
}
}
module.exports = WxController;
Service
'use strict';
const { Service } = require('egg');
const appid = 'wx5591*****e47'; // 小程序id
const secret = '670c7*****1a7de971c3'; // 小程序secret
const mch_id = '1607****01'; // 商户id
const mch_key = '52052*****2522ghi52352'; // 商户key
class Wx extends Service {
// 登录获取openid,session_key并返回给前端
async login() {
const { ctx } = this;
const { code } = ctx.request.body;
const url = `https://api.weixin.qq.com/sns/jscode2session?appid=${appid}&secret=${secret}&js_code=${code}&grant_type=authorization_code`;
const res = await ctx.curl(url, {
method: 'POST',
dataType: 'json',
});
console.log(res, res.data);
return ctx.helper.success({ ctx, res: res.data });
}
async unifiedorder() {
const { ctx } = this;
// 自己封装的微信支付工具
const wxpayUitls = ctx.helper.wxpayUitls;
const { openid, total, orderCode } = ctx.request.body;
const url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
// params 参数参考地址:https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=9_1&index=1
// 所有能使用到到的参数都罗列了,根据实际需求自己选择
const params = {
appid,
mch_id,
// 自定义参数,可以为终端设备号(门店号或收银设备ID),PC网页或公众号内支付可以传"WEB"
// device_info: '',
// 随机字符串,长度要求在32位以内。推荐随机数生成算法
nonce_str: wxpayUitls.createNonceStr(),
// 通过签名算法计算得出的签名值,详见签名生成算法
sign: '',
// 签名类型,默认为MD5,支持HMAC-SHA256和MD5。
// sign_type: 'MD5',
// 商品简单描述,该字段请按照规范传递,具体请见参数规定
body: '支付测试',
// 商品详细描述,对于使用单品优惠的商户,该字段必须按照规范上传,详见“单品优惠参数说明”
// detail: '',
// 附加数据,在查询API和支付通知中原样返回,可作为自定义参数使用。
// attach: '',
// 商户系统内部订单号,要求32个字符内,只能是数字、大小写字母_-|*且在同一个商户号下唯一。详见商户订单号
out_trade_no: orderCode,
// 符合ISO 4217标准的三位字母代码,默认人民币:CNY,详细列表请参见货币类型
fee_type: 'CNY',
// 订单总金额,单位为分,详见支付金额
total_fee: wxpayUitls.getmoney(total),
// 支持IPV4和IPV6两种格式的IP地址。调用微信支付API的机器IP
spbill_create_ip: '192.168.43.187' || ctx.request.ip,
// 订单生成时间,格式为yyyyMMddHHmmss,如2009年12月25日9点10分10秒表示为20091225091010。其他详见时间规则
// time_start: '',
// 订单失效时间,格式为yyyyMMddHHmmss,如2009年12月27日9点10分10秒表示为20091227091010。
// 订单失效时间是针对订单号而言的,由于在请求支付的时候有一个必传参数prepay_id只有两小时的有效期,
// 所以在重入时间超过2小时的时候需要重新请求下单接口获取新的prepay_id。其他详见时间规则
// 建议:最短失效时间间隔大于1分钟
// time_expire: '',
// 订单优惠标记,使用代金券或立减优惠功能时需要的参数,说明详见代金券或立减优惠
// goods_tag: '',
// 异步接收微信支付结果通知的回调地址,通知url必须为外网可访问的url,不能携带参数。
notify_url: 'http://cxbly.top',
// 小程序取值如下:JSAPI,详细说明见参数规定
trade_type: 'JSAPI',
// trade_type=NATIVE时,此参数必传。此参数为二维码中包含的商品ID,商户自行定义。
// product_id: '',
// 上传此参数no_credit--可限制用户不能使用信用卡支付
// limit_pay: 'no_credit',
// trade_type=JSAPI,此参数必传,用户在商户appid下的唯一标识。openid如何获取,可参考【获取openid】。
openid,
// Y,传入Y时,支付成功消息和支付详情页将出现开票入口。
// 需要在微信支付商户平台或微信公众平台开通电子发票功能,传此字段才可生效
// receipt: '',
// Y-是,需要分账
// N-否,不分账
// 字母要求大写,不传默认不分账
// profit_sharing: '',
// 该字段常用于线下活动时的场景信息上报,支持上报实际门店信息,
// 商户也可以按需求自己上报相关信息。该字段为JSON对象数据,
// 对象格式为{"store_info":{"id": "门店ID","name": "名称","area_code": "编码","address": "地址" }} ,
// 字段详细说明请点击行前的+展开
// scene_info: ''
};
//动态生成统一下单需要的签名MD5字符串
params.sign = wxpayUitls.createSign(params, mch_key);
const res = await ctx.curl(url, {
method: 'POST',
dataType: 'text/xml', // 注意接口数据类型
data: wxpayUitls.createXML(params) //动态生成的xml
});
// console.log('统一下单结果:', res, 'data:', res.data.toString());
// 解析统一下单后返回的xml
const {xml} = await wxpayUitls.parserXML(res.data.toString())
const r={
appId:appid,
timeStamp:Date.now().toString(), //注意类型String
nonceStr:xml.nonce_str,
package:'prepay_id='+xml.prepay_id,
signType:'MD5'
}
// paySign:wxpayUitls.createSign(r,mch_key) 动态生成wx.requestPayment使用的签名paySign MD5字符串
return ctx.helper.success({ctx,res:{...r,paySign:wxpayUitls.createSign(r,mch_key)}})
}
}
module.exports = Wx;
注意事项
- 统一下单 签名规则
将统一下单所需的所有的参数进行签名计算(空值,sgin除外)- wx.requestPayment 签名规则
参与签名计算的参数有:appId,timeStamp,nonceStr,package,signType,注意 timeStamp转字符串。
wxpayUitls工具
const crypto = require('crypto');
const Xml2 = require('xml2js');
const MD5 = require('md5');
exports.wxpayUitls = {
//把金额转为分
getmoney: function(money) {
return parseFloat(money) * 100;
},
// 随机字符串产生函数
createNonceStr: function() {
return Math.random()
.toString(36)
.substr(2, 15)
.toUpperCase();
},
// 时间戳产生函数
createTimeStamp: function() {
return parseInt(new Date().getTime() / 1000) + '';
},
// 动态生成签名方法
createSign: function(params, mchkey) {
let str = raw(params);
str += `&key=${mchkey}`;
console.log('str=====', str);
// 第一种方法
// return MD5(str).toUpperCase()
// 第二种方法
return crypto.createHash('md5')
.update(str, 'utf8')
.digest('hex')
.toUpperCase();
},
// 根据对象生成xml
createXML: function(params) {
// let xml = `<xml>`;
// Object.keys(params)
// .map(key => {
// xml += `<${key}>${params[ key ]}</${key}>`;
// });
// xml += `</xml>`;
// return xml;
const builder = new Xml2.Builder();
return builder.buildObject(params);
},
// 解析xml
parserXML: function(xml) {
const Parser = new Xml2.Parser({ explicitArray: false, ignoreAttrs: false });
return new Promise((resolve, reject) => {
Parser.parseString(xml, function(err, result) {
if (err) reject(err);
resolve(result);
});
});
}
};
订单查询,关闭,申退,查询退款等后续流程 等爬完之后再更新!