diff --git a/README.md b/README.md index 1ba9e11c..de8f7e00 100644 --- a/README.md +++ b/README.md @@ -59,36 +59,61 @@ itchat.run() ## 进阶应用 +### 特殊的字典使用方式 + +通过打印itchat的用户以及注册消息的参数,可以发现这些值都是字典。 + +但实际上itchat精心构造了相应的消息、用户、群聊、公众号类。 + +其所有的键值都可以通过这一方式访问: + +```python +@itchat.msg_register(TEXT) +def _(msg): + # equals to print(msg['FromUserName']) + print(msg.fromUserName) +``` + +属性名为键值首字母小写后的内容。 + +```python +author = itchat.search_friends(nickName='LittleCoder')[0] +author.send('greeting, littlecoder!') +``` + ### 各类型消息的注册 通过如下代码,微信已经可以就日常的各种信息进行获取与回复。 ```python -#coding=utf8 import itchat, time from itchat.content import * @itchat.msg_register([TEXT, MAP, CARD, NOTE, SHARING]) def text_reply(msg): - itchat.send('%s: %s' % (msg['Type'], msg['Text']), msg['FromUserName']) + msg.user.send('%s: %s' % (msg.type, msg.text)) @itchat.msg_register([PICTURE, RECORDING, ATTACHMENT, VIDEO]) def download_files(msg): - msg['Text'](msg['FileName']) - return '@%s@%s' % ({'Picture': 'img', 'Video': 'vid'}.get(msg['Type'], 'fil'), msg['FileName']) + msg.download(msg.fileName) + typeSymbol = { + PICTURE: 'img', + VIDEO: 'vid', }.get(msg.type, 'fil') + return '@%s@%s' % (typeSymbol, msg.fileName) @itchat.msg_register(FRIENDS) def add_friend(msg): - itchat.add_friend(**msg['Text']) # 该操作会自动将新好友的消息录入,不需要重载通讯录 - itchat.send_msg('Nice to meet you!', msg['RecommendInfo']['UserName']) + msg.user.verify() + msg.user.send('Nice to meet you!') @itchat.msg_register(TEXT, isGroupChat=True) def text_reply(msg): - if msg['isAt']: - itchat.send(u'@%s\u2005I received: %s' % (msg['ActualNickName'], msg['Content']), msg['FromUserName']) + if msg.isAt: + msg.user.send(u'@%s\u2005I received: %s' % ( + msg.actualNickName, msg.text)) itchat.auto_login(True) -itchat.run() +itchat.run(True) ``` ### 命令行二维码 @@ -234,6 +259,8 @@ A: 有些账号是天生无法给自己的账号发送信息的,建议使用`f ## 类似项目 +[youfou/wxpy][youfou-wxpy]: 优秀的api包装和配套插件,微信机器人/优雅的微信个人号API + [liuwons/wxBot][liuwons-wxBot]: 类似的基于Python的微信机器人 [zixia/wechaty][zixia-wechaty]: 基于Javascript(ES6)的微信个人账号机器人NodeJS框架/库 @@ -267,6 +294,7 @@ A: 有些账号是天生无法给自己的账号发送信息的,建议使用`f [littlecodersh]: https://github.com/littlecodersh [tempdban]: https://github.com/tempdban [Chyroc]: https://github.com/Chyroc +[youfou-wxpy]: https://github.com/youfou/wxpy [liuwons-wxBot]: https://github.com/liuwons/wxBot [zixia-wechaty]: https://github.com/zixia/wechaty [Mojo-Weixin]: https://github.com/sjdy521/Mojo-Weixin diff --git a/itchat/components/contact.py b/itchat/components/contact.py index 1fc98bec..95bd3b60 100644 --- a/itchat/components/contact.py +++ b/itchat/components/contact.py @@ -6,7 +6,7 @@ from .. import config, utils from ..returnvalues import ReturnValue -from ..storage import contact_change +from ..storage import contact_change, templates logger = logging.getLogger('itchat') @@ -99,8 +99,8 @@ def update_friend(self, userName): return r if len(r) != 1 else r[0] def update_info_dict(oldInfoDict, newInfoDict): - ''' - only normal values will be updated here + ''' only normal values will be updated here + because newInfoDict is normal dict, so it's not necessary to consider templates ''' for k, v in newInfoDict.items(): if any((isinstance(v, t) for t in (tuple, list, dict))): @@ -130,8 +130,8 @@ def update_local_chatrooms(core, l): if oldChatroom: update_info_dict(oldChatroom, chatroom) # - update other values - memberList, oldMemberList = (c.get('MemberList', []) - for c in (chatroom, oldChatroom)) + memberList = chatroom.get('MemberList', []) + oldMemberList = oldChatroom.memberList if memberList: for member in memberList: oldMember = utils.search_dict_list( @@ -141,17 +141,19 @@ def update_local_chatrooms(core, l): else: oldMemberList.append(member) else: - oldChatroom = chatroom - core.chatroomList.append(chatroom) + oldChatroom = templates.wrap_user_dict(chatroom) + core.chatroomList.append(oldChatroom) # delete useless members if len(chatroom['MemberList']) != len(oldChatroom['MemberList']) and \ chatroom['MemberList']: existsUserNames = [member['UserName'] for member in chatroom['MemberList']] delList = [] for i, member in enumerate(oldChatroom['MemberList']): - if member['UserName'] not in existsUserNames: delList.append(i) + if member['UserName'] not in existsUserNames: + delList.append(i) delList.sort(reverse=True) - for i in delList: del oldChatroom['MemberList'][i] + for i in delList: + del oldChatroom['MemberList'][i] # - update OwnerUin if oldChatroom.get('ChatRoomOwner') and oldChatroom.get('MemberList'): oldChatroom['OwnerUin'] = utils.search_dict_list(oldChatroom['MemberList'], @@ -184,8 +186,8 @@ def update_local_friends(core, l): utils.emoji_formatter(friend, 'NickName') if 'DisplayName' in friend: utils.emoji_formatter(friend, 'DisplayName') - if 'RemarkName' in member: - utils.emoji_formatter(member, 'RemarkName') + if 'RemarkName' in friend: + utils.emoji_formatter(friend, 'RemarkName') oldInfoDict = utils.search_dict_list( fullList, 'UserName', friend['UserName']) if oldInfoDict is None: diff --git a/itchat/components/hotreload.py b/itchat/components/hotreload.py index f0a40d04..d34d25ee 100644 --- a/itchat/components/hotreload.py +++ b/itchat/components/hotreload.py @@ -5,6 +5,7 @@ from ..config import VERSION from ..returnvalues import ReturnValue +from ..storage import templates from .contact import update_local_chatrooms, update_local_friends from .messages import produce_msg @@ -50,6 +51,8 @@ def load_login_status(self, fileDir, 'ErrMsg': 'cached status ignored because of version', 'Ret': -1005, }}) self.loginInfo = j['loginInfo'] + self.loginInfo['User'] = templates.User(self.loginInfo['User']) + self.loginInfo['User'].core = self self.s.cookies = requests.utils.cookiejar_from_dict(j['cookies']) self.storageClass.loads(j['storage']) msgList, contactList = self.get_msg() diff --git a/itchat/components/login.py b/itchat/components/login.py index c35d1ecd..7736427c 100644 --- a/itchat/components/login.py +++ b/itchat/components/login.py @@ -9,6 +9,7 @@ from .. import config, utils from ..returnvalues import ReturnValue +from ..storage.templates import wrap_user_dict from .contact import update_local_chatrooms, update_local_friends from .messages import produce_msg @@ -182,7 +183,7 @@ def web_init(self): # deal with login info utils.emoji_formatter(dic['User'], 'NickName') self.loginInfo['InviteStartCount'] = int(dic['InviteStartCount']) - self.loginInfo['User'] = utils.struct_friend_info(dic['User']) + self.loginInfo['User'] = wrap_user_dict(utils.struct_friend_info(dic['User'])) self.memberList.append(self.loginInfo['User']) self.loginInfo['SyncKey'] = dic['SyncKey'] self.loginInfo['synckey'] = '|'.join(['%s_%s' % (item['Key'], item['Val']) @@ -247,6 +248,7 @@ def maintain_loop(): else: otherList.append(contact) chatroomMsg = update_local_chatrooms(self, chatroomList) + chatroomMsg['User'] = self.loginInfo['User'] self.msgList.put(chatroomMsg) update_local_friends(self, otherList) retryCount = 0 diff --git a/itchat/components/messages.py b/itchat/components/messages.py index 9f3cd10d..c376dc59 100644 --- a/itchat/components/messages.py +++ b/itchat/components/messages.py @@ -1,438 +1,509 @@ -import os, time, re, io -import json -import mimetypes, hashlib -import traceback, logging -from collections import OrderedDict - -import requests - -from .. import config, utils -from ..returnvalues import ReturnValue -from .contact import update_local_uin - -logger = logging.getLogger('itchat') - -def load_messages(core): - core.send_raw_msg = send_raw_msg - core.send_msg = send_msg - core.upload_file = upload_file - core.send_file = send_file - core.send_image = send_image - core.send_video = send_video - core.send = send - -def get_download_fn(core, url, msgId): - def download_fn(downloadDir=None): - params = { - 'msgid': msgId, - 'skey': core.loginInfo['skey'],} - headers = { 'User-Agent' : config.USER_AGENT } - r = core.s.get(url, params=params, stream=True, headers = headers) - tempStorage = io.BytesIO() - for block in r.iter_content(1024): - tempStorage.write(block) - if downloadDir is None: - return tempStorage.getvalue() - with open(downloadDir, 'wb') as f: - f.write(tempStorage.getvalue()) - tempStorage.seek(0) - return ReturnValue({'BaseResponse': { - 'ErrMsg': 'Successfully downloaded', - 'Ret': 0, }, - 'PostFix': utils.get_image_postfix(tempStorage.read(20)), }) - return download_fn - -def produce_msg(core, msgList): - ''' for messages types - * 40 msg, 43 videochat, 50 VOIPMSG, 52 voipnotifymsg - * 53 webwxvoipnotifymsg, 9999 sysnotice - ''' - rl = [] - srl = [40, 43, 50, 52, 53, 9999] - for m in msgList: - if '@@' in m['FromUserName'] or '@@' in m['ToUserName']: - produce_group_chat(core, m) - else: - utils.msg_formatter(m, 'Content') - if m['MsgType'] == 1: # words - if m['Url']: - regx = r'(.+?\(.+?\))' - data = re.search(regx, m['Content']) - data = 'Map' if data is None else data.group(1) - msg = { - 'Type': 'Map', - 'Text': data,} - else: - msg = { - 'Type': 'Text', - 'Text': m['Content'],} - elif m['MsgType'] == 3 or m['MsgType'] == 47: # picture - download_fn = get_download_fn(core, - '%s/webwxgetmsgimg' % core.loginInfo['url'], m['NewMsgId']) - msg = { - 'Type' : 'Picture', - 'FileName' : '%s.%s' % (time.strftime('%y%m%d-%H%M%S', time.localtime()), - 'png' if m['MsgType'] == 3 else 'gif'), - 'Text' : download_fn, } - elif m['MsgType'] == 34: # voice - download_fn = get_download_fn(core, - '%s/webwxgetvoice' % core.loginInfo['url'], m['NewMsgId']) - msg = { - 'Type': 'Recording', - 'FileName' : '%s.mp3' % time.strftime('%y%m%d-%H%M%S', time.localtime()), - 'Text': download_fn,} - elif m['MsgType'] == 37: # friends - msg = { - 'Type': 'Friends', - 'Text': { - 'status' : m['Status'], - 'userName' : m['RecommendInfo']['UserName'], - 'verifyContent' : m['Ticket'], - 'autoUpdate' : m['RecommendInfo'], }, } - elif m['MsgType'] == 42: # name card - msg = { - 'Type': 'Card', - 'Text': m['RecommendInfo'], } - elif m['MsgType'] in (43, 62): # tiny video - msgId = m['MsgId'] - def download_video(videoDir=None): - url = '%s/webwxgetvideo' % core.loginInfo['url'] - params = { - 'msgid': msgId, - 'skey': core.loginInfo['skey'],} - headers = {'Range': 'bytes=0-', 'User-Agent' : config.USER_AGENT } - r = core.s.get(url, params=params, headers=headers, stream=True) - tempStorage = io.BytesIO() - for block in r.iter_content(1024): - tempStorage.write(block) - if videoDir is None: - return tempStorage.getvalue() - with open(videoDir, 'wb') as f: - f.write(tempStorage.getvalue()) - return ReturnValue({'BaseResponse': { - 'ErrMsg': 'Successfully downloaded', - 'Ret': 0, }}) - msg = { - 'Type': 'Video', - 'FileName' : '%s.mp4' % time.strftime('%y%m%d-%H%M%S', time.localtime()), - 'Text': download_video, } - elif m['MsgType'] == 49: # sharing - if m['AppMsgType'] == 6: - rawMsg = m - cookiesList = {name:data for name,data in core.s.cookies.items()} - def download_atta(attaDir=None): - url = core.loginInfo['fileUrl'] + '/webwxgetmedia' - params = { - 'sender': rawMsg['FromUserName'], - 'mediaid': rawMsg['MediaId'], - 'filename': rawMsg['FileName'], - 'fromuser': core.loginInfo['wxuin'], - 'pass_ticket': 'undefined', - 'webwx_data_ticket': cookiesList['webwx_data_ticket'],} - headers = { 'User-Agent' : config.USER_AGENT } - r = core.s.get(url, params=params, stream=True, headers=headers) - tempStorage = io.BytesIO() - for block in r.iter_content(1024): - tempStorage.write(block) - if attaDir is None: - return tempStorage.getvalue() - with open(attaDir, 'wb') as f: - f.write(tempStorage.getvalue()) - return ReturnValue({'BaseResponse': { - 'ErrMsg': 'Successfully downloaded', - 'Ret': 0, }}) - msg = { - 'Type': 'Attachment', - 'Text': download_atta, } - elif m['AppMsgType'] == 8: - download_fn = get_download_fn(core, - '%s/webwxgetmsgimg' % core.loginInfo['url'], m['NewMsgId']) - msg = { - 'Type' : 'Picture', - 'FileName' : '%s.gif' % ( - time.strftime('%y%m%d-%H%M%S', time.localtime())), - 'Text' : download_fn, } - elif m['AppMsgType'] == 17: - msg = { - 'Type': 'Note', - 'Text': m['FileName'], } - elif m['AppMsgType'] == 2000: - regx = r'\[CDATA\[(.+?)\][\s\S]+?\[CDATA\[(.+?)\]' - data = re.search(regx, m['Content']) - if data: - data = data.group(2).split(u'\u3002')[0] - else: - data = 'You may found detailed info in Content key.' - msg = { - 'Type': 'Note', - 'Text': data, } - else: - msg = { - 'Type': 'Sharing', - 'Text': m['FileName'], } - elif m['MsgType'] == 51: # phone init - msg = update_local_uin(core, m) - elif m['MsgType'] == 10000: - msg = { - 'Type': 'Note', - 'Text': m['Content'],} - elif m['MsgType'] == 10002: - regx = r'\[CDATA\[(.+?)\]\]' - data = re.search(regx, m['Content']) - data = 'System message' if data is None else data.group(1).replace('\\', '') - msg = { - 'Type': 'Note', - 'Text': data, } - elif m['MsgType'] in srl: - msg = { - 'Type': 'Useless', - 'Text': 'UselessMsg', } - else: - logger.debug('Useless message received: %s\n%s' % (m['MsgType'], str(m))) - msg = { - 'Type': 'Useless', - 'Text': 'UselessMsg', } - m = dict(m, **msg) - rl.append(m) - return rl - -def produce_group_chat(core, msg): - r = re.match('(@[0-9a-z]*?):
(.*)$', msg['Content']) - if r: - actualUserName, content = r.groups() - chatroomUserName = msg['FromUserName'] - elif msg['FromUserName'] == core.storageClass.userName: - actualUserName = core.storageClass.userName - content = msg['Content'] - chatroomUserName = msg['ToUserName'] - else: - msg['ActualUserName'] = core.storageClass.userName - msg['ActualNickName'] = core.storageClass.nickName - msg['isAt'] = False - utils.msg_formatter(msg, 'Content') - return - chatroom = core.storageClass.search_chatrooms(userName=chatroomUserName) - member = utils.search_dict_list((chatroom or {}).get( - 'MemberList') or [], 'UserName', actualUserName) - if member is None: - chatroom = core.update_chatroom(msg['FromUserName']) - member = utils.search_dict_list((chatroom or {}).get( - 'MemberList') or [], 'UserName', actualUserName) - if member is None: - logger.debug('chatroom member fetch failed with %s' % actualUserName) - msg['ActualNickName'] = '' - msg['isAt'] = False - else: - msg['ActualNickName'] = member['DisplayName'] or member['NickName'] - atFlag = '@' + (chatroom['self']['DisplayName'] - or core.storageClass.nickName) - msg['isAt'] = ( - (atFlag + (u'\u2005' if u'\u2005' in msg['Content'] else ' ')) - in msg['Content'] or msg['Content'].endswith(atFlag)) - msg['ActualUserName'] = actualUserName - msg['Content'] = content - utils.msg_formatter(msg, 'Content') - -def send_raw_msg(self, msgType, content, toUserName): - url = '%s/webwxsendmsg' % self.loginInfo['url'] - data = { - 'BaseRequest': self.loginInfo['BaseRequest'], - 'Msg': { - 'Type': msgType, - 'Content': content, - 'FromUserName': self.storageClass.userName, - 'ToUserName': (toUserName if toUserName else self.storageClass.userName), - 'LocalID': int(time.time() * 1e4), - 'ClientMsgId': int(time.time() * 1e4), - }, - 'Scene': 0, } - headers = { 'ContentType': 'application/json; charset=UTF-8', 'User-Agent' : config.USER_AGENT } - r = self.s.post(url, headers=headers, - data=json.dumps(data, ensure_ascii=False).encode('utf8')) - return ReturnValue(rawResponse=r) - -def send_msg(self, msg='Test Message', toUserName=None): - logger.debug('Request to send a text message to %s: %s' % (toUserName, msg)) - r = self.send_raw_msg(1, msg, toUserName) - return r - -def upload_file(self, fileDir, isPicture=False, isVideo=False, - toUserName='filehelper'): - logger.debug('Request to upload a %s: %s' % ( - 'picture' if isPicture else 'video' if isVideo else 'file', fileDir)) - if not utils.check_file(fileDir): - return ReturnValue({'BaseResponse': { - 'ErrMsg': 'No file found in specific dir', - 'Ret': -1002, }}) - fileSize = os.path.getsize(fileDir) - fileSymbol = 'pic' if isPicture else 'video' if isVideo else'doc' - with open(fileDir, 'rb') as f: - fileMd5 = hashlib.md5(f.read()).hexdigest() - file_ = open(fileDir, 'rb') - chunks = int((fileSize - 1) / 524288) + 1 - clientMediaId = int(time.time() * 1e4) - uploadMediaRequest = json.dumps(OrderedDict([ - ('UploadType', 2), - ('BaseRequest', self.loginInfo['BaseRequest']), - ('ClientMediaId', clientMediaId), - ('TotalLen', fileSize), - ('StartPos', 0), - ('DataLen', fileSize), - ('MediaType', 4), - ('FromUserName', self.storageClass.userName), - ('ToUserName', toUserName), - ('FileMd5', fileMd5)] - ), separators = (',', ':')) - r = {'BaseResponse': {'Ret': -1005, 'ErrMsg': 'Empty file detected'}} - for chunk in range(chunks): - r = upload_chunk_file(self, fileDir, fileSymbol, fileSize, - file_, chunk, chunks, uploadMediaRequest) - file_.close() - if isinstance(r, dict): - return ReturnValue(r) - return ReturnValue(rawResponse=r) - -def upload_chunk_file(core, fileDir, fileSymbol, fileSize, - file, chunk, chunks, uploadMediaRequest): - url = core.loginInfo.get('fileUrl', core.loginInfo['url']) + \ - '/webwxuploadmedia?f=json' - # save it on server - cookiesList = {name:data for name,data in core.s.cookies.items()} - fileType = mimetypes.guess_type(fileDir)[0] or 'application/octet-stream' - files = OrderedDict([ - ('id', (None, 'WU_FILE_0')), - ('name', (None, os.path.basename(fileDir))), - ('type', (None, fileType)), - ('lastModifiedDate', (None, time.strftime('%a %b %d %Y %H:%M:%S GMT+0800 (CST)'))), - ('size', (None, str(fileSize))), - ('chunks', (None, None)), - ('chunk', (None, None)), - ('mediatype', (None, fileSymbol)), - ('uploadmediarequest', (None, uploadMediaRequest)), - ('webwx_data_ticket', (None, cookiesList['webwx_data_ticket'])), - ('pass_ticket', (None, core.loginInfo['pass_ticket'])), - ('filename' , (os.path.basename(fileDir), file.read(524288), 'application/octet-stream'))]) - if chunks == 1: - del files['chunk']; del files['chunks'] - else: - files['chunk'], files['chunks'] = (None, str(chunk)), (None, str(chunks)) - headers = { 'User-Agent' : config.USER_AGENT } - return requests.post(url, files=files, headers=headers) - -def send_file(self, fileDir, toUserName=None, mediaId=None): - logger.debug('Request to send a file(mediaId: %s) to %s: %s' % ( - mediaId, toUserName, fileDir)) - if toUserName is None: toUserName = self.storageClass.userName - if mediaId is None: - r = self.upload_file(fileDir) - if r: - mediaId = r['MediaId'] - else: - return r - url = '%s/webwxsendappmsg?fun=async&f=json' % self.loginInfo['url'] - data = { - 'BaseRequest': self.loginInfo['BaseRequest'], - 'Msg': { - 'Type': 6, - 'Content': ("%s"%os.path.basename(fileDir) + - "6" + - "%s%s"%(str(os.path.getsize(fileDir)), mediaId) + - "%s"%os.path.splitext(fileDir)[1].replace('.','')), - 'FromUserName': self.storageClass.userName, - 'ToUserName': toUserName, - 'LocalID': int(time.time() * 1e4), - 'ClientMsgId': int(time.time() * 1e4), }, - 'Scene': 0, } - headers = { - 'User-Agent': config.USER_AGENT, - 'Content-Type': 'application/json;charset=UTF-8', } - r = self.s.post(url, headers=headers, - data=json.dumps(data, ensure_ascii=False).encode('utf8')) - return ReturnValue(rawResponse=r) - -def send_image(self, fileDir, toUserName=None, mediaId=None): - logger.debug('Request to send a image(mediaId: %s) to %s: %s' % ( - mediaId, toUserName, fileDir)) - if toUserName is None: toUserName = self.storageClass.userName - if mediaId is None: - r = self.upload_file(fileDir, isPicture=not fileDir[-4:] == '.gif') - if r: - mediaId = r['MediaId'] - else: - return r - url = '%s/webwxsendmsgimg?fun=async&f=json' % self.loginInfo['url'] - data = { - 'BaseRequest': self.loginInfo['BaseRequest'], - 'Msg': { - 'Type': 3, - 'MediaId': mediaId, - 'FromUserName': self.storageClass.userName, - 'ToUserName': toUserName, - 'LocalID': int(time.time() * 1e4), - 'ClientMsgId': int(time.time() * 1e4), }, - 'Scene': 0, } - if fileDir[-4:] == '.gif': - url = '%s/webwxsendemoticon?fun=sys' % self.loginInfo['url'] - data['Msg']['Type'] = 47 - data['Msg']['EmojiFlag'] = 2 - headers = { - 'User-Agent': config.USER_AGENT, - 'Content-Type': 'application/json;charset=UTF-8', } - r = self.s.post(url, headers=headers, - data=json.dumps(data, ensure_ascii=False).encode('utf8')) - return ReturnValue(rawResponse=r) - -def send_video(self, fileDir=None, toUserName=None, mediaId=None): - logger.debug('Request to send a video(mediaId: %s) to %s: %s' % ( - mediaId, toUserName, fileDir)) - if toUserName is None: toUserName = self.storageClass.userName - if mediaId is None: - r = self.upload_file(fileDir, isVideo=True) - if r: - mediaId = r['MediaId'] - else: - return r - url = '%s/webwxsendvideomsg?fun=async&f=json&pass_ticket=%s' % ( - self.loginInfo['url'], self.loginInfo['pass_ticket']) - data = { - 'BaseRequest': self.loginInfo['BaseRequest'], - 'Msg': { - 'Type' : 43, - 'MediaId' : mediaId, - 'FromUserName' : self.storageClass.userName, - 'ToUserName' : toUserName, - 'LocalID' : int(time.time() * 1e4), - 'ClientMsgId' : int(time.time() * 1e4), }, - 'Scene': 0, } - headers = { - 'User-Agent' : config.USER_AGENT, - 'Content-Type': 'application/json;charset=UTF-8', } - r = self.s.post(url, headers=headers, - data=json.dumps(data, ensure_ascii=False).encode('utf8')) - return ReturnValue(rawResponse=r) - -def send(self, msg, toUserName=None, mediaId=None): - if not msg: - r = ReturnValue({'BaseResponse': { - 'ErrMsg': 'No message.', - 'Ret': -1005, }}) - elif msg[:5] == '@fil@': - if mediaId is None: - r = self.send_file(msg[5:], toUserName) - else: - r = self.send_file(msg[5:], toUserName, mediaId) - elif msg[:5] == '@img@': - if mediaId is None: - r = self.send_image(msg[5:], toUserName) - else: - r = self.send_image(msg[5:], toUserName, mediaId) - elif msg[:5] == '@msg@': - r = self.send_msg(msg[5:], toUserName) - elif msg[:5] == '@vid@': - if mediaId is None: - r = self.send_video(msg[5:], toUserName) - else: - r = self.send_video(msg[5:], toUserName, mediaId) - else: - r = self.send_msg(msg, toUserName) - return r +import os, time, re, io +import json +import mimetypes, hashlib +import traceback, logging +from collections import OrderedDict + +import requests + +from .. import config, utils +from ..returnvalues import ReturnValue +from ..storage import templates +from .contact import update_local_uin + +logger = logging.getLogger('itchat') + +def load_messages(core): + core.send_raw_msg = send_raw_msg + core.send_msg = send_msg + core.upload_file = upload_file + core.send_file = send_file + core.send_image = send_image + core.send_video = send_video + core.send = send + +def get_download_fn(core, url, msgId): + def download_fn(downloadDir=None): + params = { + 'msgid': msgId, + 'skey': core.loginInfo['skey'],} + headers = { 'User-Agent' : config.USER_AGENT } + r = core.s.get(url, params=params, stream=True, headers = headers) + tempStorage = io.BytesIO() + for block in r.iter_content(1024): + tempStorage.write(block) + if downloadDir is None: + return tempStorage.getvalue() + with open(downloadDir, 'wb') as f: + f.write(tempStorage.getvalue()) + tempStorage.seek(0) + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'Successfully downloaded', + 'Ret': 0, }, + 'PostFix': utils.get_image_postfix(tempStorage.read(20)), }) + return download_fn + +def produce_msg(core, msgList): + ''' for messages types + * 40 msg, 43 videochat, 50 VOIPMSG, 52 voipnotifymsg + * 53 webwxvoipnotifymsg, 9999 sysnotice + ''' + rl = [] + srl = [40, 43, 50, 52, 53, 9999] + for m in msgList: + # get actual opposite + if m['FromUserName'] == core.storageClass.userName: + actualOpposite = m['ToUserName'] + else: + actualOpposite = m['FromUserName'] + # produce basic message + if '@@' in m['FromUserName'] or '@@' in m['ToUserName']: + produce_group_chat(core, m) + else: + utils.msg_formatter(m, 'Content') + # set user of msg + if '@@' in actualOpposite: + m['User'] = core.search_chatrooms(userName=actualOpposite) or \ + templates.Chatroom({'UserName': actualOpposite}) + # we don't need to update chatroom here because we have + # updated once when producing basic message + elif actualOpposite in ('filehelper', 'fmessage'): + m['User'] = templates.User({'UserName': actualOpposite}) + else: + m['User'] = core.search_mps(userName=actualOpposite) or \ + core.search_friends(userName=actualOpposite) or \ + templates.User(userName=actualOpposite) + # by default we think there may be a user missing not a mp + m['User'].core = core + if m['MsgType'] == 1: # words + if m['Url']: + regx = r'(.+?\(.+?\))' + data = re.search(regx, m['Content']) + data = 'Map' if data is None else data.group(1) + msg = { + 'Type': 'Map', + 'Text': data,} + else: + msg = { + 'Type': 'Text', + 'Text': m['Content'],} + elif m['MsgType'] == 3 or m['MsgType'] == 47: # picture + download_fn = get_download_fn(core, + '%s/webwxgetmsgimg' % core.loginInfo['url'], m['NewMsgId']) + msg = { + 'Type' : 'Picture', + 'FileName' : '%s.%s' % (time.strftime('%y%m%d-%H%M%S', time.localtime()), + 'png' if m['MsgType'] == 3 else 'gif'), + 'Text' : download_fn, } + elif m['MsgType'] == 34: # voice + download_fn = get_download_fn(core, + '%s/webwxgetvoice' % core.loginInfo['url'], m['NewMsgId']) + msg = { + 'Type': 'Recording', + 'FileName' : '%s.mp3' % time.strftime('%y%m%d-%H%M%S', time.localtime()), + 'Text': download_fn,} + elif m['MsgType'] == 37: # friends + m['User']['UserName'] = m['RecommendInfo']['UserName'] + msg = { + 'Type': 'Friends', + 'Text': { + 'status' : m['Status'], + 'userName' : m['RecommendInfo']['UserName'], + 'verifyContent' : m['Ticket'], + 'autoUpdate' : m['RecommendInfo'], }, } + m['User'].verifyDict = msg['Text'] + elif m['MsgType'] == 42: # name card + msg = { + 'Type': 'Card', + 'Text': m['RecommendInfo'], } + elif m['MsgType'] in (43, 62): # tiny video + msgId = m['MsgId'] + def download_video(videoDir=None): + url = '%s/webwxgetvideo' % core.loginInfo['url'] + params = { + 'msgid': msgId, + 'skey': core.loginInfo['skey'],} + headers = {'Range': 'bytes=0-', 'User-Agent' : config.USER_AGENT } + r = core.s.get(url, params=params, headers=headers, stream=True) + tempStorage = io.BytesIO() + for block in r.iter_content(1024): + tempStorage.write(block) + if videoDir is None: + return tempStorage.getvalue() + with open(videoDir, 'wb') as f: + f.write(tempStorage.getvalue()) + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'Successfully downloaded', + 'Ret': 0, }}) + msg = { + 'Type': 'Video', + 'FileName' : '%s.mp4' % time.strftime('%y%m%d-%H%M%S', time.localtime()), + 'Text': download_video, } + elif m['MsgType'] == 49: # sharing + if m['AppMsgType'] == 6: + rawMsg = m + cookiesList = {name:data for name,data in core.s.cookies.items()} + def download_atta(attaDir=None): + url = core.loginInfo['fileUrl'] + '/webwxgetmedia' + params = { + 'sender': rawMsg['FromUserName'], + 'mediaid': rawMsg['MediaId'], + 'filename': rawMsg['FileName'], + 'fromuser': core.loginInfo['wxuin'], + 'pass_ticket': 'undefined', + 'webwx_data_ticket': cookiesList['webwx_data_ticket'],} + headers = { 'User-Agent' : config.USER_AGENT } + r = core.s.get(url, params=params, stream=True, headers=headers) + tempStorage = io.BytesIO() + for block in r.iter_content(1024): + tempStorage.write(block) + if attaDir is None: + return tempStorage.getvalue() + with open(attaDir, 'wb') as f: + f.write(tempStorage.getvalue()) + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'Successfully downloaded', + 'Ret': 0, }}) + msg = { + 'Type': 'Attachment', + 'Text': download_atta, } + elif m['AppMsgType'] == 8: + download_fn = get_download_fn(core, + '%s/webwxgetmsgimg' % core.loginInfo['url'], m['NewMsgId']) + msg = { + 'Type' : 'Picture', + 'FileName' : '%s.gif' % ( + time.strftime('%y%m%d-%H%M%S', time.localtime())), + 'Text' : download_fn, } + elif m['AppMsgType'] == 17: + msg = { + 'Type': 'Note', + 'Text': m['FileName'], } + elif m['AppMsgType'] == 2000: + regx = r'\[CDATA\[(.+?)\][\s\S]+?\[CDATA\[(.+?)\]' + data = re.search(regx, m['Content']) + if data: + data = data.group(2).split(u'\u3002')[0] + else: + data = 'You may found detailed info in Content key.' + msg = { + 'Type': 'Note', + 'Text': data, } + else: + msg = { + 'Type': 'Sharing', + 'Text': m['FileName'], } + elif m['MsgType'] == 51: # phone init + msg = update_local_uin(core, m) + elif m['MsgType'] == 10000: + msg = { + 'Type': 'Note', + 'Text': m['Content'],} + elif m['MsgType'] == 10002: + regx = r'\[CDATA\[(.+?)\]\]' + data = re.search(regx, m['Content']) + data = 'System message' if data is None else data.group(1).replace('\\', '') + msg = { + 'Type': 'Note', + 'Text': data, } + elif m['MsgType'] in srl: + msg = { + 'Type': 'Useless', + 'Text': 'UselessMsg', } + else: + logger.debug('Useless message received: %s\n%s' % (m['MsgType'], str(m))) + msg = { + 'Type': 'Useless', + 'Text': 'UselessMsg', } + m = dict(m, **msg) + rl.append(m) + return rl + +def produce_group_chat(core, msg): + r = re.match('(@[0-9a-z]*?):
(.*)$', msg['Content']) + if r: + actualUserName, content = r.groups() + chatroomUserName = msg['FromUserName'] + elif msg['FromUserName'] == core.storageClass.userName: + actualUserName = core.storageClass.userName + content = msg['Content'] + chatroomUserName = msg['ToUserName'] + else: + msg['ActualUserName'] = core.storageClass.userName + msg['ActualNickName'] = core.storageClass.nickName + msg['IsAt'] = False + utils.msg_formatter(msg, 'Content') + return + chatroom = core.storageClass.search_chatrooms(userName=chatroomUserName) + member = utils.search_dict_list((chatroom or {}).get( + 'MemberList') or [], 'UserName', actualUserName) + if member is None: + chatroom = core.update_chatroom(msg['FromUserName']) + member = utils.search_dict_list((chatroom or {}).get( + 'MemberList') or [], 'UserName', actualUserName) + if member is None: + logger.debug('chatroom member fetch failed with %s' % actualUserName) + msg['ActualNickName'] = '' + msg['isAt'] = False + else: + msg['ActualNickName'] = member['DisplayName'] or member['NickName'] + atFlag = '@' + (chatroom['self']['DisplayName'] + or core.storageClass.nickName) + msg['IsAt'] = ( + (atFlag + (u'\u2005' if u'\u2005' in msg['Content'] else ' ')) + in msg['Content'] or msg['Content'].endswith(atFlag)) + msg['ActualUserName'] = actualUserName + msg['Content'] = content + utils.msg_formatter(msg, 'Content') + +def send_raw_msg(self, msgType, content, toUserName): + url = '%s/webwxsendmsg' % self.loginInfo['url'] + data = { + 'BaseRequest': self.loginInfo['BaseRequest'], + 'Msg': { + 'Type': msgType, + 'Content': content, + 'FromUserName': self.storageClass.userName, + 'ToUserName': (toUserName if toUserName else self.storageClass.userName), + 'LocalID': int(time.time() * 1e4), + 'ClientMsgId': int(time.time() * 1e4), + }, + 'Scene': 0, } + headers = { 'ContentType': 'application/json; charset=UTF-8', 'User-Agent' : config.USER_AGENT } + r = self.s.post(url, headers=headers, + data=json.dumps(data, ensure_ascii=False).encode('utf8')) + return ReturnValue(rawResponse=r) + +def send_msg(self, msg='Test Message', toUserName=None): + logger.debug('Request to send a text message to %s: %s' % (toUserName, msg)) + r = self.send_raw_msg(1, msg, toUserName) + return r + +def _prepare_file(fileDir, file_=None): + fileDict = {} + if file_: + if hasattr(file_, 'read'): + file_ = file_.read() + else: + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'file_ param should be opened file', + 'Ret': -1005, }}) + else: + if not utils.check_file(fileDir): + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'No file found in specific dir', + 'Ret': -1002, }}) + with open(fileDir, 'rb') as f: + file_ = f.read() + fileDict['fileSize'] = len(file_) + fileDict['fileMd5'] = hashlib.md5(file_).hexdigest() + fileDict['file_'] = io.BytesIO(file_) + return fileDict + +def upload_file(self, fileDir, isPicture=False, isVideo=False, + toUserName='filehelper', file_=None, preparedFile=None): + logger.debug('Request to upload a %s: %s' % ( + 'picture' if isPicture else 'video' if isVideo else 'file', fileDir)) + if not preparedFile: + preparedFile = _prepare_file(fileDir, file_) + if not preparedFile: + return preparedFile + fileSize, fileMd5, file_ = \ + preparedFile['fileSize'], preparedFile['fileMd5'], preparedFile['file_'] + fileSymbol = 'pic' if isPicture else 'video' if isVideo else'doc' + chunks = int((fileSize - 1) / 524288) + 1 + clientMediaId = int(time.time() * 1e4) + uploadMediaRequest = json.dumps(OrderedDict([ + ('UploadType', 2), + ('BaseRequest', self.loginInfo['BaseRequest']), + ('ClientMediaId', clientMediaId), + ('TotalLen', fileSize), + ('StartPos', 0), + ('DataLen', fileSize), + ('MediaType', 4), + ('FromUserName', self.storageClass.userName), + ('ToUserName', toUserName), + ('FileMd5', fileMd5)] + ), separators = (',', ':')) + r = {'BaseResponse': {'Ret': -1005, 'ErrMsg': 'Empty file detected'}} + for chunk in range(chunks): + r = upload_chunk_file(self, fileDir, fileSymbol, fileSize, + file_, chunk, chunks, uploadMediaRequest) + file_.close() + if isinstance(r, dict): + return ReturnValue(r) + return ReturnValue(rawResponse=r) + +def upload_chunk_file(core, fileDir, fileSymbol, fileSize, + file_, chunk, chunks, uploadMediaRequest): + url = core.loginInfo.get('fileUrl', core.loginInfo['url']) + \ + '/webwxuploadmedia?f=json' + # save it on server + cookiesList = {name:data for name,data in core.s.cookies.items()} + fileType = mimetypes.guess_type(fileDir)[0] or 'application/octet-stream' + files = OrderedDict([ + ('id', (None, 'WU_FILE_0')), + ('name', (None, os.path.basename(fileDir))), + ('type', (None, fileType)), + ('lastModifiedDate', (None, time.strftime('%a %b %d %Y %H:%M:%S GMT+0800 (CST)'))), + ('size', (None, str(fileSize))), + ('chunks', (None, None)), + ('chunk', (None, None)), + ('mediatype', (None, fileSymbol)), + ('uploadmediarequest', (None, uploadMediaRequest)), + ('webwx_data_ticket', (None, cookiesList['webwx_data_ticket'])), + ('pass_ticket', (None, core.loginInfo['pass_ticket'])), + ('filename' , (os.path.basename(fileDir), file_.read(524288), 'application/octet-stream'))]) + if chunks == 1: + del files['chunk']; del files['chunks'] + else: + files['chunk'], files['chunks'] = (None, str(chunk)), (None, str(chunks)) + headers = { 'User-Agent' : config.USER_AGENT } + return requests.post(url, files=files, headers=headers) + +def send_file(self, fileDir, toUserName=None, mediaId=None, file_=None): + logger.debug('Request to send a file(mediaId: %s) to %s: %s' % ( + mediaId, toUserName, fileDir)) + if hasattr(fileDir, 'read'): + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'fileDir param should not be an opened file in send_file', + 'Ret': -1005, }}) + if toUserName is None: + toUserName = self.storageClass.userName + if mediaId is None: + preparedFile = _prepare_file(fileDir, file_) + if not preparedFile: + return preparedFile + fileSize = preparedFile['fileSize'] + r = self.upload_file(fileDir, preparedFile=preparedFile) + if r: + mediaId = r['MediaId'] + else: + return r + url = '%s/webwxsendappmsg?fun=async&f=json' % self.loginInfo['url'] + data = { + 'BaseRequest': self.loginInfo['BaseRequest'], + 'Msg': { + 'Type': 6, + 'Content': ("%s" % os.path.basename(fileDir) + + "6" + + "%s%s" % (str(fileSize), mediaId) + + "%s" % os.path.splitext(fileDir)[1].replace('.','')), + 'FromUserName': self.storageClass.userName, + 'ToUserName': toUserName, + 'LocalID': int(time.time() * 1e4), + 'ClientMsgId': int(time.time() * 1e4), }, + 'Scene': 0, } + headers = { + 'User-Agent': config.USER_AGENT, + 'Content-Type': 'application/json;charset=UTF-8', } + r = self.s.post(url, headers=headers, + data=json.dumps(data, ensure_ascii=False).encode('utf8')) + return ReturnValue(rawResponse=r) + +def send_image(self, fileDir=None, toUserName=None, mediaId=None, file_=None): + logger.debug('Request to send a image(mediaId: %s) to %s: %s' % ( + mediaId, toUserName, fileDir)) + if fileDir or file_: + if hasattr(fileDir, 'read'): + file_, fileDir = fileDir, None + if fileDir is None: + fileDir = 'tmp.jpg' # specific fileDir to send gifs + else: + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'Either fileDir or file_ should be specific', + 'Ret': -1005, }}) + if toUserName is None: + toUserName = self.storageClass.userName + if mediaId is None: + r = self.upload_file(fileDir, isPicture=not fileDir[-4:] == '.gif', file_=file_) + if r: + mediaId = r['MediaId'] + else: + return r + url = '%s/webwxsendmsgimg?fun=async&f=json' % self.loginInfo['url'] + data = { + 'BaseRequest': self.loginInfo['BaseRequest'], + 'Msg': { + 'Type': 3, + 'MediaId': mediaId, + 'FromUserName': self.storageClass.userName, + 'ToUserName': toUserName, + 'LocalID': int(time.time() * 1e4), + 'ClientMsgId': int(time.time() * 1e4), }, + 'Scene': 0, } + if fileDir[-4:] == '.gif': + url = '%s/webwxsendemoticon?fun=sys' % self.loginInfo['url'] + data['Msg']['Type'] = 47 + data['Msg']['EmojiFlag'] = 2 + headers = { + 'User-Agent': config.USER_AGENT, + 'Content-Type': 'application/json;charset=UTF-8', } + r = self.s.post(url, headers=headers, + data=json.dumps(data, ensure_ascii=False).encode('utf8')) + return ReturnValue(rawResponse=r) + +def send_video(self, fileDir=None, toUserName=None, mediaId=None, file_=None): + logger.debug('Request to send a video(mediaId: %s) to %s: %s' % ( + mediaId, toUserName, fileDir)) + if fileDir or file_: + if hasattr(fileDir, 'read'): + file_, fileDir = fileDir, None + if fileDir is None: + fileDir = 'tmp.mp4' # specific fileDir to send other formats + else: + return ReturnValue({'BaseResponse': { + 'ErrMsg': 'Either fileDir or file_ should be specific', + 'Ret': -1005, }}) + if toUserName is None: + toUserName = self.storageClass.userName + if mediaId is None: + r = self.upload_file(fileDir, isVideo=True, file_=file_) + if r: + mediaId = r['MediaId'] + else: + return r + url = '%s/webwxsendvideomsg?fun=async&f=json&pass_ticket=%s' % ( + self.loginInfo['url'], self.loginInfo['pass_ticket']) + data = { + 'BaseRequest': self.loginInfo['BaseRequest'], + 'Msg': { + 'Type' : 43, + 'MediaId' : mediaId, + 'FromUserName' : self.storageClass.userName, + 'ToUserName' : toUserName, + 'LocalID' : int(time.time() * 1e4), + 'ClientMsgId' : int(time.time() * 1e4), }, + 'Scene': 0, } + headers = { + 'User-Agent' : config.USER_AGENT, + 'Content-Type': 'application/json;charset=UTF-8', } + r = self.s.post(url, headers=headers, + data=json.dumps(data, ensure_ascii=False).encode('utf8')) + return ReturnValue(rawResponse=r) + +def send(self, msg, toUserName=None, mediaId=None): + if not msg: + r = ReturnValue({'BaseResponse': { + 'ErrMsg': 'No message.', + 'Ret': -1005, }}) + elif msg[:5] == '@fil@': + if mediaId is None: + r = self.send_file(msg[5:], toUserName) + else: + r = self.send_file(msg[5:], toUserName, mediaId) + elif msg[:5] == '@img@': + if mediaId is None: + r = self.send_image(msg[5:], toUserName) + else: + r = self.send_image(msg[5:], toUserName, mediaId) + elif msg[:5] == '@msg@': + r = self.send_msg(msg[5:], toUserName) + elif msg[:5] == '@vid@': + if mediaId is None: + r = self.send_video(msg[5:], toUserName) + else: + r = self.send_video(msg[5:], toUserName, mediaId) + else: + r = self.send_msg(msg, toUserName) + return r diff --git a/itchat/components/register.py b/itchat/components/register.py index a07c1111..86a37fcc 100644 --- a/itchat/components/register.py +++ b/itchat/components/register.py @@ -6,6 +6,7 @@ from ..log import set_logging from ..utils import test_connect +from ..storage import templates logger = logging.getLogger('itchat') @@ -46,19 +47,12 @@ def configured_reply(self): except Queue.Empty: pass else: - if msg['FromUserName'] == self.storageClass.userName: - actualOpposite = msg['ToUserName'] - else: - actualOpposite = msg['FromUserName'] - if '@@' in actualOpposite: - replyFn = self.functionDict['GroupChat'].get(msg['Type']) - elif self.search_mps(userName=msg['FromUserName']): - replyFn = self.functionDict['MpChat'].get(msg['Type']) - elif '@' in actualOpposite or \ - actualOpposite in ('filehelper', 'fmessage'): + if isinstance(msg['User'], templates.User): replyFn = self.functionDict['FriendChat'].get(msg['Type']) - else: + elif isinstance(msg['User'], templates.MassivePlatform): replyFn = self.functionDict['MpChat'].get(msg['Type']) + elif isinstance(msg['User'], templates.Chatroom): + replyFn = self.functionDict['GroupChat'].get(msg['Type']) if replyFn is None: r = None else: @@ -72,7 +66,7 @@ def configured_reply(self): def msg_register(self, msgType, isFriendChat=False, isGroupChat=False, isMpChat=False): ''' a decorator constructor return a specific decorator based on information given ''' - if not isinstance(msgType, list): + if not (isinstance(msgType, list) or isinstance(msgType, tuple)): msgType = [msgType] def _msg_register(fn): for _msgType in msgType: @@ -84,6 +78,7 @@ def _msg_register(fn): self.functionDict['MpChat'][_msgType] = fn if not any((isFriendChat, isGroupChat, isMpChat)): self.functionDict['FriendChat'][_msgType] = fn + return fn return _msg_register def run(self, debug=False, blockThread=True): diff --git a/itchat/config.py b/itchat/config.py index ab17ffbe..515ea463 100644 --- a/itchat/config.py +++ b/itchat/config.py @@ -1,9 +1,9 @@ -import os, platform - -VERSION = '1.2.33' -BASE_URL = 'https://login.weixin.qq.com' -OS = platform.system() #Windows, Linux, Darwin -DIR = os.getcwd() -DEFAULT_QR = 'QR.png' - -USER_AGENT = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36' +import os, platform + +VERSION = '1.3.1' +BASE_URL = 'https://login.weixin.qq.com' +OS = platform.system() #Windows, Linux, Darwin +DIR = os.getcwd() +DEFAULT_QR = 'QR.png' + +USER_AGENT = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36' diff --git a/itchat/core.py b/itchat/core.py index dea80903..ba384ced 100644 --- a/itchat/core.py +++ b/itchat/core.py @@ -18,7 +18,7 @@ def __init__(self): - failing is failing ''' self.alive, self.isLogging = False, False - self.storageClass = storage.Storage() + self.storageClass = storage.Storage(self) self.memberList = self.storageClass.memberList self.mpList = self.storageClass.mpList self.chatroomList = self.storageClass.chatroomList @@ -313,7 +313,7 @@ def send_msg(self, msg='Test Message', toUserName=None): ''' raise NotImplementedError() def upload_file(self, fileDir, isPicture=False, isVideo=False, - toUserName='filehelper'): + toUserName='filehelper', file_=None, preparedFile=None): ''' upload file to server and get mediaId for options - fileDir: dir for file ready for upload @@ -325,7 +325,7 @@ def upload_file(self, fileDir, isPicture=False, isVideo=False, it is defined in components/messages.py ''' raise NotImplementedError() - def send_file(self, fileDir, toUserName=None, mediaId=None): + def send_file(self, fileDir, toUserName=None, mediaId=None, file_=None): ''' send attachment for options - fileDir: dir for file ready for upload @@ -335,7 +335,7 @@ def send_file(self, fileDir, toUserName=None, mediaId=None): it is defined in components/messages.py ''' raise NotImplementedError() - def send_image(self, fileDir, toUserName=None, mediaId=None): + def send_image(self, fileDir=None, toUserName=None, mediaId=None, file_=None): ''' send image for options - fileDir: dir for file ready for upload @@ -346,7 +346,7 @@ def send_image(self, fileDir, toUserName=None, mediaId=None): it is defined in components/messages.py ''' raise NotImplementedError() - def send_video(self, fileDir=None, toUserName=None, mediaId=None): + def send_video(self, fileDir=None, toUserName=None, mediaId=None, file_=None): ''' send video for options - fileDir: dir for file ready for upload diff --git a/itchat/returnvalues.py b/itchat/returnvalues.py index e9643d08..b9600bd6 100644 --- a/itchat/returnvalues.py +++ b/itchat/returnvalues.py @@ -4,6 +4,23 @@ TRANSLATE = 'Chinese' class ReturnValue(dict): + ''' turn return value of itchat into a boolean value + for requests: + ..code::python + + import requests + r = requests.get('http://httpbin.org/get') + print(ReturnValue(rawResponse=r) + + for normal dict: + ..code::python + + returnDict = { + 'BaseResponse': { + 'Ret': 0, + 'ErrMsg': 'My error msg', }, } + print(ReturnValue(returnDict)) + ''' def __init__(self, returnValueDict={}, rawResponse=None): if rawResponse: try: @@ -14,7 +31,8 @@ def __init__(self, returnValueDict={}, rawResponse=None): 'Ret': -1004, 'ErrMsg': 'Unexpected return value', }, 'Data': rawResponse.content, } - for k, v in returnValueDict.items(): self[k] = v + for k, v in returnValueDict.items(): + self[k] = v if not 'BaseResponse' in self: self['BaseResponse'] = { 'ErrMsg': 'no BaseResponse in raw response', @@ -45,6 +63,7 @@ def __repr__(self): -1003: u'服务器拒绝连接', -1004: u'服务器返回异常值', -1005: u'参数错误', + -1006: u'无效操作', 0: u'请求成功', }, } diff --git a/itchat/storage.py b/itchat/storage/__init__.py similarity index 63% rename from itchat/storage.py rename to itchat/storage/__init__.py index 5bfff6f0..8010c9a1 100644 --- a/itchat/storage.py +++ b/itchat/storage/__init__.py @@ -1,10 +1,11 @@ import os, time, copy -try: - import Queue -except ImportError: - import queue as Queue from threading import Lock +from .messagequeue import Queue +from .templates import ( + ContactList, AbstractUserDict, User, + MassivePlatform, Chatroom, ChatroomMember) + def contact_change(fn): def _contact_change(core, *args, **kwargs): with core.storageClass.updateLock: @@ -12,32 +13,41 @@ def _contact_change(core, *args, **kwargs): return _contact_change class Storage(object): - def __init__(self): + def __init__(self, core): self.userName = None self.nickName = None self.updateLock = Lock() - self.memberList = [] - self.mpList = [] - self.chatroomList = [] - self.msgList = Queue.Queue(-1) + self.memberList = ContactList() + self.mpList = ContactList() + self.chatroomList = ContactList() + self.msgList = Queue(-1) self.lastInputUserName = None + self.memberList.set_default_value(contactClass=User) + self.memberList.core = core + self.mpList.set_default_value(contactClass=MassivePlatform) + self.mpList.core = core + self.chatroomList.set_default_value(contactClass=Chatroom) + self.chatroomList.core = core def dumps(self): return { 'userName' : self.userName, 'nickName' : self.nickName, - 'memberList' : self.memberList, - 'mpList' : self.mpList, - 'chatroomList' : self.chatroomList, + 'memberList' : [dict(member) for member in self.memberList], + 'mpList' : [dict(mp) for mp in self.mpList], + 'chatroomList' : [dict(chatroom) for chatroom in self.chatroomList], 'lastInputUserName' : self.lastInputUserName, } def loads(self, j): - self.userName = j.get('userName', None) - self.nickName = j.get('nickName', None) + self.userName = j.get('userName', None) + self.nickName = j.get('nickName', None) del self.memberList[:] - for i in j.get('memberList', []): self.memberList.append(i) + for i in j.get('memberList', []): + self.memberList.append(i) del self.mpList[:] - for i in j.get('mpList', []): self.mpList.append(i) + for i in j.get('mpList', []): + self.mpList.append(i) del self.chatroomList[:] - for i in j.get('chatroomList', []): self.chatroomList.append(i) + for i in j.get('chatroomList', []): + self.chatroomList.append(i) self.lastInputUserName = j.get('lastInputUserName', None) def search_friends(self, name=None, userName=None, remarkName=None, nickName=None, wechatAccount=None): @@ -75,19 +85,23 @@ def search_chatrooms(self, name=None, userName=None): with self.updateLock: if userName is not None: for m in self.chatroomList: - if m['UserName'] == userName: return copy.deepcopy(m) + if m['UserName'] == userName: + return copy.deepcopy(m) elif name is not None: matchList = [] for m in self.chatroomList: - if name in m['NickName']: matchList.append(copy.deepcopy(m)) + if name in m['NickName']: + matchList.append(copy.deepcopy(m)) return matchList def search_mps(self, name=None, userName=None): with self.updateLock: if userName is not None: for m in self.mpList: - if m['UserName'] == userName: return copy.deepcopy(m) + if m['UserName'] == userName: + return copy.deepcopy(m) elif name is not None: matchList = [] for m in self.mpList: - if name in m['NickName']: matchList.append(copy.deepcopy(m)) + if name in m['NickName']: + matchList.append(copy.deepcopy(m)) return matchList diff --git a/itchat/storage/messagequeue.py b/itchat/storage/messagequeue.py new file mode 100644 index 00000000..c1f6ae71 --- /dev/null +++ b/itchat/storage/messagequeue.py @@ -0,0 +1,26 @@ +try: + import Queue as queue +except ImportError: + import queue + +class Queue(queue.Queue): + def put(self, message): + if 'IsAt' in message: + message['isAt'] = message['IsAt'] + queue.Queue.put(self, Message(message)) + +class Message(dict): + def download(self, fileName): + if hasattr(self.text, '__call__'): + return self.text(fileName) + else: + return b'' + def __getattr__(self, value): + value = value[0].upper() + value[1:] + return self.get(value, '') + def __str__(self): + return '{%s}' % ', '.join( + ['%s: %s' % (repr(k),repr(v)) for k,v in self.items()]) + def __repr__(self): + return '<%s: %s>' % (self.__class__.__name__.split('.')[-1], + self.__str__()) diff --git a/itchat/storage/templates.py b/itchat/storage/templates.py new file mode 100644 index 00000000..70bf0d56 --- /dev/null +++ b/itchat/storage/templates.py @@ -0,0 +1,254 @@ +import logging, copy, pickle + +from ..returnvalues import ReturnValue + +logger = logging.getLogger('itchat') + +class UnInitializedItchat(object): + def _raise_error(self, *args, **kwargs): + logger.warning('An itchat instance is called before initialized') + def __getattr__(self, value): + return self._raise_error + +fakeItchat = UnInitializedItchat() + +class ContactList(list): + ''' when a dict is append, init function will be called to format that dict + ''' + def __init__(self, *args, **kwargs): + super(ContactList, self).__init__(*args, **kwargs) + self.contactInitFn = None + self.contactClass = User + self.core = fakeItchat + def set_default_value(self, initFunction=None, contactClass=None): + if hasattr(initFunction, '__call__'): + self.contactInitFn = initFunction + if hasattr(contactClass, '__call__'): + self.contactClass = contactClass + def append(self, value): + contact = self.contactClass(value) + contact.core = self.core + if self.contactInitFn is not None: + contact = self.contactInitFn(contact) or contact + super(ContactList, self).append(contact) + def __deepcopy__(self, memo): + return self.__class__([copy.deepcopy(v) for v in self]) + def __getstate__(self): + return [pickle.dumps(v) for v in self] + def __setstate__(self, state): + for v in state: + super(ContactList, self).append(pickle.loads(v)) + def __str__(self): + return '[%s]' % ', '.join([repr(v) for v in self]) + def __repr__(self): + return '<%s: %s>' % (self.__class__.__name__.split('.')[-1], + self.__str__()) + +fakeContactList = ContactList + +class AbstractUserDict(dict): + def __init__(self, *args, **kwargs): + super(AbstractUserDict, self).__init__(*args, **kwargs) + self.core = fakeItchat + def update(self): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not be updated' % \ + self.__class__.__name__, }, }) + def set_alias(self, alias): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not set alias' % \ + self.__class__.__name__, }, }) + def set_pinned(self, isPinned=True): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not be pinned' % \ + self.__class__.__name__, }, }) + def verify(self): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s do not need verify' % \ + self.__class__.__name__, }, }) + def get_head_image(self, imageDir=None): + return self.core.get_head_img(self.userName, picDir=imageDir) + def delete_member(self, userName): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not delete member' % \ + self.__class__.__name__, }, }) + def add_member(self, userName): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not add member' % \ + self.__class__.__name__, }, }) + def send_raw_msg(self, msgType, content): + return self.core.send_raw_msg(msgType, content, self.userName) + def send_msg(self, msg='Test Message'): + return self.core.send_msg(msgType, content, self.userName) + def send_file(self, fileDir, mediaId=None): + return self.core.send_file(fileDir, self.userName, mediaId) + def send_image(self, fileDir, mediaId=None): + return self.core.send_image(fileDir, self.userName, mediaId) + def send_video(self, fileDir=None, mediaId=None): + return self.core.send_video(fileDir, self.userName, mediaId) + def send(self, msg, mediaId=None): + return self.core.send(msg, self.userName, mediaId) + def search_member(self, name=None, userName=None, remarkName=None, nickName=None, + wechatAccount=None): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s do not have members' % \ + self.__class__.__name__, }, }) + def __getattr__(self, value): + value = value[0].upper() + value[1:] + return self.get(value, '') + def __deepcopy__(self, memo): + r = self.__class__({ + copy.deepcopy(k, memo): copy.deepcopy(v, memo) + for k, v in self.items()}) + r.core = self.core + return r + def __getstate__(self): + return dict(self) + def __setstate__(self, state): + for k, v in state.items(): + self[k] = v + def __str__(self): + return '{%s}' % ', '.join( + ['%s: %s' % (repr(k),repr(v)) for k,v in self.items()]) + def __repr__(self): + return '<%s: %s>' % (self.__class__.__name__.split('.')[-1], + self.__str__()) + +class User(AbstractUserDict): + def __init__(self, *args, **kwargs): + super(User, self).__init__(*args, **kwargs) + self.verifyDict = {} + self.memberList = fakeContactList + def update(self): + return self.core.update_friend(self.userName) + def set_alias(self, alias): + return self.core.set_alias(self.userName, alias) + def set_pinned(self, isPinned=True): + return self.core.set_pinned(self.userName, isPinned) + def verify(self): + return self.core.add_friend(**self.verifyDict) + def __deepcopy__(self, memo): + r = super(User, self).__deepcopy__(memo) + r.verifyDict = copy.deepcopy(self.verifyDict) + return r + +class MassivePlatform(AbstractUserDict): + def __init__(self, *args, **kwargs): + super(MassivePlatform, self).__init__(*args, **kwargs) + self.memberList = fakeContactList + +class Chatroom(AbstractUserDict): + def __init__(self, *args, **kwargs): + super(Chatroom, self).__init__(*args, **kwargs) + memberList = ContactList() + def init_fn(d): + d.chatroom = self + memberList.set_default_value(init_fn, ChatroomMember) + for rawMember in self.memberList: + memberList.append(rawMember) + self['MemberList'] = memberList + def update(self, detailedMember=False): + return self.core.update_chatroom(self.userName, detailedMember) + def set_alias(self, alias): + return self.core.set_chatroom_name(self.userName, alias) + def set_pinned(self, isPinned=True): + return self.core.set_pinned(self.userName, isPinned) + def delete_member(self, userName): + return self.core.delete_member_from_chatroom(self.userName, userName) + def add_member(self, userName): + return self.core.add_member_into_chatroom(self.userName, userName) + def search_member(self, name=None, userName=None, remarkName=None, nickName=None, + wechatAccount=None): + with self.core.storageClass.updateLock: + if (name or userName or remarkName or nickName or wechatAccount) is None: + return None + elif userName: # return the only userName match + for m in self.memberList: + if m.userName == userName: + return copy.deepcopy(m) + else: + matchDict = { + 'RemarkName' : remarkName, + 'NickName' : nickName, + 'Alias' : wechatAccount, } + for k in ('RemarkName', 'NickName', 'Alias'): + if matchDict[k] is None: + del matchDict[k] + if name: # select based on name + contact = [] + for m in self.memberList: + if any([m.get(k) == name for k in ('RemarkName', 'NickName', 'Alias')]): + contact.append(m) + else: + contact = self.memberList[:] + if matchDict: # select again based on matchDict + friendList = [] + for m in contact: + if all([m.get(k) == v for k, v in matchDict.items()]): + friendList.append(m) + return copy.deepcopy(friendList) + else: + return copy.deepcopy(contact) + +class ChatroomMember(AbstractUserDict): + def __init__(self, *args, **kwargs): + super(AbstractUserDict, self).__init__(*args, **kwargs) + self.core = fakeItchat + self.chatroom = self.fakeChatroom + def get_head_image(self, imageDir=None): + return self.core.get_head_img(self.userName, self.chatroom.userName, picDir=imageDir) + def delete_member(self, userName): + return self.core.delete_member_from_chatroom(self.chatroom.userName, self.userName) + def send_raw_msg(self, msgType, content): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def send_msg(self, msg='Test Message'): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def send_file(self, fileDir, mediaId=None): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def send_image(self, fileDir, mediaId=None): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def send_video(self, fileDir=None, mediaId=None): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def send(self, msg, mediaId=None): + return ReturnValue({'BaseResponse': { + 'Ret': -1006, + 'ErrMsg': '%s can not send message directly' % \ + self.__class__.__name__, }, }) + def __deepcopy__(self, memo): + r = super(ChatroomMember, self).__deepcopy__(memo) + r.core = self.core + return r + +ChatroomMember.fakeChatroom = Chatroom() + +def wrap_user_dict(d): + userName = d.get('UserName') + if '@@' in userName: + r = Chatroom(d) + elif d.get('VerifyFlag', 8) & 8 == 0: + r = User(d) + else: + r = MassivePlatform(d) + return r diff --git a/itchat/utils.py b/itchat/utils.py index a0a61a2e..202de6e7 100644 --- a/itchat/utils.py +++ b/itchat/utils.py @@ -106,7 +106,8 @@ def search_dict_list(l, key, value): ''' Search a list of dict * return dict with specific value & key ''' for i in l: - if i.get(key) == value: return i + if i.get(key) == value: + return i def print_line(msg, oneLine = False): if oneLine: