zoukankan      html  css  js  c++  java
  • 4.接收与发送消息

    公众号接收与发送消息

    验证URL有效性成功后即接入生效,成为开发者。如果公众号类型为服务号(订阅号只能使用普通消息接口),可以在公众平台网站中申请认证,认证成功的服务号将获得众多接口权限,以满足开发者需求。

    此后用户每次向公众号发送消息、或者产生自定义菜单点击事件时,开发者填写的服务器配置URL将得到微信服务器推送过来的消息和事件,然后开发者可以依据自身业务逻辑进行响应,例如回复消息等。

    用户向公众号发送消息时,公众号方收到的消息发送者是一个OpenID,是使用用户微信号加密后的结果,每个用户对每个公众号有一个唯一的OpenID

    接收普通消息

    当普通微信用户向公众账号发消息时,微信服务器将POST消息的XML数据包到开发者填写的URL上。

    微信服务器在五秒内收不到响应会断掉连接,并且重新发起请求,总共重试三次。假如服务器无法保证在五秒内处理并回复,可以直接回复空串,微信服务器不会对此作任何处理,并且不会发起重试。

    各消息类型的推送使用XML数据包结构,如:

    <xml>
    <ToUserName><![CDATA[gh_866835093fea]]></ToUserName>
    <FromUserName><![CDATA[ogdotwSc_MmEEsJs9-ABZ1QL_4r4]]></FromUserName>
    <CreateTime>1478317060</CreateTime>
    <MsgType><![CDATA[text]]></MsgType>
    <Content><![CDATA[你好]]></Content>
    <MsgId>6349323426230210995</MsgId>
    </xml> 

    注意:<![CDATA 与 ]]> 括起来的数据不会被xml解析器解析

    xmltodict 模块基本用法

    xmltodict 是一个用来处理xml数据的很方便的模块。包含两个常用方法parseunparse

    1. parse

    xmltodict.parse()方法可以将xml数据转为python中的dict字典数据:

    >>> import xmltodict
    >>> xml_str = """
    ... <xml>
    ... <ToUserName><![CDATA[gh_866835093fea]]></ToUserName>
    ... <FromUserName><![CDATA[ogdotwSc_MmEEsJs9-ABZ1QL_4r4]]></FromUserName>
    ... <CreateTime>1478317060</CreateTime>
    ... <MsgType><![CDATA[text]]></MsgType>
    ... <Content><![CDATA[你好]]></Content>
    ... <MsgId>6349323426230210995</MsgId>
    ... </xml>
    ... """
    >>>
    >>> xml_dict = xmltodict.parse(xml_str)
    >>> type(xml_dict)
    <class 'collections.OrderedDict'>  # 类字典型,可以按照字典方法操作
    >>>
    >>> xml_dict
    OrderedDict([(u'xml', OrderedDict([(u'ToUserName', u'gh_866835093fea'), (u'FromUserName', u'ogdotwSc_MmEEsJs9-ABZ1QL_4r4'), (u'CreateTime', u'1478317060'), (u'MsgType', u'text'), (u'Content', u'u4f60u597d'), (u'MsgId', u'6349323426230210995')]))])
    >>>
    >>> xml_dict['xml']
    OrderedDict([(u'ToUserName', u'gh_866835093fea'), (u'FromUserName', u'ogdotwSc_MmEEsJs9-ABZ1QL_4r4'), (u'CreateTime', u'1478317060'), (u'MsgType', u'text'), (u'Content', u'u4f60u597d'), (u'MsgId', u'6349323426230210995')])
    >>>
    >>> for key, val in xml_dict['xml'].items():
    ...     print key, "=", val
    ... 
    ToUserName = gh_866835093fea
    FromUserName = ogdotwSc_MmEEsJs9-ABZ1QL_4r4
    CreateTime = 1478317060
    MsgType = text
    Content = 你好
    MsgId = 6349323426230210995
    >>>

    2. unparse

    xmltodict.unparse()方法可以将字典转换为xml字符串:

    xml_dict = {
        "xml": {
            "ToUserName" : "gh_866835093fea",
            "FromUserName" : "ogdotwSc_MmEEsJs9-ABZ1QL_4r4",
            "CreateTime" : "1478317060",
            "MsgType" : "text",
            "Content" : u"你好",
            "MsgId" : "6349323426230210995",
        }
    }
    >>> xml_str = xmltodict.unparse(xml_dict) >>> print xml_str <?xml version="1.0" encoding="utf-8"?> <xml><FromUserName>ogdotwSc_MmEEsJs9-ABZ1QL_4r4</FromUserName><MsgId>6349323426230210995</MsgId><ToUserName>gh_866835093fea</ToUserName><Content>你好</Content><MsgType>text</MsgType><CreateTime>1478317060</CreateTime></xml> >>> >>> xml_str = xmltodict.unparse(xml_dict, pretty=True) # pretty表示友好输出 >>> print xml_str <?xml version="1.0" encoding="utf-8"?> <xml> <FromUserName>ogdotwSc_MmEEsJs9-ABZ1QL_4r4</FromUserName> <MsgId>6349323426230210995</MsgId> <ToUserName>gh_866835093fea</ToUserName> <Content>你好</Content> <MsgType>text</MsgType> <CreateTime>1478317060</CreateTime> </xml> >>>

    普通消息类别

    1. 文本消息
    2. 图片消息
    3. 语音消息
    4. 视频消息
    5. 小视频消息
    6. 地理位置消息
    7. 链接消息

    文本消息

     <xml>
     <ToUserName><![CDATA[toUser]]></ToUserName>
     <FromUserName><![CDATA[fromUser]]></FromUserName> 
     <CreateTime>1348831860</CreateTime>
     <MsgType><![CDATA[text]]></MsgType>
     <Content><![CDATA[this is a test]]></Content>
     <MsgId>1234567890123456</MsgId>
     </xml> 

    被动回复消息

    当用户发送消息给公众号时(或某些特定的用户操作引发的事件推送时),会产生一个POST请求,开发者可以在响应包中返回特定XML结构,来对该消息进行响应(现支持回复文本、图片、图文、语音、视频、音乐)。严格来说,发送被动响应消息其实并不是一种接口,而是对微信服务器发过来消息的一次回复。

    假如服务器无法保证在五秒内处理并回复,必须做出下述回复,这样微信服务器才不会对此作任何处理,并且不会发起重试(这种情况下,可以使用客服消息接口进行异步回复),否则,将出现严重的错误提示。详见下面说明:

    1. (推荐方式)直接回复success
    2. 直接回复空串(指字节长度为0的空字符串,而不是XML结构体中content字段的内容为空)

    一旦遇到以下情况,微信都会在公众号会话中,向用户下发系统提示“该公众号暂时无法提供服务,请稍后再试”:

    1. 开发者在5秒内未回复任何内容
    2. 开发者回复了异常数据,比如JSON数据等

    回复的消息类型

    1. 文本消息
    2. 图片消息
    3. 语音消息
    4. 视频消息
    5. 音乐消息
    6. 图文消息

    回复文本消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>12345678</CreateTime>
    <MsgType><![CDATA[text]]></MsgType>
    <Content><![CDATA[你好]]></Content>
    </xml>

    鹦鹉学舌代码实现

    我们现在来实现一个针对文本消息的收发程序。实现的业务逻辑类似与“鹦鹉学舌”,粉丝发什么内容,我们就传回给粉丝什么内容。

    # coding:utf-8
    
    import tornado.web
    import tornado.httpserver
    import tornado.ioloop
    import tornado.options
    import hashlib
    import xmltodict
    import time
    
    from tornado.web import RequestHandler
    from tornado.options import define, options
    
    WECHAT_TOKEN = "itcast"
    
    define("port", default=8080, type=int)
    
    class WeChatHandler(RequestHandler):
        """微信接入接口"""
        def get(self):
            """开发者验证接口"""
            signature = self.get_argument("signature")
            timestamp = self.get_argument("timestamp")
            nonce = self.get_argument("nonce")
            echostr = self.get_argument("echostr")
            tmp = [WECHAT_TOKEN, timestamp, nonce]
            tmp.sort()
            tmp = "".join(tmp)
            tmp = hashlib.sha1(tmp).hexdigest()
            if tmp == signature:
                self.write(echostr)
            else:
                self.write("error")
    
        def post(self):
            """收发消息接口"""
            req_xml = self.request.body
            req = xmltodict.parse(req_xml)['xml']
            if "text" == req.get("MsgType"):
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Content", "")
                }
            else:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":"I love you, itcast!"
                }
            resp_xml = xmltodict.unparse({"xml":resp})
            self.write(resp_xml)
    
    
    def main():
        tornado.options.parse_command_line()
        app = tornado.web.Application([
                (r"/wechat", WeChatHandler),
            ])
        http_server = tornado.httpserver.HTTPServer(app)
        http_server.listen(options.port)
        tornado.ioloop.IOLoop.current().start()
    
    
    if __name__ == '__main__':
        main()

    有趣的表情

    QQ表情

    实际是字符串转义,如 /::D/::P 等,仍属于文本信息。

    emoji

    绘文字(日语:絵文字/えもじ emoji)是日本在无线通信中所使用的视觉情感符号,绘意指图形,文字则是图形的隐喻,可用来代表多种表情,如笑脸表示笑、蛋糕表示食物等。

    在NTTDoCoMo的i-mode系统电话系统中,绘文字的尺寸是12x12 像素,在传送时,一个图形有2个字节。Unicode编码为E63E到E757,而在Shift-JIS编码则是从F89F到F9FC。基本的绘文字共有176个符号,在C-HTML4.0的编程语言中,则另增添了76个情感符号。

    最早由栗田穰崇(Shigetaka Kurita)创作,并在日本网络及手机用户中流行。

    自苹果公司发布的iOS 5输入法中加入了emoji后,这种表情符号开始席卷全球,目前emoji已被大多数现代计算机系统所兼容的Unicode编码采纳,普遍应用于各种手机短信和社交网络中。

    本质是Unicode字符,也属于文本消息。

    自定表情

    微信的自定义表情不是文本,也不是图片,而是一种不支持的格式,微信未提供处理此消息的接口。

    改写代码

    微信发送的请求中会携带签名验证信息(正如验证服务器有效性一章节所示),我们需要对收到的请求进行验证是否来自微信服务器,所以在处理请求前都要按照验证算法进行检验。

    class WeChatBaseHandler(RequestHandler):
        def prepare(self):
            """验证请求是否来自微信服务器"""
            signature = self.get_argument("signature")
            timestamp = self.get_argument("timestamp")
            nonce = self.get_argument("nonce")
            tmp = [WECHAT_TOKEN, timestamp, nonce]
            tmp.sort()
            tmp = "".join(tmp)
            tmp = hashlib.sha1(tmp).hexdigest()
            if tmp != signature:
                self.send_error(403) # 若是非法请求,返回403错误
    
    class WeChatHandler(WeChatBaseHandler):
        """微信接入接口"""
        def get(self):
            """开发者验证接口"""
            echostr = self.get_argument("echostr")
            self.write(echostr)
    
        def post(self):
            """收发消息接口"""
            req_xml = self.request.body
            req = xmltodict.parse(req_xml)['xml']
            if "text" == req.get("MsgType"):
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Content", "")
                }
            else:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":"I love you, itcast!"
                }
            resp_xml = xmltodict.unparse({"xml":resp})
            self.write(resp_xml)

    接收其他普通消息

    接收图片消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>1348831860</CreateTime>
    <MsgType><![CDATA[image]]></MsgType>
    <PicUrl><![CDATA[this is a url]]></PicUrl>
    <MediaId><![CDATA[media_id]]></MediaId>
    <MsgId>1234567890123456</MsgId>
    </xml>
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType image
    PicUrl 图片链接
    MediaId 图片消息媒体id,可以调用多媒体文件下载接口拉取数据。
    MsgId 消息id,64位整型

    接收视频消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>1357290913</CreateTime>
    <MsgType><![CDATA[video]]></MsgType>
    <MediaId><![CDATA[media_id]]></MediaId>
    <ThumbMediaId><![CDATA[thumb_media_id]]></ThumbMediaId>
    <MsgId>1234567890123456</MsgId>
    </xml> 
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 视频为video
    MediaId 视频消息媒体id,可以调用多媒体文件下载接口拉取数据。
    ThumbMediaId 视频消息缩略图的媒体id,可以调用多媒体文件下载接口拉取数据。
    MsgId 消息id,64位整型

    接收小视频消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>1357290913</CreateTime>
    <MsgType><![CDATA[shortvideo]]></MsgType>
    <MediaId><![CDATA[media_id]]></MediaId>
    <ThumbMediaId><![CDATA[thumb_media_id]]></ThumbMediaId>
    <MsgId>1234567890123456</MsgId>
    </xml> 
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 小视频为shortvideo
    MediaId 视频消息媒体id,可以调用多媒体文件下载接口拉取数据。
    ThumbMediaId 视频消息缩略图的媒体id,可以调用多媒体文件下载接口拉取数据。
    MsgId 消息id,64位整型

    接收语音消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>1357290913</CreateTime>
    <MsgType><![CDATA[voice]]></MsgType>
    <MediaId><![CDATA[media_id]]></MediaId>
    <Format><![CDATA[Format]]></Format>
    <MsgId>1234567890123456</MsgId>
    </xml>
    

      

    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 语音为voice
    MediaId 语音消息媒体id,可以调用多媒体文件下载接口拉取数据。
    Format 语音格式,如amr,speex等
    MsgID 消息id,64位整型

    请注意,开通语音识别后,用户每次发送语音给公众号时,微信会在推送的语音消息XML数据包中,增加一个Recongnition字段(注:由于客户端缓存,开发者开启或者关闭语音识别功能,对新关注者立刻生效,对已关注用户需要24小时生效。开发者可以重新关注此帐号进行测试)。开启语音识别后的语音XML数据包如下:

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>1357290913</CreateTime>
    <MsgType><![CDATA[voice]]></MsgType>
    <MediaId><![CDATA[media_id]]></MediaId>
    <Format><![CDATA[Format]]></Format>
    <Recognition><![CDATA[腾讯微信团队]]></Recognition>
    <MsgId>1234567890123456</MsgId>
    </xml>

    多出的字段中,Format为语音格式,一般为amr,Recognition为语音识别结果,使用UTF8编码。

    回复其他普通消息

    回复图片消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>12345678</CreateTime>
    <MsgType><![CDATA[image]]></MsgType>
    <Image>
    <MediaId><![CDATA[media_id]]></MediaId>
    </Image>
    </xml>
    参数是否必须说明
    ToUserName 接收方帐号(收到的OpenID)
    FromUserName 开发者微信号
    CreateTime 消息创建时间 (整型)
    MsgType image
    MediaId 通过素材管理接口上传多媒体文件,得到的id。

    回复语音消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>12345678</CreateTime>
    <MsgType><![CDATA[voice]]></MsgType>
    <Voice>
    <MediaId><![CDATA[media_id]]></MediaId>
    </Voice>
    </xml>
    参数是否必须说明
    ToUserName 接收方帐号(收到的OpenID)
    FromUserName 开发者微信号
    CreateTime 消息创建时间戳 (整型)
    MsgType 语音,voice
    MediaId 通过素材管理接口上传多媒体文件,得到的id

    回复视频消息

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[fromUser]]></FromUserName>
    <CreateTime>12345678</CreateTime>
    <MsgType><![CDATA[video]]></MsgType>
    <Video>
    <MediaId><![CDATA[media_id]]></MediaId>
    <Title><![CDATA[title]]></Title>
    <Description><![CDATA[description]]></Description>
    </Video> 
    </xml>
    参数是否必须说明
    ToUserName 接收方帐号(收到的OpenID)
    FromUserName 开发者微信号
    CreateTime 消息创建时间 (整型)
    MsgType video
    MediaId 通过素材管理接口上传多媒体文件,得到的id
    Title 视频消息的标题
    Description 视频消息的描述

    回复用户语音消息识别

    class WeChatHandler(WeChatBaseHandler):
        """微信接入接口"""
        def get(self):
            """开发者验证接口"""
            echostr = self.get_argument("echostr")
            self.write(echostr)
    
        def post(self):
            """收发消息接口"""
            req_xml = self.request.body
            req = xmltodict.parse(req_xml)['xml']
            msg_type = req.get("MsgType")
            if "text" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Content", "")
                }
            elif "voice" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Recognition", u"未识别")
                }
            else:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":"I love you, itcast!"
                }
            resp_xml = xmltodict.unparse({"xml":resp})
            self.write(resp_xml)

    关注/取消关注事件

    用户在关注与取消关注公众号时,微信会把这个事件推送到开发者填写的URL。

    微信服务器在五秒内收不到响应会断掉连接,并且重新发起请求,总共重试三次。

    假如服务器无法保证在五秒内处理并回复,可以直接回复空串,微信服务器不会对此作任何处理,并且不会发起重试。

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[FromUser]]></FromUserName>
    <CreateTime>123456789</CreateTime>
    <MsgType><![CDATA[event]]></MsgType>
    <Event><![CDATA[subscribe]]></Event>
    </xml>
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 消息类型,event
    Event 事件类型,subscribe(订阅)、unsubscribe(取消订阅)
    class WeChatHandler(WeChatBaseHandler):
        """微信接入接口"""
        def get(self):
            """开发者验证接口"""
            echostr = self.get_argument("echostr")
            self.write(echostr)
    
        def post(self):
            """收发消息接口"""
            req_xml = self.request.body
            req = xmltodict.parse(req_xml)['xml']
            msg_type = req.get("MsgType")
            if "text" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Content", "")
                }
            elif "voice" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Recognition", u"未识别")
                }
            elif "event" == msg_type:
                if "subscribe" == req.get("Event"):
                    resp = {
                         "ToUserName":req.get("FromUserName", ""),
                        "FromUserName":req.get("ToUserName", ""),
                        "CreateTime":int(time.time()),
                        "MsgType":"text",
                        "Content":u"感谢您的关注!"
                    }
                else:
                    resp = None
            else:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":"I love you, itcast!"
                }
            if resp:
                resp_xml = xmltodict.unparse({"xml":resp})
            else:
                resp_xml = ""
            self.write(resp_xml)

    获取接口调用凭据

    access_token是公众号的全局唯一票据,公众号调用各接口时都需使用access_token。开发者需要进行妥善保存。access_token的存储至少要保留512个字符空间。access_token的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的access_token失效。

    接口说明

    请求方法

    https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET

    参数说明

    返回值

    正确时返回的JSON数据包如下:

    {
        "access_token":"ACCESS_TOKEN",
        "expires_in":7200
    }

    错误时微信会返回JSON数据包如下:

    {
        "errcode":40013,
        "errmsg":"invalid appid"
    }

    代码实现

    class AccessToken(object):
        """微信接口调用Token"""
        _access_token = {
            "token":"",
            "updatetime":datetime.datetime.now()
        }
    
        @classmethod
        @tornado.gen.coroutine
        def update_access_token(cls):
            """更新access_token"""
            client = AsyncHTTPClient()
            url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s" % (WECHAT_APPID, WECHAT_APPSECRET)
            resp = yield client.fetch(url)
            print resp.body
            ret = json.loads(resp.body)
            token = ret.get("access_token")
            if token:
                cls._access_token["token"] = token
                cls._access_token['updatetime'] = datetime.datetime.now()
    
        @classmethod
        @tornado.gen.coroutine
        def get_access_token(cls):
            """获取access_token"""
            if not cls._access_token["token"] or (datetime.datetime.now()-cls._access_token["updatetime"]).seconds>=6600:
                yield cls.update_access_token()
            raise tornado.gen.Return(cls._access_token["token"])

    带参数的二维码

    为了满足用户渠道推广分析和用户帐号绑定等场景的需要,公众平台提供了生成带参数二维码的接口。使用该接口可以获得多个带不同场景值的二维码,用户扫描后,公众号可以接收到事件推送。

    目前有2种类型的二维码:

    1. 临时二维码,是有过期时间的,最长可以设置为在二维码生成后的30天(即2592000秒)后过期,但能够生成较多数量。临时二维码主要用于帐号绑定等不要求二维码永久保存的业务场景

    2. 永久二维码,是无过期时间的,但数量较少(目前为最多10万个)。永久二维码主要用于适用于帐号绑定、用户来源统计等场景。

    获取带参数的二维码的过程包括两步,首先创建二维码ticket,然后凭借ticket到指定URL换取二维码。

    创建二维码ticket

    每次创建二维码ticket需要提供一个开发者自行设定的参数(scene_id),分别介绍临时二维码和永久二维码的创建二维码ticket过程。

    临时二维码请求说明

    http请求方式: POST
    URL: https://api.weixin.qq.com/cgi-bin/qrcode/create?access_token=TOKEN
    POST数据格式:json
    POST数据例子:{"expire_seconds": 604800, "action_name": "QR_SCENE", "action_info": {"scene": {"scene_id": 123}}}

    永久二维码请求说明

    http请求方式: POST
    URL: https://api.weixin.qq.com/cgi-bin/qrcode/create?access_token=TOKEN
    POST数据格式:json
    POST数据例子:{"action_name": "QR_LIMIT_SCENE", "action_info": {"scene": {"scene_id": 123}}}
    或者也可以使用以下POST数据创建字符串形式的二维码参数:
    {"action_name": "QR_LIMIT_STR_SCENE", "action_info": {"scene": {"scene_str": "123"}}}
    参数说明
    expire_seconds 该二维码有效时间,以秒为单位。 最大不超过2592000(即30天),此字段如果不填,则默认有效期为30秒。
    action_name 二维码类型,QR_SCENE为临时,QR_LIMIT_SCENE为永久,QR_LIMIT_STR_SCENE为永久的字符串参数值
    action_info 二维码详细信息
    scene_id 场景值ID,临时二维码时为32位非0整型,永久二维码时最大值为100000(目前参数只支持1--100000)
    scene_str 场景值ID(字符串形式的ID),字符串类型,长度限制为1到64,仅永久二维码支持此字段

    返回说明

    正确的Json返回结果:

    {"ticket":"gQH47joAAAAAAAAAASxodHRwOi8vd2VpeGluLnFxLmNvbS9xL2taZ2Z3TVRtNzJXV1Brb3ZhYmJJAAIEZ23sUwMEmm3sUw==","expire_seconds":60,"url":"http://weixin.qq.com/q/kZgfwMTm72WWPkovabbI"
    参数说明
    ticket 获取的二维码ticket,凭借此ticket可以在有效时间内换取二维码。
    expire_seconds 该二维码有效时间,以秒为单位。 最大不超过2592000(即30天)。
    url 二维码图片解析后的地址,开发者可根据该地址自行生成需要的二维码图片

    错误的Json返回示例:

    {"errcode":40013,"errmsg":"invalid appid"}

    通过ticket换取二维码

    获取二维码ticket后,开发者可用ticket换取二维码图片。请注意,本接口无须登录态即可调用。

    请求说明

    HTTP GET请求(请使用https协议)
    https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket=TICKET

    代码实例

    class QRCodeHandler(RequestHandler):
        @tornado.gen.coroutine
        def get(self):
            access_token = yield AccessToken.get_access_token()
            print "access_token", access_token
            url = "https://api.weixin.qq.com/cgi-bin/qrcode/create?access_token=%s" % access_token
            scene_id = self.get_argument("scene_id")
            req_body = '{"expire_seconds": 7200, "action_name": "QR_SCENE", "action_info": {"scene": {"scene_id": %s}}}' % scene_id
            client = AsyncHTTPClient()
            req = HTTPRequest(url, method="POST", body=req_body)
            resp = yield client.fetch(req)
            if "errcode" in resp.body:
                self.write("error")
            else:
                resp_data = json.loads(resp.body)
                ticket = resp_data['ticket']
                self.write('<img src="https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket=%s">' % ticket)

    扫描带参数二维码

    用户扫描带场景值二维码时,可能推送以下两种事件:

    • 如果用户还未关注公众号,则用户可以关注公众号,关注后微信会将带场景值关注事件推送给开发者。

    • 如果用户已经关注公众号,则微信会将带场景值扫描事件推送给开发者。

    1. 用户未关注时,进行关注后的事件推送

    推送XML数据包示例:

    <xml><ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[FromUser]]></FromUserName>
    <CreateTime>123456789</CreateTime>
    <MsgType><![CDATA[event]]></MsgType>
    <Event><![CDATA[subscribe]]></Event>
    <EventKey><![CDATA[qrscene_123123]]></EventKey>
    <Ticket><![CDATA[TICKET]]></Ticket>
    </xml>
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 消息类型,event
    Event 事件类型,subscribe
    EventKey 事件KEY值,qrscene_为前缀,后面为二维码的参数值
    Ticket 二维码的ticket,可用来换取二维码图片

    2. 用户已关注时的事件推送

    推送XML数据包示例:

    <xml>
    <ToUserName><![CDATA[toUser]]></ToUserName>
    <FromUserName><![CDATA[FromUser]]></FromUserName>
    <CreateTime>123456789</CreateTime>
    <MsgType><![CDATA[event]]></MsgType>
    <Event><![CDATA[SCAN]]></Event>
    <EventKey><![CDATA[SCENE_VALUE]]></EventKey>
    <Ticket><![CDATA[TICKET]]></Ticket>
    </xml>
    参数描述
    ToUserName 开发者微信号
    FromUserName 发送方帐号(一个OpenID)
    CreateTime 消息创建时间 (整型)
    MsgType 消息类型,event
    Event 事件类型,SCAN
    EventKey 事件KEY值,是一个32位无符号整数,即创建二维码时的二维码scene_id
    Ticket 二维码的ticket,可用来换取二维码图片

    代码

    class WeChatHandler(WeChatBaseHandler):
        """微信接入接口"""
        def get(self):
            """开发者验证接口"""
            echostr = self.get_argument("echostr")
            self.write(echostr)
    
        def post(self):
            """收发消息接口"""
            req_xml = self.request.body
            req = xmltodict.parse(req_xml)['xml']
            msg_type = req.get("MsgType")
            if "text" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Content", "")
                }
            elif "voice" == msg_type:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":req.get("Recognition", u"未识别")
                }
            elif "event" == msg_type:
                if "subscribe" == req.get("Event"):
                    resp = {
                        "ToUserName":req.get("FromUserName", ""),
                        "FromUserName":req.get("ToUserName", ""),
                        "CreateTime":int(time.time()),
                        "MsgType":"text",
                        "Content":u"感谢您的关注!"
                    }
                    if None != req.get("EventKey"):
                        resp["Content"] += u"场景值:"
                        resp["Content"] += req.get("EventKey")[8:]
                elif "SCAN" == req.get("Event"):
                    resp = {
                        "ToUserName":req.get("FromUserName", ""),
                        "FromUserName":req.get("ToUserName", ""),
                        "CreateTime":int(time.time()),
                        "MsgType":"text",
                        "Content":u"您扫描的场景值为:%s" % req.get("EventKey")
                    } 
                else:
                    resp = None
            else:
                resp = {
                    "ToUserName":req.get("FromUserName", ""),
                    "FromUserName":req.get("ToUserName", ""),
                    "CreateTime":int(time.time()),
                    "MsgType":"text",
                    "Content":"I love you, itcast!"
                }
            if resp:
                resp_xml = xmltodict.unparse({"xml":resp})
            else:
                resp_xml = ""
            self.write(resp_xml)
  • 相关阅读:
    KMP算法与字符串匹配问题
    贪婪算法(贪心算法)
    普里姆算法(Prim)与最小生成树问题
    克鲁斯卡尔算法(Kruskal算法)与最小生成树问题
    Dijkstra算法与最短路径问题
    SpringCloud(十一)使用actuator和dashborad、turbine对微服务进行监控
    博客美化——页面白天黑夜切换
    Spring5学习笔记——day01
    Mybatis学习笔记——day06
    Mybatis学习笔记——day05
  • 原文地址:https://www.cnblogs.com/alexzhang92/p/9380205.html
Copyright © 2011-2022 走看看