update python_wechat

master
xianhu 2017-06-25 11:23:36 +08:00
parent a44c05f913
commit 77803dc215
1 changed files with 347 additions and 50 deletions

View File

@ -4,7 +4,9 @@
python_wechat.py by xianhu python_wechat.py by xianhu
主要包括如下功能 主要包括如下功能
(1) 自动提醒群红包 (1) 自动提醒群红包
(2) 自动保存被撤回消息包括内容文件图片语音视频等 (2) 自动监测被撤回消息
(3) 群统计发言人数发言次数等
(4) 群关键字提醒群被@提醒
""" """
import time import time
@ -15,7 +17,7 @@ from itchat.content import *
# 初始化 # 初始化
my = itchat.new_instance() my = itchat.new_instance()
my.auto_login(hotReload=False, enableCmdQR=2) my.auto_login(hotReload=False, enableCmdQR=-2)
my.global_keys = ["创业", "算法", "人工智能", "机器学习"] my.global_keys = ["创业", "算法", "人工智能", "机器学习"]
my.to_user_name = "filehelper" my.to_user_name = "filehelper"
@ -38,24 +40,15 @@ def get_msg_list(msg):
""" """
提取消息内容消息来源分类 提取消息内容消息来源分类
1来自好友的消息 1来自好友的消息
2来自自己的消息 2来自的消息
3来自文件传输助手的消息 3来自文件传输助手的消息
4来自群聊的消息
提取消息内容消息类型分类 提取消息内容消息类型分类
1文字 1文字2图片3语音4视频5地址6名片7提醒8分享9附件
2图片
3语音
4视频
5地址
6名片
7Note
8分享
9附件
""" """
logging.warning("message: %s", msg) # logging.warning("message: %s", msg)
msg_id = msg["MsgId"] # 消息ID msg_id = msg["MsgId"] # 消息ID
from_user_name = msg["FromUserName"] # 消息发送者ID from_user_name = msg["FromUserName"] # 消息发送者ID如果为群消息则为群ID
to_user_name = msg["ToUserName"] # 消息接受者ID to_user_name = msg["ToUserName"] # 消息接受者ID如果为群消息则为群ID
msg_type = msg["MsgType"] # 消息类型 msg_type = msg["MsgType"] # 消息类型
msg_content = msg["Content"] # 消息内容 msg_content = msg["Content"] # 消息内容
@ -70,34 +63,35 @@ def get_msg_list(msg):
wind_name = msg["User"]["RemarkName"] if msg["User"].get("RemarkName") else ( wind_name = msg["User"]["RemarkName"] if msg["User"].get("RemarkName") else (
msg["User"]["NickName"] if msg["User"].get("NickName") else to_user_name msg["User"]["NickName"] if msg["User"].get("NickName") else to_user_name
) ) # 窗口名称:群名或好友名
if from_user_name.startswith("@@"): msg_user_name = from_user_name # 消息发送者的ID
nick_name = msg["ActualNickName"] if (msg["ActualUserName"] not in my.friends) or \ msg_nick_name = wind_name # 消息发送者的昵称
(not my.friends[msg["ActualUserName"]]["RemarkName"]) else my.friends[msg["ActualUserName"]]["RemarkName"] if from_user_name.startswith("@@") or to_user_name.startswith("@@"):
else: msg_user_name = msg["ActualUserName"]
nick_name = wind_name msg_nick_name = msg["ActualNickName"] if (msg_user_name not in my.friends) or (not my.friends[msg_user_name]["RemarkName"]) else my.friends[msg_user_name]["RemarkName"]
is_at = msg.get("IsAt", None) # 是否在群内被@
we_type = msg["Type"] # 消息类型 we_type = msg["Type"] # 消息类型
we_text = msg["Text"] # 消息内容 we_text = msg["Text"] # 消息内容
logging.warning("show: nick_name=%s, wind_name=%s, we_type=%s, we_text=%s", nick_name, wind_name, we_type, we_text) logging.warning("show: nick_name=%s, wind_name=%s, we_type=%s, we_text=%s, msg_time=%s", msg_nick_name, wind_name, we_type, we_text, msg_time)
return msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, \ return msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, \
msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, wind_name, nick_name, we_type, we_text wind_name, msg_user_name, msg_nick_name, is_at, we_type, we_text
# 消息注册,主要处理群消息 # 消息注册,主要处理群消息
@my.msg_register([TEXT, MAP, CARD, NOTE, SHARING, PICTURE, RECORDING, ATTACHMENT, VIDEO], isFriendChat=True, isGroupChat=True) @my.msg_register([TEXT, PICTURE, RECORDING, VIDEO, MAP, CARD, NOTE, SHARING, ATTACHMENT], isFriendChat=True, isGroupChat=True)
def text_reply(msg): def text_reply(msg):
""" """
消息自动接收, 接受全部的消息 消息自动接收, 接受全部的消息
""" """
# 消息提取 # 消息提取
msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, \ msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, \
msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, wind_name, nick_name, we_type, we_text = get_msg_list(msg) wind_name, msg_user_name, msg_nick_name, is_at, we_type, we_text = get_msg_list(msg)
# 消息过滤, 只监测文字、注解、分享、图片、语音等 # 消息过滤, 只监测文字、图片、语音、注解、分享
if we_type not in ["Text", "Note", "Sharing", "Picture", "Recording"]: if we_type not in ["Text", "Picture", "Recording", "Note", "Sharing"]:
logging.warning("message type isn't included, ignored") logging.warning("message type isn't included, ignored")
return return
@ -109,40 +103,343 @@ def text_reply(msg):
# 消息存储,删除过期消息 # 消息存储,删除过期消息
my.msg_store[msg_id] = msg my.msg_store[msg_id] = msg
for _id in [_id for _id in my.msg_store if time.time() - my.msg_store[_id]["CreateTime"] > 120]: for _id in [_id for _id in my.msg_store if time.time() - my.msg_store[_id]["CreateTime"] > 120]:
logging.warning("delete message, message_id = %s", _id)
my.msg_store.pop(_id) my.msg_store.pop(_id)
# 处理群消息 # 保存消息中的内容(图片、语音等),不保存动态图片
if from_user_name.startswith("@@"): if (we_type in ["Picture", "Recording"]) and (not msg_file.endswith(".gif")):
# 红包消息处理 try:
if we_type == "Note" and we_text.find("收到红包,请在手机上查看") >= 0: we_text(".Cache/" + msg_file)
my.send("%s】中有红包,快抢!" % wind_name, toUserName=my.to_user_name) logging.warning("downloading %s to .Cache/", msg_file)
except Exception as excep:
logging.error("downloading %s to .Cache/ error: %s", msg_file, excep)
# 撤回消息处理 # ==== 处理红包消息 ====
if from_user_name.startswith("@@"):
# ==== 处理红包消息 ====
if we_type == "Note" and we_text.find("收到红包,请在手机上查看") >= 0:
my.send("%s】中【%s】发红包啦,快抢!" % (wind_name, msg_nick_name), toUserName=my.to_user_name)
# ==== 处理关键词消息 ====
for key in my.global_keys:
if we_type == "Text" and we_text.find(key) >= 0:
my.send("%s】中【%s】提及关键字:%s" % (wind_name, msg_nick_name, key), toUserName=my.to_user_name)
my.send(we_text, toUserName=my.to_user_name)
break
# ==== 群内是否被@ ====
if we_type == "Text" and is_at:
my.send("%s】中【%s】@了你" % (wind_name, msg_nick_name), toUserName=my.to_user_name)
my.send(we_text, toUserName=my.to_user_name)
# ==== 撤回消息处理(必须为最后一步) ====
if we_type == "Note" and we_text.find("撤回了一条消息") >= 0: if we_type == "Note" and we_text.find("撤回了一条消息") >= 0:
old_msg = my.msg_store.get(msg_content[msg_content.find("<msgid>")+7: msg_content.find("</msgid>")]) old_msg = my.msg_store.get(msg_content[msg_content.find("<msgid>")+7: msg_content.find("</msgid>")])
if not old_msg: if not old_msg:
logging.warning("not message id in my.msg_store") logging.warning("not message id in my.msg_store")
return return
msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, \ msg_id, from_user_name, to_user_name, msg_type, msg_content, msg_time, msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, \
msg_file, msg_file_length, msg_voice_length, msg_play_length, msg_url, wind_name, nick_name, we_type, we_text = get_msg_list(old_msg) wind_name, msg_user_name, msg_nick_name, is_at, we_type, we_text = get_msg_list(old_msg)
my.send("%s】中【%s】撤回了自己发送的消息:\nType: %s\nTime: %s\n%s" % (wind_name, msg_nick_name, we_type, msg_time, msg_file), toUserName=my.to_user_name)
if we_type == "Text": if we_type == "Text":
msg_content = we_text my.send(we_text, toUserName=my.to_user_name)
elif we_type == "Sharing": elif we_type == "Sharing":
msg_content = we_text + ": " + msg_url my.send(we_text + "\n" + msg_url, toUserName=my.to_user_name)
elif we_type in ["Picture", "Recording"]: elif (we_type in ["Picture", "Recording"]) and (not msg_file.endswith(".gif")):
try: my.send_image(".Cache/" + msg_file, toUserName=my.to_user_name)
we_text(".Cache/" + msg_file)
logging.warning("downloading %s to .Cache/", msg_file)
except Exception as excep:
logging.error("downloading %s to .Cache/ error: %s", msg_file, excep)
msg_content = msg_file
my.send("%s】中有消息被撤回:\nFrom: %s\nType: %s\nTime: %s\n%s" % (wind_name, nick_name, we_type, msg_time, msg_content), toUserName=my.to_user_name)
return return
# 运行程序 # 运行程序
my.run() my.run(debug=False)
"""
好友消息
{
'MsgId': '5254859004542036569',
'FromUserName': '@f3b7fdc54717ea8dc22cb3edef59688e82ef34874e3236801537b94f6cd73e1e',
'ToUserName': '@e79dde912b8f817514c01f399ca9ba12',
'MsgType': 1,
'Content': '[微笑]己改',
'Status': 3,
'ImgStatus': 1,
'CreateTime': 1498448860,
'VoiceLength': 0,
'PlayLength': 0,
'FileName': '',
'FileSize': '',
'MediaId': '',
'Url': '',
'AppMsgType': 0,
'StatusNotifyCode': 0,
'StatusNotifyUserName': '',
'HasProductId': 0,
'Ticket': '',
'ImgHeight': 0,
'ImgWidth': 0,
'SubMsgType': 0,
'NewMsgId': 5254859004542036569,
'OriContent': '',
'User': <User: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@f3b7fdc54717ea8dc22cb3edef59688e82ef34874e3236801537b94f6cd73e1e',
'NickName': '付贵吉祥',
'HeadImgUrl': '/cgi-bin/mmwebwx-bin/webwxgeticon?seq=688475226&username=@f3b7fdc54717ea8dc22cb3edef59688e82ef34874e3236801537b94f6cd73e1e&skey=@',
'ContactFlag': 3,
'MemberCount': 0,
'RemarkName': '付贵吉祥@中建5号楼',
'HideInputBarFlag': 0,
'Sex': 1,
'Signature': '漫漫人生路...',
'VerifyFlag': 0,
'OwnerUin': 0,
'PYInitial': 'FGJX',
'PYQuanPin': 'fuguijixiang',
'RemarkPYInitial': 'FGJXZJ5HL',
'RemarkPYQuanPin': 'fuguijixiangzhongjian5haolou',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 135205,
'Province': '山东',
'City': '',
'Alias': '',
'SnsFlag': 17,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': '',
'EncryChatRoomId': '',
'IsOwner': 0
}>,
'Type': 'Text',
'Text': '[微笑]己改'
}
"""
"""
群消息来自别人
{
'MsgId': '7844877618948840992',
'FromUserName': '@@8dc5df044444d1fb8e3972e755b47adf9d07f5a032cae90a4d822b74ee1e4880',
'ToUserName': '@e79dde912b8f817514c01f399ca9ba12',
'MsgType': 1,
'Content': '就是那个,那个协议我们手上有吗',
'Status': 3,
'ImgStatus': 1,
'CreateTime': 1498448972,
'VoiceLength': 0,
'PlayLength': 0,
'FileName': '',
'FileSize': '',
'MediaId': '',
'Url': '',
'AppMsgType': 0,
'StatusNotifyCode': 0,
'StatusNotifyUserName': '',
'HasProductId': 0,
'Ticket': '',
'ImgHeight': 0,
'ImgWidth': 0,
'SubMsgType': 0,
'NewMsgId': 7844877618948840992,
'OriContent': '',
'ActualNickName': '5-1-1003',
'IsAt': False,
'ActualUserName': '@a0922f18795e4c3b6d7d09c492ace233',
'User': <Chatroom: {
'MemberList': <ContactList: [
<ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@e79dde912b8f817514c01f399ca9ba12',
'NickName': '齐现虎',
'AttrStatus': 2147600869,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '5-1-1601',
'KeyWord': 'qix'
}>,
<ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@a9620e3d4b82eab2521ccdbb985afc37',
'NickName': 'A高佳祥15069179911',
'AttrStatus': 102503,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '5-2-220315069179911',
'KeyWord': 'gao'
}>,
.......
]>,
'Uin': 0,
'UserName': '@@8dc5df044444d1fb8e3972e755b47adf9d07f5a032cae90a4d822b74ee1e4880',
'NickName': '中建锦绣澜庭二期5#楼',
'HeadImgUrl': '/cgi-bin/mmwebwx-bin/webwxgetheadimg?seq=0&username=@@8dc5df044444d1fb8e3972e755b47adf9d07f5a032cae90a4d822b74ee1e4880&skey=@',
'ContactFlag': 3,
'MemberCount': 106,
'RemarkName': '',
'HideInputBarFlag': 0,
'Sex': 0,
'Signature': '',
'VerifyFlag': 0,
'OwnerUin': 0,
'PYInitial': 'ZJJXLTEJ5L',
'PYQuanPin': 'zhongjianjinxiulantingerji5lou',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 0,
'Province': '',
'City': '',
'Alias': '',
'SnsFlag': 0,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': '',
'EncryChatRoomId': '@d1e510bc8cbd192468e9c85c6f5a9d81',
'IsOwner': 1,
'IsAdmin': None,
'Self': <ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@e79dde912b8f817514c01f399ca9ba12',
'NickName': '齐现虎',
'AttrStatus': 2147600869,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '5-1-1601',
'KeyWord': 'qix'
}>,
'HeadImgUpdateFlag': 1,
'ContactType': 0,
'ChatRoomOwner': '@e79dde912b8f817514c01f399ca9ba12'
}>,
'Type': 'Text',
'Text': '就是那个,那个协议我们手上有吗'
}
群消息来自自己
WARNING: root: message: {
'MsgId': '6658361167561279652',
'FromUserName': '@e79dde912b8f817514c01f399ca9ba12',
'ToUserName': '@@cbd264b76a28d4f5bad27197de60735bc082c95ab49891cf64d745ff4be17e30',
'MsgType': 1,
'Content': '看来最近是没有线下活动了',
'Status': 3,
'ImgStatus': 1,
'CreateTime': 1498455090,
'VoiceLength': 0,
'PlayLength': 0,
'FileName': '',
'FileSize': '',
'MediaId': '',
'Url': '',
'AppMsgType': 0,
'StatusNotifyCode': 0,
'StatusNotifyUserName': '',
'HasProductId': 0,
'Ticket': '',
'ImgHeight': 0,
'ImgWidth': 0,
'SubMsgType': 0,
'NewMsgId': 6658361167561279652,
'OriContent': '',
'ActualNickName': '齐现虎',
'IsAt': False,
'ActualUserName': '@e79dde912b8f817514c01f399ca9ba12',
'User': <Chatroom: {
'MemberList': <ContactList: [
<ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@143b0d468a5be892768e372aae5d3c97f20dd73c6f28a99092fb96d4fc7862e3',
'NickName': '李小盛',
'AttrStatus': 235557,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '',
'KeyWord': ''
}>,
<ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@edb9eb66b235ef72237ad025290813159b6d3e9cdaa15284ad7f6748b38c2c1f',
'NickName': '小z@德研社',
'AttrStatus': 103461,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '小z研值512',
'KeyWord': ''
}>,
]>,
'Uin': 0,
'UserName': '@@cbd264b76a28d4f5bad27197de60735bc082c95ab49891cf64d745ff4be17e30',
'NickName': '德研社z战队',
'HeadImgUrl': '/cgi-bin/mmwebwx-bin/webwxgetheadimg?seq=688463282&username=@@cbd264b76a28d4f5bad27197de60735bc082c95ab49891cf64d745ff4be17e30&skey=',
'ContactFlag': 2,
'MemberCount': 70,
'RemarkName': '',
'HideInputBarFlag': 0,
'Sex': 0,
'Signature': '',
'VerifyFlag': 0,
'OwnerUin': 0,
'PYInitial': 'DYSZZD',
'PYQuanPin': 'deyanshezzhandui',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 0,
'Province': '',
'City': '',
'Alias': '',
'SnsFlag': 0,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': '',
'EncryChatRoomId': '@be08ab93d4d5440069d6617df937b689',
'IsOwner': 0,
'IsAdmin': None,
'Self': <ChatroomMember: {
'MemberList': <ContactList: []>,
'Uin': 0,
'UserName': '@e79dde912b8f817514c01f399ca9ba12',
'NickName': '齐现虎',
'AttrStatus': 2147600869,
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'MemberStatus': 0,
'DisplayName': '',
'KeyWord': 'qix'
}>
}>,
'Type': 'Text',
'Text': '看来最近是没有线下活动了'
}
"""