使用koa2對接微信公眾平臺

這幾天心血來潮,想玩下微信公眾平臺,看看它是如何對接到個人服務(wù)器的。這里簡單記錄下。

環(huán)境:
  node v7.7.1
  koa  v2.0.1

流程

1.在微信公眾平臺的 開發(fā) -> 基本配置 那填寫服務(wù)器信息;

2.點擊啟用后,微信會發(fā) Get 請求到填寫的服務(wù)器,檢查服務(wù)器是否有效;

3.服務(wù)器驗證通過后,公眾號每次接收到新消息,都會發(fā) POST 請求到服務(wù)器,然后我們就可以在服務(wù)器里進行各種處理。

驗證服務(wù)器

微信開發(fā)文檔里有寫驗證的規(guī)則(這不是廢話么):

verify.png

上面的文字已經(jīng)寫的很直白了,不多說,代碼實現(xiàn)如下:

// 驗證消息來自微信服務(wù)器
const crypto = require('crypto')
module.exports = (ctx) => {
    const token = 'xxxx', // 自定義,與公眾號設(shè)置的一致
          signature = ctx.query.signature,
          timestamp = ctx.query.timestamp,
          nonce = ctx.query.nonce
        
    // 字典排序
    const arr = [token, timestamp, nonce].sort()

    const sha1 = crypto.createHash('sha1')
    sha1.update(arr.join(''))
    const result = sha1.digest('hex')

    if (result === signature) {
        ctx.body = ctx.query.echostr
    } else {
        ctx.body = { code: -1, msg: "fail"}
    }
}

處理 POST 請求

下面要處理接收消息的 POST 請求。因為消息的格式都是 XML ,所以這里需要引入 xml2js 。

接收 XML 數(shù)據(jù)

koa2 沒有對 XML 格式的參數(shù)進行處理,這里需要我們自己來處理下,寫的中間件如下:

// xmlTool.js
const xml2js = require('xml2js')

exports.xmlToJson = (str) => {
     return new Promise((resolve, reject) => {
        const parseString = xml2js.parseString
        parseString(str, (err, result) => {
            if (err) {
                reject(err)
            } else {
                resolve(result)
            }
        })
     })
}

exports.jsonToXml = (obj) => {
    const builder = new xml2js.Builder()
    return builder.buildObject(obj)
}

// xmlParse.js
const xml = require('./xmlTool')

module.exports = () => {
    return async (ctx, next) => {
        if (ctx.method == 'POST' && ctx.is('text/xml')) {
            let promise = new Promise(function (resolve, reject) {
                let buf = ''
                ctx.req.setEncoding('utf8')
                ctx.req.on('data', (chunk) => {
                    buf += chunk
                })
                ctx.req.on('end', () => {
                    xml.xmlToJson(buf)
                        .then(resolve)
                        .catch(reject)
                })
            })

            await promise.then((result) => {
                    ctx.req.body = result
                })
                .catch((e) => {
                    e.status = 400
                })

            next()
        } else {
            await next()
        }
    }
}

加上了這個中間件,我們就可以正確接收到 XML 格式的參數(shù)了。接收到的參數(shù)如下:

// console.log(buf) 
<xml>\n<ToUserName><![CDATA[toUser]]></ToUserName>\n<FromUserName><![CDATA[fromUser]]></FromUserName>\n<C
reateTime>12345678</CreateTime>\n<MsgType><![CDATA[text]]></MsgType>\n<Content><![CDATA[你好]]></Content>\n</x
ml>\n\n

// 轉(zhuǎn)為JSON后
{ ToUserName: [ 'toUser' ],
  FromUserName: [ 'fromUser' ],
  CreateTime: [ '12345678' ],
  MsgType: [ 'text' ],
  Content: [ '你好' ] }

發(fā)送消息

接收到消息后,服務(wù)器需要在5s內(nèi)返回消息,如果沒內(nèi)容返回,可以返回 success 或空字符串。下面是返回文本信息的例子:

// wx.js
exports.message = {
    text (msg, content) {
        return xml.jsonToXml({
            xml: {
                ToUserName: msg.FromUserName,
                FromUserName: msg.ToUserName,
                CreateTime: Date.now(),
                MsgType: msg.MsgType,
                Content: content
            }
        })
    }
}

// index.js
const wx = require('./wx')
exports.postHandle = (ctx, next) => {
    let msg,
        MsgType,
        result

    msg = ctx.req.body ? ctx.req.body.xml : ''

    if (!msg) {
        ctx.body = 'error request.'
        return;
    }
    
    MsgType = msg.MsgType[0]

    switch (MsgType) {
        case 'text':
            result = wx.message.text(msg, msg.Content)
            break;
        default: 
            result = 'success'
    }
    ctx.res.setHeader('Content-Type', 'application/xml')
    ctx.res.end(result)
}

這樣子就可以返回文本信息啦~到這里已經(jīng)把基本的流程走完。其他更多的操作可以看官方文檔,加以修改就好。

公眾號例子如下,目前只是實現(xiàn)了回復(fù)相同的文本內(nèi)容:

qrcode_430.jpg

上面的代碼已開源到https://github.com/cirplan/koa2-wechat 上,歡迎圍觀。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,533評論 19 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,872評論 25 709
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,637評論 18 399
  • Http協(xié)議詳解 標簽(空格分隔): Linux 聲明:本片文章非原創(chuàng),內(nèi)容來源于博客園作者MIN飛翔的HTTP協(xié)...
    Sivin閱讀 5,344評論 3 82
  • 大家轟隆隆相聚在一起,各捧手機關(guān)注各自喜好的小領(lǐng)域小群落,有八卦有笑話有惡搞,想關(guān)注什么就會有什么,應(yīng)有盡有,豐富...
    你在微笑嗎閱讀 1,551評論 3 1

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