这个系列文章里曾经说过,若是好友短期发送多条消息而后撤回会难以判断究竟撤回的是哪条信息,只能靠猜。后来我以为“猜”这个事情特别不
Pythonic
,研究一段时间后找到了解决方案,不得不惊叹ItChat
真的好强大。html
以前的不能肯定的状况大概是这样:短期内同一位好友发送了多条消息,当他随便撤回一条消息时,咱们不能肯定他到底撤回的究竟是哪一条消息。只能猜他多是撤回了最近的一条消息,而后将其余消息贴出来做为备选。代码以下:python
target_msg_pattern = '"{}" 撤回了一条消息'.format(sender_name)
if content == target_msg_pattern:
return_msg = '【{}】撤回了一条消息:\n'.format(sender_name)
if len(log[sender_name].items()) == 0:
return_msg = '缓存信息列表为空!'
else:
return_msg += log[sender_name].items()[-1][-1] + '\n'
if len(log[sender_name].items()) > 1:
msgs = [msg for timestamp, msg in log[sender_name].items()[:-1]]
return_msg += '也有多是下列信息中的某一条:\n' + '\n'.join(msgs)
复制代码
实际效果是这样:git
我这个强迫症简直受不了这么不肯定的说法。github
msg
信息要想肯定撤回了哪一条信息,就必须先熟悉普通msg
和撤回的msg
里面都有哪些信息,他们的相同点和不一样点。下面就来看看这两种状况下msg
都是怎么样的,不须要仔细的看每一行,后面会做具体分析。web
先是用机器人“小帮帮”发送过来的信息获得的msg
信息:缓存
{
'MsgId': '2018511155698964390',
'FromUserName': '@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3',
'ToUserName': '@**********c2e61fdb47b5c241553a2f',
'MsgType': 1,
'Content': 'msg里面到底有什么?',
'Status': 3,
'ImgStatus': 1,
'CreateTime': 1578069291,
'VoiceLength': 0,
'PlayLength': 0,
'FileName': '',
'FileSize': '',
'MediaId': '',
'Url': '',
'AppMsgType': 0,
'StatusNotifyCode': 0,
'StatusNotifyUserName': '',
'RecommendInfo': {
'UserName': '',
'NickName': '',
'QQNum': 0,
'Province': '',
'City': '',
'Content': '',
'Signature': '',
'Alias': '',
'Scene': 0,
'VerifyFlag': 0,
'AttrStatus': 0,
'Sex': 0,
'Ticket': '',
'OpCode': 0
},
'ForwardFlag': 0,
'AppInfo': {
'AppID': '',
'Type': 0
},
'HasProductId': 0,
'Ticket': '',
'ImgHeight': 0,
'ImgWidth': 0,
'SubMsgType': 0,
'NewMsgId': 2018511155698964390,
'OriContent': '',
'EncryFileName': '',
'User': < User: {
'MemberList': < ContactList: [] > ,
'Uin': 0,
'UserName': '@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3',
'NickName': '小帮帮',
'HeadImgUrl': '/cgi-bin/mmwebwx-bin/webwxgeticon?seq=699837854&username=@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3&skey=@crypt_****c00c_92668c8ba7d285c221a85e**********',
'ContactFlag': 2049,
'MemberCount': 0,
'RemarkName': '小帮帮',
'HideInputBarFlag': 0,
'Sex': 2,
'Signature': '',
'VerifyFlag': 0,
'OwnerUin': 0,
'PYInitial': 'XBB',
'PYQuanPin': 'xiaobangbang',
'RemarkPYInitial': 'XBB',
'RemarkPYQuanPin': 'xiaobangbang',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 33658937,
'Province': '浙江',
'City': '台州',
'Alias': '',
'SnsFlag': 17,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': '',
'EncryChatRoomId': '',
'IsOwner': 0
} > ,
'Type': 'Text',
'Text': 'msg里面到底有什么?'
}
复制代码
下面是机器人撤回刚才的信息获得的msg信息:微信
{
'MsgId': '4056955577161654067',
'FromUserName': '@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3',
'ToUserName': '@**********c2e61fdb47b5c241553a2f',
'MsgType': 10002,
'Content': '<sysmsg type="revokemsg"><revokemsg><session>wxid_4gngrr04aqjn21</session><oldmsgid>1123721956</oldmsgid><msgid>2018511155698964390</msgid><replacemsg><![CDATA["小帮帮" 撤回了一条消息]]></replacemsg></revokemsg></sysmsg>',
'Status': 4,
'ImgStatus': 1,
'CreateTime': 1578069381,
'VoiceLength': 0,
'PlayLength': 0,
'FileName': '',
'FileSize': '',
'MediaId': '',
'Url': '',
'AppMsgType': 0,
'StatusNotifyCode': 0,
'StatusNotifyUserName': '',
'RecommendInfo': {
'UserName': '',
'NickName': '',
'QQNum': 0,
'Province': '',
'City': '',
'Content': '',
'Signature': '',
'Alias': '',
'Scene': 0,
'VerifyFlag': 0,
'AttrStatus': 0,
'Sex': 0,
'Ticket': '',
'OpCode': 0
},
'ForwardFlag': 0,
'AppInfo': {
'AppID': '',
'Type': 0
},
'HasProductId': 0,
'Ticket': '',
'ImgHeight': 0,
'ImgWidth': 0,
'SubMsgType': 0,
'NewMsgId': 4056955577161654067,
'OriContent': '',
'EncryFileName': '',
'User': < User: {
'MemberList': < ContactList: [] > ,
'Uin': 0,
'UserName': '@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3',
'NickName': '小帮帮',
'HeadImgUrl': '/cgi-bin/mmwebwx-bin/webwxgeticon?seq=699837854&username=@**********f511363f8200853d724137bb31236a7ea81e5183cc06cb4ec978e3&skey=@crypt_****c00c_92668c8ba7d285c221a85e**********',
'ContactFlag': 2049,
'MemberCount': 0,
'RemarkName': '小帮帮',
'HideInputBarFlag': 0,
'Sex': 2,
'Signature': '',
'VerifyFlag': 0,
'OwnerUin': 0,
'PYInitial': 'XBB',
'PYQuanPin': 'xiaobangbang',
'RemarkPYInitial': 'XBB',
'RemarkPYQuanPin': 'xiaobangbang',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 33658937,
'Province': '浙江',
'City': '台州',
'Alias': '',
'SnsFlag': 17,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': '',
'EncryChatRoomId': '',
'IsOwner': 0
} > ,
'Type': 'Note',
'Text': '"小帮帮" 撤回了一条消息'
}
复制代码
获得了两种类型的msg
,下面是对比(高亮的部分是不一样处,省略了部分相同内容。能够点击放大查看大图):session
如今来分析几条关键信息:并发
MsgId(与下面的NewMsgId同)ide
消息编号。这个很好理解,每条消息都是经过一个独一无二的编号来与其余消息区分,因此这两条消息的编号不一样很正常。若是咱们能拿到好友撤回消息的编号,也就能锁定这条消息了。
MsgType(与下面的Type同)
消息类型。以下图,左边是普通的对话消息,右边相似于系统提示消息。是否是能够根据这条信息来判断是否是有好友撤回了消息?
Content
消息内容,注意与下面的Text
区分,这两种消息类型在内容上最大的区别可能就在这里了。
来看一下撤回消息的Content
是怎么样的(为了便于查看,已经通过格式化):
<sysmsg type="revokemsg">
<revokemsg>
<session>wxid_4gngrr04aqjn21</session>
<oldmsgid>1123721956</oldmsgid>
<msgid>2018511155698964390</msgid>
<replacemsg><![CDATA["小帮帮" 撤回了一条消息]]></replacemsg>
</revokemsg>
</sysmsg>
复制代码
一眼就能发现关键点:撤回的那条消息属于系统消息(sysmsg
),类型是撤回消息(revokemsg
),对应的消息编号是2018511155698964390
。
细心的读者已经发现,这个消息编号正好就是左边那条消息的编号。
经过这个推理,猜想Content
字段是系统内部传输的内容,而Text
字段则是用户看到的内容。
根据上述分析,有三个地方帮助肯定收到的某条信息是不是撤回的消息:
MsgType
是1
就是普通消息,是10002
则可能为撤回消息。
Content
若是Content里有包含type="revokemsg"
则可能为撤回消息,不然不是撤回消息。
Type
是Text
就是普通消息,是Note
则可能为撤回消息。
精确起见,消息还要同时知足上面三种状况才可认定为撤回消息。
因为要锁定撤回消息必需要MsgId
才能肯定,因此在存储临时消息时须要加上这一字段。
log[sender_name][cur_timestamp] = msg['MsgId'] + '|||' + content
复制代码
为了简化数据复杂度,我经过分隔符|||
直接把MsgId
加在前面。
因而,锁定并发送撤回消息的代码就时这样:
content = str(msg['Text'])
revoke_info = msg['Content']
print('{}, {} 发来消息: {}'.format(formatted_timestamp, sender_name, content))
target_msg_pattern = '"{}" 撤回了一条消息'.format(sender_name)
if target_msg_pattern == content and msg['Type'] == 'Note' and str(msg['MsgType']) == '10002' and 'type="revokemsg"' in revoke_info:
return_msg = ''
return_msg_head = '{},【{}】撤回了一条消息:\n'.format(formatted_timestamp, sender_name)
revoke_msg_id = revoke_info.split('<msgid>')[-1].split('</msgid>')[0]
for _, value in log[sender_name].items():
if value.split('|||')[0] == revoke_msg_id:
return_msg = value.split('|||')[1]
if return_msg == '':
return_msg = '缓存信息列表为空!'
return_msg = return_msg_head + return_msg
print(return_msg)
itchat.send_msg(return_msg, 'filehelper')
复制代码
测试一下,为便于查看,将撤回提醒直接发给机器人“小帮帮”(掘金上不会上传动图,须要看动图版本的能够到这里):
一个完美的微信防撤回脚本大功告成!
系列结语 Python有不少好用好玩的库,能够慢慢发掘。本期咱们利用ItChat库编写了一个微信防撤回脚本。其实ItChat功能远远不止这些,它还能够处理微信群消息以及各类其余类型的消息,咱们讲到的只是九牛一毛,更多的还要你们本身去探索。
这个系列就到此为止了,若是有想要了解交流的能够在公众号主页联系我,这个系列的代码在这里:
https://github.com/TitusWongCN/AntiInfoWithdrawal
复制代码
你们有什么想了解的,或者有什么想作的也能够在文章后面留言,后面说不定就会作了哦~
无论写什么,但愿能跟更多人沟通,有问题或者需求随时欢迎交流。
我全部的项目源码都会放在下面的github仓库里面,有须要能够参考,有问题欢迎指正,谢谢!
https://github.com/TitusWongCN/
复制代码
【Python写微信防撤回脚本】往期推荐:
第一期:【Python写微信防撤回脚本】01 熟悉ItChat库
第二期:【Python写微信防撤回脚本】02 接收记录聊天信息
第三期:【Python写微信防撤回脚本】03 获取撤回信息并整理
第四期:【Python写微信防撤回脚本】04完结 发送被撤回消息
下面是个人公众号,有兴趣能够扫一下: