文档章节

NodeJS微信支付签名代码

cooleone
 cooleone
发布于 2017/07/25 16:16
字数 563
阅读 5
收藏 0
//统一下单签名参数顺序
let ss1Str = ['appid', 'body', 'mch_id', 'nonce_str', 'notify_url', 'out_trade_no', 'spbill_create_ip', 'total_fee', 'trade_type'];
//支付签名参数顺序(公众号)
let ss2Str = ['appId', 'nonceStr', 'package', 'signType', 'timeStamp'];
//支付签名参数顺序(APP)
let ss3Str = ['appid', 'noncestr', 'package', 'partnerid', 'prepayid', 'timestamp'];
/**
 * 微信签名
 */
router.get('/wxpay/sign2', function (req, res) {
    let statementId = req.queryData.statementId;
    let userId = req.queryData.userId;
    let ids = statementId.split(',');
    Statement.findAll({
        where: {
            id: {$in: ids}
        }
    }).then((list) => {
        let sum = 0;
        let ids = '';
        let contractId = '';
        let remark = '';
        list.forEach(function (st) {
            if (st.state == 0) {
                sum += st.sum;
                ids += st.id + ',';
                contractId = st.contractId;
                if (st.num > 0) {
                    remark += '第' + st.num + '期' + st.remark + ',';
                } else if (st.num == 0) {
                    remark += st.remark + ',';
                } else if (st.num == -1) {
                    remark += st.title + ' ' + st.remark + ',';
                }
            }
        });

        if (sum == 0 || ids.length == 0) {
            res.json({result: false, message: '未找到可以支付的记录'});
        } else {
            OrderPay.build({
                sum: sum,
                type: 1,
                contractId: contractId,
                statementIds: ids,
                payerId: userId,
                remark: remark
            }).save().then(function (orderPay) {
                let ipAddress = getClientIp(req);
                console.log('ipAddress===' + ipAddress);
                ipAddress = '218.205.52.3';
                let cost = parseInt(orderPay.sum*100);
                console.log('cost=' + cost);
                //调用微信接口下单
                let map = initMap();
                map['notify_url'] = config.jhkj.weixin.notify_url; //通知地址
                map['spbill_create_ip'] = ipAddress;
                map['body'] = config.jhkj.weixin.appname + '-' + orderPay.remark; //商品描述
                map['out_trade_no'] = orderPay.id; //商户订单号
                map['total_fee'] = cost; //总金额

                let signData = signup(map, ss1Str);

                unifiedorder(signData, function(body) {
                    xml2js.parseString(body, function(err, result) {
                        if (result.xml.return_code[0] == 'FAIL') {
                            res.json({result: false, message: result.xml.return_msg[0]});
                            return;
                        } else if (result.xml.result_code[0] == 'FAIL') {
                            if (result.xml.err_code[0] == 'OUT_TRADE_NO_USED') { //商户订单号重复
                            } else if (result.xml.err_code[0] == 'ORDERPAID') { //商户订单已支付
                            }
                            res.json({result: false, message: result.xml.err_code_des[0]});
                        } else {
                            //统一下单成功,返回签名信息
                            let prepayId = result.xml.prepay_id[0];
                            let map2 = {
                                "appid": config.jhkj.weixin.appid
                                ,"timestamp": new Date().getTime().toString().substring(0, 10)
                                ,"noncestr": uuid.v1().replace(/\-/g,"")
                                ,"package": "Sign=WXPay"
                                ,"prepayid": prepayId
                                ,"partnerid": config.jhkj.weixin.mch_id
                            };
                            map2.paySign = getSignStr(map2, ss3Str);
                            res.json({result: true, payJSON: map2, prepayId: prepayId, out_trade_no: orderPay.id });
                        }
                    });
                });
            });
        }
    });
});

/**
 * 微信统一下单
 * @param data
 * @param callback
 */
function unifiedorder(data, callback) {
    let opt = {
        method: 'POST'
        ,host: 'api.mch.weixin.qq.com'
        ,port: 443
        ,path: '/pay/unifiedorder'
        ,headers: {
            'Content-Type': 'text/xml;charset=utf-8'
            ,'Content-Length': iconv.encode(data, 'utf8').length
        }
    };

    let reqVideo = https.request(opt, function(serverFeedback) {
        serverFeedback.setEncoding('utf-8');
        console.log('getVideo:' + serverFeedback.statusCode);
        if (serverFeedback.statusCode == 200) {
            let body = "";
            serverFeedback
                .on('data', function (data) { body += data; })
                .on('end', function () {
                    console.log(body);
                    callback(body);
                });
        } else {
            console.log(serverFeedback.statusCode);
        }
    });
    reqVideo.write(iconv.encode(data, 'utf8'));
    reqVideo.end();
}

/**
 * 生成签名字符串
 * @param map
 * @param arrStr
 * @returns {string}
 */
function getSignStr(map, arrStr) {
    let stringA = '';
    let index = 0;
    arrStr.forEach(function(key) {
        stringA += (index++ != 0 ? '&' : '') + key + '=' + map[key];
    });
    stringA += '&key=' + config.jhkj.weixin.paykey;
    return utils.md5(stringA).toUpperCase();
}

/**
 * 获取签名数据
 * @param map
 * @param arrStr
 */
function signup(map, arrStr) {
    let sign = getSignStr(map, arrStr);
    let data = '<xml>';
    arrStr.forEach(function(key) {
        if (key == 'body') {
            data += '<' + key + '><![CDATA[' + map[key] + ']]></' + key + '>';
        } else {
            data += '<' + key + '>' + map[key] + '</' + key + '>';
        }
    });
    data += '<sign>' + sign + '</sign>';
    data += '</xml>';

    return data;
}

/**
 * 初始化基础参数map
 * @returns {{}}
 */
function initMap() {
    let map = {};
    map['appid'] = config.jhkj.weixin.appid;
    map['mch_id'] = config.jhkj.weixin.mch_id;
    map['trade_type'] = 'APP';
    map['nonce_str'] = uuid.v1().replace(/\-/g,"");
    return map;
}

/**
 * 获取用户端IP地址
 * @param req
 * @returns {*}
 */
function getClientIp(req) {
    let ipaddress = req.headers['x-forwarded-for'] ||
        req.connection.remoteAddress ||
        req.socket.remoteAddress ||
        req.connection.socket.remoteAddress;
    if (ipaddress.indexOf(',') >= 0) {
        ipaddress = ipaddress.split(',')[0];
    }
    return ipaddress;
}

© 著作权归作者所有

共有 人打赏支持
cooleone

cooleone

粉丝 1
博文 3
码字总数 982
作品 0
杭州
程序员
anziguoer/wechatPay

#wechatPay nodejs 微信公众号支付开发 NodeJs 微信公众号功能开发,移动端 H5页面调用微信的支付功能。这几天根据公司的需要使用 node 和 h5页面调用微信的支付功能完成支付需求。现在把开发...

anziguoer
2016/09/15
0
0
nodejs 微信公众号支付开发

nodejs 微信公众号支付开发 NodeJs 微信公众号功能开发,移动端 H5页面调用微信的支付功能。这几天根据公司的需要使用 node 和 h5页面调用微信的支付功能完成支付需求。现在把开发过程重新捋一...

anziguoer
2016/09/14
980
0
高效编码习惯训练班(vsc为主,含Nodejs调试技巧)

技巧网上多的是,习惯哪里有?于是有了这个课程 高效编码习惯训练班(vsc为主,含Nodejs调试技巧) 本课程主要讲如何高效编码,训练良好编码习惯,以目前最潮的vsc编辑器的用法(含nodejs调试...

i5ting
2015/11/02
0
0
Phonegap桌面开发工具 Desktop-App与手机调试工具

天猫phonegap Desktop-App:可以在不搭建andrid 环境 不配置android sdk 或者不搭建xcode环境,或者不安装 node.js npm 的情况下就可以创建phoengap应用 Phonegap Desktop-App:它绕过了全部...

liaolzy2
2015/05/29
0
1
Android 微信支付开发流程

首先奉上调起支付页面截图 002.png 一:介绍 项目中要用到支付功能,需要支付宝支付、微信支付、银联支付,所以打算总结一下,方便以后的查阅,也方便大家, 用到的地方避免再次被坑。 今天我...

展菲
2017/11/10
0
0

没有更多内容

加载失败,请刷新页面

加载更多

python做文本内容指定区域字符串替换

需求: 因为公司项目需要做SEO优化,所以对项目中的各种长连接做优化,比如本文中提到的精简路径;之前已经批量吧文本的路径名字等做过修改,这里不再赘述;这里的问题是外部的路径修改了,文...

坦途abc
13分钟前
0
0
MySQL 关键字模糊匹配,并按照匹配度排序

MySQL 关键字模糊匹配,并按照匹配度排序。 方式一、按照关键字搜索,然后根据关键字所占比例排序 SELECTdrug_name,pinyinFROMtbl_drugWHEREpinyin LIKE '%AM%'ORDER BY...

yh32
23分钟前
0
0
虚拟机学习之一:java内存区域与内存溢出异常

1.运行时数据区域 java虚拟机在执行java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域都有各自的用途和创建、销毁时间,有的区域伴随虚拟机进程的启动而存在,有些区...

贾峰uk
24分钟前
0
0
Spring加载properties文件的两种方式

在项目中如果有些参数经常需要修改,或者后期可能需要修改,那我们最好把这些参数放到properties文件中,源代码中读取properties里面的配置,这样后期只需要改动properties文件即可,不需要修...

架构师springboot
40分钟前
0
0
分布式事务,原来可以这么玩?

多个数据要同时操作,如何保证数据的完整性,以及一致性? 答 : 事务 ,是常见的做法。 举个栗子: 用户下了一个订单,需要修改 余额表 , 订单 表 , 流水 表 ,于是会有类似的伪代码: st...

微笑向暖wx
43分钟前
2
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部