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