wxbot.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. #!/usr/bin/env python
  2. # coding: utf-8
  3. import pyqrcode
  4. import requests
  5. import json
  6. import xml.dom.minidom
  7. import urllib
  8. import time
  9. import re
  10. import random
  11. class WXBot:
  12. def __init__(self):
  13. self.DEBUG = False
  14. self.uuid = ''
  15. self.base_uri = ''
  16. self.redirect_uri = ''
  17. self.uin = ''
  18. self.sid = ''
  19. self.skey = ''
  20. self.pass_ticket = ''
  21. self.device_id = 'e' + repr(random.random())[2:17]
  22. self.base_request = {}
  23. self.sync_key_str = ''
  24. self.sync_key = []
  25. self.user = {}
  26. self.member_list = []
  27. self.contact_list = [] # contact list
  28. self.public_list = [] # public account list
  29. self.group_list = [] # group chat list
  30. self.special_list = [] # special list account
  31. self.sync_host = ''
  32. self.session = requests.Session()
  33. self.session.headers.update({'User-Agent': 'Mozilla/5.0 (X11; Linux i686; U;) Gecko/20070322 Kazehakase/0.4.5'})
  34. self.conf = {'qr': 'png'}
  35. def get_uuid(self):
  36. url = 'https://login.weixin.qq.com/jslogin'
  37. params = {
  38. 'appid': 'wx782c26e4c19acffb',
  39. 'fun': 'new',
  40. 'lang': 'zh_CN',
  41. '_': int(time.time())*1000 + random.randint(1, 999),
  42. }
  43. r = self.session.get(url, params=params)
  44. r.encoding = 'utf-8'
  45. data = r.text
  46. regx = r'window.QRLogin.code = (\d+); window.QRLogin.uuid = "(\S+?)"'
  47. pm = re.search(regx, data)
  48. if pm:
  49. code = pm.group(1)
  50. self.uuid = pm.group(2)
  51. return code == '200'
  52. return False
  53. def gen_qr_code(self, qr_file_path):
  54. string = 'https://login.weixin.qq.com/l/' + self.uuid
  55. qr = pyqrcode.create(string)
  56. if self.conf['qr'] == 'png':
  57. qr.png(qr_file_path)
  58. elif self.conf['qr'] == 'tty':
  59. print(qr.terminal(quiet_zone=1))
  60. def wait4login(self, tip):
  61. time.sleep(tip)
  62. url = 'https://login.weixin.qq.com/cgi-bin/mmwebwx-bin/login?tip=%s&uuid=%s&_=%s' \
  63. % (tip, self.uuid, int(time.time()))
  64. r = self.session.get(url)
  65. r.encoding = 'utf-8'
  66. data = r.text
  67. param = re.search(r'window.code=(\d+);', data)
  68. code = param.group(1)
  69. if code == '201':
  70. return True
  71. elif code == '200':
  72. param = re.search(r'window.redirect_uri="(\S+?)";', data)
  73. redirect_uri = param.group(1) + '&fun=new'
  74. self.redirect_uri = redirect_uri
  75. self.base_uri = redirect_uri[:redirect_uri.rfind('/')]
  76. return True
  77. elif code == '408':
  78. print '[ERROR] WeChat login timeout .'
  79. else:
  80. print '[ERROR] WeChat login exception .'
  81. return False
  82. def login(self):
  83. r = self.session.get(self.redirect_uri)
  84. r.encoding = 'utf-8'
  85. data = r.text
  86. doc = xml.dom.minidom.parseString(data)
  87. root = doc.documentElement
  88. for node in root.childNodes:
  89. if node.nodeName == 'skey':
  90. self.skey = node.childNodes[0].data
  91. elif node.nodeName == 'wxsid':
  92. self.sid = node.childNodes[0].data
  93. elif node.nodeName == 'wxuin':
  94. self.uin = node.childNodes[0].data
  95. elif node.nodeName == 'pass_ticket':
  96. self.pass_ticket = node.childNodes[0].data
  97. if '' in (self.skey, self.sid, self.uin, self.pass_ticket):
  98. return False
  99. self.base_request = {
  100. 'Uin': self.uin,
  101. 'Sid': self.sid,
  102. 'Skey': self.skey,
  103. 'DeviceID': self.device_id,
  104. }
  105. return True
  106. def init(self):
  107. url = self.base_uri + '/webwxinit?r=%i&lang=en_US&pass_ticket=%s' % (int(time.time()), self.pass_ticket)
  108. params = {
  109. 'BaseRequest': self.base_request
  110. }
  111. r = self.session.post(url, data=json.dumps(params))
  112. r.encoding = 'utf-8'
  113. dic = json.loads(r.text)
  114. self.sync_key = dic['SyncKey']
  115. self.user = dic['User']
  116. self.sync_key_str = '|'.join([str(keyVal['Key']) + '_' + str(keyVal['Val'])
  117. for keyVal in self.sync_key['List']])
  118. return dic['BaseResponse']['Ret'] == 0
  119. def status_notify(self):
  120. url = self.base_uri + '/webwxstatusnotify?lang=zh_CN&pass_ticket=%s' % self.pass_ticket
  121. self.base_request['Uin'] = int(self.base_request['Uin'])
  122. params = {
  123. 'BaseRequest': self.base_request,
  124. "Code": 3,
  125. "FromUserName": self.user['UserName'],
  126. "ToUserName": self.user['UserName'],
  127. "ClientMsgId": int(time.time())
  128. }
  129. r = self.session.post(url, data=json.dumps(params))
  130. r.encoding = 'utf-8'
  131. dic = json.loads(r.text)
  132. return dic['BaseResponse']['Ret'] == 0
  133. def get_contact(self):
  134. url = self.base_uri + '/webwxgetcontact?pass_ticket=%s&skey=%s&r=%s' \
  135. % (self.pass_ticket, self.skey, int(time.time()))
  136. r = self.session.post(url, data='{}')
  137. r.encoding = 'utf-8'
  138. if self.DEBUG:
  139. with open('contacts.json', 'w') as f:
  140. f.write(r.text.encode('utf-8'))
  141. dic = json.loads(r.text)
  142. self.member_list = dic['MemberList']
  143. special_users = ['newsapp', 'fmessage', 'filehelper', 'weibo', 'qqmail',
  144. 'fmessage', 'tmessage', 'qmessage', 'qqsync', 'floatbottle',
  145. 'lbsapp', 'shakeapp', 'medianote', 'qqfriend', 'readerapp',
  146. 'blogapp', 'facebookapp', 'masssendapp', 'meishiapp',
  147. 'feedsapp', 'voip', 'blogappweixin', 'weixin', 'brandsessionholder',
  148. 'weixinreminder', 'wxid_novlwrv3lqwv11', 'gh_22b87fa7cb3c',
  149. 'officialaccounts', 'notification_messages', 'wxid_novlwrv3lqwv11',
  150. 'gh_22b87fa7cb3c', 'wxitil', 'userexperience_alarm', 'notification_messages']
  151. self.contact_list = []
  152. self.public_list = []
  153. self.special_list = []
  154. self.group_list = []
  155. for contact in self.member_list:
  156. if contact['VerifyFlag'] & 8 != 0: # public account
  157. self.public_list.append(contact)
  158. elif contact['UserName'] in special_users: # special account
  159. self.special_list.append(contact)
  160. elif contact['UserName'].find('@@') != -1: # group
  161. self.group_list.append(contact)
  162. elif contact['UserName'] == self.user['UserName']: # self
  163. pass
  164. else:
  165. self.contact_list.append(contact)
  166. if self.DEBUG:
  167. with open('contact_list.json', 'w') as f:
  168. f.write(json.dumps(self.contact_list))
  169. with open('special_list.json', 'w') as f:
  170. f.write(json.dumps(self.special_list))
  171. with open('group_list.json', 'w') as f:
  172. f.write(json.dumps(self.group_list))
  173. with open('public_list.json', 'w') as f:
  174. f.write(json.dumps(self.public_list))
  175. with open('member_list.json', 'w') as f:
  176. f.write(json.dumps(self.member_list))
  177. return True
  178. def batch_get_contact(self):
  179. url = self.base_uri + '/webwxbatchgetcontact?type=ex&r=%s&pass_ticket=%s' % (int(time.time()), self.pass_ticket)
  180. params = {
  181. 'BaseRequest': self.base_request,
  182. "Count": len(self.group_list),
  183. "List": [{"UserName": g['UserName'], "EncryChatRoomId":""} for g in self.group_list]
  184. }
  185. r = self.session.post(url, data=params)
  186. r.encoding = 'utf-8'
  187. dic = json.loads(r.text)
  188. return dic
  189. def test_sync_check(self):
  190. for host in ['webpush', 'webpush2']:
  191. self.sync_host = host
  192. retcode = self.sync_check()[0]
  193. if retcode == '0':
  194. return True
  195. return False
  196. def sync_check(self):
  197. params = {
  198. 'r': int(time.time()),
  199. 'sid': self.sid,
  200. 'uin': self.uin,
  201. 'skey': self.skey,
  202. 'deviceid': self.device_id,
  203. 'synckey': self.sync_key_str,
  204. '_': int(time.time()),
  205. }
  206. url = 'https://' + self.sync_host + '.weixin.qq.com/cgi-bin/mmwebwx-bin/synccheck?' + urllib.urlencode(params)
  207. r = self.session.get(url)
  208. r.encoding = 'utf-8'
  209. data = r.text
  210. pm = re.search(r'window.synccheck=\{retcode:"(\d+)",selector:"(\d+)"\}', data)
  211. retcode = pm.group(1)
  212. selector = pm.group(2)
  213. return [retcode, selector]
  214. def sync(self):
  215. url = self.base_uri + '/webwxsync?sid=%s&skey=%s&lang=en_US&pass_ticket=%s' \
  216. % (self.sid, self.skey, self.pass_ticket)
  217. params = {
  218. 'BaseRequest': self.base_request,
  219. 'SyncKey': self.sync_key,
  220. 'rr': ~int(time.time())
  221. }
  222. r = self.session.post(url, data=json.dumps(params))
  223. r.encoding = 'utf-8'
  224. dic = json.loads(r.text)
  225. if dic['BaseResponse']['Ret'] == 0:
  226. self.sync_key = dic['SyncKey']
  227. self.sync_key_str = '|'.join([str(keyVal['Key']) + '_' + str(keyVal['Val'])
  228. for keyVal in self.sync_key['List']])
  229. return dic
  230. def get_icon(self, uid):
  231. url = self.base_uri + '/webwxgeticon?username=%s&skey=%s' % (uid, self.skey)
  232. r = self.session.get(url)
  233. data = r.content
  234. fn = 'img_'+uid+'.jpg'
  235. with open(fn, 'wb') as f:
  236. f.write(data)
  237. return fn
  238. def get_head_img(self, uid):
  239. url = self.base_uri + '/webwxgetheadimg?username=%s&skey=%s' % (uid, self.skey)
  240. r = self.session.get(url)
  241. data = r.content
  242. fn = 'img_'+uid+'.jpg'
  243. with open(fn, 'wb') as f:
  244. f.write(data)
  245. return fn
  246. def get_msg_img_url(self, msgid):
  247. return self.base_uri + '/webwxgetmsgimg?MsgID=%s&skey=%s' % (msgid, self.skey)
  248. def get_msg_img(self, msgid):
  249. url = self.base_uri + '/webwxgetmsgimg?MsgID=%s&skey=%s' % (msgid, self.skey)
  250. r = self.session.get(url)
  251. data = r.content
  252. fn = 'img_'+msgid+'.jpg'
  253. with open(fn, 'wb') as f:
  254. f.write(data)
  255. return fn
  256. def get_voice_url(self, msgid):
  257. return self.base_uri + '/webwxgetvoice?msgid=%s&skey=%s' % (msgid, self.skey)
  258. def get_voice(self, msgid):
  259. url = self.base_uri + '/webwxgetvoice?msgid=%s&skey=%s' % (msgid, self.skey)
  260. r = self.session.get(url)
  261. data = r.content
  262. fn = 'voice_'+msgid+'.mp3'
  263. with open(fn, 'wb') as f:
  264. f.write(data)
  265. return fn
  266. # Get the NickName or RemarkName of an user by user id
  267. def get_user_remark_name(self, uid):
  268. name = 'unknown group' if uid[:2] == '@@' else 'stranger'
  269. for member in self.member_list:
  270. if member['UserName'] == uid:
  271. name = member['RemarkName'] if member['RemarkName'] else member['NickName']
  272. return name
  273. # Get user id of an user
  274. def get_user_id(self, name):
  275. for member in self.member_list:
  276. if name == member['RemarkName'] or name == member['NickName'] or name == member['UserName']:
  277. return member['UserName']
  278. return None
  279. def get_user_type(self, wx_user_id):
  280. for account in self.contact_list:
  281. if wx_user_id == account['UserName']:
  282. return 'contact'
  283. for account in self.public_list:
  284. if wx_user_id == account['UserName']:
  285. return 'public'
  286. for account in self.special_list:
  287. if wx_user_id == account['UserName']:
  288. return 'special'
  289. for account in self.group_list:
  290. if wx_user_id == account['UserName']:
  291. return 'group'
  292. return 'unknown'
  293. def is_contact(self, uid):
  294. for account in self.contact_list:
  295. if uid == account['UserName']:
  296. return True
  297. return False
  298. def is_public(self, uid):
  299. for account in self.public_list:
  300. if uid == account['UserName']:
  301. return True
  302. return False
  303. def is_special(self, uid):
  304. for account in self.special_list:
  305. if uid == account['UserName']:
  306. return True
  307. return False
  308. '''
  309. msg:
  310. user
  311. type
  312. data
  313. detail
  314. '''
  315. def handle_msg_all(self, msg):
  316. pass
  317. '''
  318. content_type_id:
  319. 0 -> Text
  320. 1 -> Location
  321. 3 -> Image
  322. 4 -> Voice
  323. 5 -> Recommend
  324. 6 -> Animation
  325. 7 -> Share
  326. 8 -> Video
  327. 9 -> VideoCall
  328. 10 -> Redraw
  329. 11 -> Empty
  330. 99 -> Unknown
  331. '''
  332. def extract_msg_content(self, msg_type_id, msg):
  333. mtype = msg['MsgType']
  334. content = msg['Content'].replace('&lt;', '<').replace('&gt;', '>')
  335. msg_id = msg['MsgId']
  336. msg_content = {}
  337. if msg_type_id == 0:
  338. return {'type': 11, 'data': ''}
  339. elif msg_type_id == 2: # File Helper
  340. return {'type': 0, 'data': content.replace('<br/>', '\n')}
  341. elif msg_type_id == 3: # Group
  342. sp = content.find('<br/>')
  343. uid = content[:sp]
  344. content = content[sp:]
  345. content = content.replace('<br/>', '')
  346. uid = uid[:-1]
  347. msg_content['user'] = {'id': uid, 'name': self.get_user_remark_name(uid)}
  348. if self.DEBUG:
  349. print msg_content['user']['name']
  350. else: # Self, Contact, Special, Public, Unknown
  351. pass
  352. if mtype == 1:
  353. if content.find('http://weixin.qq.com/cgi-bin/redirectforward?args=') != -1:
  354. r = self.session.get(content)
  355. r.encoding = 'gbk'
  356. data = r.text
  357. pos = self.search_content('title', data, 'xml')
  358. msg_content['type'] = 1
  359. msg_content['data'] = pos
  360. msg_content['detail'] = data
  361. if self.DEBUG:
  362. print ' [Location] I am at %s ' % pos
  363. else:
  364. msg_content['type'] = 0
  365. msg_content['data'] = content.replace(u'\u2005', '')
  366. if self.DEBUG:
  367. print ' [Text] %s' % msg_content['data']
  368. elif mtype == 3:
  369. msg_content['type'] = 3
  370. msg_content['data'] = self.get_msg_img_url(msg_id)
  371. if self.DEBUG:
  372. image = self.get_msg_img(msg_id)
  373. print ' [Image] %s' % image
  374. elif mtype == 34:
  375. msg_content['type'] = 4
  376. msg_content['data'] = self.get_voice_url(msg_id)
  377. if self.DEBUG:
  378. voice = self.get_voice(msg_id)
  379. print ' [Voice] %s' % voice
  380. elif mtype == 42:
  381. msg_content['type'] = 5
  382. info = msg['RecommendInfo']
  383. msg_content['data'] = {'nickname': info['NickName'],
  384. 'alias': info['Alias'],
  385. 'province': info['Province'],
  386. 'city': info['City'],
  387. 'gender': ['unknown', 'male', 'female'][info['Sex']]}
  388. if self.DEBUG:
  389. print ' [Recommend]'
  390. print ' -----------------------------'
  391. print ' | NickName: %s' % info['NickName']
  392. print ' | Alias: %s' % info['Alias']
  393. print ' | Local: %s %s' % (info['Province'], info['City'])
  394. print ' | Gender: %s' % ['unknown', 'male', 'female'][info['Sex']]
  395. print ' -----------------------------'
  396. elif mtype == 47:
  397. msg_content['type'] = 6
  398. msg_content['data'] = self.search_content('cdnurl', content)
  399. if self.DEBUG:
  400. print ' [Animation] %s' % msg_content['data']
  401. elif mtype == 49:
  402. msg_content['type'] = 7
  403. app_msg_type = ''
  404. if msg['AppMsgType'] == 3:
  405. app_msg_type = 'music'
  406. elif msg['AppMsgType'] == 5:
  407. app_msg_type = 'link'
  408. elif msg['AppMsgType'] == 7:
  409. app_msg_type = 'weibo'
  410. else:
  411. app_msg_type = 'unknown'
  412. msg_content['data'] = {'type': app_msg_type,
  413. 'title': msg['FileName'],
  414. 'desc': self.search_content('des', content, 'xml'),
  415. 'url': msg['Url'],
  416. 'from': self.search_content('appname', content, 'xml')}
  417. if self.DEBUG:
  418. print ' [Share] %s' % app_msg_type
  419. print ' --------------------------'
  420. print ' | title: %s' % msg['FileName']
  421. print ' | desc: %s' % self.search_content('des', content, 'xml')
  422. print ' | link: %s' % msg['Url']
  423. print ' | from: %s' % self.search_content('appname', content, 'xml')
  424. print ' --------------------------'
  425. elif mtype == 62:
  426. msg_content['type'] = 8
  427. msg_content['data'] = content
  428. if self.DEBUG:
  429. print ' [Video] Please check on mobiles'
  430. elif mtype == 53:
  431. msg_content['type'] = 9
  432. msg_content['data'] = content
  433. if self.DEBUG:
  434. print ' [Video Call]'
  435. elif mtype == 10002:
  436. msg_content['type'] = 10
  437. msg_content['data'] = content
  438. if self.DEBUG:
  439. print ' [Redraw]'
  440. else:
  441. msg_content['type'] = 99
  442. msg_content['data'] = content
  443. if self.DEBUG:
  444. print ' [Unknown]'
  445. return msg_content
  446. '''
  447. msg_type_id:
  448. 0 -> Init
  449. 1 -> Self
  450. 2 -> FileHelper
  451. 3 -> Group
  452. 4 -> Contact
  453. 5 -> Public
  454. 6 -> Special
  455. 99 -> Unknown
  456. '''
  457. def handle_msg(self, r):
  458. for msg in r['AddMsgList']:
  459. msg_type_id = 99
  460. user = {'id': msg['FromUserName']}
  461. if msg['MsgType'] == 51: # init message
  462. msg_type_id = 0
  463. elif msg['FromUserName'] == self.user['UserName']: # Self
  464. msg_type_id = 1
  465. user['name'] = 'self'
  466. elif msg['ToUserName'] == 'filehelper': # File Helper
  467. msg_type_id = 2
  468. user['name'] = 'file_helper'
  469. elif msg['FromUserName'][:2] == '@@': # Group
  470. msg_type_id = 3
  471. user['name'] = self.get_user_remark_name(user['id'])
  472. if self.DEBUG:
  473. print '[From] %s' % user['name']
  474. elif self.is_contact(msg['FromUserName']): # Contact
  475. msg_type_id = 4
  476. user['name'] = self.get_user_remark_name(user['id'])
  477. elif self.is_public(msg['FromUserName']): # Public
  478. msg_type_id = 5
  479. user['name'] = self.get_user_remark_name(user['id'])
  480. elif self.is_special(msg['FromUserName']): # Special
  481. msg_type_id = 6
  482. user['name'] = self.get_user_remark_name(user['id'])
  483. else:
  484. user['name'] = 'unknown' # Unknown
  485. if self.DEBUG and msg_type_id != 0:
  486. print '[MSG] %s:' % user['name']
  487. content = self.extract_msg_content(msg_type_id, msg)
  488. message = {'msg_type_id': msg_type_id,
  489. 'msg_id': msg['MsgId'],
  490. 'content': content,
  491. 'user': user}
  492. self.handle_msg_all(message)
  493. def schedule(self):
  494. pass
  495. def proc_msg(self):
  496. self.test_sync_check()
  497. while True:
  498. [retcode, selector] = self.sync_check()
  499. if retcode == '1100': # User have login on mobile
  500. pass
  501. elif retcode == '0':
  502. if selector == '2':
  503. r = self.sync()
  504. if r is not None:
  505. self.handle_msg(r)
  506. elif selector == '7': # Play WeChat on mobile
  507. r = self.sync()
  508. if r is not None:
  509. self.handle_msg(r)
  510. elif selector == '0':
  511. time.sleep(1)
  512. self.schedule()
  513. def send_msg_by_uid(self, word, dst='filehelper'):
  514. url = self.base_uri + '/webwxsendmsg?pass_ticket=%s' % self.pass_ticket
  515. msg_id = str(int(time.time()*1000)) + str(random.random())[:5].replace('.', '')
  516. if type(word) == 'str':
  517. word = word.decode('utf-8')
  518. params = {
  519. 'BaseRequest': self.base_request,
  520. 'Msg': {
  521. "Type": 1,
  522. "Content": word,
  523. "FromUserName": self.user['UserName'],
  524. "ToUserName": dst,
  525. "LocalID": msg_id,
  526. "ClientMsgId": msg_id
  527. }
  528. }
  529. headers = {'content-type': 'application/json; charset=UTF-8'}
  530. data = json.dumps(params, ensure_ascii=False).encode('utf8')
  531. r = self.session.post(url, data=data, headers=headers)
  532. dic = r.json()
  533. return dic['BaseResponse']['Ret'] == 0
  534. def send_msg(self, name, word, isfile=False):
  535. uid = self.get_user_id(name)
  536. if uid:
  537. if isfile:
  538. with open(word, 'r') as f:
  539. result = True
  540. for line in f.readlines():
  541. line = line.replace('\n', '')
  542. print '-> '+name+': '+line
  543. if self.send_msg_by_uid(line, uid):
  544. pass
  545. else:
  546. result = False
  547. time.sleep(1)
  548. return result
  549. else:
  550. if self.send_msg_by_uid(word, uid):
  551. return True
  552. else:
  553. return False
  554. else:
  555. if self.DEBUG:
  556. print '[ERROR] This user does not exist .'
  557. return True
  558. @staticmethod
  559. def search_content(key, content, fmat='attr'):
  560. if fmat == 'attr':
  561. pm = re.search(key+'\s?=\s?"([^"<]+)"', content)
  562. if pm:
  563. return pm.group(1)
  564. elif fmat == 'xml':
  565. pm = re.search('<{0}>([^<]+)</{0}>'.format(key), content)
  566. if pm:
  567. return pm.group(1)
  568. return 'unknown'
  569. def run(self):
  570. self.get_uuid()
  571. self.gen_qr_code('qr.png')
  572. print '[INFO] Please use WeCaht to scan the QR code .'
  573. self.wait4login(1)
  574. print '[INFO] Please confirm to login .'
  575. self.wait4login(0)
  576. if self.login():
  577. print '[INFO] Web WeChat login succeed .'
  578. else:
  579. print '[ERROR] Web WeChat login failed .'
  580. return
  581. if self.init():
  582. print '[INFO] Web WeChat init succeed .'
  583. else:
  584. print '[INFO] Web WeChat init failed'
  585. return
  586. self.status_notify()
  587. self.get_contact()
  588. print '[INFO] Get %d contacts' % len(self.contact_list)
  589. print '[INFO] Start to process messages .'
  590. self.proc_msg()