ngrok+express解决本地环境中微信接口调试问题

  在微信项目的开发的时候,常常须要对微信jssdk提供的接口进行调试,好比说录音, 分享 ,上传图像等接口,可是微信jssdk要求绑定安全域名才能使用其提供的一系列功能 , 而在开发环境中使用localhost或者本地ip没法完成域名的认证和绑定, 因此没法在本地调试 。固然有一种无可奈何方法 ,就是在本地开发完 ,打包发到公司的测试服务器上 ,利用测试服务器认证后的域名进行调试,每次改动,调试都要发一遍测试,显然这种方法很是麻烦且很不科学,因此这篇文章就针对这个问题介绍一下如何利用ngrok和express解决开发环境中微信接口的调试问题。前端

  一:首先介绍一下 ngrok,ngrok主要的功能就是将本地的ip映射到外网 ,而且分配给你一个可用的域名,经过这个域名可让外网用户打开你的本地的web服务,使用起来也很简单,官网也有文档也有详细介绍 。这里简单的介绍一下使用方法,首先去ngrok 的官网下载ngrok的对应的客户端 ,而且注册用户 ,能够经过你的github帐号或者google帐号注册 ,注册完成后再我的中心打开auth选项,复制这里的authtoken,以下图:node


(这里就以window版本为例),而后下载完成解压,会有一个ngrok.exe文件,双击运行会出现下面的命令行:webpack

首先咱们须要完成ngrok的token认证,不然运行会发生错误,运行一下命令git

ngrok authtoken ***************** //*号就是我的中心中的token ,复制下来就能够了复制代码

认证完成后,就能够操做了,上图中的examples就是一些经常使用的示例命令,咱们用到的就是ngrok http,后面接的参数就是你本地web服务的端口号,运行后会分配一个外网域名,经过这个域名就能够访问到你的本地web服务,github


  不过,这个域名在重启后就会从新分配一个新域名,致使重启后须要去微信公众平台从新设置一下安全域名和token认证 。比较遗憾的是在ngrok1.0的时候能够经过 ngrok http subdomain=***(自定义域名) 80 固定每次的分配的域名,可是在2.0版本后,免费用户没法固定域名,只有付费用户才能够,虽然每月只须要$5,可是对于不是常常测试的人来讲仍是彻底没有购买欲望,关键是好像只支持visaa......。不过对于想要免费固定域名的胖友来讲,解决办法仍是有的,国内有个sunny-ngrok ,能够免费申请一个自定义的固定域名 ,具体教程能够去其官网查看,也不是很复杂,有问题话能够在评论里面问我,就不详细讲了。固然想要实现外网映射的话还有不少其余方法,好比使用npm安装的Localtunnel和花生壳等等,能够自行了解一下。web

  二:获得域名后,接下来咱们要作的就是使用该域名完成微信安全域名绑定啦,咱们能够去微信公众平台申请一个测试号,不过这时候填写时没法经过的,由于微信认证须要拥有一个本身的服务器正确响应配置请求算法


测试号申请的时候填写配置信息的url,微信服务器会发送一个get请求到这个地址上,get请求会携带一些参数,咱们须要用这些参数生成一个签名和微信参数的签名进行对比,对比成功接口才会配置成功。express

  由于微信认证须要拥有一个本身的服务器 ,因此这里咱们就须要用到express搭建一个简单的服务器,用来完成微信的token认证和生成signature(签名),搭建的过程也很简单,参照express中文文档,下面就贴一下官网的步骤:npm


安装完成事后,进入myapp目录,建立一个app.js的文件 ,json

var express = require('express');
var crypto = require('crypto') //使用npm安装后引入,用来生成签名
var http = require('request') //express的中间件,使用npm安装,用来发出请求
var jsSHA = require('jssha')//jssha是微信官网提供的nodejs版本签名算法,能够去官网下载官网的sample包
var app = express();
app.use(express.static('./review'))

app.get('/weixin',function (req, res) {//这个get接口就是测试号填写的接口,用来响应微信服务器的请求
    var token = 'weixin' //注意这里填写token,与微信测试号申请时候填写的token要保持一致    
    var signature = req.query.signature;
    var timestamp = req.query.timestamp;    
    var nonce = req.query.nonce;    
    var echostr = req.query.echostr;   
     /*  加密/校验流程以下: */   
     //1. 将token、timestamp、nonce三个参数进行字典序排序   
     var array = new Array(token,timestamp,nonce);   
     array.sort();   
     var str = array.toString().replace(/,/g,"");      
    //2. 将三个参数字符串拼接成一个字符串进行sha1加密    
    var sha1Code = crypto.createHash("sha1");    
    var code = sha1Code.update(str,'utf-8').digest("hex");   
     //3. 开发者得到加密后的字符串可与signature对比,标识该请求来源于微信    
    if(code===signature){        
        res.send(echostr)   
    }else{
        res.send("error");
    }  
});
var server = app.listen(80, function () {
    var host = server.address().address;
    var port = server.address().port;
    console.log('Example app listening at http://%s:%s', host, port);
});复制代码

建立完成后,运行

node app.js复制代码

服务器就开启好了,上面要注意的几点就是:

1:jssha不能用npm安装,由于npm安装的运行时候会报Chosen SHA variant is not supported

,必须使用官网提供的sample包,下载解压后,选择node版本,打开后将node_module里面jssha文件复制到项目内的node_module里面便可;

2:这里的token值须要和微信测试号中填写的token值一致;

如今咱们就能够开始填写测试号的参数了,填写完成微信服务器就会发送请求给你填写的接口了,都正确响应的话就会弹出配置成功。

  固然到这尚未结束,由于前端想要调用jssdk的接口还须要经过接口请求完成权限配置,这里你们能够看一下微信jssdk的说明文档,具体引用步骤这里就不赘述了,接口请求大概以下:

这个接口主要就是提交当前的url请求服务端拿到相应的参数,完成权限配置,因此在express中还须要在写一个响应post请求的接口,这个接口作的主要的工做就是拿appid和appSerect(测试号提供)去请求微信提供的接口生成access_token,而后拿这个access_token再去请求微信提供的接口生成tiket,关于这二者文档上都有详细说明。最后生成签名,代码以下

// noncestr生成var createNonceStr = function() {
    return Math.random().toString(36).substr(2, 15);
};
// timestamp时间戳生成var createTimeStamp = function () {
    return parseInt(new Date().getTime() / 1000) + '';
};
//获取tiket
var getTiket= function (data) { //经过access_token获取tiket
   return new Promise((reslove,reject)=>{
        http.get(`https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=${data}&type=jsapi`,
         function(err,res,body){
             if(res.body.tiket){
                resoleve(res.body.ticket)
             }else{
                reject(err)
             }         })      })}
// 计算签名方法
var calcSignature = function (ticket, noncestr, ts, url) {//使用jssha
    var str = 'jsapi_ticket=' + ticket + '&noncestr=' + noncestr + '&timestamp='+ ts +'&url=' + url;
    shaObj = new jsSHA(str, 'TEXT');    return shaObj.getHash('SHA-1', 'HEX');
}
//返回给前端配置信息的post接口
app.post('/weixin',function(req,res,next){
     let appId = '******'
     let appSecret = '******'
     let url = `https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=${appId}&secret=${appSecret }`
     http.get(url, function (err, response, body) {
       getTiket(response.body).then(resolve=>{
          let tiket = resolve//tiket
          let nonceStr = createNonceStr()//随机字符串
          let timeStamp = createTimeStamp()//时间戳
          let signature = calcSignature(tiket,nonceStr,timeStamp,req.body.url)
          let obj = { //将前端须要的参数返回
            data:{
                appId:appId,
                timestamp:timeStamp,
                nonceStr:nonceStr,
                signature:signature
            } 
         } 
         res.end(JSON.stringify(obj))
        }).catch(err=>{})
          res.end(JSON.stringify(err))
     });})复制代码

这里要注意的是微信返回的access_token 和tiket的都有7200s的有效期,因此要进行缓存,个人代码中没有写缓存的操做代码了,你们有两种方法:

1.拿到access_token和tiket后,直接写在变量中存下来,有效期内就不用继续请求接口了,直接进行签名操做就能够了;过时后,在请求一次就行了,虽然这种方法有点笨,不过好歹有效期还算长。

2.在服务器拿到access_token和tiket后,写入本地的json文件中,具体步骤也不赘述了,而后判断是否过时,过时后就从新请求,没过时就直接读取json文件中的数据进行签名。

  最后呢,有两种选择:

第一:把咱们的前端项目执行npm run build后,把dist文件放入咱们的服务器文件夹中,能够直接用express的static中间件

app.use(express.static('./dist'))复制代码

而后微信开发者工具,输入分配的域名打开咱们的项目,这样咱们不用设置代理了,不过须要执行build,项目大一点的话仍是有点浪费时间的;

第二:就是为咱们的开发环境在申请一个域名,由于如今脚手架的热更新其实就是启动了一个webpack-dev-sever的微服务器,申请域名是后填写开发的端口号就能够了,使得开发地址和咱们的服务器地址的二级域名相同,不过对于服务器的接口,开发环境须要设置一下代理,并且热更新也会失效,须要手动刷新一下,不过相对于第一种方法可能会更好一点。

两种方法运行成功后查看发出请求若是配置成功,控制台会出现配置成功的信息以下:

而后咱们就能够愉快的在使用jssdk的接口了,不再求后端,能够本身在本地测试好全部的接口了,且不是美滋滋。

  写的过程当中已经尽可能详细说明了,不过不免会漏掉一些细节问题,欢迎指正和讨论。

相关文章
相关标签/搜索