詳解nodejs微信公眾號開發(fā)——2.自動回復
上一篇文章:nodejs微信公眾號開發(fā)(1)接入微信公眾號,本篇文章將在此基礎上實現(xiàn)簡單的回復功能。
1. 接入代碼的優(yōu)化
之前我們簡單粗暴的實現(xiàn)了微信公眾號的接入,接入的代碼直接寫在了app.js
文件里面,從項目開發(fā)的角度而言,不便于日后代碼的維護,所以將這部分代碼獨立出來,按照koa
的風格,寫成一個中間件。
在根目錄下新建wechat
文件夾,新建generator.js
文件,
var sha1 = require('sha1'); module.exports = function(opts){ return function *(next){ var token = opts.token; var signature = this.query.signature; var nonce = this.query.nonce; var timestamp = this.query.timestamp; var echostr = this.query.echostr; var str = [token,timestamp,nonce].sort().join(''); var sha = sha1(str); this.body = (sha === signature) ? echostr + '' : 'failed'; }; }
此時app.js
的內(nèi)容變成:
'use strict' var Koa = require('koa'); var wechat = require('./wechat/generator'); var config = { wechat:{ appID:'...', appSecret:'...', token:'...' } }; var app = new Koa(); app.use(wechat(config.wechat)); app.listen(8080); console.log('Listening 8080...')
2. 獲取access_token
access_token
是開發(fā)程序與wexin公眾平臺交互的一把鑰匙,調(diào)用絕大部分接口都需要用到access_token
。
access_token
的特點:
- 有效期為2小時(7200s),過期自動失效,需要重新獲??;
- 只要更新了access_token,之前的access_token自動失效;
解決方案:
- 系統(tǒng)每隔2小時自動去獲取一下access_token的值,確保access_token始終是有效的;
- 為了方便頻繁調(diào)用,將access_token存儲在唯一的一個地方(數(shù)據(jù)庫、文件等),所有子系統(tǒng)都能訪問。
程序中采用構(gòu)造函數(shù)的方式,在生成實例,完成初始化工作的的過程中,讀取存儲在config/wechat.txt
文件中的票據(jù),判斷是否為空且是否過期,選擇性的重新獲取數(shù)字并且保存在原文件里面,關于獲取access_token的官方文檔介紹可見:獲取access_token。
function Wechat(opts){ //構(gòu)造函數(shù),用以生成實例,完成初始化工作,讀寫票據(jù) var that = this; this.appID = opts.appID; this.appSecret = opts.appSecret; this.getAccessToken = opts.getAccessToken; this.saveAccessToken = opts.saveAccessToken; this.getAccessToken().then(function(data){ try{ data = JSON.parse(data); }catch(e){ return that.updateAccessToken(); } if(that.isvalidAccessToken(data)){ Promise.resolve(data); }else{ return that.updateAccessToken(); } }).then(function(data){ that.access_token = data.access_token; that.expires_in = data.expires_in; that.saveAccessToken(JSON.stringify(data)); }); }
我們在moudle.exports
中實例化一個Wechat
:
var wechat = new Wechat(opts);
這樣確保了每次程序啟動都會獲取對access_token的有效性進行檢驗,并且每個一段時間會自動獲取一個新的access_token。
3. 處理微信消息的步驟
無論是事件推送還是消息推送,微信服務器都是以post的方式發(fā)送請求,推送的數(shù)據(jù)類型不是json
而是xml
,處理推送消息一般分為五個步驟:
- 處理POST類型的控制邏輯,接收xml數(shù)據(jù)包;
- 解析數(shù)據(jù)包,獲取數(shù)據(jù)包的消息類型或數(shù)據(jù)類型;
- 拼裝自定義的消息;
- 包裝成xml格式;
- 在5秒鐘內(nèi)返回消息。
3.1 接收xml數(shù)據(jù)
通過raw-body模塊可以獲取http模塊中的request對象,并且可以對數(shù)據(jù)進行拼裝,從而拿到一個buffer的xml對象
var data = yield rawBody(this.req,{ length:this.length, limit:'1mb', encoding:this.charset }); console.log('data:'+data);
3.2 解析xml數(shù)據(jù)
使用xml2js模塊,將xml數(shù)據(jù)解析成對象格式
var content = yield util.parseXMLAsync(data); util中的parseXMLAsync方法: exports.parseXMLAsync = function(xml){ return new Promise(function(resolve,reject){ xml2js.parseString(xml,{trim:true},function(err,content){ err ? reject(err) : resolve(content); }) }); }
3.3 格式化xml數(shù)據(jù)
從解析的xml數(shù)據(jù)來看,數(shù)據(jù)雖然已經(jīng)呈現(xiàn)鍵值對的形式,但是其值是數(shù)組的形式,需要進行扁平化處理:
var message = util.formatMessage(content.xml);
其本質(zhì)就是遍歷數(shù)組中的值,因為在多圖文的消息中存在嵌套的情況:
function formatMessage(result){ var message = {}; if(typeof result === 'object'){ var keys = Object.keys(result); for(var i=0;i<keys.length;i++){ var key = keys[i]; var item = result[key]; if(!(item instanceof Array) || item.length === 0) continue; if (item.length === 1){ var val = item[0]; if (typeof val === 'object') message[key] = formatMessage(val); else message[key] = (val || '').trim(); }else{ message[key] = []; for(var j=0,k=item.length;j<k;j++) message[key].push(formatMessage(item[j])); } } } return message; }
3.4 判斷消息類型并回復
這里針對subscribe
事件,新關注后自動回復文本消息終于等到你,還好我沒放棄
if(message.MsgType === 'event'){ if(message.Event === 'subscribe'){ var createTime = new Date().getTime(); that.status = 200; that.type = 'application/xml'; that.body = '<xml>'+ '<ToUserName><![CDATA['+ message.FromUserName +']]></ToUserName>'+ '<FromUserName><![CDATA['+ message.ToUserName +']]></FromUserName>'+ '<CreateTime>'+createTime+'</CreateTime>'+ '<MsgType><![CDATA[text]]></MsgType>'+ '<Content><![CDATA[終于等到你,還好我沒放棄]]></Content>'+ '</xml>' return; } }
注:這里只是簡單地實現(xiàn)一下自動回復功能,這種拼接字符串的方式還是很不方便的,后面會封裝成接口。
使用手機微信掃描測試賬號的二維碼,即可關注,同時接收到測試公眾號推送的消息!
啦啦,一個簡單的關注回復就完成了。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Windows系統(tǒng)下Node.js的簡單入門教程
這篇文章主要介紹了Windows系統(tǒng)下Node.js的簡單入門教程,Node.js是用于后端編程的JavaScript框架,需要的朋友可以參考下2015-06-06