基于koa2開發(fā)微信公眾號(hào)(擼一個(gè)消息回復(fù))

當(dāng)準(zhǔn)備工作完成的時(shí)候,先來(lái)一個(gè)簡(jiǎn)單的功能,回復(fù)一個(gè)固定的消息,大致就是當(dāng)用戶關(guān)注該公眾號(hào)的時(shí)候,公眾號(hào)會(huì)回復(fù)一則文本消息。

在開始之前呢,首先概述一下這篇文章的重點(diǎn),第一,公眾號(hào)的取消、關(guān)注、用戶發(fā)送的內(nèi)容都是POST請(qǐng)求;第二,公眾號(hào)發(fā)送的內(nèi)容不是json格式,而是xml格式,是不是有點(diǎn)腦殼疼;第三,返回給用戶的消息格式也是xml格式。

在這里,我采用raw-body及xml2js模塊來(lái)將xml數(shù)據(jù)格式轉(zhuǎn)換為json。

//g.js部分代碼
const sha1=require('sha1');
const getRawBody=require('raw-body');
const util=require('./util');
module.exports=function(opts){
    var wechat=new WeChat(opts);
    return async function(ctx){
        const signature=ctx.query.signature,
              timestamp=ctx.query.timestamp,
              nonce=ctx.query.nonce,
              token=opts.token;
        //字典排序
        const str=[token,timestamp,nonce].sort().join('');
        const result=sha1(str);
        if(result===signature){
            if(ctx.request.method==="GET"){
                ctx.body=ctx.query.echostr;
            }else if(ctx.request.method==="POST"){
                try{
                    var data=await getRawBody(ctx.req,{
                        length: ctx.request.length,
                        limit: "1mb",
                        encoding: ctx.request.charset
                    });
                    var content=await util.parseXMLAsync(data);
                    console.log(content);
                }catch(err){
                    console.log(err);
                }
            }
        }else{
            ctx.body={
                code:-1,
                msg:"fail"
            }
        }
    }   
}


//util.js部分代碼
var xml2js=require('xml2js');
exports.parseXMLAsync=function(xml){
    return new Promise((resolve,reject)=>{
        xml2js.parseString(xml,{trim:true},function(err,content){
            if(err){
                reject(err);
            }
            resolve(content);
        });
    });
}

當(dāng)我關(guān)注公眾號(hào)的時(shí)候,可以發(fā)現(xiàn)控制臺(tái)打印結(jié)果如下圖所示:


打印結(jié)果

很容易發(fā)現(xiàn)這個(gè)數(shù)據(jù)仍然不是最終想要的json格式,因?yàn)樗拿恳粋€(gè)值都是個(gè)數(shù)組格式,所以需要再在util.js當(dāng)中增加一個(gè)方法來(lái)將這種數(shù)據(jù)轉(zhuǎn)換為需要的格式。

function formatMessage(result){
    var message={};
    if(typeof result ==='object'){
        var keys=Object.keys(result);
        for(let i=0;i<keys.length;i++){
            var item=result[keys[i]];
            var key=keys[i];
            if(!(item instanceof Array) || item.length===0){
                continue;
            }
            else if(item.length===1){
                var val=item[0];
                if(typeof val === 'object'){
                    messgae[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;
}

exports.formatMessage=formatMessage

最后就是返回消息內(nèi)容了,具體的返回格式可參考公眾號(hào)官方文檔

if(result===signature){
    if(ctx.request.method==="GET"){
        ctx.body=ctx.query.echostr;
    }else if(ctx.request.method==="POST"){
        try{
            var data=await getRawBody(ctx.req,{
                length: ctx.request.length,
                limit: "1mb",
                encoding: ctx.request.charset
            });
            var content=await util.parseXMLAsync(data);
            var message=util.formatMessage(content.xml);
            if(message.MsgType==='event'){
                if(message.Event==='subscribe'){//關(guān)注
                    var now = Date.parse(new Date())/1000;
                    ctx.type='application/xml';
                    var reply=`<xml> 
                                <ToUserName><![CDATA[${message.FromUserName}]]></ToUserName>
                                <FromUserName><![CDATA[${message.ToUserName}]]></FromUserName>
                                <CreateTime>${now}</CreateTime>
                                <MsgType><![CDATA[text]]></MsgType>
                                <Content><![CDATA[終于等到你,還好我沒放棄,歡迎關(guān)注lk儒家測(cè)試公眾號(hào)!]]></Content>
                            </xml>`;
                    ctx.body=reply;
                }
            }
        }catch(err){
            console.log(err);
        }
    }
}else{
    ctx.body={
        code:-1,
        msg:"fail"
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,534評(píng)論 19 139
  • 點(diǎn)擊查看原文 Web SDK 開發(fā)手冊(cè) SDK 概述 網(wǎng)易云信 SDK 為 Web 應(yīng)用提供一個(gè)完善的 IM 系統(tǒng)...
    layjoy閱讀 14,299評(píng)論 0 15
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,881評(píng)論 25 709
  • 博物館是個(gè)奇妙的地方,文化遺跡也是個(gè)奇妙的地方,壯美自然燈紅酒綠同樣是奇妙的地方,但是——去奇妙的地方,不一定有奇...
    dragonfly蜻蜓點(diǎn)水閱讀 369評(píng)論 0 1
  • 世界給我一個(gè)心情,音樂給我一個(gè)心情!身處異世!鐘愛音樂
    遲繪閱讀 162評(píng)論 0 0

友情鏈接更多精彩內(nèi)容