描述

微信公众号开发基本分为2大种类型

1.用户直接做了某些操作(回复信息、订阅、扫码、发语音、点按钮等),此时这些信息微信会发送到微信服务器的80端口,这是一种开发类型;
2.通过连接(按钮、文章)引导用户到另一个网站,网站通过oauth实现微信的三方登录做到用户打通,然后提供更多服务

注意第二种类型的其他网站可以跟第一种类型(只提供80端口服务)的网站是一个server

80端口的服务开发

代码位置weixin_server tag:80-port-20160415

wechat对象以及缓存access_token

wechat sdk操作都需要一个wechat的东西,生成这玩意儿需要加载一个wechat_config,config里面包含了你的appid,appsecret,以及你消息加密解密的配置,为了简少根据加密方式造成的代码变更,我在配置里面添加了settings.WEIXIN_ENCRYPT_MODE,这样直接填写你的加密模式,就可以在任何用到wechat的地方直接import一个wechat变量,之所以要用get_wechat的方式实时生成是为了做access_token的缓存,下面会讲。

from .wechat import get_wechat

wechat = get_wechat()

微信在做一些操作是需要用到access_token(生成二维码等),而每天access_token接口的调用上限为2000,需要自己做缓存, sdk文档讲了几种方式,我觉得放在cache中拿比较妥当。

代码我不贴了,这是位置
weixin/config.py
weixin/wechat.py

handler各种微信post到80的事件

微信会在用户对公众号交互时像我们的服务器80端口post一些事件,sdk文档 message源码,大体看了下我写了个mixin

# -*- coding: utf-8 -*-
from weixin.wechat import get_wechat
from wechat_sdk.messages import MESSAGE_TYPES, EventMessage

wechat = get_wechat()

REVERSED_MESSAGE_TYPES = {value:key for key, value in MESSAGE_TYPES.iteritems()}

class WeixinDispatchMixin(object):

    def dispatch_weixin(self, request, *args, **kwargs):
        content = request.body
        signature = request.GET.get(\'signature\', \'\')
        msg_signature = request.GET.get(\'msg_signature\', \'\')
        timestamp = request.GET.get(\'timestamp\', \'\')
        nonce = request.GET.get(\'nonce\', \'\')
        try:
            wechat.parse_data(
                    content,
                    msg_signature=msg_signature,
                    timestamp=timestamp,
                    nonce=nonce)
        except ParseError:
            return HttpResponse(\'Invalid Body Text\')
        handler_name = self.get_weixin_handler_name(request, wechat, *args, **kwargs)
        handler = getattr(self, handler_name, self.http_method_not_allowed)
        return handler(request, wechat, *args, **kwargs)

    def get_weixin_handler_name(self, request, parsed_wechat, *args, **kwargs):
        message = parsed_wechat.message
        if isinstance(message, EventMessage):
            event_name = REVERSED_MESSAGE_TYPES[type(message)]
            event_detail_name = \'weixin_handler_{}_{}\'.format(event_name, message.type)
            if hasattr(self, event_detail_name):
                event_name = event_detail_name
                return event_name
        return u\'weixin_handler_{}\'.format(REVERSED_MESSAGE_TYPES.get(type(message), \'unsupport\'))
        

mixin参考django的dispatch,这样在继承的类里面直接实现weixin_handler_xxx方法即可,然而event有更多的类型,如果是通用处理则直接实现weixin_handler_event,如果要更加细化,例如扫码的event,则实现weixin_handler_event_scan方法,可以参考weixin_server/views.py

菜单儿

菜单可以直接在admin定制,由于菜单类似一种配置,同一时间最多且只有1个,我就把edx的config_model拿过来了,自己定制了下admin,这样你可以直接在admin里面修改菜单,微信会生效。微信菜单会缓存5分钟,你可以取消关注,然后在关注查看菜单变化效果。

菜单这块儿的代码有点意思,感兴趣的可以看下。

weixin/models.py
weixin/admin.py

生成二维码

二维码的逻辑略有不同(相对于sdk的其他response_xxx),所以我写了个qrcode.py封装了一下,注意永久二维码只能生产10万张,业务场景不要乱用,二维码的这些id、url都是需要做本地存储的,我没接model就丢到缓存里了。