国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

詳解NodeJs支付寶移動(dòng)支付簽名及驗(yàn)簽

2019-11-19 18:06:44
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

非常感謝 :http://www.jianshu.com/p/8513e995ff3a?utm_campaign=hugo&utm_medium=reader_share&utm_content=note&utm_source=weibo 的文章,如果不是找到這篇文章我可能還要繼續(xù)坑幾天,代碼也基本都是照著他的搬過(guò)來(lái)的,不過(guò)支付寶移動(dòng)支付文檔寫的非常糟糕而且沒(méi)有node的SDK和demo,寫起來(lái)異常痛苦..好在找到了這篇文章順便折騰了一下午支付寶的技術(shù)人員總算把移動(dòng)支付整個(gè)流程給做完了,所以就順便記錄一下自己遇到的坑,和對(duì)移動(dòng)支付整個(gè)流程的梳理。

支付寶給的流程圖還是很清晰的,其實(shí)基本流程就是

  1. 用戶向服務(wù)器請(qǐng)求一個(gè)付款
  2. 服務(wù)器生成一個(gè)帶簽名的訂單發(fā)送給客戶端
  3. 客戶端通過(guò)這個(gè)訂單向app sdk請(qǐng)求付款
  4. sdk把用戶引入支付寶付款界面進(jìn)行支付
  5. 支付成功后支付寶向前端返回支付成功結(jié)果,并且向服務(wù)器發(fā)送一個(gè)支付通知
  6. 服務(wù)器接收通知并且驗(yàn)證是否是支付寶發(fā)送的成功結(jié)果

app客戶端需要做的很簡(jiǎn)單:

  1. 向自己的服務(wù)器請(qǐng)求一個(gè)訂單,
  2. 接收到訂單后,向支付寶sdk發(fā)情一個(gè)支付請(qǐng)求
  3. 交易結(jié)束后返回一個(gè)成功或者失敗

服務(wù)器做的事情稍微多一點(diǎn)(注意:服務(wù)端需要存放應(yīng)用的私鑰進(jìn)行簽名,還有支付寶的公鑰進(jìn)行驗(yàn)簽):

1.接收到客戶端請(qǐng)求時(shí)候,生成一個(gè)帶簽名訂單返回給客戶端,中間的步奏有

1) 把相應(yīng)的配置數(shù)據(jù)生成一個(gè)數(shù)組,再把數(shù)組的數(shù)據(jù)生成一個(gè)有序的字符串

//將支付寶發(fā)來(lái)的數(shù)據(jù)生成有序數(shù)列function getVerifyParams(params) { var sPara = []; if(!params) return null; for(var key in params) { if((!params[key]) || key == "sign" || key == "sign_type") { continue; }; sPara.push([key, params[key]]); } sPara = sPara.sort(); var prestr = ''; for(var i2 = 0; i2 < sPara.length; i2++) { var obj = sPara[i2]; if(i2 == sPara.length - 1) { prestr = prestr + obj[0] + '=' + obj[1] + ''; } else { prestr = prestr + obj[0] + '=' + obj[1] + '&'; } } return prestr;}

  2) 將這組支付串進(jìn)行RSA-SHA1算法,得到的結(jié)果再與存在服務(wù)端的私鑰進(jìn)行簽名

//驗(yàn)簽function veriySign(params) { try { var publicPem = fs.readFileSync('./rsa_public_key.pem'); var publicKey = publicPem.toString(); var prestr = getVerifyParams(params); var sign = params['sign'] ? params['sign'] : ""; var verify = crypto.createVerify('RSA-SHA1'); verify.update(prestr); return verify.verify(publicKey, sign, 'base64') } catch(err) { console.log('veriSign err', err) }}

  3) 有序的字符串+得到的簽名+簽名方法就是生成的訂單,將這組訂單返回給客戶端

//發(fā)送訂單號(hào) sendAlipay: function(req, res) { var code = "" for(var i = 0; i < 4; i++) { code += Math.floor(Math.random() * 10); } //訂單號(hào)暫時(shí)由時(shí)間戳與四位隨機(jī)碼生成 AlipayConfig.out_trade_no = Date.now().toString() + code; var myParam = getParams(AlipayConfig); var mySign = getSign(AlipayConfig) var last = myParam + '&sign="' + mySign + '"&sign_type="RSA"'; console.log(last) return res.send(last) }

2.前半段的工作就做完了,接下來(lái)如果前端支付成功,支付寶會(huì)向我們預(yù)留好的回調(diào)接口發(fā)送一個(gè)POST請(qǐng)求,讓我們驗(yàn)證用戶是否支付成功

  1) 將支付寶發(fā)送過(guò)來(lái)的數(shù)據(jù)生成一個(gè)有序的字符串

//將支付寶發(fā)來(lái)的數(shù)據(jù)生成有序數(shù)列function getVerifyParams(params) { var sPara = []; if(!params) return null; for(var key in params) { if((!params[key]) || key == "sign" || key == "sign_type") { continue; }; sPara.push([key, params[key]]); } sPara = sPara.sort(); var prestr = ''; for(var i2 = 0; i2 < sPara.length; i2++) { var obj = sPara[i2]; if(i2 == sPara.length - 1) { prestr = prestr + obj[0] + '=' + obj[1] + ''; } else { prestr = prestr + obj[0] + '=' + obj[1] + '&'; } } return prestr;}

  2) 將獲取的數(shù)據(jù)進(jìn)行hash然后根據(jù)公鑰進(jìn)行對(duì)簽名的有效應(yīng)驗(yàn)證,返回true和false

//驗(yàn)簽function veriySign(params) { try { var publicPem = fs.readFileSync('./rsa_public_key.pem'); var publicKey = publicPem.toString(); var prestr = getVerifyParams(params); var sign = params['sign'] ? params['sign'] : ""; var verify = crypto.createVerify('RSA-SHA1'); verify.update(prestr); return verify.verify(publicKey, sign, 'base64') } catch(err) { console.log('veriSign err', err) }}

  3) 如果驗(yàn)簽成功再生成支付寶通知url,來(lái)驗(yàn)證是否是支付寶發(fā)來(lái)的通知(支付寶的驗(yàn)證一大堆,腦殼都痛了),如果有數(shù)據(jù)則說(shuō)明確實(shí)是支付寶發(fā)來(lái)的通知,這次交易有效

//回調(diào)驗(yàn)簽 getAlipay: function(req, res) { console.log(req.body) var params = req.body var mysign = veriySign(params); //驗(yàn)證支付寶簽名mysign為true表示簽名正確 console.log(mysign) try { //驗(yàn)簽成功 if(mysign) { if(params['notify_id']) { var partner = AlipayConfig.partner; //生成驗(yàn)證支付寶通知的url var url = 'https://mapi.alipay.com/gateway.do?service=notify_verify&' + 'partner=' + partner + '¬ify_id=' + params['notify_id']; console.log('url:' + url) //驗(yàn)證是否是支付寶發(fā)來(lái)的通知 https.get(url, function(text) { //有數(shù)據(jù)表示是由支付寶發(fā)來(lái)的通知 if(text) { //交易成功 console.log('success') } else { //交易失敗 console.log('err') } }) } } } catch(err) { console.log(err); } }

這樣整個(gè)流程就跑完了,代碼原博客都有,這里最多只是有些改成了sails的寫法,主要寫一下這次遇到的幾個(gè)坑和值得注意的幾個(gè)地方

1. 由于移動(dòng)支付的文檔描述不清楚,私鑰其實(shí)上上傳到賬戶信息的mapi網(wǎng)管產(chǎn)品密鑰里:

而不是上傳到應(yīng)用的密鑰里

2. 移動(dòng)支付只支持RSA(SHA1)

3.  ./是在sails里獲取的到根目錄下的密鑰(有點(diǎn)搞不懂sails的這個(gè)路徑)

4. 生成訂單時(shí)候的有序字符串格式是body="測(cè)試" ,有雙引號(hào),但是驗(yàn)簽生成的有序字符串里不能有雙引號(hào)

以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,同時(shí)也希望多多支持武林網(wǎng)!

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 普格县| 武功县| 新河县| 岗巴县| 银川市| 抚宁县| 靖西县| 锦屏县| 南开区| 呼伦贝尔市| 明光市| 钟祥市| 镇赉县| 常州市| 阿合奇县| 云林县| 米泉市| 鄱阳县| 运城市| 集贤县| 金昌市| 家居| 丹阳市| 吉木萨尔县| 宣化县| 四平市| 绥宁县| 公安县| 东乡县| 兰州市| 宣恩县| 分宜县| 庆安县| 乌拉特前旗| 容城县| 濮阳市| 宜良县| 常德市| 庆城县| 桑日县| 贞丰县|