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端用户零钱. 服务 ...
随机推荐
- SpringBoot + Log4j2使用配置 (转)
前言 后台程序开发及上线时,一般都会用到Log信息打印及Log日志记录,开发时通过Log信息打印可以快速的定位问题所在,帮助我们快捷开发.程序上线后如遇到Bug或错误,此时则需要日志记录来查找发现问题 ...
- UIScrollView的代理方法(delegate)
// ViewController.m // 05-UIScrollView的代理(delegate) #import "ViewController.h" @interface ...
- 如何使Label显示时,一行顶部居中,两行靠左显示----董鑫
有时我们会碰到这种情况,一个要根据内容显示一行还是两行,一行时还要靠着顶部再居中,比如下面 最左边的名称,要求是靠上的,如果按照正常的方式写的话,可能一行的话就会出现居中显示了,不会顶着头部显示. 我 ...
- 03.python语法入门--注释、变量
注释 """多行注释加在整个py文件的开头用来对该py文件进行一个整体性的说明通常一个py中只有一个多行注释"""# 这是一行贼溜的代码p ...
- 基于TMS320C6670的软件无线电核心板
一.板卡概述 北京太速科技自主研发的TMS320C6670核心板,采用TI KeyStone系列的四核定点/浮点DSP TMS320C6670作主处理器.板卡引出处理器的全部信号引脚,便于客户二次开发 ...
- Zookeeper开源客户端Curator之创建会话
前面Zookeeper的链接使用的都是其提供的原生代码,实际开发过程中非常底层的细节开发工作如连接重连,反复注册等耗费开发人员大量的工作精力并且重复工作.而开源客户端Curator的出现解决了该类问题 ...
- RocketMQ的invokeSync call timeout异常的解决办法
缘起 在RocketMQ客户端的DefaultMQPushConsumer的start方法被执行时,时不时会报出invokeSync call timeout异常,如下: Caused by: jav ...
- Learning Schedules
近期目标 1. 争取搞定小论文 2. Java SE 学习 3. 剑指Offer每日1-2题
- 3款自助型BI分析平台功能盘点,帮助你预测商业发展方向
在快速发展的今天,商业智能BI已经不同于传统的商业智能BI,商业智能BI已经逐渐转变为自助和业务主导的模式,自助BI分析平台应运而生,自助BI分析平台逐渐成为许多企业的选择. 自助式BI分析平台与传统 ...
- Smartbi代替Alteryx+Tableau,用1份投入如何获得2份回报?
Smartbi是国内一家知名的BI厂商,Alteryx.Tableau是国外两款重要的BI工具,它们都是在BI领域内提供特定的功能,以满足企业的数据分析需求.那么,对于用户来说,在选择BI工具的时候要 ...