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端用户零钱. 服务 ...
随机推荐
- HBase安装教程
一.版本介绍 linux : CentOS7 Hadoop : 2.7.6 zookeeper : 3.4.6 hbase : 1.4.6 jdk : jdk1.8.0_171 三个节点的主机名分别为 ...
- VC 窗口置顶并激活为当前窗体
转载请注明来源:https://www.cnblogs.com/hookjc/ if (this != GetForegroundWindow()) ...
- Solution -「多校联训」古老的序列问题
\(\mathcal{Description}\) Link. 给定序列 \(\{a_n\}\),和 \(q\) 次形如 \([L,R]\) 的询问,每次回答 \[\sum_{[l,r]\su ...
- Solution -「洛谷 P4389」付公主的背包
\(\mathcal{Description}\) Link. 容量为 \(n\),\(m\) 种物品的无限背包,求凑出每种容量的方案数,对 \(998244353\) 取模. \(n,m ...
- CoaXPress 接口相机的控制方法--2
接上一篇 <CoaXPress 接口相机的控制方法--1> https://www.cnblogs.com/xingce/p/15902246.html 这里再介绍一下具体是如何完成相机寄 ...
- suse 12 二进制部署 Kubernetets 1.19.7 - 第04章 - 部署docker服务
文章目录 1.4.部署docker 1.4.0.下载docker二进制文件 1.4.1.配置docker镜像加速 1.4.2.配置docker为systemctl管理 1.4.3.启动docker服务 ...
- RPC 技术及其框架 Sekiro 在爬虫逆向中的应用,加密数据一把梭!
什么是 RPC RPC,英文 RangPaCong,中文让爬虫,旨在为爬虫开路,秒杀一切,让爬虫畅通无阻! 开个玩笑,实际上 RPC 为远程过程调用,全称 Remote Procedure Call, ...
- vue实例中的watch属性
watch 就是监听,当数据发生改变的时候就执行 data:{ num1:1, num2:2 }, watch:{ num1(newValue){ }, num2(newValue,oldValue) ...
- 学着去看开 你不是一个low程序员
http://www.cocoachina.com/programmer/20160627/16835.html 看完这篇文章 感谢作者
- Windows禁用445端口
今天来公司有好多电脑感染了0day病毒, 写个脚本,一键执行禁用445,135-139端口.Windows7测试没有问题. reg add "HKEY_LOCAL_MACHINE\SYSTE ...