文档章节

NodeJS微信支付签名代码

cooleone
 cooleone
发布于 2017/07/25 16:16
字数 563
阅读 2
收藏 0
点赞 0
评论 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

nodejs 微信公众号支付开发

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

anziguoer ⋅ 2016/09/14 ⋅ 0

Android 微信支付开发流程

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

展菲 ⋅ 2017/11/10 ⋅ 0

nodejs实现微信公众平台开发

基本原理   用nodejs怎样来实现对微信公众平台的开发呢?   别的就不多说了,先来简单介绍微信公众平台的基本原理。   微信服务器就相当于一个转发服务器,终端(手机、Pad等)发起请求...

笔阁 ⋅ 2016/03/07 ⋅ 3

微信支付之h5网页支付

先简单说下微信公众号支付的流程首先获得code,然后获得openid,然后根据openid获取到预支付ID(prepayid),获取到prepayid,自然得到packages ,这个最难获得的参数获得之后,基本上就做好一半...

扎西多顿 ⋅ 2016/01/21 ⋅ 3

Java微信扫码支付

前言:让我用Java写个微信扫码支付,身为小白,网上搜了好多文章,终于找到一个看得明白的,链接。表示人家讲的够详细了,现在自己要是实现一个,我觉得吧,可能入手比较乱。其实后来发现,代...

ioufev ⋅ 2016/11/07 ⋅ 0

微信h5支付 靠谱版

林伟琨 ⋅ 2016/10/18 ⋅ 0

mayubao/Android-Pay

Android-Pay 支持微信和支付宝两种主流支付的集成库, 两行代码实现微信支付, 三行代码实现支付宝支付 引入 gradle 对应的项目中的build.gradle文件添加依赖: dependencies { } maven io....

mayubao ⋅ 2017/03/01 ⋅ 0

简单的聊一聊微信支付的那些事

前往官网注册账号创建应用 微信商户平台商家查看一些交易记录和余额提现 微信开放平台给开发者提供的网站 需要使用微信的支付功能肯定是要先去他的申请账号的,这些步骤这里就不讲了。 注册成...

a_zhon ⋅ 2017/09/21 ⋅ 0

Node.js VS PHP:Web 开发的史诗级对决,你支持谁?

现在,Web开发公司和开发人员可以选择多种技术栈来构建Web应用程序。早期网络发展,不同的技术被用于前端和后端开发。但是,随着Node.js的发布,布局发生了变化,因为它允许开发人员使用 Ja...

uzv80px5v412ne ⋅ 2017/12/29 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

高并发之Nginx的限流

首先Nginx的版本号有要求,最低为1.11.5 如果低于这个版本,在Nginx的配置中 upstream web_app { server 到达Ip1:端口 max_conns=10; server 到达Ip2:端口 max_conns=10; } server { listen ...

算法之名 ⋅ 今天 ⋅ 0

Spring | IOC AOP 注解 简单使用

写在前面的话 很久没更新笔记了,有人会抱怨:小冯啊,你是不是在偷懒啊,没有学习了。老哥,真的冤枉:我觉得我自己很菜,还在努力学习呢,正在学习Vue.js做管理系统呢。即便这样,我还是不...

Wenyi_Feng ⋅ 今天 ⋅ 0

博客迁移到 https://www.jianshu.com/u/aa501451a235

博客迁移到 https://www.jianshu.com/u/aa501451a235 本博客不再更新

为为02 ⋅ 今天 ⋅ 0

win10怎么彻底关闭自动更新

win10自带的更新每天都很多,每一次下载都要占用大量网络,而且安装要等得时间也蛮久的。 工具/原料 Win10 方法/步骤 单击左下角开始菜单点击设置图标进入设置界面 在设置窗口中输入“服务”...

阿K1225 ⋅ 今天 ⋅ 0

Elasticsearch 6.3.0 SQL功能使用案例分享

The best elasticsearch highlevel java rest api-----bboss Elasticsearch 6.3.0 官方新推出的SQL检索插件非常不错,本文一个实际案例来介绍其使用方法。 1.代码中的sql检索 @Testpu...

bboss ⋅ 今天 ⋅ 0

informix数据库在linux中的安装以及用java/c/c++访问

一、安装前准备 安装JDK(略) 到IBM官网上下载informix软件:iif.12.10.FC9DE.linux-x86_64.tar放在某个大家都可以访问的目录比如:/mypkg,并解压到该目录下。 我也放到了百度云和天翼云上...

wangxuwei ⋅ 今天 ⋅ 0

PHP语言系统ZBLOG或许无法重现月光博客的闪耀历史[图]

最近在写博客,希望通过自己努力打造一个优秀的教育类主题博客,名动江湖,但是问题来了,现在写博客还有前途吗?面对强大的自媒体站点围剿,还有信心和可能型吗? 至于程序部分,我选择了P...

原创小博客 ⋅ 今天 ⋅ 0

IntelliJ IDEA 2018.1新特性

工欲善其事必先利其器,如果有一款IDE可以让你更高效地专注于开发以及源码阅读,为什么不试一试? 本文转载自:netty技术内幕 3月27日,jetbrains正式发布期待已久的IntelliJ IDEA 2018.1,再...

Romane ⋅ 今天 ⋅ 0

浅谈设计模式之工厂模式

工厂模式(Factory Pattern)是 Java 中最常用的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。 在工厂模式中,我们在创建对象时不会对客户端暴露创建逻...

佛系程序猿灬 ⋅ 今天 ⋅ 0

Dockerfile基础命令总结

FROM 指定使用的基础base image FROM scratch # 制作base image ,不使用任何基础imageFROM centos # 使用base imageFROM ubuntu:14.04 尽量使用官方的base image,为了安全 LABEL 描述作...

ExtreU ⋅ 昨天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部