• 媒体品牌
    爱范儿
    关注明日产品的数字潮牌
    APPSO
    先进工具,先知先行,AIGC 的灵感指南
    董车会
    造车新时代,明日出行家
    玩物志
    探索城市新生活方式,做你的明日生活指南
  • 知晓云
  • 制糖工厂
    扫描小程序码,了解更多

开发 | 只需 3 步,教你在小程序中接入微信支付

小程序

2017-03-26 12:37

文 | 隔壁陈叔叔

最近需要在微信小程序中用到在线支付功能,发现要在小程序里实现微信支付还是很方便的。

如果你以前开发过服务号下的微信支付,那么你会发现其实小程序里的微信支付和服务号里的开发过程如出一辙。

下面,知晓程序(微信号 zxcx0101)将会与大家分享,小程序里微信支付的开发流程和注意点。

1. 开通微信支付和微信商户号

这个过程就和开通服务号的微信支付过程一样,没有什么可以说的。

2. 获得用户的 OpenID

首先,我们需要在小程序的客户端的逻辑层中,获取当前用户的 OpenID。

通过调用 wx.login 方法,可以得到用户的 code,然后开发者服务器使用登录凭证 code 获取 openid

wx.login({
  success: function (res) {
    if (res.code) {
      //发起网络请求
      wx.request({
        url: 'https://yourwebsit/onLogin',
        method: 'POST',
        data: {
          code: res.code
        },
        success: function (res) {
          var openid = res.data.openid;
        },
        fail: function (err) {
          console.log(err)
        }
      })
    }
    else {
      console.log('获取用户登录态失败!' + res.errMsg)
    }
  }
});
var code = req.param("code");
request({
  url: "https://api.weixin.qq.com/sns/jscode2session?appid=" + appid + "&secret=" + secret + "&js_code=" + code + "&grant_type=authorization_code",
  method: 'GET'
}, function (err, response, body) {
  if (!err && response.statusCode == 200) {
    res.json(JSON.parse(body));
  }
});

3. 获取 prepay_id 和支付签名验证 paySign

这一步的过程,和服务号里的微信支付过程一样。这个过程区分为客户端操作,以及服务器端的操作。

首先来看一下客户端逻辑层实现。

在服务号里,我们是通过如下的代码来调起支付功能:

function jsApiCall() {
  WeixinJSBridge.invoke('getBrandWCPayRequest', {
    "appId": "", //公众号名称,由商户传入     
    "timeStamp": "", //时间戳,自 1970 年以来的秒数     
    "nonceStr": "", //随机串     
    "package": "prepay_id=<%=prepay_id%>",
    "signType": "MD5", //微信签名方式:     
    "paySign": "<%=_paySignjs%>" //微信签名
  }, function (res) {
    WeixinJSBridge.log(res.err_msg);
    if (res.err_msg == "get_brand_wcpay_request:ok") {
      alert("支付成功!");
    }
    else {
      alert("支付失败!");
    }
  });
}

在小程序里,我们是通过 wx.requestPayment 方法调起支付功能。当然,在这之前,我们先要获取 prepay_id

wx.request({
 url: 'https://yourwebsit/service/getPay',
 method: 'POST',
 data: {
   bookingNo: bookingNo, /*订单号*/
   total_fee: total_fee, /*订单金额*/
   openid: openid
 },
 header: {
   'content-type': 'application/json'
 }, success: function (res) {
   wx.requestPayment({
     'timeStamp': timeStamp
     , 'nonceStr': nonceStr
     , 'package': 'prepay_id=' + res.data.prepay_id
     , 'signType': 'MD5'
     , 'paySign': res.data._paySignjs
     , 'success': function (res) {
       console.log(res);
     }
     , 'fail': function (res) {
       console.log('fail:' + JSON.stringify(res));
     }
   })
 },
 fail: function (err) {
   console.log(err)
 }
})

在服务器端,我们需要实现的是 prepay_id 的获取,以及签名 paySign 的获取。

var bookingNo = req.param("bookingNo");
var total_fee = req.param("total_fee");
var openid = req.param("openid");
var body = "费用说明";
var url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
var formData = "";
formData += "appid"; //appid
formData += "test";
formData += "" + body + "";
formData += "mch_id"; //商户号
formData += "nonce_str";
formData += "notify_url";
formData += "" + openid + "";
formData += "" + bookingNo + "";
formData += "spbill_create_ip";
formData += "" + total_fee + "";
formData += "JSAPI";
formData += "" + paysignjsapi(appid, attach, body, mch_id, nonce_str, notify_url, openid, bookingNo, spbill_create_ip, total_fee, 'JSAPI') + "";
formData += "";
request({
  url: url,
  method: 'POST',
  body: formData
}, function (err, response, body) {
  if (!err && response.statusCode == 200) {
    var prepay_id = getXMLNodeValue('prepay_id', body.toString("utf-8"));
    var tmp = prepay_id.split('[');
    var tmp1 = tmp[2].split(']');
    //签名
    var _paySignjs = paysignjs(appid, mch_id, 'prepay_id=' + tmp1[0], 'MD5', timeStamp);
    var o = {
      prepay_id: tmp1[0]
      , _paySignjs: _paySignjs
    }
    res.send(o);
  }
});

下面是用到的函数:

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, attach, body, mch_id, nonce_str, notify_url, openid, out_trade_no, spbill_create_ip, total_fee, trade_type) {
  var ret = {
    appid: appid,
    attach: attach,
    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 crypto = require('crypto');
  return crypto.createHash('md5').update(string, 'utf8').digest('hex');
};

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

function getXMLNodeValue(node_name, xml) {
  var tmp = xml.split("<" + node_name + ">");
  var _tmp = tmp[1].split("");
  return _tmp[0];
}

这样的简单 3 步,小程序就能对接微信支付功能了。

下面是测试的支付效果图:

原文地址:http://www.jianshu.com/p/72f5c1e3f8a5

本文由知晓程序授权转载,关注微信号 zxcx0101,在微信后台回复「出行」,获取出行必备的小程序推荐。

zxcx_0208

登录,参与讨论前请先登录

评论在审核通过后将对所有人可见

正在加载中

净干文科生事情的理科生,悲观的乐观主义者,希望成为路人的铺路石,不矛盾的矛盾体。

本篇来自栏目

解锁订阅模式,获得更多专属优质内容