真机测试已通过。你照抄就行,保证可通过。
最新完美版本可供参考:
https://developers.weixin.qq.com/community/develop/article/doc/0004c4a50a03107eaa79f03cc56c13
小程序端:
wx.cloud.callFunction({
name: 'getPay' ,
data: {
total_fee: parseFloat(0.01).toFixed(2) * 100,
attach: 'anything',
body: 'whatever'
}
})
.then( res => {
wx.requestPayment({
appId: res.result.appid,
timeStamp: res.result.timeStamp,
nonceStr: res.result.nonce_str,
package: 'prepay_id=' + res.result.prepay_id,
signType: 'MD5',
paySign: res.result.paySign,
success: res => {
console.log(res)
}
})
})
云函数:getPay
getPay目录下共两个文件:
1、index.js
2、package.json
index.js代码如下:
const key = "YOURKEY1234YOURKEY1234YOURKEY123"//这是商户的key,不是小程序的密钥,32位。
const mch_id = "1413090000" //你的商户号
//将以上的两个参数换成你的,然后以下可以不用改一个字照抄
const rp = require('request-promise')
const crypto = require('crypto')
function paysign({ ...args }) {
let sa = []
for (let k in args) sa.push( k + '=' + args[k])
sa.push( 'key=' + key)
return crypto.createHash('md5').update(sa.join('&'), 'utf8').digest('hex').toUpperCase()
}
exports.main = async (event, context) => {
const appid = event.userInfo.appId
const openid = event.userInfo.openId
const attach = event.attach
const body = event.body
const total_fee = event.total_fee
const notify_url = "https://whatever.com/notify"
const spbill_create_ip = "118.89.40.200"
const nonce_str = Math.random().toString(36).substr(2, 15)
const timeStamp = parseInt(Date.now() / 1000) + ''
const out_trade_no = "otn" + nonce_str + timeStamp
let formData = "<xml>"
formData += "<appid>" + appid + "</appid>"
formData += "<attach>" + attach + "</attach>"
formData += "<body>" + body + "</body>"
formData += "<mch_id>" + mch_id + "</mch_id>"
formData += "<nonce_str>" + nonce_str + "</nonce_str>"
formData += "<notify_url>" + notify_url + "</notify_url>"
formData += "<openid>" + openid + "</openid>"
formData += "<out_trade_no>" + out_trade_no + "</out_trade_no>"
formData += "<spbill_create_ip>" + spbill_create_ip + "</spbill_create_ip>"
formData += "<total_fee>" + total_fee + "</total_fee>"
formData += "<trade_type>JSAPI</trade_type>"
formData += "<sign>" + paysign({ appid, attach, body, mch_id, nonce_str, notify_url, openid, out_trade_no, spbill_create_ip, total_fee, trade_type: 'JSAPI' }) + "</sign>"
formData += "</xml>"
let res = await rp({ url: "https://api.mch.weixin.qq.com/pay/unifiedorder", method: 'POST',body: formData})
let xml = res.toString("utf-8")
if (xml.indexOf('prepay_id')<0) return xml
let prepay_id = xml.split("<prepay_id>")[1].split("</prepay_id>")[0].split('[')[2].split(']')[0]
let paySign = paysign({ appId: appid, nonceStr: nonce_str, package: ('prepay_id=' + prepay_id), signType: 'MD5', timeStamp: timeStamp })
return { appid, nonce_str, timeStamp, prepay_id, paySign }
}
package.json 代码如下:
{
"name": "getPay",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "youself",
"license": "ISC",
"dependencies": {
"crypto": "^1.0.1",
"request-promise": "^4.2.2"
}
}
最后选择:上传和部署:云端安装依赖。
老张你好,感谢你提供这么一个解决方案。我有两个问题。
我的问题1: 支付后回调
// 问题1
{
notify_url:
'https://whatever.com/notify'
}
支付通知回调,填写这样一个公网地址,有什么风险吗?我打开一看,发现跳转到了以下链接
https:
//www.youtube.com/user/whatever?sub_confirmation=1
我个人认为风险有2个。1、回调信息被暴露(虽然他们可能不会拿来怎么样);2、这个公网地址发现访问异常,直接把微信域名的所有回调禁掉,这样子所有用了这段代码、这个库的所有小程序可能都会支付失败(因为微信可能会先发option请求这个url是否可用,一旦被禁将不能正常访问)
我的问题2:wx.requestPayment
当仅仅使用云开发时,这个支付api的success的回调,能完全保证100%支付成功吗?会不会存在跟微信回调返回结果不一致的场景?
whatever === 随便啥都行。
云开发是解决不了回调的,你自己去实现回调吧,用自己的服务器。
谢谢老张,等小程序认证成功后。
一定要用你的代码实践一下支付的内容!
我用get方式取access_token,在云函数里这么写对吗
这是我跑通的,你可以参考:
https://developers.weixin.qq.com/community/develop/doc/00048279e2c5f07923b71efaf51804
谢谢
const
cloud = require(
'wx-server-sdk'
)
var
rp = require(
'request-promise'
)
var
Pay = require(
'@sigodenh/wechatpay'
);
cloud.init({
})
const
db = cloud.database()
const
_ = db.command
const
MAX_LIMIT =
100
exports.main = async (event, context) => {
const
openId = event.userInfo.openId
const
appId = event.userInfo.appId
const
tokenData = await db.collection(
'token'
).
get
() //读取token数据表
const
nowTime =
parseInt
(
new
Date
().getTime() /
1000
)
const
merchatId = tokenData.data[
0
].merchatId
const
APIKey = tokenData.data[
0
].APIKey
//微信支付api key
const
server = tokenData.data[
0
].server
// cloud://XXXXXX.XXXXXX/ 微信云存储地址
const
fileID = server +
'cert/apiclient_cert.p12'
//微信支付证书,放在云存储cert目录里面
let pfxContent =
''
await cloud.downloadFile({
fileID: fileID,
}).then(res => {
pfxContent = res.fileContent.toString(
'utf8'
)
})
let pay =
new
Pay(appId, merchatId, APIKey, pfxContent)
const
result = await pay.unifiedOrder({
body: title,
out_trade_no: orderId,
total_fee: amount *
100
,
spbill_create_ip:
'8.8.8.8'
,
notify_url:
'https://www.baidu.com/notify'
,
trade_type:
'JSAPI'
,
openid: openId
})
return
await pay.tidyOrderResult(result)
}
唐总 往token的集合插入的代码 能不能也发下出来,感激不尽!!!!
找到你这个插件的地址了 有更新了 用法变了
这是你用的那个插件的最新地址 https://github.com/sigoden/wechatpay/tree/master
大神,请问你写没写退款的代码?
还没有,业务里没有涉及到,还没去研究。
话说为啥评论的兄弟给的代码需要配置证书?统一下单过程不是不需要证书吗?
他们说的是企业付款到零钱。
别管他们说的,照我的代码抄就行了。
给力!我已经跑通了
兄弟,你退款的,怎么样了?
mine.js? [sm]:37 Uncaught (in promise) TypeError: Cannot read property 'out_trade_no' of null这是为啥
把所有注释都删除了试试。或者把out_trade_no = "otn" + nonce_str + timeStamp里的三个值都log出来看看结果。
谢谢
老张我用了你的方法可是云函数返回值是空的,能帮我看下是什么原因吗?
// 云函数入口文件
const cloud = require('wx-server-sdk')
const request = require('request-promise')
const crypto = require('crypto')
cloud.init()
const key = "qwerewq1234qewr123dfdadsdfasdfas"
const mch_id = "10060011"
const notify_url = "https://www.yoursite.com/notify"
const url = "https://api.mch.weixin.qq.com/pay/unifiedorder"
const spbill_create_ip = "118.89.40.123"
function paysign({ ...args }) {
let str = ''
for (let k in args) {
str += '&' + k + '=' + args[k]
}
str += '&key=' + key
str = str.substr(1)
return crypto.createHash('md5').update(str, 'utf8').digest('hex')
}
function getPrepayId(xml) {
let tmp = xml.split("<prepay_id>")
tmp = tmp[1].split("</prepay_id>")
tmp = tmp[0].split('[')
tmp = tmp[2].split(']')
return tmp[0]
}
exports.main = async (event, context) => {
const attach = event.attach
const body = event.body
const total_fee = event.total_fee
const openid = event.userInfo.openId
const appid = event.userInfo.appId
let nonce_str = Math.random().toString(36).substr(2, 15)
let timeStamp = parseInt(Date.now() / 1000) + ''
let out_trade_no = "otn" + nonce_str + timeStamp
let formData = "<xml>"
formData += "<appid>" + appid + "</appid>"
formData += "<attach>" + attach + "</attach>"
formData += "<body>" + body + "</body>"
formData += "<mch_id>" + mch_id + "</mch_id>"
formData += "<nonce_str>" + nonce_str + "</nonce_str>"
formData += "<notify_url>" + notify_url + "</notify_url>"
formData += "<openid>" + openid + "</openid>"
formData += "<out_trade_no>" + out_trade_no + "</out_trade_no>"
formData += "<spbill_create_ip>" + spbill_create_ip + "</spbill_create_ip>"
formData += "<total_fee>" + total_fee + "</total_fee>"
formData += "<trade_type>JSAPI</trade_type>"
formData += "<sign>" + paysign({ appid, attach, body, mch_id, nonce_str, notify_url, openid, out_trade_no, spbill_create_ip, total_fee, trade_type: 'JSAPI' }) + "</sign>"
formData += "</xml>"
let pr = await request({ url: url, method: 'POST', body: formData })
let xml = pr.toString("utf-8")
if (xml.indexOf('prepay_id') < 0) return
let prepay_id = getPrepayId(xml)
return {
appid: appid,
nonce_str: nonce_str,
timeStamp: timeStamp,
out_trade_no: out_trade_no,
prepay_id: prepay_id,
paySign: paysign({ appId: appid, nonceStr: nonce_str, package: ('prepay_id=' + prepay_id), signType: 'MD5', timeStamp: timeStamp })
}
}
你用的代码是从小程序“客博”上拷贝下来的吧,谢谢你的付费。那个代码虽然也没问题,但是有点老,这里的我更精简了。
最好用我在这篇里的代码,是最新的。
你是运行到这里了:if (xml.indexOf('prepay_id') < 0) return
你最好把xml打印出来看看:console.log(xml)
看看是什么原因的错误。
我把后面的代码注释在这返回res
j结果
在后面返回xml是nulll
秘钥错了
key是自己写,32位,记得好象必须大写。
这种方式,代码比较少,可以参考一下。
服务端关键代码:
var rp = require('request-promise')
var Pay = require('@sigodenh/wechatpay');
const fileID = server + 'wxCert/apiclient_cert.p12'
let pfxContent = ''
await cloud.downloadFile({
fileID: fileID,
}).then(res => {
pfxContent = res.fileContent.toString('utf8')
})
let pay = new Pay(appId, merchatId, APIKey, pfxContent)
const result = await pay.unifiedOrder({
body: title,
out_trade_no: orderId,
total_fee: amount * 100,
spbill_create_ip: '8.8.8.8',
notify_url: 'https://www.baidu.com/notify',
trade_type: 'JSAPI',
openid: openId
})
return await pay.tidyOrderResult(result)
客户端关键代码:
wx.cloud.callFunction({
name: 'token',
data: {
usage: "pay",
orderId: orderId,
title: '小程序购物',
amount: amount,
},
success: res => {
var formId = res.result.package.slice(10)
wx.requestPayment(
{
'timeStamp': res.result.timeStamp.toString(),
'nonceStr': res.result.nonceStr,
'package': res.result.package,
'signType': 'MD5',
'paySign': res.result.paySign,
'success': res => {
wx.showToast({
title: '支付成功',
})
//支付成功后,更新订单
that.updateOrder("payOrder", orderId, app.globalData.userInfo.nickName, formId);
},
'fail': res => {
wx.showToast({
title: '未支付',
icon: 'none'
})
},
'complete': res => {
console.log(res)
}
})
不错不错,有人专门做了微信支付的插件,有时间我走一遍看看。
server + 'wxCert/apiclient_cert.p12' 证书具体放在什么地方?
放在你的云存储里面,server是你的云存储地址,cloud://XXXXXXXXXXX
这个好,我正在做企业付款到零钱,是不是按你你上面的提到证书那样配置就行了?
是的,没问题的。
您知道是什么原因么
返回结果:
{"errorCode":1,"errorMessage":"user code exception caught","stackTrace":"Cannot find module 'request-promise'"}
大神求解 运行环境Nodejs8.9
汗。你这是连基本的nodejs都不懂了。
看我补充的package.json吧。
复制到云函数getPay的目录下,选择上传和部署:云端安装依赖。
大哥,万分感谢,我的问题是因为idea没有更新。所以没有云端安装依赖
很无奈,谢谢了
原因很明显嘛,就是没有安装'request-promise'
碰到这种情况,那就在本机 npm install request-promise (不要说这个你也不会哦)
然后选择上传并部署:所有文件。
试试看吧。
再不行,只能怪你人品了。