强大的Itchat
itchat是一个开源的微信个人号接口,使用python封装接入微信网页版接口,通过调用itchat来登录微信网页版收发消息。
项目简介 - itchat
掌握itchat之后,只要你有兴趣可以随意开发自己的专属微信网页版客户端。目前比较流行的就是利用它来找回他人撤回的微信消息,和设置自动聊天机器人
撤回的微信消息
找回逻辑非常简单,对方撤回消息的时候,我们会收到一个[note]
类型的推送消息,里面包含了撤回的消息idmsg_id
。我们可以对所有收到的消息进行监控并暂时保存,当收到撤回的通知时从存储中找到这条消息记录通过某种方式告知自己。简单归纳如下
- 暂存微信消息,以消息ID作为主键,建议直接存内存里。如果是图片等其他资源,需要下载到一个临时目录
- 监控撤回通知。通过撤回消息关键字找到撤回的消息id,到暂存的历史消息里找到对应记录,通过文件转发助手直接发送到手机客户端。如果是图片等资源,则从临时目录找到对应文件发送
- 因为微信消息撤回时限为2分钟,因此可以把2分钟之前的所有历史记录删除,以节省内存(或者保存到其他地方)
前提是2分钟内不会收到太多消息以至内存耗尽
代码示例
import os
import re
import shutil
import time
import itchat
from itchat.content import *
# 说明:可以撤回的有文本文字、语音、视频、图片、位置、名片、分享、附件
# {msg_id:(msg_from,msg_to,msg_time,msg_time_rec,msg_type,msg_content,msg_share_url)}
msg_dict = {}
# 文件存储临时目录
# rev_tmp_dir = "/home/wechat"
rev_tmp_dir = "F:\private files\wechat\\"
qr_dir = "F:\\qrcode.png"
if not os.path.exists(rev_tmp_dir):
os.mkdir(rev_tmp_dir)
# 表情有一个问题 | 接受信息和接受note的msg_id不一致 巧合解决方案
#face_bug = None
# 将接收到的消息存放在字典中,当接收到新消息时对字典中超时的消息进行清理 | 不接受不具有撤回功能的信息
# [TEXT, PICTURE, MAP, CARD, SHARING, RECORDING, ATTACHMENT, VIDEO, FRIENDS, NOTE]
@ itchat.msg_register([TEXT, PICTURE, MAP, CARD, SHARING, RECORDING, ATTACHMENT, VIDEO])
def handler_receive_msg(msg):
# 获取的是本地时间戳并格式化本地时间戳 e: 2017-04-21 21:30:08
msg_time_rec = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
# 消息ID
msg_id = msg['MsgId']
#print(msg_id)
# 消息时间
msg_time = msg['CreateTime']
# 消息发送人昵称 | 这里也可以使用RemarkName备注 但是自己或者没有备注的人为None
msg_from = (itchat.search_friends(userName=msg['FromUserName']))["NickName"]
#print(msg_from)
# 消息内容
msg_content = None
# 分享的链接
msg_share_url = None
if msg['Type'] == 'Text' or msg['Type'] == 'Friends':
msg_content = msg['Text']
elif msg['Type'] == 'Recording' or msg['Type'] == 'Attachment' or msg['Type'] == 'Video' or msg['Type'] == 'Picture':
msg_content = r"" + msg['FileName']
# 保存文件
msg['Text'](rev_tmp_dir + msg['FileName'])
elif msg['Type'] == 'Card':
msg_content = msg['RecommendInfo']['NickName'] + r" 的名片"
# itchat.send(msg_content, toUserName='filehelper')
elif msg['Type'] == 'Map':
x, y, location = re.search('<location x="(.*?)" y="(.*?)".*label="(.*?)".*', msg['OriContent']).group(1, 2, 3)
if location is None:
msg_content = r"纬度->" + x.__str__() + " 经度->" + y.__str__()
else:
msg_content = r"" + location
elif msg['Type'] == 'Sharing':
msg_content = msg['Text']
msg_share_url = msg['Url']
# 更新字典
msg_dict.update({
msg_id: {
"msg_from": msg_from, "msg_time": msg_time, "msg_time_rec": msg_time_rec,
"msg_type": msg["Type"],
"msg_content": msg_content, "msg_share_url": msg_share_url
}
})
# 删除过期记录
for mid in list(msg_dict.keys()):
if (msg_dict[mid].get('msg_time') < time.time()-120):
# 删除记录,以及对应的文件(略)
del msg_dict[mid]
else:
break
# 收到note通知类消息,判断是不是撤回并进行相应操作
@ itchat.msg_register([NOTE])
def send_msg_helper(msg):
# 匹配撤回消息
if re.search(r"\<\!\[CDATA\[.*撤回了一条消息\]\]\>", msg['Content']) is not None:
# 获取消息的id
old_msg_id = re.search("<msgid>(.*?)</msgid>", msg['Content']).group(1)
print("发现撤回消息 " + old_msg_id)
old_msg = msg_dict.get(old_msg_id, {})
# print(old_msg)
msg_body = "[撤回]" + " " + old_msg.get('msg_from') + " " + old_msg.get('msg_time_rec') + "\n" + old_msg.get('msg_content')
# 如果是分享
if old_msg['msg_type'] == "Sharing":
msg_body += old_msg.get('msg_share_url')
# 将撤回消息发送到文件助手
itchat.send(msg_body, toUserName='filehelper')
# 有文件的话也要将文件发送回去
if old_msg["msg_type"] == "Picture" or old_msg["msg_type"] == "Recording" or old_msg["msg_type"] == "Video" or old_msg["msg_type"] == "Attachment":
file = '@%s@%s' % ('img' if msg['Type'] == 'Picture' else 'fil', rev_tmp_dir + old_msg['msg_content'])
itchat.send(msg=file, toUserName='filehelper')
if __name__ == '__main__':
# 打开 hotReload 短时间内可自动重新登录
# 命令行查看二维码使用 enableCmdQR=True,关闭则会调用系统的文件查看命令(win 使用 open, linux 使用 xdg-open,需要yum install xdg-utils)
itchat.auto_login(hotReload=True, enableCmdQR=1)
itchat.run()
其他应用
- 聊天机器人自动回复
- 发送定时消息
- 斗图(调用图像识别api)
- 聊天记录备份,分析
- 微信好友数据分析