微信公众平台开发

本门课程主要讲的是如何通过Nodejs来开发与微信公众平台的接口。

基本原理

  先来简单介绍微信公众平台的基本原理。

  微信服务器就相当于一个转发服务器,终端(手机、Pad等)发起请求至微信服务器,微信服务器,然后将请求转发给自定义服务(这里就是我们的具体实现)。服务处理完毕,然后转发给微信服务器,微信服务器再将具体响应回复到终端;通信协议为:HTTP;数据格式为:XML

具体的流程如下图所示:

微信公众平台开发_第1张图片

  其实,我们需要做的事情,就是对HTTP请求,做出响应。具体的请求内容,我们按照特定的XML格式去解析,处理完毕后,也要按照特定的XML格式返回。

平台注册

  要想完成对微信公众平台的开发,我们需要注册一个微信公众平台帐号。注册步骤如下:

  1. 打开微信公共平台的官网,https://mp.weixin.qq.com/,点击“立即注册”。微信公众平台开发_第2张图片
  2. 然后根据提示,填写基本信息,邮箱激活,选择类型,信息登记,公众号信息,完成注册。

  在注册完成以后,我们要对公众号进行一些基本的设置。登录公众号,找到【公众号设置】,然后设置头像以及其它信息。

微信公众平台开发_第3张图片

nodejs环境搭建

  我们需要在公网上找一台服务器,以便可以启动我们的nodejs的环境,启动环境后通过设置访问地址,我们就可以接收微信服务器发送的消息了,并且我们也可以向微信服务器发送消息了。

  在公网的服务器中安装完成nodejs以后,我们还需要安装一些nodejs所用到的模块,如:expressnode-xmljssha等模块。可以通过npm命令进行安装。

  我们通过nodejs来实现向微信服务器消息的发送与接收,以及与微信服务器的签名认证。

  在我们右面的编辑环境中已经为同学们安装了nodejs环境。我们在接下来内容中就为同学们来实现微信服务器的签名认证。

创建express框架

  我们在前面的课程中已经安装了express模块,并且在我们右面的环境中已经创建了一个名为app.js的文件。现在我们就在这个文件中完成express框架。如下代码:

  
  
  
  
  1. var express = require("express");
  2. var path=require('path');
  3. var app = express();
  4. server = require('http').Server(app);
  5. app.set('views',__dirname); // 设置视图
  6. app.set('view engine', 'html');
  7. app.engine( '.html', require( 'ejs' ).__express );
  8. require('./index')(app); //路由配置文件
  9. server.listen(80,function(){
  10. console.log('App start,port 80.');
  11. });

  然后再添加一个名为test.html的文件。写入以下内容

  
  
  
  
  1. <!DOCTYPE html>
  2. <html>
  3. <head lang="en">
  4. <meta charset="UTF-8">
  5. <title>中文网</title>
  6. </head>
  7. <body>
  8. <div><%=issuccess%></div>
  9. </body>
  10. </html>

  我们还要添加一个名为index.js的文件,来实现我们的路由。点击编辑环境中的添加文件按钮,添加文件,然后我们写入以下代码,其中GET请求用来验证配置的URL合法性,POST请求用来处理微信消息。

  
  
  
  
  1. module.exports = function(app){
  2.   app.get('/',function(req,res){
  3. res.render('test',{issuccess:"success"})
  4. });
  5. app.get('/interface',function(req,res){});
  6. app.post('/interface',function(req,res){});
  7. }

  这样我们需要的express框架就完成了,当然我们还可以添加public公共文件夹以及我们要用到的中间件。保存文件,点击【提交运行】,然后点击【访问测试】,去试试吧。记下访问测试的地址,我们将在下一节中会用到该地址。

微信服务器配置

  我们登录微信公众平台,在开发者模式下面找到基本配置,然后修改服务器配置。如图所示:

微信公众平台开发_第4张图片

  首先URL要填写公网上我们安装nodejs接收与发送数据的路径。我们可以填写上节中【访问测试】的地址,然后加上对应的路由就可以了。

  
  
  
  
  1. http://724515db515222a9efffd6b092aa955d.me.hubwiz.com/interface

  上面代码是我的访问测试的地址,然后加上前面课程中的路由,同学们要根据自己的访问测试地址与路由来填写。

  Token要与我们自定义服务器端的token一致。填写完成以后,就可以点击提交了,在提交以前,我们启动app.js(点击【提交运行】)。这样根据我们的路由匹配就可以验证签名是否有效了。

  当配置完成以后,一定要启用配置。

微信公众平台开发_第5张图片

网址接入

  公众平台用户提交信息后,微信服务器将发送GET请求到填写的URL上,并且带上四个参数:

参数 描述
signature 微信加密签名
timestamp 时间戳
nonce 随机数
echostr 随机字符串

  开发者通过检验signature对请求进行校验(下面有校验方式)。若确认此次GET请求来自微信服务器,请原样返回echostr参数内容,则接入生效,否则接入失败。

  signature结合了开发者填写的token参数和请求中的timestamp参数、nonce参数。

  加密/校验流程:

  1. tokentimestampnonce三个参数进行字典序排序;
  2. 将三个参数字符串拼接成一个字符串进行sha1加密;
  3. 开发者获得加密后的字符串可与signature对比,标识该请求来源于微信。

参数排序

  首先我们确认请求是来自微信服务器的get请求,那么就可以在index.js文件中进行添加代码了。然后在app.get('/interface',function(req,res){});function中进行添加。

  先来获取各个参数的值,如下代码:

  
  
  
  
  1. var token="weixin";
  2. var signature = req.query.signature;
  3. var timestamp = req.query.timestamp;
  4. var echostr = req.query.echostr;
  5. var nonce = req.query.nonce;

  我们在这里对token进行设置,让其与微信服务器中设置的token一致。

  然后对其中的tokentimestampnonce进行排序,如下代码:

  
  
  
  
  1. var oriArray = new Array();
  2. oriArray[0] = nonce;
  3. oriArray[1] = timestamp;
  4. oriArray[2] = token;
  5. oriArray.sort();

这样我们就完成了排序。

参数加密

  在上节中我们已经对参数进行了排序,然后我们在这一节中要将参数组成一个字符串,进行SH-1加密。在加密以前要用到jssha模块,在我们的文件中要引用该模块。

  
  
  
  
  1. var jsSHA = require('jssha');

  在上一节课中我们已经对参数排序完成,并存放在数组中,我们可以通过join方法来生成一个字符串,如下代码:

  
  
  
  
  1. var original = oriArray.join('');

  最后对该数据进行加密,如下代码:

  
  
  
  
  1. var jsSHA = require('jssha');
  2. var shaObj = new jsSHA(original, 'TEXT');
  3. var scyptoString=shaObj.getHash('SHA-1', 'HEX');

  好了这样就生成了我们需要的签名字符串scyptoString

签名对比

  我们已经得到了我们想要的签名字符串scyptoString,然后我们就可以与来自微信服务器的签名进行对比了,对比通过,则我们就可以接收与发送消息了。

  
  
  
  
  1. if(signature == scyptoString){
  2. //验证成功
  3. } else {
  4. //验证失败
  5. }

接收消息

  在接收消息之前,我们还需要验证与微信平台的签名,前面几节已经讲过了,这里不在详述。现在就来讲解一下接收微信平台的消息。在签名认证以后,我们就可以接收消息了。在index.jsapp.post方法中添加以下功能。

  1. 签名认证;
  2. 认证成功后,接收消息;
  3. 解析消息
  4. 回复消息(文本,图片……)

接收消息的代码如下:

  
  
  
  
  1. var post_data="";
  2. req.on("data",function(data){post_data=data;});
  3. req.on("end",function(){
  4. var xmlStr=post_data.toString('utf-8',0,post_data.length);
  5. //解析消息代码
  6. //回发消息代码
  7. });

上面代码中的xmlStr就是我们收到的微信平台发过来的消息。

解析消息

  在解析消息之前,我们需要在index.js中引入一个解析消息的模块node-xml,通过该模块来解析消息。

  
  
  
  
  1. var xml=require('node-xml');

如果我们发送给微信平台的消息为文本消息,则收到的消息格式如下:

  
  
  
  
  1. <xml>
  2. <ToUserName><![CDATA[toUser]]></ToUserName>
  3. <FromUserName><![CDATA[fromUser]]></FromUserName>
  4. <CreateTime>1348831860</CreateTime>
  5. <MsgType><![CDATA[text]]></MsgType>
  6. <Content><![CDATA[this is a test]]></Content>
  7. <MsgId>1234567890123456</MsgId>
  8. </xml>

微信公众平台开发_第6张图片

然后我们就通过node-xml模块来解析,解析的代码如下:

  
  
  
  
  1. // 定义解析存储变量
  2. var ToUserName="";
  3. var FromUserName="";
  4. var CreateTime="";
  5. var MsgType="";
  6. var Content="";
  7. var tempName="";
  8. //开始解析消息
  9. var parse=new xml.SaxParser(function(cb){
  10. cb.onStartElementNS(function(elem,attra,prefix,uri,namespaces){
  11. tempName=elem;
  12. });
  13. cb.onCharacters(function(chars){
  14. chars=chars.replace(/(^\s*)|(\s*$)/g, "");
  15. if(tempName=="CreateTime"){
  16. CreateTime=chars;
  17. }
  18. });
  19. cb.onCdata(function(cdata){
  20. if(tempName=="ToUserName"){
  21. ToUserName=cdata;
  22. }else if(tempName=="FromUserName"){
  23. FromUserName=cdata;
  24. }else if(tempName=="MsgType"){
  25. MsgType=cdata;
  26. }else if(tempName=="Content"){
  27. Content=cdata;
  28. }
  29. console.log(tempName+":"+cdata);
  30. });
  31. cb.onEndElementNS(function(elem,prefix,uri){
  32. tempName="";
  33. });
  34. cb.onEndDocument(function(){
  35. //按收到的消息格式回复消息
  36. });
  37. });
  38. parse.parseString(xmlStr);

通过以上代码的实现,这样收到的消息就解析完成了。 大家想了解更多收到数据的格式,请查看微信平台的官方文档 。

文本消息

  根据前一节中我们解析出来的内容,然后对应组织我们的发送数据包,数据包格式如下:

  
  
  
  
  1. <xml>
  2. <ToUserName><![CDATA[toUser]]></ToUserName>
  3. <FromUserName><![CDATA[fromUser]]></FromUserName>
  4. <CreateTime>12345678</CreateTime>
  5. <MsgType><![CDATA[text]]></MsgType>
  6. <Content><![CDATA[你好]]></Content>
  7. </xml>

微信公众平台开发_第7张图片

  关于更多被动回复数据格式请查看微信官方文档。

回复文本消息的代码如下:

  
  
  
  
  1. CreateTime=parseInt(new Date().getTime() / 1000);
  2. var msg="";
  3. if(MsgType=="text"){
  4. msg="谢谢关注,你说的是:"+Content;
  5. //组织返回的数据包
  6. var sendMessage='
  7. <xml>
  8. <ToUserName><![CDATA['+FromUserName+']]></ToUserName>
  9. <FromUserName><![CDATA['+ToUserName+']]></FromUserName>
  10. <CreateTime>'+CreateTime+'</CreateTime>
  11. <MsgType><![CDATA[text]]></MsgType>
  12. <Content><![CDATA['+msg+']]></Content>
  13. </xml>';
  14. res.send(sendMessage);
  15. }

  好了,这样我们的服务端就完成了,点击【提交运行】。现在拿起手机,扫描二维码关注我们的微信公众号,发送消息,看一看结果吧。

  关于其它消息的回复,同学们按照微信官方说明组织数据包进行回复就可以了。

你可能感兴趣的:(微信公众平台开发)