egg微信小程序支付(服务商)插件封装
下单
通过下单获取prepay_id,然后返回给小程序发起支付
若是服务商,mch_id:传入服务的商户号;sub_mch_id:传入子商户的商户号,算法签名的秘钥是服务商的秘钥。
'use strict';
const Service = require('egg').Service;
class wxPayment extends Service {
// 小程序支付
async unifiedorder(data) {
const { ctx } = this;
const wxpayUitls = ctx.service.payment.helper.paymentUitls;
// 微信支付调用ip
const ip = ctx.request.ip.replace(/::ffff:/, '');
const params = {
// 小程序appid
appid: this.config.wxampPayConfig.appid,
// 商户号
mch_id: this.config.wechatPayConfig.mch_id,
// 子商户号
sub_mch_id: this.config.wechatPayConfig.sub_mch_id,
// 32位以内随机字符串
nonce_str: wxpayUitls.createNonceStr(),
// 商品描述
body: data.body,
// 商户订单号,32个字符内
out_trade_no: wxpayUitls.getTradeNo(),
// 金额,分;正式上线时启用
// total_fee: wxpayUitls.regYuanToFen(data.total, 100),
total_fee: 1,
// 终端ip
spbill_create_ip: ip,
// 通知地址
notify_url: 'http://baidu.com',
// 交易类型
trade_type: 'JSAPI',
// 用户标识
openid: data.openID,
};
// 算法签名,传入参数及服务商商户的秘钥
params.sign = wxpayUitls.getSign(params, this.config.wechatPayConfig.wxPaykey);
const url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
const res = await ctx.curl(url, {
method: 'POST',
dataType: 'text/xml', // 注意接口数据类型
data: wxpayUitls.createXML(params), // 动态生成的xml
});
const result = await wxpayUitls.parserXML(res.data.toString());
// 返回的参数
const r = {
appId: this.config.wxampPayConfig.appid, // 当前小程序的appid
timeStamp: wxpayUitls.createTimeStamp(), // 注意类型String
nonceStr: result.xml.nonce_str,
package: 'prepay_id=' + result.xml.prepay_id,
signType: 'MD5',
};
r.paySign = wxpayUitls.getSign(r, this.config.wechatPayConfig.wxPaykey);
r.tradeNo = params.out_trade_no;
return r;
}
}
module.exports = wxPayment;
helper.js
'use strict';
// eslint-disable-next-line strict
const QueryString = require('querystring');
const Crypto = require('crypto');
const XmlToJs = require('xml2js');
exports.paymentUitls = {
// 订单号
getTradeNo: () => {
let No = '';
for (let i = 0; i < 5; i++) {
No += Math.floor(Math.random() * 10);
}
No = new Date().getTime() + No;
return No;
},
/**
* 元的转化
* @param yuan {number} 金额
* @param digit {number} 放大倍数,如果元转分则为100
* */
regYuanToFen(yuan, digit) {
let m = 0;
const s1 = yuan.toString();
const s2 = digit.toString();
try {
m += s1.split('.')[1].length;
m += s2.split('.')[1].length;
} catch (e) {
// console.log(e);
}
return Number(s1.replace('.', '')) * Number(s2.replace('.', '')) / Math.pow(10, m);
},
// 时间戳产生函数
createTimeStamp() {
return parseInt(new Date().getTime() / 1000) + '';
},
// 生成随机的NonceStr,随机字符串,不长于32位。
createNonceStr: () => {
return Math.random()
.toString(36)
.substr(2, 15);
},
// 获取微信支付的签名
/**
* @param signParams {object} 请求参数的集合
* @param key {string} 商户号
* */
getSign: (signParams, key) => {
// 按 key 值的ascll 排序
let keys = Object.keys(signParams);
keys = keys.sort();
const newArguments = {};
// eslint-disable-next-line no-unused-vars
keys.forEach(function(val, key) {
if (signParams[val]) {
newArguments[val] = signParams[val];
}
});
const string = QueryString.stringify(newArguments) + '&key=' + key;
// 生成签名
return Crypto.createHash('md5')
.update(QueryString.unescape(string), 'utf8')
.digest('hex')
.toUpperCase();
},
// 根据对象生成xml
createXML: params => {
const builder = new XmlToJs.Builder();
return builder.buildObject(params);
},
// 解析xml
parserXML: xml => {
const Parser = new XmlToJs.Parser({ explicitArray: false, ignoreAttrs: false });
return new Promise((resolve, reject) => {
Parser.parseString(xml, function(err, result) {
if (err) reject(err);
resolve(result);
});
});
},
};
微信小程序获取返回的prepay_id,发起支付
// 发起支付
wx.requestPayment({
timeStamp: r.data.result.timeStamp,
nonceStr: r.data.result.nonceStr,
package: r.data.result.package,
signType: 'MD5',
paySign: r.data.result.paySign,
success(res) {
console.log('成功', res)
},
fail(res) {
}
})
欢迎关注公众号【无聊猿】,共同学习探讨

egg微信小程序支付(服务商)插件封装的更多相关文章
- 微信小程序支付及退款流程详解
微信小程序的支付和退款流程 近期在做微信小程序时,涉及到了小程序的支付和退款流程,所以也大概的将这方面的东西看了一个遍,就在这篇博客里总结一下. 首先说明一下,微信小程序支付的主要逻辑集中在后端,前端 ...
- php对接微信小程序支付
前言:这里我就假装你已经注册了微信小程序,并且基本的配置都已经好了.注: 个人注册小程序不支持微信支付,所以我还是假装你是企业或者个体工商户的微信小程序,其他的商户号注册,二者绑定,授权,支付开通,就 ...
- 微信小程序支付接入注意点
一.微信支付后台服务器部署 服务器采用ubuntu16.04 + php7.0 + apache2.0. 微信支付后台服务使用了curl 和 samplexml ,因此php.ini配置中必须开启这两 ...
- 微信小程序支付开发之申请退款
微信小程序支付跟微信公众号支付类似,这里不另做记录,如果没有开发过支付,可以查看我关于微信支付的文章 重点记录微信小程序申请退款开发过程中遇到一些坑. 退款接口比支付接口接口多了一个 双向证书 证书介 ...
- 微信小程序支付接入实战
1. 微信小程序支付接入实战 1.1. 需求 最近接到一个小程序微信支付的需求,需要我写后台支持,本着能不自己写就不自己写的cv原则,在网上找到了些第三方程序,经过尝试后,最后决定了这不要脸作者的 ...
- Java实现微信小程序支付(完整版)
在开发微信小程序支付的功能前,我们先熟悉下微信小程序支付的业务流程图: 不熟悉流程的建议还是仔细阅读微信官方的开发者文档. 一,准备工作 事先需要申请企业版小程序,并开通“微信支付”(即商户功能).并 ...
- php 微信小程序支付
php 微信小程序支付 直接贴代码: 前端测试按钮wxml: <view class="container"> <text class="name&qu ...
- .Net后台实现微信小程序支付
最近一直再研究微信支付和支付宝支付,官方支付文档中一直在讲与第三方支付打交道的原理,却没有介绍我们自己项目中的APP与后台该怎么交互(哈哈,人家也没必要介绍这一块).拜读了官方文档和前辈们的佳作,自己 ...
- 微信小程序支付(企业支付给用户零钱)
内容摘要:本案例客户端支付方式为微信小程序支付(JSAPI).商户运营一段时间后,在微信商户平台开通企业支付服务后,即可调用微信支付提供的企业付款接口将佣金等金额通过微信零钱返现给C端用户零钱. 服务 ...
随机推荐
- Java-方法的递归调用
方法的递归是指在一个方法的内部调用自身的过程.递归必须要有结束条件,否则将陷入无限递归的状态,永远无法结束调用. 代码 public class Example24{ public static vo ...
- 跨域 CORS 详解 (转)
CORS是一个W3C标准,全称是"跨域资源共享"(Cross-origin resource sharing). 它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从 ...
- uniap tab list 滑动
效果如下 <uni-popup ref="bankListAll" type="dialog"> <ty-mutiple-select :mu ...
- CSS 圆角框
转载请注明来源:https://www.cnblogs.com/hookjc/ 其实这种圆角框是靠一个个容器堆砌而成的,每一个容器的宽度不同,这个宽度是由margin外边距来实现的,如:margin: ...
- CSS 3D的魅力
用户1093975发表于Web项目聚集地订阅 151 在这篇文章中: 前言: demo1 demo2 结语: 本文介绍了CSS来实现3D效果,并且有详细代码和解释.建议大家只字不差的阅读.本文的作者是 ...
- go基础——基本数据类型
GO语言的数据类型: /* GO语言的数据类型: 1.基本数据类型: 布尔类型:true,false 数值类型:整数,浮点,复数complex 字符串:string 2.复合数据类型 array,sl ...
- shell脚本命令
http://man.linuxde.net/shell-script 从键盘或文件中获取标准输入:read命令 文件的描述符和重定向 数组.关联数组和别名的使用 函数的定义.执行.传参和递归函数 ...
- find+grep+正则表达式
目录 find+grep+正则表达式 1.find 2.grep 3.正则表达式 find+grep+正则表达式 1.find 根据文件的名称或者属性查找文件. # 自己在 /root/adc目录下长 ...
- 字符编码和Python代码操作文件
字符编码和Python代码操作文件 读写模式之a模式 # a模式 只追加模式 # 路径不存在:自动创建 with open(r'a.txt','a',encoding='utf8') as f: pa ...
- 前端表单标签form 及 简单应用
今日内容 form 表单(重点) 后端框架之 flask 简介 内容详细 form 表单 1.作用 form 表单可以在前端获取用户输入的数据并发送给后端(服务端) 2.input 标签 获取用户数据 ...