这小节的主要内容,是得到微信服务器传过来的文本消息,然后逆序回给微信服务器。
根据这个需求,你可以思考下有什么好的方法?
最简单的实现思路是使用 for 循环,申请同等大小的字符串,逆序赋值然后返回给服务器,但使用 Python 最方便的一点就是可以使用字符串的切片来实现逆序。
首先来复习下 Python 列表切片知识,打开 Xfce 终端,输入 python
启动进入命令行:
>>> s = 'shiyanlou'
>>> s[1:3] #这里呢,是左闭右开,是角标大于等于1并且小于3,就是1,2咯,所以是'hi'
'hi'
>>> s[0:7:1] #这里呢,等于s[0:7],因为递增的值1是默认的,你也可以改
'shiyanl'
>>> s[0:7:2] #递增2,看下结果,对上了吧
'sial'
>>> s[-9:-3] #其实s是不仅仅是'shiyanlou',更是'shiyanloushiyanlou',中间的s角标是0,而0前面的就是负数了
'shiyan'
>>> s[9:0:-1] #少了个s,不太好,因为右开,下标为0是不输出的
'uolnayih'
>>> s[::-1] #这样就可以了,不填,默认全部咯,ok,完成了
'uolnayihs'
本节主要参考的文档是 微信公众平台开发者文档。
微信公众平台发过来的 XML 消息如下:
# 这是微信公众平台发过来的 xml 格式的信息
<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>
其中每个字段的描述:
参数 | 描述 |
---|---|
ToUserName | 开发者微信号 |
FromUserName | 发送方帐号(一个OpenID) |
CreateTime | 消息创建时间 (整型) |
MsgType | text |
Content | 文本消息内容 |
MsgId | 消息id,64位整型 |
Python 分析 XML 格式的信息,可以在 Python 中引入:
import xml.etree.ElementTree as ET
从 XML 消息中读取 Content
的内容,然后倒序,就可以发回了。
下面我们看下如何按照规定的格式回复文本信息。
本节参考的文档:微信公众平台开发者文档。
回复的文本内容如下:
<xml>
<ToUserName><![CDATA[toUser]]></ToUserName>
<FromUserName><![CDATA[fromUser]]></FromUserName>
<CreateTime>12345678</CreateTime>
<MsgType><![CDATA[text]]></MsgType>
<Content><![CDATA[你好]]></Content>
</xml>
其中每一项字段的介绍:
参数 | 是否必须 | 描述 |
---|---|---|
ToUserName | 是 | 接收方帐号(收到的OpenID) |
FromUserName | 是 | 开发者微信号 |
CreateTime | 是 | 消息创建时间 (整型) |
MsgType | 是 | text |
Content | 是 | 回复的消息内容(换行:在content中能够换行,微信客户端就支持换行显示) |
接收微信发来的文本消息,不是 GET 请求,而是 POST 请求,因此在 POST
处理流程中进行操作:
if request.method == 'POST':
xml_str = request.stream.read()
xml = ET.fromstring(xml_str)
toUserName=xml.find('ToUserName').text
fromUserName = xml.find('FromUserName').text
createTime = xml.find('CreateTime').text
msgType = xml.find('MsgType').text
一个正常的流程,先读取发信者收信者和写信时间,还有一个特重要的就是信的类型,微信支持文本、图片、视频、音频等... 我们是回复文本信息,如果发来的不是文本信息,那我们就回复一句话提醒一下:
if msgType != 'text':
reply = '''
<xml>
<ToUserName><![CDATA[%s]]></ToUserName>
<FromUserName><![CDATA[%s]]></FromUserName>
<CreateTime>%s</CreateTime>
<MsgType><![CDATA[%s]]></MsgType>
<Content><![CDATA[%s]]></Content>
</xml>
''' % (
fromUserName,
toUserName,
createTime,
'text',
'Unknow Format, Please check out'
)
return reply
如果信息不是文本类,返回 Unknow Format, Please check out
。
如果是文本呢,我们就继续操作,读取文本内容和文本信息的 ID,然后逆序,赋值,最后回复:
content = xml.find('Content').text
msgId = xml.find('MsgId').text
else:
content = content[::-1]
reply = '''
<xml>
<ToUserName><![CDATA[%s]]></ToUserName>
<FromUserName><![CDATA[%s]]></FromUserName>
<CreateTime>%s</CreateTime>
<MsgType><![CDATA[%s]]></MsgType>
<Content><![CDATA[%s]]></Content>
</xml>
''' % (fromUserName, toUserName, createTime, msgType, content)
return reply
这里并没有完成,回复会出现乱码。可以通过 content.encode('UTF-8')
编码成 UTF-8
解决乱码问题。
附上本节可运行代码 wechat.py
:
def wechat_auth():
if request.method == 'GET':
print 'coming Get'
data = request.args
token = '*****************'
signature = data.get('signature','')
timestamp = data.get('timestamp','')
nonce = data.get('nonce','')
echostr = data.get('echostr','')
s = [timestamp,nonce,token]
s.sort()
s = ''.join(s)
if (hashlib.sha1(s).hexdigest() == signature):
return make_response(echostr)
if request.method == 'POST':
xml_str = request.stream.read()
xml = ET.fromstring(xml_str)
toUserName=xml.find('ToUserName').text
fromUserName = xml.find('FromUserName').text
createTime = xml.find('CreateTime').text
msgType = xml.find('MsgType').text
if msgType != 'text':
reply = '''
<xml>
<ToUserName><![CDATA[%s]]></ToUserName>
<FromUserName><![CDATA[%s]]></FromUserName>
<CreateTime>%s</CreateTime>
<MsgType><![CDATA[%s]]></MsgType>
<Content><![CDATA[%s]]></Content>
</xml>
''' % (
fromUserName,
toUserName,
createTime,
'text',
'Unknow Format, Please check out'
)
return reply
content = xml.find('Content').text
msgId = xml.find('MsgId').text
if type(content).__name__ == "unicode":
content = content[::-1]
content = content.encode('UTF-8')
elif type(content).__name__ == "str":
print type(content).__name__
content = content.decode('utf-8')
content = content[::-1]
reply = '''
<xml>
<ToUserName><![CDATA[%s]]></ToUserName>
<FromUserName><![CDATA[%s]]></FromUserName>
<CreateTime>%s</CreateTime>
<MsgType><![CDATA[%s]]></MsgType>
<Content><![CDATA[%s]]></Content>
</xml>
''' % (fromUserName, toUserName, createTime, msgType, content)
return reply
if __name__ == "__main__":
app.run(host='0.0.0.0', port=8080)
代码执行方式可以在实验楼桌面上打开 XFce,执行下面的程序:
python wechat.py
本节运行后进行对话的效果截图:
这节主要介绍如何实现机器人自动回复逆序文本的功能,程序调通后,我们可以进入到下一个环节,爬取笑话内容并自动回复。