微信公眾號(hào)有個(gè)規(guī)則,一旦開(kāi)啟了開(kāi)發(fā)者模式,其他的常規(guī)功能就都必須通過(guò)接口調(diào)用完成。比如說(shuō)自定義菜單功能,必須通過(guò)發(fā)送post請(qǐng)求的方式生成。本章就通過(guò)關(guān)注到取消關(guān)注的整個(gè)過(guò)程來(lái)談一談nodejs是怎么樣與微信交互的。這些功能的入口就是你在測(cè)試公眾號(hào)里面填寫(xiě)的URL(以下用/login/wechat代替)。
事件交互
掃碼關(guān)注微信公眾號(hào)后,微信會(huì)調(diào)用你的接口/login/wechat,并且附帶一段xml信息,首先你需要獲取一些簽名,通過(guò)加密、排序比對(duì)是否與你填寫(xiě)的TOKEN一致,如果一致則進(jìn)行xml的解析。node解析xml時(shí)必須先引用模塊。所以,先引入xml解析模塊
//xml解析模塊
var XMLJS = require('xml2js');
//解析,將xml解析為json
var parser = new XMLJS.Parser();
//重組,將json重組為xml
var builder = new XMLJS.Builder();
通過(guò)req的監(jiān)聽(tīng)data,來(lái)獲取微信發(fā)送過(guò)來(lái)的xml包。以下是某個(gè)新用戶(hù)關(guān)注公眾號(hào)后微信向你的后臺(tái)接口(上一篇中提到的/yourapi)發(fā)送的xml包數(shù)據(jù),經(jīng)過(guò)解析后,他的結(jié)構(gòu)如下:
tousername:收信人【此處為公眾微信號(hào)】
fromusername:發(fā)信人【此處為用戶(hù)openid】
createTime:發(fā)送時(shí)間
msgtype:消息類(lèi)型【event(響應(yīng)事件)、text(推送消息)、image(推送圖文消息)等】
event:消息名稱(chēng)【此處為關(guān)注】
eventkey:自定義的key,在設(shè)置網(wǎng)頁(yè)時(shí)可以自定義后文中會(huì)講到
以上就是當(dāng)一個(gè)用戶(hù)關(guān)注后微信往你接口發(fā)送的數(shù)據(jù)包。上面對(duì)我們有用的是fromusername,即關(guān)注人的openid,我們?cè)陉P(guān)注時(shí)獲取了用戶(hù)的該openid后可以通過(guò)微信提供的特定接口(https://api.weixin.qq.com/cgi-bin/user/info?access_token=ACCESS_TOKEN&openid=OPENID&lang=zh_CN)獲取用戶(hù)的頭像,性別,昵稱(chēng)等信息,為你的app建立一個(gè)可靠的資料庫(kù)。
代碼實(shí)現(xiàn)
//微信事件推送的入口
app.post('/yourapi', function(req, res, next) {
//獲取參數(shù)
var query = req.query;
//簽名
var signature = query.signature;
//輸出的字符,你填寫(xiě)的TOKEN
var echostr = query.echostr;
//時(shí)間戳
var timestamp = query['timestamp'];
//隨機(jī)字符串
var nonce = query.nonce;
var oriArray = new Array();
oriArray[] = nonce;
oriArray[] = timestamp;
oriArray[] = appConfig.token;
//排序參數(shù)
oriArray.sort();
var original = oriArray[]+oriArray[]+oriArray[];
//加密
var scyptoString = sha(original);
//判斷是否與你填寫(xiě)TOKEN相等
if (signature == scyptoString) {
//獲取xml數(shù)據(jù)
req.on("data", function(data) {
//將xml解析
parser.parseString(data.toString(), function(err, result) {
var body = result.xml;
var messageType = body.MsgType[];
//用戶(hù)點(diǎn)擊菜單響應(yīng)事件
if(messageType === 'event') {
var eventName = body.Event[];
(EventFunction[eventName]||function(){})(body, req, res);
//自動(dòng)回復(fù)消息
}else if(messageType === 'text') {
EventFunction.responseNews(body, res);
//第一次填寫(xiě)URL時(shí)確認(rèn)接口是否有效
}else {
res.send(echostr);
}
});
});
} else {
//認(rèn)證失敗,非法操作
res.send("Bad Token!");
}
});
//微信客戶(hù)端各類(lèi)回調(diào)用接口
var EventFunction = {
//關(guān)注
subscribe: function(result, req, res) {
//存入openid 通過(guò)微信的接口獲取用戶(hù)的信息同時(shí)存入數(shù)據(jù)庫(kù)。
},
//注銷(xiāo)
unsubscribe: function(openid, req, res) {
//刪除對(duì)應(yīng)id
},
//打開(kāi)某個(gè)網(wǎng)頁(yè)
VIEW: function() {
//根據(jù)需求,處理不同的業(yè)務(wù)
},
//自動(dòng)回復(fù)
responseNews: function(body, res) {
//組裝微信需要的json
var xml = {xml: {
ToUserName: body.FromUserName,
FromUserName: body.ToUserName,
CreateTime: + new Date(),
MsgType: 'text',
Content: '編輯@+您想說(shuō)的話(huà),我們可以收到'
}};
var reciviMessage = body.Content[]
if(/^\@.*/.test(reciviMessage)) {
xml.xml.Content = '已經(jīng)收到您的建議,會(huì)及時(shí)處理!'
}<br>//將json轉(zhuǎn)為xml
xml = builder.buildObject(xml);<br>//發(fā)送給微信
res.send(xml);
}
}
此處,適合采用JS設(shè)計(jì)模式中的策略模式,在subscribe方法里面寫(xiě)上你自己的業(yè)務(wù),通過(guò)發(fā)送帶openid參數(shù)的請(qǐng)求,可以在用戶(hù)關(guān)注微信號(hào)的時(shí)候?qū)⑵鋷妆举Y料存入數(shù)據(jù)庫(kù),并且建立會(huì)話(huà)。這樣在用戶(hù)接下來(lái)打開(kāi)你的網(wǎng)頁(yè)的時(shí)候就無(wú)需再次認(rèn)證,只需要比對(duì)openid然后查詢(xún)數(shù)據(jù)庫(kù)就行了。