微信公眾號(hào)開發(fā)蜡饵,微信官方只開放了個(gè)人訂閱號(hào)給個(gè)人開發(fā)者,很多事件推送接口都有限制黔夭,所以自己申請(qǐng)了一個(gè)測(cè)試號(hào)宏胯,相對(duì)于個(gè)人未認(rèn)證的訂閱號(hào),多了一些接口本姥,下面就微信開發(fā)者文檔肩袍,用Nodejs的express框架實(shí)現(xiàn)了基本的事件回復(fù),利用一些普通的api實(shí)現(xiàn)微信公眾號(hào)常見(jiàn)的功能婚惫。代碼方面沒(méi)有使用市面比較流行的wechat和wechat-api開發(fā)框架氛赐,純粹使用原生的消息模板進(jìn)行事件的回復(fù)。
-
準(zhǔn)備工作:
- 微信公眾號(hào)的開發(fā)需要一臺(tái)公網(wǎng)可以訪問(wèn)的服務(wù)器先舷,現(xiàn)在百度BAE艰管,騰訊的服務(wù)器,阿里云的ECS對(duì)于學(xué)生都具有優(yōu)惠活動(dòng)蒋川,國(guó)外的AWS更有新用戶免費(fèi)一年的活動(dòng)牲芋。自己買的是阿里云學(xué)生優(yōu)惠ECS,價(jià)格是¥9.9尔破;關(guān)于阿里云服務(wù)器的配置可以直接參考其他教程街图。
- 微信公眾號(hào)的域名驗(yàn)證,可以參考上一遍文件懒构,關(guān)于微信接入指南餐济。
-
對(duì)于微信各種事件回復(fù)的代碼如下:
- 對(duì)于微信事件回復(fù)的代碼如下:
var express = require('express');
var path = require('path');
var favicon = require('serve-favicon');
var logger = require('morgan');
var xmlParse=require('xml2js').parseString;
var cookieParser = require('cookie-parser');
var bodyParser = require('body-parser');
var sha1=require('sha1');
var routes = require('./routes/index');
var users = require('./routes/users');
var common=require('./common');
var request=require('request');
var app = express();
var data='';
// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'jade');
// uncomment after placing your favicon in /public
//app.use(favicon(path.join(__dirname, 'public', 'favicon.ico')));
app.use(logger('dev'));
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({ extended: false }));
app.use(cookieParser());
app.use(express.static(path.join(__dirname, 'public')));
app.use('/', routes);
app.use('/users', users);
app.get('/validate',(req,res,next)=>{
let token='wechat';
let signature=req.query.signature;
let timestamp=req.query.timestamp;
let echostr=req.query.echostr;
let nonce=req.query.nonce;
let oriArray=new Array();
oriArray.push(nonce);
oriArray.push(timestamp);
oriArray.push(token);
let original=oriArray.sort().join('');
let combineStr=sha1(original);
if(signature==combineStr){
res.send(echostr);
}else{
console.log('error');
}
next();
});
//對(duì)于各種微信服務(wù)器的事件推送進(jìn)行回復(fù)
app.post('/validate',(req,res,next)=>{
var data='';
req.on('data',(chunk)=>{
data+=chunk;
});
req.on('end',()=>{
xmlParse(data,(err,result)=>{
if(result.xml.MsgType=='event'){
if(result.xml.Event=='subscribe'){
common.dealText('歡迎關(guān)注聊天小喵;\n 1.發(fā)送地址定位信息可以獲得當(dāng)?shù)靥鞖馇闆r喲\n 2.發(fā)送歌曲名字可以獲取想要聽(tīng)的歌喲\n3.發(fā)送語(yǔ)音信息可以直接小喵聊天喲,她會(huì)很多東西喲\n圖片,視頻識(shí)別功能還在開發(fā)中\(zhòng)n',res,result);
}else if(result.xml.Event=='CLICK' && result.xml.EventKey=='V1001_IT_NEWS'){
common.requestMsg('https://api.tianapi.com/it/?key=APIkey&num=2',result,res);//API
key是天行數(shù)據(jù)的新聞apikey
}else if(result.xml.Event=='CLICK' && result.xml.EventKey=='V1001_TRAVEL_NEWS'){
common.requestMsg('https://api.tianapi.com/travel/?key=APIkey&num=2',result,res);//如上
}else if(result.xml.Event=='CLICK' && result.xml.EventKey=='V1001_VR_NEWS'){
common.requestMsg('https://api.tianapi.com/vr/?key=APIkey&num=2',result,res);//如上
}else if(result.xml.Event=='CLICK' && result.xml.EventKey=='V1001_AMUSE_NEWS'){
common.requestMsg('https://api.tianapi.com/huabian/?key=APIkey&num=2',result,res);
}
}
if(result.xml.MsgType=='location'){
let lat=(result.xml.Location_X);
let log=(result.xml.Location_Y);
const url='http://api.yytianqi.com/forecast7d?city='+lat+','+log+'&key=APIKEY';
common.requestWeather(url,result,res);
}
if(result.xml.MsgType=='image'){
let str= '<xml><ToUserName><![CDATA['+result.xml.FromUserName+']]></ToUserName><FromUserName><![CDATA['+result.xml.ToUserName+']]></FromUserName><CreateTime>'+new Date().getTime()+'</CreateTime><MsgType><![CDATA['+'image'+']]></MsgType><Image><MediaId><![CDATA['+result.xml.MediaId+']]></MediaId></Image></xml>';
res.send(str);
}
if(result.xml.MsgType=='text'){
let responseMSg=(result.xml.Content).toString();
let url=encodeURI("http://s.music.163.com/search/get?type=1&limit=10&offset=0&s="+responseMSg);
common.requestSong(url,result,res);
}
if(result.xml.MsgType=='voice'){
let url='http://www.tuling123.com/openapi/api?key=APIKEY&info='+encodeURI(result.xml.Recognition.toString());
common.requestRobot(url,result,res);
}
if(result.xml.MsgType=='video'){
let str='<xml><ToUserName><![CDATA['+result.xml.FromUserName+']]></ToUserName><FromUserName><![CDATA['+result.xml.ToUserName+']]></FromUserName><CreateTime>'+new Date().getTime()+'</CreateTime><MsgType><![CDATA['+'video'+']]></MsgType><Video><MediaId><![CDATA['+result.xml.MediaId+']]></MediaId><Title><![CDATA['+'video_info'+']]></Title><Description><![CDATA['+'information'+']]></Description></Video></xml>'
res.send(str);
}
});
});
});
// catch 404 and forward to error handler
app.use(function(req, res, next) {
var err = new Error('Not Found');
err.status = 404;
next(err);
});
// error handlers
// development error handler
// will print stacktrace
if (app.get('env') === 'development') {
app.use(function(err, req, res, next) {
res.status(err.status || 500);
res.render('error', {
message: err.message,
error: err
});
});
}
// production error handler
// no stacktraces leaked to user
app.use(function(err, req, res, next) {
res.status(err.status || 500);
res.render('error', {
message: err.message,
error: {}
});
});
module.exports = app;
- 各種事件回復(fù)函數(shù)模塊文件
var request=require('request');
//click事件推送
function requestMsg(url,result,res){
request(url,(err,response,body)=>{
let data=JSON.parse(body);
let arr=data.newslist;
let str='<xml><ToUserName><![CDATA['+result.xml.FromUserName+']]></ToUserName><FromUserName><![CDATA['+result.xml.ToUserName+']]></FromUserName><CreateTime>'+new Date().getTime()+'</CreateTime><MsgType><![CDATA['+'news'+']]></MsgType><ArticleCount>'+'2'+'</ArticleCount><Articles><item><Title><![CDATA['+arr[0].title+']]></Title> <Description><![CDATA['+arr[0].description+']]></Description><PicUrl><![CDATA['+arr[0].picUrl+']]></PicUrl><Url><![CDATA['+arr[0].url+']]></Url></item><item><Title><![CDATA['+arr[1].title+']]></Title><Description><![CDATA['+arr[1].deacription+']]></Description><PicUrl><![CDATA['+arr[1].picUrl+']]></PicUrl><Url><![CDATA['+arr[1].url+']]></Url></item></Articles></xml>';
res.send(str);
});
}
//地理位置事件處理
function requestWeather(url,result,res){
request(url,(err,response,body)=>{
if(err){
console.log(err);
}
data=JSON.parse(body);
dealText('城市: '+data.data.cityName+'\n時(shí)間:'+data.data.sj+'\n天氣情況:'+data.data.list[0].tq1+'\n白天溫度:'+data.data.list[0].qw1+'度\n夜間溫度'+data.data.list[0].qw2+'度\n白天風(fēng)向 :'+data.data.list[0].fx1+'\n夜間風(fēng)向:'+data.data.list[0].fx2+'\n',res,result);
});
}
//處理文本點(diǎn)歌
function requestSong(url,result,res){
request(url, function (error, response, body) {
if(error){
dealText('出錯(cuò)請(qǐng)重試胆剧!',res,result);
}else if (!error && response.statusCode == 200) {
var data=JSON.parse(body);
if(data.result){
let url=data.result.songs[0].audio;
console.log(url);
request(url,(error, response, body)=>{
if(response["headers"]["content-type"]=='text/html'){
dealText('沒(méi)有資源絮姆,請(qǐng)選擇其他歌曲',res,result);
}else{
let picUrl=data.result.songs[0].album.picUrl;
let str='<xml><ToUserName><![CDATA['+result.xml.FromUserName+']]></ToUserName><FromUserName><![CDATA['+result.xml.ToUserName+']]></FromUserName><CreateTime>'+new Date().getTime()+'</CreateTime><MsgType><![CDATA['+'music'+']]></MsgType><Music><Title><![CDATA['+data.result.songs[0].name+']]></Title><Description><![CDATA['+'悠悠音樂(lè),縷縷動(dòng)聽(tīng)'+']]></Description><MusicUrl><![CDATA['+url+']]></MusicUrl><HQMusicUrl><![CDATA['+url+']]></HQMusicUrl><ThumbMediaId><![CDATA['+'ojH9Q9iDl50J6PjyFmQcYYKg51COLtm2SJpFdDFzR0jYNYt4JOjtfee0LKNDYzQa'+']]></ThumbMediaId></Music></xml>';
res.send(str);
}
});
}else{
dealText('發(fā)送歌曲名字不合法,請(qǐng)重試',res,result);
}
}
});
}
function requestRobot(url,result,res){
request(url,(error,response,body)=>{
if(!error && response.statusCode==400){
dealText('智能小喵我病了秩霍,病好了再和你聊天篙悯。。铃绒。',res,result);
}
let data=JSON.parse(body);
if(data.url){
dealText(data.text+data.url,res,result);
}else{
dealText(data.text,res,result);
}
});
}
function dealText(responseMSg,res,result){
let str='<xml><ToUserName><![CDATA['+result.xml.FromUserName+']]></ToUserName><FromUserName><![CDATA['+result.xml.ToUserName+']]></FromUserName><CreateTime>'+new Date().getTime()+'</CreateTime><MsgType><![CDATA['+'text'+']]></MsgType><Content><![CDATA['+responseMSg+']]></Content></xml>';
res.send(str);
}
exports.requestSong=requestSong;
exports.requestWeather=requestWeather;
exports.requestMsg=requestMsg;
exports.dealText=dealText;
exports.requestRobot=requestRobot;
- 實(shí)現(xiàn)的功能效果圖如下所示:
微信公眾號(hào)效果圖
- 微信測(cè)試號(hào)開發(fā)實(shí)例代碼如上所示現(xiàn)在已經(jīng)實(shí)現(xiàn)衛(wèi)星自定義菜單鸽照,文本消息回復(fù),語(yǔ)音消息回復(fù),地理位置事件回復(fù)颠悬,已經(jīng)圖靈機(jī)器人接入矮燎,語(yǔ)音識(shí)別功能的實(shí)現(xiàn)
源代碼地址微信公眾號(hào)開發(fā).覺(jué)得有幫助定血,點(diǎn)個(gè)star喲。