收藏
评论

云开发支付的代码,有需要的进。

真机测试已通过。你照抄就行,保证可通过。

最新完美版本可供参考:

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"

    }

}


最后选择:上传和部署:云端安装依赖。

最后一次编辑于  2019-12-14
收藏

30 个评论

  • Pt
    Pt
    2018-12-18

    老张你好,感谢你提供这么一个解决方案。我有两个问题。


    我的问题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%支付成功吗?会不会存在跟微信回调返回结果不一致的场景?


    2018-12-18
    赞同
    回复 1
    • 老张
      老张
      2018-12-18

      whatever === 随便啥都行。

      云开发是解决不了回调的,你自己去实现回调吧,用自己的服务器。


      2018-12-18
      回复
  • 刘涛
    刘涛
    2018-12-10

    谢谢老张,等小程序认证成功后。

    一定要用你的代码实践一下支付的内容!

    2018-12-10
    赞同
    回复
  • MCC
    MCC
    2018-12-06


    我用get方式取access_token,在云函数里这么写对吗

    2018-12-06
    赞同
    回复 2
  • 唐全
    唐全
    2018-12-05
    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)
            
        }







    2018-12-05
    赞同
    回复 2
    • 北城以北
      北城以北
      2018-12-11

      唐总 往token的集合插入的代码 能不能也发下出来,感激不尽!!!!

      2018-12-11
      回复
    • 北城以北
      北城以北
      2018-12-11

      找到你这个插件的地址了 有更新了 用法变了

      这是你用的那个插件的最新地址 https://github.com/sigoden/wechatpay/tree/master




      2018-12-11
      回复
  • Xingyu YE
    Xingyu YE
    2018-12-04

    大神,请问你写没写退款的代码?

    2018-12-04
    赞同
    回复 6
    • 老张
      老张
      2018-12-04

      还没有,业务里没有涉及到,还没去研究。

      2018-12-04
      回复
    • Xingyu YE
      Xingyu YE
      2018-12-05回复老张

      话说为啥评论的兄弟给的代码需要配置证书?统一下单过程不是不需要证书吗?

      2018-12-05
      回复
    • 老张
      老张
      2018-12-05回复Xingyu YE

      他们说的是企业付款到零钱。

      别管他们说的,照我的代码抄就行了。

      2018-12-05
      回复
    • Xingyu YE
      Xingyu YE
      2018-12-05回复老张

      给力!我已经跑通了

      2018-12-05
      回复
    • Wing-Li_
      Wing-Li_
      2019-01-25回复Xingyu YE

      兄弟,你退款的,怎么样了?

      2019-01-25
      回复
    查看更多(1)
  • 齐兴云陈利雄
    齐兴云陈利雄
    2018-12-01

    mine.js? [sm]:37 Uncaught (in promise) TypeError: Cannot read property 'out_trade_no' of null这是为啥

    2018-12-01
    赞同
    回复 2
    • 老张
      老张
      2018-12-01

      把所有注释都删除了试试。或者把out_trade_no = "otn" + nonce_str + timeStamp里的三个值都log出来看看结果。

      2018-12-01
      回复
    • 齐兴云陈利雄
      齐兴云陈利雄
      2018-12-01回复老张

      谢谢

      2018-12-01
      回复
  • 2018-11-27

    老张我用了你的方法可是云函数返回值是空的,能帮我看下是什么原因吗?

    // 云函数入口文件

    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 })

    }

    }



    2018-11-27
    赞同
    回复 9
    • 老张
      老张
      2018-11-27

      你用的代码是从小程序“客博”上拷贝下来的吧,谢谢你的付费。那个代码虽然也没问题,但是有点老,这里的我更精简了。

      最好用我在这篇里的代码,是最新的。

      你是运行到这里了:if (xml.indexOf('prepay_id') < 0return

      你最好把xml打印出来看看:console.log(xml)

      看看是什么原因的错误。

      2018-11-27
      回复
    • 2018-11-27回复老张

      我把后面的代码注释在这返回res

      j结果


      2018-11-27
      回复
    • 2018-11-27

      在后面返回xml是nulll

      2018-11-27
      回复
    • 2018-11-27

      秘钥错了

      2018-11-27
      回复
    • 老张
      老张
      2018-11-28回复

      key是自己写,32位,记得好象必须大写。

      2018-11-28
      回复
    查看更多(4)
  • 唐全
    唐全
    2018-11-21

    这种方式,代码比较少,可以参考一下。


    服务端关键代码:

    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)

    }

    })


    2018-11-21
    赞同
    回复 10
    • 老张
      老张
      2018-11-22

      不错不错,有人专门做了微信支付的插件,有时间我走一遍看看。

      2018-11-22
      回复
    • 杉星
      杉星
      2018-11-22

       server + 'wxCert/apiclient_cert.p12'    证书具体放在什么地方?

      2018-11-22
      回复
    • 唐全
      唐全
      2018-11-26回复杉星

      放在你的云存储里面,server是你的云存储地址,cloud://XXXXXXXXXXX

      2018-11-26
      回复
    • 老张
      老张
      2018-11-27回复唐全

      这个好,我正在做企业付款到零钱,是不是按你你上面的提到证书那样配置就行了?

      2018-11-27
      回复
    • 唐全
      唐全
      2018-11-28回复老张

      是的,没问题的。

      2018-11-28
      回复
    查看更多(5)
  • UNDO
    UNDO
    2018-11-21

    您知道是什么原因么

    2018-11-21
    赞同
    回复
  • UNDO
    UNDO
    2018-11-21

    返回结果:

    {"errorCode":1,"errorMessage":"user code exception caught","stackTrace":"Cannot find module 'request-promise'"}



    2018-11-21
    赞同
    回复 6
    • UNDO
      UNDO
      2018-11-21

      大神求解 运行环境Nodejs8.9

      2018-11-21
      回复
    • 老张
      老张
      2018-11-21回复UNDO

      汗。你这是连基本的nodejs都不懂了。

      看我补充的package.json吧。

      复制到云函数getPay的目录下,选择上传和部署:云端安装依赖。

      2018-11-21
      回复
    • UNDO
      UNDO
      2018-11-21回复老张

      大哥,万分感谢,我的问题是因为idea没有更新。所以没有云端安装依赖

      2018-11-21
      回复
    • UNDO
      UNDO
      2018-11-21回复老张

      很无奈,谢谢了

      2018-11-21
      回复
    • 老张
      老张
      2018-11-21回复UNDO

      原因很明显嘛,就是没有安装'request-promise'

      碰到这种情况,那就在本机 npm install request-promise (不要说这个你也不会哦)

      然后选择上传并部署:所有文件。

      试试看吧。

      再不行,只能怪你人品了。

      2018-11-21
      回复
    查看更多(1)

正在加载...

登录 后发表内容