对微信公众号接口的单元测试(Django)

先简要概述微信公众号如何进行第三方配置(Django)

配置

menu = {
        'button': [
            {
                "name": "服务",
                "sub_button": [
                    {
                        "type": "click",
                        "name": "抢啥",
                        "key": event_keys['book_what'],
                    },
                    {
                        "type": "click",
                        "name": "查票",
                        "key": event_keys['get_ticket'],
                    },
                    {
                        "type": "click",
                        "name": "绑定",
                        "key": event_keys['account_bind'],
                    },
                    {
                        "type": "click",
                        "name": "帮助",
                        "key": event_keys['help'],
                    }
                ]
            },
            {
                "name": "抢票",
                "sub_button": []
            }
        ]
    }
  • 用户事件响应:通过对微信发过来的POST事件进行解析、认证后,直接发送给对应的handler列表进行检查,让他们对这些数据进行对应的相应。

单元测试

如果已经进行好相关的配置,又已经实现相关的Handler处理,那就该进行单元测试了。

此时的单元测试最大的困难就是模拟按键以及输入信息,那如何做到呢?

修改配置

在单元测试中,首先设置settings.IGNORE_WECHAT_SIGNATURE = True,防止微信的验证阻拦等

xml格式

实际上微信公众号规定了按键操作以及输入操作的信息格式,均为xml,在官方文档里有说明。此处仅仅介绍自定义菜单的操作以及发送文本消息的xml格式:

文本消息xml模板:xxxx是你的微信公众号的id,下面的{}内的为变量,可以在微信公众开发平台上查找:
fromUser是用户的id,curTime是当前时间,textMsg是文本消息的内容
<xml>  
<ToUserName>< ![CDATA[xxxx] ]></ToUserName>  
<FromUserName>< ![CDATA[{fromUser}] ]></FromUserName>  
<CreateTime>{curTime}</CreateTime>  
<MsgType>< ![CDATA[text] ]></MsgType>  
<Content>< ![CDATA[{textMsg}] ]></Content>  
<MsgId>{msgId}</MsgId>  
</xml>

自定义菜单事件xml模板:xxxx是你的微信公众号的id,下面的{}内的为变量,可以在微信公众开发平台上查找:
fromUser是用户的id,curTime是当前时间,EVENTKEY是菜单中设置的键值,例如上述的event_keys['help']
<xml>
<ToUserName><![CDATA[xxxx]]></ToUserName>
<FromUserName><![CDATA[{fromUser}]]></FromUserName>
<CreateTime>{curTime}</CreateTime>
<MsgType><![CDATA[event]]></MsgType>
<Event><![CDATA[CLICK]]></Event>
<EventKey><![CDATA[{EVENTKEY}]]></EventKey>
</xml>

既然有了模板,我们就可以模拟输入了,接下来需要解决的就是如何将输入post到后端:
注意需要post到/wechat/,同时制定xml格式,其中的self是TestCase子类下的

response = self.client.post(
                    path='/wechat/',
                    content_type='application/xml',
                    data=data
                )

以下是一个完整模拟访问帮助功能的测试类:

class TestHelpOrSubscribe(TestCase):

    def setUp(self):
        # 设置配置
        settings.IGNORE_WECHAT_SIGNATURE = True

        # user1 => not bind
        # user2 => bind
        User.objects.create(open_id='abc')
        User.objects.create(open_id='a', student_id='2016013265')

        # textMsgs => 用户一般可能输入(成功)
        self.textMsgs = ['帮助', 'Help', 'help', 'HELP']

        # clickEvents => 用户一般可能点击事件
        self.clickEvents = [CustomWeChatView.event_keys['help']]

    # 是否返回帮助
    def is_help(self, content):
        pattern = '“紫荆之声”使用指南'
        return content.find(pattern) != -1

    def test_text(self):
        users = User.objects.all()

        for user in users:
            for textMsg in self.textMsgs:
                fromUser = user.open_id
                curTime = str(getTimeStamp(datetime.datetime.now()))
                msgId = str(random.randint(0, 99999)) + curTime
                data = getTextXml(fromUser, curTime, textMsg, msgId)

                response = self.client.post(
                    path='/wechat/',
                    content_type='application/xml',
                    data=data
                )

                content = str(response.content.decode('utf-8'))
                self.assertEqual(self.is_help(content), True)

    def test_event_click(self):
        users = User.objects.all()

        for user in users:
            for clickEvent in self.clickEvents:
                fromUser = user.open_id
                curTime = str(getTimeStamp(datetime.datetime.now()))
                data = getClickXml(fromUser, curTime, clickEvent)

                response = self.client.post(
                    path='/wechat/',
                    content_type='application/xml',
                    data=data
                )

                content = str(response.content.decode('utf-8'))
                self.assertEqual(self.is_help(content), True)

如果有对测试不了解的,请看我之前的Django单元测试文档:https://www.jianshu.com/p/5533c866453a

这样就完成了对微信公众号接口的测试。

参考

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • “文章千古事”,可惜提笔忘字,文字是需要不断的输出才能够有所增进的,可惜这年头,今年许下的是去年誓死要完成前年的...
    律人阅读 183评论 0 0
  • 工作琐碎又复杂,可是我干了那么多的活,又能怎么样,身边没人,领导不重视,不知道自己一天都为了撒 ,无头苍蝇
    一朵太阳花shl阅读 173评论 0 0
  • “古书有言:以擎穹为左,以厚土为右,向前百里跨溪而行,三日不止 。遇烈日不畏赤炙,遭泛洪不惧溺息 ,身坚而心澄者,...
    晴岩阅读 231评论 0 0