var express=require('express');
var router = express.Router();
var request = require('request');
var crypto = require('crypto');
var key = 'xxx'; //商戶后臺支付的key
router.get('/notifypay', (req, res) => {
console.log(req)
})
router.post('/wxpay', (req, res) => {
var apiUrl = "https://api.mch.weixin.qq.com/pay/unifiedorder";
var total_fee = req.body.total_fee *100
var openid= req.body.openid
var body = req.body.body;
var bookingNo = req.body.bookingNo
var wxConfig = {
appid:'xxx', //appid
mch_id : 'xxx' //商戶號
};
var timeStamp = createTimeStamp(); //時間節(jié)點
var nonce_str = createNonceStr() + createTimeStamp(); //隨機字符串
var create_ip = get_client_ip(req); //請求ip
var notify_url ='https://xxx.com';
var formData = "<xml>";
formData += "<appid>"+wxConfig['appid']+"</appid>"; //appid
formData += "<mch_id>"+wxConfig['mch_id']+"</mch_id>"; //商戶號
formData += "<nonce_str>"+nonce_str+"</nonce_str>"; //隨機字符串
formData += "<body>" + body + "</body>"; //商品描述
formData += "<notify_url>"+notify_url+"</notify_url>";
formData += "<openid>" + openid + "</openid>";
formData += "<out_trade_no>" + bookingNo + "</out_trade_no>";//訂單號
formData += "<spbill_create_ip>"+create_ip+"</spbill_create_ip>";
formData += "<total_fee>" + total_fee + "</total_fee>";
formData += "<trade_type>JSAPI</trade_type>";
formData += "<sign>" + paysignjsapi(wxConfig['appid'],body,wxConfig['mch_id'],nonce_str,notify_url,openid,bookingNo,create_ip,total_fee,'JSAPI') + "</sign>";
formData += "</xml>";
console.log(formData);
request({
url: apiUrl,
method: 'POST',
body: formData
},function (err, response, body) {
if (!err && response.statusCode === 200){
// console.log(body);
var result_code = getXMLNodeValue('result_code', body.toString("utf-8"));
var resultCode = result_code.split('[')[2].split(']')[0];
if(resultCode === 'SUCCESS'){
//成功
var prepay_id = getXMLNodeValue('prepay_id', body.toString("utf-8")).split('[')[2].split(']')[0]; //獲取到prepay_id
//簽名
var _paySignjs = paysignjs(wxConfig['appid'], nonce_str, 'prepay_id='+ prepay_id,'MD5',timeStamp);
var args = {
appId: wxConfig['appid'],
timeStamp: timeStamp,
nonceStr: nonce_str,
signType: "MD5",
package: prepay_id,
paySign: _paySignjs,
status:200
};
res.write(JSON.stringify(args));
res.end();
}else{
//失敗
var err_code_des = getXMLNodeValue('err_code_des',body.toString("utf-8"));
var errDes = err_code_des.split('[')[2].split(']')[0];
var errArg = {
status:400,
errMsg: errDes
};
res.write(JSON.stringify(errArg));
res.end();
}
// console.log('prepay_id是'+resultCode)
}
})
})
function paysignjs(appid, nonceStr, package, signType, timeStamp) {
var ret = {
appId: appid,
nonceStr: nonceStr,
package: package,
signType: signType,
timeStamp: timeStamp
};
var string = raw1(ret);
string = string + '&key='+key;
// console.log(string);
var crypto = require('crypto');
return crypto.createHash('md5').update(string, 'utf8').digest('hex');
}
function raw1(args) {
var keys = Object.keys(args);
keys = keys.sort()
var newArgs = {};
keys.forEach(function(key) {
newArgs[key] = args[key];
});
var string = '';
for(var k in newArgs) {
string += '&' + k + '=' + newArgs[k];
}
string = string.substr(1);
return string;
}
//生成簽名
function paysignjsapi(appid,body,mch_id,nonce_str,notify_url,openid,out_trade_no,spbill_create_ip,total_fee,trade_type) {
var ret = {
appid: appid,
body: body,
mch_id: mch_id,
nonce_str: nonce_str,
notify_url: notify_url,
openid: openid,
out_trade_no: out_trade_no,
spbill_create_ip: spbill_create_ip,
total_fee: total_fee,
trade_type: trade_type
};
var string = raw(ret);
string = string + '&key='+ key;
var sign = crypto.createHash('md5').update(string, 'utf8').digest('hex');
return sign.toUpperCase()
}
function raw(args) {
var keys = Object.keys(args);
keys = keys.sort();
var newArgs = {};
keys.forEach(function(key) {
newArgs[key.toLowerCase()] = args[key];
});
var string = '';
for(var k in newArgs) {
string += '&' + k + '=' + newArgs[k];
}
string = string.substr(1);
return string;
}
//解析xml
function getXMLNodeValue(node_name, xml) {
var tmp = xml.split("<" + node_name + ">");
var _tmp = tmp[1].split("</" + node_name + ">");
return _tmp[0];
}
//獲取url請求客戶端ip
var get_client_ip = function(req) {
var ip = req.headers['x-forwarded-for'] ||
req.ip ||
req.connection.remoteAddress ||
req.socket.remoteAddress ||
req.connection.socket.remoteAddress || '';
if(ip.split(',').length>0){
ip = ip.split(',')[0]
}
return ip;
};
// 隨機字符串產(chǎn)生函數(shù)
function createNonceStr() {
return Math.random().toString(36).substr(2, 15)
}
// 時間戳產(chǎn)生函數(shù)
function createTimeStamp() {
return parseInt(new Date().getTime() / 1000) + ''
}
module.exports = router; //暴露這個 router模塊
nodejs微信小程序支付
最后編輯于 :
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
- 文/潘曉璐 我一進(jìn)店門哲鸳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來臣疑,“玉大人,你說我怎么就攤上這事徙菠⊙渡颍” “怎么了?”我有些...
- 文/不壞的土叔 我叫張陵婿奔,是天一觀的道長缺狠。 經(jīng)常有香客問我,道長萍摊,這世上最難降的妖魔是什么挤茄? 我笑而不...
- 正文 為了忘掉前任,我火速辦了婚禮冰木,結(jié)果婚禮上穷劈,老公的妹妹穿的比我還像新娘。我一直安慰自己踊沸,他們只是感情好歇终,可當(dāng)我...
- 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著逼龟,像睡著了一般评凝。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上腺律,一...
- 文/蒼蘭香墨 我猛地睜開眼朱浴,長吁一口氣:“原來是場噩夢啊……” “哼吊圾!你這毒婦竟也來了达椰?” 一聲冷哼從身側(cè)響起,我...
- 正文 年R本政府宣布狭莱,位于F島的核電站,受9級特大地震影響概作,放射性物質(zhì)發(fā)生泄漏腋妙。R本人自食惡果不足惜,卻給世界環(huán)境...
- 文/蒙蒙 一讯榕、第九天 我趴在偏房一處隱蔽的房頂上張望骤素。 院中可真熱鬧,春花似錦瘩扼、人聲如沸谆甜。這莊子的主人今日做“春日...
- 文/蒼蘭香墨 我抬頭看了看天上的太陽规辱。三九已至,卻和暖如春栽燕,著一層夾襖步出監(jiān)牢的瞬間罕袋,已是汗流浹背。 一陣腳步聲響...
推薦閱讀更多精彩內(nèi)容
- 支付的大致流程如下: 小程序端用 wx.request 向服務(wù)器請求支付參數(shù)(相當(dāng)于ajax) 服務(wù)器根據(jù)客戶端的...
- 微信小程序開發(fā)之---微信免密支付 很多時候我們都要用到免密支付奈籽,也就是常說的自動扣款饥侵,但是如何才能自動扣費,這一...
- 接著教程1寫的 一些通用的接口函數(shù)衣屏,接收解析參數(shù)等我去掉了躏升,你們自己加,給微信傳輸?shù)膮?shù)看我上一個教程中鏈接...
- 早上爬不起來佃却,昨天還是抑制不住的點了三斤小龍蝦,真是罪惡啊斧蜕。我今年的十斤肉就是這么來的双霍。早上孩子起床后說肚子難受不...