Advertisement
Guest User

Untitled

a guest
May 2nd, 2017
116
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 62.71 KB | None | 0 0
  1. ################################################################
  2. # File: ch.py
  3. # Title: Chatango Library
  4. # Original Author: Lumirayz/Lumz <lumirayz@gmail.com>
  5. # Current Maintainers and Contributors:
  6. # Nullspeaker <import codecs;codecs.encode('aunzzbaq129@tznvy.pbz','rot_13')>
  7. # asl97 <asl97@outlook.com>
  8. # pystub
  9. # dani87
  10. # domzy
  11. # kamijoutouma
  12. # piks
  13. # Version: 1.3.7
  14. # Description:
  15. # An event-based library for connecting to one or multiple Chatango rooms, has
  16. # support for several things including: messaging, message font,
  17. # name color, deleting, banning, recent history, 2 userlist modes,
  18. # flagging, avoiding flood bans, detecting flags.
  19. # Contact Info:
  20. # Any question, comment, or suggestion should be directed to the current
  21. # maintainers and contributors, located at:
  22. # https://github.com/Nullspeaker/ch.py
  23. # Where a more satisfactory response to all bug reports (which can be made on the
  24. # issues page) and other statements can be garnered. For things not specific or
  25. # in direct reference to this library, 'ch.py', a direct response can be filed
  26. # to the individual persons listed above as 'Current Maintainers and Contributors.'
  27. ################################################################
  28.  
  29. ################################################################
  30. # License
  31. ################################################################
  32. # Copyright 2011 Lumirayz
  33. # This program is distributed under the terms of the GNU GPL.
  34.  
  35. ################################################################
  36. # Imports
  37. ################################################################
  38. import socket
  39. import threading
  40. import time
  41. import random
  42. import re
  43. import sys
  44. import select
  45. import queue
  46.  
  47. ################################################################
  48. # Debug stuff
  49. ################################################################
  50. debug = False
  51.  
  52. ################################################################
  53. # Python 2 compatibility
  54. ################################################################
  55. if sys.version_info[0] < 3:
  56. class urllib:
  57. parse = __import__("urllib")
  58. request = __import__("urllib2")
  59. input = raw_input
  60. import codecs
  61. else:
  62. import urllib.request
  63. import urllib.parse
  64.  
  65. ################################################################
  66. # Constants
  67. ################################################################
  68. Userlist_Recent = 0
  69. Userlist_All = 1
  70.  
  71. BigMessage_Multiple = 0
  72. BigMessage_Cut = 1
  73.  
  74. # minimum of 1 thread needed
  75. Number_of_Threads = 1
  76. ################################################################
  77. # Struct class
  78. ################################################################
  79. class Struct:
  80. def __init__(self, **entries):
  81. self.__dict__.update(entries)
  82.  
  83. ################################################################
  84. # Tagserver stuff
  85. ################################################################
  86. specials = {'mitvcanal': 56, 'animeultimacom': 34, 'cricket365live': 21, 'pokemonepisodeorg': 22, 'animelinkz': 20, 'sport24lt': 56, 'narutowire': 10, 'watchanimeonn': 22, 'cricvid-hitcric-': 51, 'narutochatt': 70, 'leeplarp': 27, 'stream2watch3': 56, 'ttvsports': 56, 'ver-anime': 8, 'vipstand': 21, 'eafangames': 56, 'soccerjumbo': 21, 'myfoxdfw': 67, 'kiiiikiii': 21, 'de-livechat': 5, 'rgsmotrisport': 51, 'dbzepisodeorg': 10, 'watch-dragonball': 8, 'peliculas-flv': 69, 'tvanimefreak': 54, 'tvtvanimefreak': 54}
  87. tsweights = [['5', 75], ['6', 75], ['7', 75], ['8', 75], ['16', 75], ['17', 75], ['18', 75], ['9', 95], ['11', 95], ['12', 95], ['13', 95], ['14', 95], ['15', 95], ['19', 110], ['23', 110], ['24', 110], ['25', 110], ['26', 110], ['28', 104], ['29', 104], ['30', 104], ['31', 104], ['32', 104], ['33', 104], ['35', 101], ['36', 101], ['37', 101], ['38', 101], ['39', 101], ['40', 101], ['41', 101], ['42', 101], ['43', 101], ['44', 101], ['45', 101], ['46', 101], ['47', 101], ['48', 101], ['49', 101], ['50', 101], ['52', 110], ['53', 110], ['55', 110], ['57', 110], ['58', 110], ['59', 110], ['60', 110], ['61', 110], ['62', 110], ['63', 110], ['64', 110], ['65', 110], ['66', 110], ['68', 95], ['71', 116], ['72', 116], ['73', 116], ['74', 116], ['75', 116], ['76', 116], ['77', 116], ['78', 116], ['79', 116], ['80', 116], ['81', 116], ['82', 116], ['83', 116], ['84', 116]]
  88.  
  89. def getServer(group):
  90. """
  91. Get the server host for a certain room.
  92.  
  93. @type group: str
  94. @param group: room name
  95.  
  96. @rtype: str
  97. @return: the server's hostname
  98. """
  99. try:
  100. sn = specials[group]
  101. except KeyError:
  102. group = group.replace("_", "q")
  103. group = group.replace("-", "q")
  104. fnv = float(int(group[0:min(5, len(group))], 36))
  105. lnv = group[6: (6 + min(3, len(group) - 5))]
  106. if(lnv):
  107. lnv = float(int(lnv, 36))
  108. lnv = max(lnv,1000)
  109. else:
  110. lnv = 1000
  111. num = (fnv % lnv) / lnv
  112. maxnum = sum(map(lambda x: x[1], tsweights))
  113. cumfreq = 0
  114. sn = 0
  115. for wgt in tsweights:
  116. cumfreq += float(wgt[1]) / maxnum
  117. if(num <= cumfreq):
  118. sn = int(wgt[0])
  119. break
  120. return "s" + str(sn) + ".chatango.com"
  121.  
  122. ################################################################
  123. # Uid
  124. ################################################################
  125. def _genUid():
  126. """
  127. generate a uid
  128. """
  129. return str(random.randrange(10 ** 15, 10 ** 16))
  130.  
  131. ################################################################
  132. # Message stuff
  133. ################################################################
  134. def _clean_message(msg):
  135. """
  136. Clean a message and return the message, n tag and f tag.
  137.  
  138. @type msg: str
  139. @param msg: the message
  140.  
  141. @rtype: str, str, str
  142. @returns: cleaned message, n tag contents, f tag contents
  143. """
  144. n = re.search("<n(.*?)/>", msg)
  145. if n: n = n.group(1)
  146. f = re.search("<f(.*?)>", msg)
  147. if f: f = f.group(1)
  148. msg = re.sub("<n.*?/>", "", msg)
  149. msg = re.sub("<f.*?>", "", msg)
  150. msg = _strip_html(msg)
  151. msg = msg.replace("&lt;", "<")
  152. msg = msg.replace("&gt;", ">")
  153. msg = msg.replace("&quot;", "\"")
  154. msg = msg.replace("&apos;", "'")
  155. msg = msg.replace("&amp;", "&")
  156. return msg, n, f
  157.  
  158. def _strip_html(msg):
  159. """Strip HTML."""
  160. li = msg.split("<")
  161. if len(li) == 1:
  162. return li[0]
  163. else:
  164. ret = list()
  165. for data in li:
  166. data = data.split(">", 1)
  167. if len(data) == 1:
  168. ret.append(data[0])
  169. elif len(data) == 2:
  170. ret.append(data[1])
  171. return "".join(ret)
  172.  
  173. def _parseNameColor(n):
  174. """This just returns its argument, should return the name color."""
  175. #probably is already the name
  176. return n
  177.  
  178. def _parseFont(f):
  179. """Parses the contents of a f tag and returns color, face and size."""
  180. #' xSZCOL="FONT"'
  181. try: #TODO: remove quick hack
  182. sizecolor, fontface = f.split("=", 1)
  183. sizecolor = sizecolor.strip()
  184. size = int(sizecolor[1:3])
  185. col = sizecolor[3:6]
  186. if col == "": col = None
  187. face = f.split("\"", 2)[1]
  188. return col, face, size
  189. except:
  190. return None, None, None
  191.  
  192. ################################################################
  193. # Anon id
  194. ################################################################
  195. def _getAnonId(n, ssid):
  196. """Gets the anon's id."""
  197. if n == None: n = "5504"
  198. try:
  199. return "".join(list(
  200. map(lambda x: str(x[0] + x[1])[-1], list(zip(
  201. list(map(lambda x: int(x), n)),
  202. list(map(lambda x: int(x), ssid[4:]))
  203. )))
  204. ))
  205. except ValueError:
  206. return "NNNN"
  207.  
  208. ################################################################
  209. # ANON PM class
  210. ################################################################
  211.  
  212. class _ANON_PM_OBJECT:
  213. """Manages connection with Chatango anon PM."""
  214. def __init__(self, mgr, name):
  215. self._connected = False
  216. self._mgr = mgr
  217. self._wlock = False
  218. self._firstCommand = True
  219. self._wbuf = b""
  220. self._wlockbuf = b""
  221. self._rbuf = b""
  222. self._pingTask = None
  223. self._name = name
  224.  
  225. def _auth(self):
  226. self._sendCommand("mhs","mini","unknown","%s" % (self._name))
  227. self._setWriteLock(True)
  228. return True
  229.  
  230. def disconnect(self):
  231. """Disconnect the bot from PM"""
  232. self._disconnect()
  233. self._callEvent("onAnonPMDisconnect", User(self._name))
  234.  
  235. def _disconnect(self):
  236. self._connected = False
  237. self._sock.close()
  238. self._sock = None
  239.  
  240. def ping(self):
  241. """send a ping"""
  242. self._sendCommand("")
  243. self._callEvent("onPMPing")
  244.  
  245. def message(self, user, msg):
  246. """send a pm to a user"""
  247. if msg!=None:
  248. self._sendCommand("msg", user.name, msg)
  249.  
  250. ####
  251. # Feed
  252. ####
  253. def _feed(self, data):
  254. """
  255. Feed data to the connection.
  256.  
  257. @type data: bytes
  258. @param data: data to be fed
  259. """
  260. self._rbuf += data
  261. while self._rbuf.find(b"\x00") != -1:
  262. data = self._rbuf.split(b"\x00")
  263. for food in data[:-1]:
  264. self._process(food.decode(errors="replace").rstrip("\r\n"))
  265. self._rbuf = data[-1]
  266.  
  267. def _process(self, data):
  268. """
  269. Process a command string.
  270.  
  271. @type data: str
  272. @param data: the command string
  273. """
  274. self._callEvent("onRaw", data)
  275. data = data.split(":")
  276. cmd, args = data[0], data[1:]
  277. func = "_rcmd_" + cmd
  278. if hasattr(self, func):
  279. getattr(self, func)(args)
  280. else:
  281. if debug:
  282. print("unknown data: "+str(data))
  283.  
  284. def _getManager(self): return self._mgr
  285.  
  286. mgr = property(_getManager)
  287.  
  288. ####
  289. # Received Commands
  290. ####
  291.  
  292. def _rcmd_mhs(self, args):
  293. """
  294. note to future maintainers
  295.  
  296. args[1] is ether "online" or "offline"
  297. """
  298. self._connected = True
  299. self._setWriteLock(False)
  300.  
  301. def _rcmd_msg(self, args):
  302. user = User(args[0])
  303. body = _strip_html(":".join(args[5:]))
  304. self._callEvent("onPMMessage", user, body)
  305.  
  306. ####
  307. # Util
  308. ####
  309. def _callEvent(self, evt, *args, **kw):
  310. getattr(self.mgr, evt)(self, *args, **kw)
  311. self.mgr.onEventCalled(self, evt, *args, **kw)
  312.  
  313. def _write(self, data):
  314. if self._wlock:
  315. self._wlockbuf += data
  316. else:
  317. self.mgr._write(self, data)
  318.  
  319. def _setWriteLock(self, lock):
  320. self._wlock = lock
  321. if self._wlock == False:
  322. self._write(self._wlockbuf)
  323. self._wlockbuf = b""
  324.  
  325. def _sendCommand(self, *args):
  326. """
  327. Send a command.
  328.  
  329. @type args: [str, str, ...]
  330. @param args: command and list of arguments
  331. """
  332. if self._firstCommand:
  333. terminator = b"\x00"
  334. self._firstCommand = False
  335. else:
  336. terminator = b"\r\n\x00"
  337. self._write(":".join(args).encode() + terminator)
  338.  
  339. class ANON_PM:
  340. """Comparable wrapper for anon Chatango PM"""
  341. ####
  342. # Init
  343. ####
  344. def __init__(self, mgr):
  345. self._mgr = mgr
  346. self._wlock = False
  347. self._firstCommand = True
  348. self._persons = dict()
  349. self._wlockbuf = b""
  350. self._pingTask = None
  351.  
  352. ####
  353. # Connections
  354. ####
  355. def _connect(self,name):
  356. self._persons[name] = _ANON_PM_OBJECT(self._mgr,name)
  357. sock = socket.socket()
  358. sock.connect((self._mgr._anonPMHost, self._mgr._PMPort))
  359. sock.setblocking(False)
  360. self._persons[name]._sock = sock
  361. if not self._persons[name]._auth(): return
  362. self._persons[name]._pingTask = self._mgr.setInterval(self._mgr._pingDelay, self._persons[name].ping)
  363. self._persons[name]._connected = True
  364.  
  365. def message(self, user, msg):
  366. """send a pm to a user"""
  367. if not user.name in self._persons:
  368. self._connect(user.name)
  369. self._persons[user.name].message(user,msg)
  370.  
  371. def getConnections(self):
  372. return list(self._persons.values())
  373.  
  374. ################################################################
  375. # PM class
  376. ################################################################
  377. class PM:
  378. """Manages a connection with Chatango PM."""
  379. ####
  380. # Init
  381. ####
  382. def __init__(self, mgr):
  383. self._auth_re = re.compile(r"auth\.chatango\.com ?= ?([^;]*)", re.IGNORECASE)
  384. self._connected = False
  385. self._mgr = mgr
  386. self._auid = None
  387. self._blocklist = set()
  388. self._contacts = set()
  389. self._status = dict()
  390. self._wlock = False
  391. self._firstCommand = True
  392. self._wbuf = b""
  393. self._wlockbuf = b""
  394. self._rbuf = b""
  395. self._pingTask = None
  396. self._connect()
  397.  
  398. ####
  399. # Connections
  400. ####
  401. def _connect(self):
  402. self._wbuf = b""
  403. self._sock = socket.socket()
  404. self._sock.connect((self._mgr._PMHost, self._mgr._PMPort))
  405. self._sock.setblocking(False)
  406. self._firstCommand = True
  407. if not self._auth(): return
  408. self._pingTask = self.mgr.setInterval(self._mgr._pingDelay, self.ping)
  409. self._connected = True
  410.  
  411.  
  412. def _getAuth(self, name, password):
  413. """
  414. Request an auid using name and password.
  415.  
  416. @type name: str
  417. @param name: name
  418. @type password: str
  419. @param password: password
  420.  
  421. @rtype: str
  422. @return: auid
  423. """
  424. data = urllib.parse.urlencode({
  425. "user_id": name,
  426. "password": password,
  427. "storecookie": "on",
  428. "checkerrors": "yes"
  429. }).encode()
  430. try:
  431. resp = urllib.request.urlopen("http://chatango.com/login", data)
  432. headers = resp.headers
  433. except Exception:
  434. return None
  435. for header, value in headers.items():
  436. if header.lower() == "set-cookie":
  437. m = self._auth_re.search(value)
  438. if m:
  439. auth = m.group(1)
  440. if auth == "":
  441. return None
  442. return auth
  443. return None
  444.  
  445. def _auth(self):
  446. self._auid = self._getAuth(self._mgr.name, self._mgr.password)
  447. if self._auid == None:
  448. self._sock.close()
  449. self._callEvent("onLoginFail")
  450. self._sock = None
  451. return False
  452. self._sendCommand("tlogin", self._auid, "2")
  453. self._setWriteLock(True)
  454. return True
  455.  
  456. def disconnect(self):
  457. """Disconnect the bot from PM"""
  458. self._disconnect()
  459. self._callEvent("onPMDisconnect")
  460.  
  461. def _disconnect(self):
  462. self._connected = False
  463. self._sock.close()
  464. self._sock = None
  465.  
  466. ####
  467. # Feed
  468. ####
  469. def _feed(self, data):
  470. """
  471. Feed data to the connection.
  472.  
  473. @type data: bytes
  474. @param data: data to be fed
  475. """
  476. self._rbuf += data
  477. while self._rbuf.find(b"\x00") != -1:
  478. data = self._rbuf.split(b"\x00")
  479. for food in data[:-1]:
  480. self._process(food.decode(errors="replace").rstrip("\r\n"))
  481. self._rbuf = data[-1]
  482.  
  483. def _process(self, data):
  484. """
  485. Process a command string.
  486.  
  487. @type data: str
  488. @param data: the command string
  489. """
  490. self._callEvent("onRaw", data)
  491. data = data.split(":")
  492. cmd, args = data[0], data[1:]
  493. func = "_rcmd_" + cmd
  494. if hasattr(self, func):
  495. getattr(self, func)(args)
  496. else:
  497. if debug:
  498. print("unknown data: "+str(data))
  499.  
  500. ####
  501. # Properties
  502. ####
  503. def _getManager(self): return self._mgr
  504. def _getContacts(self): return self._contacts
  505. def _getBlocklist(self): return self._blocklist
  506.  
  507. mgr = property(_getManager)
  508. contacts = property(_getContacts)
  509. blocklist = property(_getBlocklist)
  510.  
  511. ####
  512. # Received Commands
  513. ####
  514. def _rcmd_OK(self, args):
  515. self._setWriteLock(False)
  516. self._sendCommand("wl")
  517. self._sendCommand("getblock")
  518. self._callEvent("onPMConnect")
  519.  
  520. def _rcmd_wl(self, args):
  521. self._contacts = set()
  522. for i in range(len(args) // 4):
  523. name, last_on, is_on, idle = args[i * 4: i * 4 + 4]
  524. user = User(name)
  525. if last_on=="None":pass#in case chatango gives a "None" as data argument
  526. elif not is_on == "on": self._status[user] = [int(last_on), False, 0]
  527. elif idle == '0': self._status[user] = [int(last_on), True, 0]
  528. else: self._status[user] = [int(last_on), True, time.time() - int(idle) * 60]
  529. self._contacts.add(user)
  530. self._callEvent("onPMContactlistReceive")
  531.  
  532. def _rcmd_block_list(self, args):
  533. self._blocklist = set()
  534. for name in args:
  535. if name == "": continue
  536. self._blocklist.add(User(name))
  537.  
  538. def _rcmd_idleupdate(self, args):
  539. user = User(args[0])
  540. last_on, is_on, idle = self._status[user]
  541. if args[1] == '1':
  542. self._status[user] = [last_on, is_on, 0]
  543. else:
  544. self._status[user] = [last_on, is_on, time.time()]
  545.  
  546. def _rcmd_track(self, args):
  547. user = User(args[0])
  548. if user in self._status:
  549. last_on = self._status[user][0]
  550. else:
  551. last_on = 0
  552. if args[1] == '0':
  553. idle = 0
  554. else:
  555. idle = time.time() - int(args[1]) * 60
  556. if args[2] == "online":
  557. is_on = True
  558. else:
  559. is_on = False
  560. self._status[user] = [last_on, is_on, idle]
  561.  
  562. def _rcmd_DENIED(self, args):
  563. self._disconnect()
  564. self._callEvent("onLoginFail")
  565.  
  566. def _rcmd_msg(self, args):
  567. user = User(args[0])
  568. body = _strip_html(":".join(args[5:]))
  569. self._callEvent("onPMMessage", user, body)
  570.  
  571. def _rcmd_msgoff(self, args):
  572. user = User(args[0])
  573. body = _strip_html(":".join(args[5:]))
  574. self._callEvent("onPMOfflineMessage", user, body)
  575.  
  576. def _rcmd_wlonline(self, args):
  577. user = User(args[0])
  578. last_on = float(args[1])
  579. self._status[user] = [last_on,True,last_on]
  580. self._callEvent("onPMContactOnline", user)
  581.  
  582. def _rcmd_wloffline(self, args):
  583. user = User(args[0])
  584. last_on = float(args[1])
  585. self._status[user] = [last_on,False,0]
  586. self._callEvent("onPMContactOffline", user)
  587.  
  588. def _rcmd_kickingoff(self, args):
  589. self.disconnect()
  590.  
  591. def _rcmd_toofast(self, args):
  592. self.disconnect()
  593.  
  594. def _rcmd_unblocked(self, user):
  595. """call when successfully unblocked"""
  596. if user in self._blocklist:
  597. self._blocklist.remove(user)
  598. self._callEvent("onPMUnblock", user)
  599.  
  600.  
  601. ####
  602. # Commands
  603. ####
  604. def ping(self):
  605. """send a ping"""
  606. self._sendCommand("")
  607. self._callEvent("onPMPing")
  608.  
  609. def message(self, user, msg):
  610. """send a pm to a user"""
  611. if msg!=None:
  612. self._sendCommand("msg", user.name, msg)
  613.  
  614. def addContact(self, user):
  615. """add contact"""
  616. if user not in self._contacts:
  617. self._sendCommand("wladd", user.name)
  618. self._contacts.add(user)
  619. self._callEvent("onPMContactAdd", user)
  620.  
  621. def removeContact(self, user):
  622. """remove contact"""
  623. if user in self._contacts:
  624. self._sendCommand("wldelete", user.name)
  625. self._contacts.remove(user)
  626. self._callEvent("onPMContactRemove", user)
  627.  
  628. def block(self, user):
  629. """block a person"""
  630. if user not in self._blocklist:
  631. self._sendCommand("block", user.name, user.name, "S")
  632. self._blocklist.add(user)
  633. self._callEvent("onPMBlock", user)
  634.  
  635. def unblock(self, user):
  636. """unblock a person"""
  637. if user in self._blocklist:
  638. self._sendCommand("unblock", user.name)
  639.  
  640. def track(self, user):
  641. """get and store status of person for future use"""
  642. self._sendCommand("track", user.name)
  643.  
  644. def checkOnline(self, user):
  645. """return True if online, False if offline, None if unknown"""
  646. if user in self._status:
  647. return self._status[user][1]
  648. else:
  649. return None
  650.  
  651. def getIdle(self, user):
  652. """return last active time, time.time() if isn't idle, 0 if offline, None if unknown"""
  653. if not user in self._status: return None
  654. if not self._status[user][1]: return 0
  655. if not self._status[user][2]: return time.time()
  656. else: return self._status[user][2]
  657.  
  658. ####
  659. # Util
  660. ####
  661. def _callEvent(self, evt, *args, **kw):
  662. getattr(self.mgr, evt)(self, *args, **kw)
  663. self.mgr.onEventCalled(self, evt, *args, **kw)
  664.  
  665. def _write(self, data):
  666. if self._wlock:
  667. self._wlockbuf += data
  668. else:
  669. self.mgr._write(self, data)
  670.  
  671. def _setWriteLock(self, lock):
  672. self._wlock = lock
  673. if self._wlock == False:
  674. self._write(self._wlockbuf)
  675. self._wlockbuf = b""
  676.  
  677. def _sendCommand(self, *args):
  678. """
  679. Send a command.
  680.  
  681. @type args: [str, str, ...]
  682. @param args: command and list of arguments
  683. """
  684. if self._firstCommand:
  685. terminator = b"\x00"
  686. self._firstCommand = False
  687. else:
  688. terminator = b"\r\n\x00"
  689. self._write(":".join(args).encode() + terminator)
  690.  
  691. def getConnections(self):
  692. return [self]
  693.  
  694. ################################################################
  695. # Room class
  696. ################################################################
  697. class Room:
  698. """Manages a connection with a Chatango room."""
  699. ####
  700. # Init
  701. ####
  702. def __init__(self, room, uid = None, server = None, port = None, mgr = None):
  703. """init, don't overwrite"""
  704. # Basic stuff
  705. self._name = room
  706. self._server = server or getServer(room)
  707. self._port = port or 443
  708. self._mgr = mgr
  709.  
  710. # Under the hood
  711. self._connected = False
  712. self._reconnecting = False
  713. self._uid = uid or _genUid()
  714. self._rbuf = b""
  715. self._wbuf = b""
  716. self._wlockbuf = b""
  717. self._owner = None
  718. self._mods = set()
  719. self._mqueue = dict()
  720. self._history = list()
  721. self._userlist = list()
  722. self._firstCommand = True
  723. self._connectAmmount = 0
  724. self._premium = False
  725. self._userCount = 0
  726. self._pingTask = None
  727. self._botname = None
  728. self._currentname = None
  729. self._users = dict()
  730. self._msgs = dict()
  731. self._wlock = False
  732. self._silent = False
  733. self._banlist = dict()
  734. self._unbanlist = dict()
  735.  
  736. # Inited vars
  737. if self._mgr: self._connect()
  738.  
  739. ####
  740. # Connect/disconnect
  741. ####
  742. def _connect(self):
  743. """Connect to the server."""
  744. self._sock = socket.socket()
  745. self._sock.connect((self._server, self._port))
  746. self._sock.setblocking(False)
  747. self._firstCommand = True
  748. self._wbuf = b""
  749. self._auth()
  750. self._pingTask = self.mgr.setInterval(self.mgr._pingDelay, self.ping)
  751. if not self._reconnecting: self.connected = True
  752.  
  753. def reconnect(self):
  754. """Reconnect."""
  755. self._reconnect()
  756.  
  757. def _reconnect(self):
  758. """Reconnect."""
  759. self._reconnecting = True
  760. if self.connected:
  761. self._disconnect()
  762. self._uid = _genUid()
  763. self._connect()
  764. self._reconnecting = False
  765.  
  766. def disconnect(self):
  767. """Disconnect."""
  768. self._disconnect()
  769. self._callEvent("onDisconnect")
  770.  
  771. def _disconnect(self):
  772. """Disconnect from the server."""
  773. if not self._reconnecting: self.connected = False
  774. for user in self._userlist:
  775. user.clearSessionIds(self)
  776. self._userlist = list()
  777. self._pingTask.cancel()
  778. self._sock.close()
  779. if not self._reconnecting: del self.mgr._rooms[self.name]
  780.  
  781. def _auth(self):
  782. """Authenticate."""
  783. # login as name with password
  784. if self.mgr.name and self.mgr.password:
  785. self._sendCommand("bauth", self.name, self._uid, self.mgr.name, self.mgr.password)
  786. self._currentname = self.mgr.name
  787. # login as anon
  788. else:
  789. self._sendCommand("bauth", self.name)
  790.  
  791. self._setWriteLock(True)
  792.  
  793. ####
  794. # Properties
  795. ####
  796. def _getName(self): return self._name
  797. def _getBotName(self):
  798. if self.mgr.name and self.mgr.password:
  799. return self.mgr.name
  800. elif self.mgr.name and self.mgr.password == None:
  801. return "#" + self.mgr.name
  802. elif self.mgr.name == None:
  803. return self._botname
  804. def _getCurrentname(self): return self._currentname
  805. def _getManager(self): return self._mgr
  806. def _getUserlist(self, mode = None, unique = None, memory = None):
  807. ul = None
  808. if mode == None: mode = self.mgr._userlistMode
  809. if unique == None: unique = self.mgr._userlistUnique
  810. if memory == None: memory = self.mgr._userlistMemory
  811. if mode == Userlist_Recent:
  812. ul = map(lambda x: x.user, self._history[-memory:])
  813. elif mode == Userlist_All:
  814. ul = self._userlist
  815. if unique:
  816. return list(set(ul))
  817. else:
  818. return ul
  819. def _getUserNames(self):
  820. ul = self.userlist
  821. return list(map(lambda x: x.name, ul))
  822. def _getUser(self): return self.mgr.user
  823. def _getOwner(self): return self._owner
  824. def _getOwnerName(self): return self._owner.name
  825. def _getMods(self):
  826. newset = set()
  827. for mod in self._mods:
  828. newset.add(mod)
  829. return newset
  830. def _getModNames(self):
  831. mods = self._getMods()
  832. return [x.name for x in mods]
  833. def _getUserCount(self): return self._userCount
  834. def _getSilent(self): return self._silent
  835. def _setSilent(self, val): self._silent = val
  836. def _getBanlist(self): return list(self._banlist.keys())
  837. def _getUnBanlist(self): return [[record["target"], record["src"]] for record in self._unbanlist.values()]
  838.  
  839. name = property(_getName)
  840. botname = property(_getBotName)
  841. currentname = property(_getCurrentname)
  842. mgr = property(_getManager)
  843. userlist = property(_getUserlist)
  844. usernames = property(_getUserNames)
  845. user = property(_getUser)
  846. owner = property(_getOwner)
  847. ownername = property(_getOwnerName)
  848. mods = property(_getMods)
  849. modnames = property(_getModNames)
  850. usercount = property(_getUserCount)
  851. silent = property(_getSilent, _setSilent)
  852. banlist = property(_getBanlist)
  853. unbanlist = property(_getUnBanlist)
  854.  
  855. ####
  856. # Feed/process
  857. ####
  858. def _feed(self, data):
  859. """
  860. Feed data to the connection.
  861.  
  862. @type data: bytes
  863. @param data: data to be fed
  864. """
  865. self._rbuf += data
  866. while self._rbuf.find(b"\x00") != -1:
  867. data = self._rbuf.split(b"\x00")
  868. for food in data[:-1]:
  869. self._process(food.decode(errors="replace").rstrip("\r\n"))
  870. self._rbuf = data[-1]
  871.  
  872. def _process(self, data):
  873. """
  874. Process a command string.
  875.  
  876. @type data: str
  877. @param data: the command string
  878. """
  879. self._callEvent("onRaw", data)
  880. data = data.split(":")
  881. cmd, args = data[0], data[1:]
  882. func = "_rcmd_" + cmd
  883. if hasattr(self, func):
  884. getattr(self, func)(args)
  885. else:
  886. if debug:
  887. print("unknown data: "+str(data))
  888.  
  889. ####
  890. # Received Commands
  891. ####
  892. def _rcmd_ok(self, args):
  893. # if no name, join room as anon and no password
  894. if args[2] == "N" and self.mgr.password == None and self.mgr.name == None:
  895. n = args[4].rsplit('.', 1)[0]
  896. n = n[-4:]
  897. aid = args[1][0:8]
  898. pid = "!anon" + _getAnonId(n, aid)
  899. self._botname = pid
  900. self._currentname = pid
  901. self.user._nameColor = n
  902. # if got name, join room as name and no password
  903. elif args[2] == "N" and self.mgr.password == None:
  904. self._sendCommand("blogin", self.mgr.name)
  905. self._currentname = self.mgr.name
  906. # if got password but fail to login
  907. elif args[2] != "M": #unsuccesful login
  908. self._callEvent("onLoginFail")
  909. self.disconnect()
  910. self._owner = User(args[0])
  911. self._uid = args[1]
  912. self._aid = args[1][4:8]
  913. self._mods = set(map(lambda x: User(x), args[6].split(";")))
  914. self._i_log = list()
  915.  
  916. def _rcmd_denied(self, args):
  917. self._disconnect()
  918. self._callEvent("onConnectFail")
  919.  
  920. def _rcmd_inited(self, args):
  921. self._sendCommand("g_participants", "start")
  922. self._sendCommand("getpremium", "1")
  923. self.requestBanlist()
  924. self.requestUnBanlist()
  925. if self._connectAmmount == 0:
  926. self._callEvent("onConnect")
  927. for msg in reversed(self._i_log):
  928. user = msg.user
  929. self._callEvent("onHistoryMessage", user, msg)
  930. self._addHistory(msg)
  931. del self._i_log
  932. else:
  933. self._callEvent("onReconnect")
  934. self._connectAmmount += 1
  935. self._setWriteLock(False)
  936.  
  937. def _rcmd_premium(self, args):
  938. if float(args[1]) > time.time():
  939. self._premium = True
  940. if self.user._mbg: self.setBgMode(1)
  941. if self.user._mrec: self.setRecordingMode(1)
  942. else:
  943. self._premium = False
  944.  
  945. def _rcmd_mods(self, args):
  946. modnames = args
  947. mods = set(map(lambda x: User(x), modnames))
  948. premods = self._mods
  949. for user in mods - premods: #modded
  950. self._mods.add(user)
  951. self._callEvent("onModAdd", user)
  952. for user in premods - mods: #demodded
  953. self._mods.remove(user)
  954. self._callEvent("onModRemove", user)
  955. self._callEvent("onModChange")
  956.  
  957. def _rcmd_b(self, args):
  958. mtime = float(args[0])
  959. puid = args[3]
  960. ip = args[6]
  961. name = args[1]
  962. rawmsg = ":".join(args[9:])
  963. msg, n, f = _clean_message(rawmsg)
  964. if name == "":
  965. nameColor = None
  966. name = "#" + args[2]
  967. if name == "#":
  968. name = "!anon" + _getAnonId(n, puid)
  969. else:
  970. if n: nameColor = _parseNameColor(n)
  971. else: nameColor = None
  972. i = args[5]
  973. unid = args[4]
  974. user = User(name)
  975. if puid:
  976. user.updatePuid(puid)
  977. #Create an anonymous message and queue it because msgid is unknown.
  978. if f: fontColor, fontFace, fontSize = _parseFont(f)
  979. else: fontColor, fontFace, fontSize = None, None, None
  980. msg = Message(
  981. time = mtime,
  982. user = user,
  983. body = msg,
  984. raw = rawmsg,
  985. ip = ip,
  986. nameColor = nameColor,
  987. fontColor = fontColor,
  988. fontFace = fontFace,
  989. fontSize = fontSize,
  990. unid = unid,
  991. room = self
  992. )
  993. self._mqueue[i] = msg
  994.  
  995. def _rcmd_u(self, args):
  996. temp = Struct(**self._mqueue)
  997. if hasattr(temp, args[0]):
  998. msg = getattr(temp, args[0])
  999. if msg.user != self.user:
  1000. msg.user._fontColor = msg.fontColor
  1001. msg.user._fontFace = msg.fontFace
  1002. msg.user._fontSize = msg.fontSize
  1003. msg.user._nameColor = msg.nameColor
  1004. del self._mqueue[args[0]]
  1005. msg.attach(self, args[1])
  1006. self._addHistory(msg)
  1007. self._callEvent("onMessage", msg.user, msg)
  1008.  
  1009. def _rcmd_i(self, args):
  1010. mtime = float(args[0])
  1011. puid = args[3]
  1012. ip = args[6]
  1013. name = args[1]
  1014. rawmsg = ":".join(args[9:])
  1015. msg, n, f = _clean_message(rawmsg)
  1016. if name == "":
  1017. nameColor = None
  1018. name = "#" + args[2]
  1019. if name == "#":
  1020. name = "!anon" + _getAnonId(n, puid)
  1021. else:
  1022. if n: nameColor = _parseNameColor(n)
  1023. else: nameColor = None
  1024. i = args[5]
  1025. unid = args[4]
  1026. user = User(name)
  1027. if puid:
  1028. user.updatePuid(puid)
  1029. #Create an anonymous message and queue it because msgid is unknown.
  1030. if f: fontColor, fontFace, fontSize = _parseFont(f)
  1031. else: fontColor, fontFace, fontSize = None, None, None
  1032. msg = Message(
  1033. time = mtime,
  1034. user = user,
  1035. body = msg,
  1036. raw = rawmsg,
  1037. ip = ip,
  1038. nameColor = nameColor,
  1039. fontColor = fontColor,
  1040. fontFace = fontFace,
  1041. fontSize = fontSize,
  1042. unid = unid,
  1043. room = self
  1044. )
  1045. self._i_log.append(msg)
  1046.  
  1047. def _rcmd_g_participants(self, args):
  1048. args = ":".join(args)
  1049. args = args.split(";")
  1050. for data in args:
  1051. data = data.split(":")
  1052. name = data[3].lower()
  1053. if name == "none": continue
  1054. user = User(
  1055. name = name,
  1056. room = self
  1057. )
  1058. user.addSessionId(self, data[0])
  1059. self._userlist.append(user)
  1060.  
  1061. def _rcmd_participant(self, args):
  1062. name = args[3].lower()
  1063. if name == "none": return
  1064. user = User(name)
  1065. puid = args[2]
  1066. if puid:
  1067. user.updatePuid(puid)
  1068.  
  1069. if args[0] == "0": #leave
  1070. user.removeSessionId(self, args[1])
  1071. self._userlist.remove(user)
  1072. if user not in self._userlist or not self.mgr._userlistEventUnique:
  1073. self._callEvent("onLeave", user)
  1074. else: #join
  1075. user.addSessionId(self, args[1])
  1076. if user not in self._userlist: doEvent = True
  1077. else: doEvent = False
  1078. self._userlist.append(user)
  1079. if doEvent or not self.mgr._userlistEventUnique:
  1080. self._callEvent("onJoin", user)
  1081.  
  1082. def _rcmd_show_fw(self, args):
  1083. self._callEvent("onFloodWarning")
  1084.  
  1085. def _rcmd_show_tb(self, args):
  1086. self._callEvent("onFloodBan")
  1087.  
  1088. def _rcmd_tb(self, args):
  1089. self._callEvent("onFloodBanRepeat")
  1090.  
  1091. def _rcmd_delete(self, args):
  1092. msg = self._msgs.get(args[0])
  1093. if msg:
  1094. if msg in self._history:
  1095. self._history.remove(msg)
  1096. self._callEvent("onMessageDelete", msg.user, msg)
  1097. msg.detach()
  1098.  
  1099. def _rcmd_deleteall(self, args):
  1100. for msgid in args:
  1101. self._rcmd_delete([msgid])
  1102.  
  1103. def _rcmd_n(self, args):
  1104. self._userCount = int(args[0], 16)
  1105. self._callEvent("onUserCountChange")
  1106.  
  1107. def _rcmd_blocklist(self, args):
  1108. self._banlist = dict()
  1109. sections = ":".join(args).split(";")
  1110. for section in sections:
  1111. params = section.split(":")
  1112. if len(params) != 5: continue
  1113. if params[2] == "": continue
  1114. user = User(params[2])
  1115. self._banlist[user] = {
  1116. "unid":params[0],
  1117. "ip":params[1],
  1118. "target":user,
  1119. "time":float(params[3]),
  1120. "src":User(params[4])
  1121. }
  1122. self._callEvent("onBanlistUpdate")
  1123.  
  1124. def _rcmd_unblocklist(self, args):
  1125. self._unbanlist = dict()
  1126. sections = ":".join(args).split(";")
  1127. for section in sections:
  1128. params = section.split(":")
  1129. if len(params) != 5: continue
  1130. if params[2] == "": continue
  1131. user = User(params[2])
  1132. self._unbanlist[user] = {
  1133. "unid":params[0],
  1134. "ip":params[1],
  1135. "target":user,
  1136. "time":float(params[3]),
  1137. "src":User(params[4])
  1138. }
  1139. self._callEvent("onUnBanlistUpdate")
  1140.  
  1141. def _rcmd_blocked(self, args):
  1142. if args[2] == "": return
  1143. target = User(args[2])
  1144. user = User(args[3])
  1145. self._banlist[target] = {"unid":args[0], "ip":args[1], "target":target, "time":float(args[4]), "src":user}
  1146. self._callEvent("onBan", user, target)
  1147.  
  1148. def _rcmd_unblocked(self, args):
  1149. if args[2] == "": return
  1150. target = User(args[2])
  1151. user=User(args[3])
  1152. del self._banlist[target]
  1153. self._unbanlist[user] = {"unid":args[0], "ip":args[1], "target":target, "time":float(args[4]), "src":user}
  1154. self._callEvent("onUnban", user, target)
  1155.  
  1156. ####
  1157. # Commands
  1158. ####
  1159. def login(self, NAME, PASS = None):
  1160. """login as a user or set a name in room"""
  1161. if PASS:
  1162. self._sendCommand("blogin", NAME, PASS)
  1163. else:
  1164. self._sendCommand("blogin", NAME)
  1165. self._currentname = NAME
  1166.  
  1167. def logout(self):
  1168. """logout of user in a room"""
  1169. self._sendCommand("blogout")
  1170. self._currentname = self._botname
  1171.  
  1172. def ping(self):
  1173. """Send a ping."""
  1174. self._sendCommand("")
  1175. self._callEvent("onPing")
  1176.  
  1177. def rawMessage(self, msg):
  1178. """
  1179. Send a message without n and f tags.
  1180.  
  1181. @type msg: str
  1182. @param msg: message
  1183. """
  1184. if not self._silent:
  1185. self._sendCommand("bmsg:tl2r", msg)
  1186.  
  1187. def message(self, msg, html = False):
  1188. """
  1189. Send a message.
  1190.  
  1191. @type msg: str
  1192. @param msg: message
  1193. """
  1194. if msg==None:
  1195. return
  1196. msg = msg.rstrip()
  1197. if not html:
  1198. msg = msg.replace("<", "&lt;").replace(">", "&gt;")
  1199. if len(msg) > self.mgr._maxLength:
  1200. if self.mgr._tooBigMessage == BigMessage_Cut:
  1201. self.message(msg[:self.mgr._maxLength], html = html)
  1202. elif self.mgr._tooBigMessage == BigMessage_Multiple:
  1203. while len(msg) > 0:
  1204. sect = msg[:self.mgr._maxLength]
  1205. msg = msg[self.mgr._maxLength:]
  1206. self.message(sect, html = html)
  1207. return
  1208. msg = "<n" + self.user.nameColor + "/>" + msg
  1209. if self._currentname != None and not self._currentname.startswith("!anon"):
  1210. msg = "<f x%0.2i%s=\"%s\">" %(self.user.fontSize, self.user.fontColor, self.user.fontFace) + msg
  1211. msg.replace("~","&#126;")
  1212. self.rawMessage(msg)
  1213.  
  1214. def setBgMode(self, mode):
  1215. """turn on/off bg"""
  1216. self._sendCommand("msgbg", str(mode))
  1217.  
  1218. def setRecordingMode(self, mode):
  1219. """turn on/off rcecording"""
  1220. self._sendCommand("msgmedia", str(mode))
  1221.  
  1222. def addMod(self, user):
  1223. """
  1224. Add a moderator.
  1225.  
  1226. @type user: User
  1227. @param user: User to mod.
  1228. """
  1229. if self.getLevel(User(self.currentname)) == 2:
  1230. self._sendCommand("addmod", user.name)
  1231.  
  1232. def removeMod(self, user):
  1233. """
  1234. Remove a moderator.
  1235.  
  1236. @type user: User
  1237. @param user: User to demod.
  1238. """
  1239. if self.getLevel(User(self.currentname)) == 2:
  1240. self._sendCommand("removemod", user.name)
  1241.  
  1242. def flag(self, message):
  1243. """
  1244. Flag a message.
  1245.  
  1246. @type message: Message
  1247. @param message: message to flag
  1248. """
  1249. self._sendCommand("g_flag", message.msgid)
  1250.  
  1251. def flagUser(self, user):
  1252. """
  1253. Flag a user.
  1254.  
  1255. @type user: User
  1256. @param user: user to flag
  1257.  
  1258. @rtype: bool
  1259. @return: whether a message to flag was found
  1260. """
  1261. msg = self.getLastMessage(user)
  1262. if msg:
  1263. self.flag(msg)
  1264. return True
  1265. return False
  1266.  
  1267. def delete(self, message):
  1268. """
  1269. Delete a message. (Moderator only)
  1270.  
  1271. @type message: Message
  1272. @param message: message to delete
  1273. """
  1274. if self.getLevel(self.user) > 0:
  1275. self._sendCommand("delmsg", message.msgid)
  1276.  
  1277. def rawClearUser(self, unid):
  1278. self._sendCommand("delallmsg", unid)
  1279.  
  1280. def clearUser(self, user):
  1281. """
  1282. Clear all of a user's messages. (Moderator only)
  1283.  
  1284. @type user: User
  1285. @param user: user to delete messages of
  1286.  
  1287. @rtype: bool
  1288. @return: whether a message to delete was found
  1289. """
  1290. if self.getLevel(self.user) > 0:
  1291. msg = self.getLastMessage(user)
  1292. if msg:
  1293. self.rawClearUser(msg.unid)
  1294. return True
  1295. return False
  1296.  
  1297. def clearall(self):
  1298. """Clear all messages. (Owner only)"""
  1299. if self.getLevel(self.user) == 2:
  1300. self._sendCommand("clearall")
  1301.  
  1302. def rawBan(self, name, ip, unid):
  1303. """
  1304. Execute the block command using specified arguments.
  1305. (For advanced usage)
  1306.  
  1307. @type name: str
  1308. @param name: name
  1309. @type ip: str
  1310. @param ip: ip address
  1311. @type unid: str
  1312. @param unid: unid
  1313. """
  1314. self._sendCommand("block", unid, ip, name)
  1315.  
  1316. def ban(self, msg):
  1317. """
  1318. Ban a message's sender. (Moderator only)
  1319.  
  1320. @type message: Message
  1321. @param message: message to ban sender of
  1322. """
  1323. if self.getLevel(self.user) > 0:
  1324. self.rawBan(msg.user.name, msg.ip, msg.unid)
  1325.  
  1326. def banUser(self, user):
  1327. """
  1328. Ban a user. (Moderator only)
  1329.  
  1330. @type user: User
  1331. @param user: user to ban
  1332.  
  1333. @rtype: bool
  1334. @return: whether a message to ban the user was found
  1335. """
  1336. msg = self.getLastMessage(user)
  1337. if msg:
  1338. self.ban(msg)
  1339. return True
  1340. return False
  1341.  
  1342. def requestBanlist(self):
  1343. """Request an updated banlist."""
  1344. self._sendCommand("blocklist", "block", "", "next", "500")
  1345.  
  1346. def requestUnBanlist(self):
  1347. """Request an updated banlist."""
  1348. self._sendCommand("blocklist", "unblock", "", "next", "500")
  1349.  
  1350. def rawUnban(self, name, ip, unid):
  1351. """
  1352. Execute the unblock command using specified arguments.
  1353. (For advanced usage)
  1354.  
  1355. @type name: str
  1356. @param name: name
  1357. @type ip: str
  1358. @param ip: ip address
  1359. @type unid: str
  1360. @param unid: unid
  1361. """
  1362. self._sendCommand("removeblock", unid, ip, name)
  1363.  
  1364. def unban(self, user):
  1365. """
  1366. Unban a user. (Moderator only)
  1367.  
  1368. @type user: User
  1369. @param user: user to unban
  1370.  
  1371. @rtype: bool
  1372. @return: whether it succeeded
  1373. """
  1374. rec = self._getBanRecord(user)
  1375. if rec:
  1376. self.rawUnban(rec["target"].name, rec["ip"], rec["unid"])
  1377. return True
  1378. else:
  1379. return False
  1380.  
  1381. ####
  1382. # Util
  1383. ####
  1384. def _getBanRecord(self, user):
  1385. if user in self._banlist:
  1386. return self._banlist[user]
  1387. return None
  1388.  
  1389. def _callEvent(self, evt, *args, **kw):
  1390. getattr(self.mgr, evt)(self, *args, **kw)
  1391. self.mgr.onEventCalled(self, evt, *args, **kw)
  1392.  
  1393. def _write(self, data):
  1394. if self._wlock:
  1395. self._wlockbuf += data
  1396. else:
  1397. self.mgr._write(self, data)
  1398.  
  1399. def _setWriteLock(self, lock):
  1400. self._wlock = lock
  1401. if self._wlock == False:
  1402. self._write(self._wlockbuf)
  1403. self._wlockbuf = b""
  1404.  
  1405. def _sendCommand(self, *args):
  1406. """
  1407. Send a command.
  1408.  
  1409. @type args: [str, str, ...]
  1410. @param args: command and list of arguments
  1411. """
  1412. if self._firstCommand:
  1413. terminator = b"\x00"
  1414. self._firstCommand = False
  1415. else:
  1416. terminator = b"\r\n\x00"
  1417. self._write(":".join(args).encode() + terminator)
  1418.  
  1419. def getLevel(self, user):
  1420. """get the level of user in a room"""
  1421. if user == self._owner: return 2
  1422. if user in self._mods: return 1
  1423. return 0
  1424.  
  1425. def getLastMessage(self, user = None):
  1426. """get last message said by user in a room"""
  1427. if user:
  1428. try:
  1429. i = 1
  1430. while True:
  1431. msg = self._history[-i]
  1432. if msg.user == user:
  1433. return msg
  1434. i += 1
  1435. except IndexError:
  1436. return None
  1437. else:
  1438. try:
  1439. return self._history[-1]
  1440. except IndexError:
  1441. return None
  1442. return None
  1443.  
  1444. def findUser(self, name):
  1445. """check if user is in the room
  1446.  
  1447. return User(name) if name in room else None"""
  1448. name = name.lower()
  1449. ul = self._getUserlist()
  1450. udi = dict(zip([u.name for u in ul], ul))
  1451. cname = None
  1452. for n in udi.keys():
  1453. if name in n:
  1454. if cname: return None #ambiguous!!
  1455. cname = n
  1456. if cname: return udi[cname]
  1457. else: return None
  1458.  
  1459. ####
  1460. # History
  1461. ####
  1462. def _addHistory(self, msg):
  1463. """
  1464. Add a message to history.
  1465.  
  1466. @type msg: Message
  1467. @param msg: message
  1468. """
  1469. self._history.append(msg)
  1470. if len(self._history) > self.mgr._maxHistoryLength:
  1471. rest, self._history = self._history[:-self.mgr._maxHistoryLength], self._history[-self.mgr._maxHistoryLength:]
  1472. for msg in rest: msg.detach()
  1473.  
  1474. ################################################################
  1475. # RoomManager class
  1476. ################################################################
  1477. class RoomManager:
  1478. """Class that manages multiple connections."""
  1479. ####
  1480. # Config
  1481. ####
  1482. _Room = Room
  1483. _PM = PM
  1484. _ANON_PM = ANON_PM
  1485. _anonPMHost = "b1.chatango.com"
  1486. _PMHost = "c1.chatango.com"
  1487. _PMPort = 5222
  1488. _TimerResolution = 0.2 #at least x times per second
  1489. _pingDelay = 20
  1490. _userlistMode = Userlist_Recent
  1491. _userlistUnique = True
  1492. _userlistMemory = 50
  1493. _userlistEventUnique = False
  1494. _tooBigMessage = BigMessage_Multiple
  1495. _maxLength = 1800
  1496. _maxHistoryLength = 150
  1497.  
  1498. ####
  1499. # Init
  1500. ####
  1501. def __init__(self, name = None, password = None, pm = True):
  1502. self._name = name
  1503. self._password = password
  1504. self._running = False
  1505. self._tasks = set()
  1506. self._rooms = dict()
  1507. self._rooms_queue = queue.Queue()
  1508. self._rooms_lock = threading.Lock()
  1509. if pm:
  1510. if self._password:
  1511. self._pm = self._PM(mgr = self)
  1512. else:
  1513. self._pm = self._ANON_PM(mgr = self)
  1514. else:
  1515. self._pm = None
  1516.  
  1517. def _joinThread(self):
  1518. while True:
  1519. room = self._rooms_queue.get()
  1520. with self._rooms_lock:
  1521. con = self._Room(room, mgr = self)
  1522. self._rooms[room] = con
  1523.  
  1524. ####
  1525. # Join/leave
  1526. ####
  1527. def joinRoom(self, room):
  1528. """
  1529. Join a room or return None if already joined.
  1530.  
  1531. @type room: str
  1532. @param room: room to join
  1533.  
  1534. @rtype: Room or None
  1535. @return: True or nothing
  1536. """
  1537. room = room.lower()
  1538. if room not in self._rooms:
  1539. self._rooms_queue.put(room)
  1540. return True
  1541. else:
  1542. return None
  1543.  
  1544. def leaveRoom(self, room):
  1545. """
  1546. Leave a room.
  1547.  
  1548. @type room: str
  1549. @param room: room to leave
  1550. """
  1551. room = room.lower()
  1552. if room in self._rooms:
  1553. with self._rooms_lock:
  1554. con = self._rooms[room]
  1555. con.disconnect()
  1556.  
  1557. def getRoom(self, room):
  1558. """
  1559. Get room with a name, or None if not connected to this room.
  1560.  
  1561. @type room: str
  1562. @param room: room
  1563.  
  1564. @rtype: Room
  1565. @return: the room
  1566. """
  1567. room = room.lower()
  1568. if room in self._rooms:
  1569. return self._rooms[room]
  1570. else:
  1571. return None
  1572.  
  1573. ####
  1574. # Properties
  1575. ####
  1576. def _getUser(self): return User(self._name)
  1577. def _getName(self): return self._name
  1578. def _getPassword(self): return self._password
  1579. def _getRooms(self): return set(self._rooms.values())
  1580. def _getRoomNames(self): return set(self._rooms.keys())
  1581. def _getPM(self): return self._pm
  1582.  
  1583. user = property(_getUser)
  1584. name = property(_getName)
  1585. password = property(_getPassword)
  1586. rooms = property(_getRooms)
  1587. roomnames = property(_getRoomNames)
  1588. pm = property(_getPM)
  1589.  
  1590. ####
  1591. # Virtual methods
  1592. ####
  1593. def onInit(self):
  1594. """Called on init."""
  1595. pass
  1596.  
  1597. def safePrint(self, text):
  1598. """Use this to safely print text with unicode"""
  1599. while True:
  1600. try:
  1601. print(text)
  1602. break
  1603. except UnicodeEncodeError as ex:
  1604. text = (text[0:ex.start]+'(unicode)'+text[ex.end:])
  1605.  
  1606. def onConnect(self, room):
  1607. """
  1608. Called when connected to the room.
  1609.  
  1610. @type room: Room
  1611. @param room: room where the event occured
  1612. """
  1613. pass
  1614.  
  1615. def onReconnect(self, room):
  1616. """
  1617. Called when reconnected to the room.
  1618.  
  1619. @type room: Room
  1620. @param room: room where the event occured
  1621. """
  1622. pass
  1623.  
  1624. def onConnectFail(self, room):
  1625. """
  1626. Called when the connection failed.
  1627.  
  1628. @type room: Room
  1629. @param room: room where the event occured
  1630. """
  1631. pass
  1632.  
  1633. def onDisconnect(self, room):
  1634. """
  1635. Called when the client gets disconnected.
  1636.  
  1637. @type room: Room
  1638. @param room: room where the event occured
  1639. """
  1640. pass
  1641.  
  1642. def onLoginFail(self, room):
  1643. """
  1644. Called on login failure, disconnects after.
  1645.  
  1646. @type room: Room
  1647. @param room: room where the event occured
  1648. """
  1649. pass
  1650.  
  1651. def onFloodBan(self, room):
  1652. """
  1653. Called when either flood banned or flagged.
  1654.  
  1655. @type room: Room
  1656. @param room: room where the event occured
  1657. """
  1658. pass
  1659.  
  1660. def onFloodBanRepeat(self, room):
  1661. """
  1662. Called when trying to send something when floodbanned.
  1663.  
  1664. @type room: Room
  1665. @param room: room where the event occured
  1666. """
  1667. pass
  1668.  
  1669. def onFloodWarning(self, room):
  1670. """
  1671. Called when an overflow warning gets received.
  1672.  
  1673. @type room: Room
  1674. @param room: room where the event occured
  1675. """
  1676. pass
  1677.  
  1678. def onMessageDelete(self, room, user, message):
  1679. """
  1680. Called when a message gets deleted.
  1681.  
  1682. @type room: Room
  1683. @param room: room where the event occured
  1684. @type user: User
  1685. @param user: owner of deleted message
  1686. @type message: Message
  1687. @param message: message that got deleted
  1688. """
  1689. pass
  1690.  
  1691. def onModChange(self, room):
  1692. """
  1693. Called when the moderator list changes.
  1694.  
  1695. @type room: Room
  1696. @param room: room where the event occured
  1697. """
  1698. pass
  1699.  
  1700. def onModAdd(self, room, user):
  1701. """
  1702. Called when a moderator gets added.
  1703.  
  1704. @type room: Room
  1705. @param room: room where the event occured
  1706. """
  1707. pass
  1708.  
  1709. def onModRemove(self, room, user):
  1710. """
  1711. Called when a moderator gets removed.
  1712.  
  1713. @type room: Room
  1714. @param room: room where the event occured
  1715. """
  1716. pass
  1717.  
  1718. def onMessage(self, room, user, message):
  1719. """
  1720. Called when a message gets received.
  1721.  
  1722. @type room: Room
  1723. @param room: room where the event occured
  1724. @type user: User
  1725. @param user: owner of message
  1726. @type message: Message
  1727. @param message: received message
  1728. """
  1729. pass
  1730.  
  1731. def onHistoryMessage(self, room, user, message):
  1732. """
  1733. Called when a message gets received from history.
  1734.  
  1735. @type room: Room
  1736. @param room: room where the event occured
  1737. @type user: User
  1738. @param user: owner of message
  1739. @type message: Message
  1740. @param message: the message that got added
  1741. """
  1742. pass
  1743.  
  1744. def onJoin(self, room, user):
  1745. """
  1746. Called when a user joins. Anonymous users get ignored here.
  1747.  
  1748. @type room: Room
  1749. @param room: room where the event occured
  1750. @type user: User
  1751. @param user: the user that has joined
  1752. """
  1753. pass
  1754.  
  1755. def onLeave(self, room, user):
  1756. """
  1757. Called when a user leaves. Anonymous users get ignored here.
  1758.  
  1759. @type room: Room
  1760. @param room: room where the event occured
  1761. @type user: User
  1762. @param user: the user that has left
  1763. """
  1764. pass
  1765.  
  1766. def onRaw(self, room, raw):
  1767. """
  1768. Called before any command parsing occurs.
  1769.  
  1770. @type room: Room
  1771. @param room: room where the event occured
  1772. @type raw: str
  1773. @param raw: raw command data
  1774. """
  1775. pass
  1776.  
  1777. def onPing(self, room):
  1778. """
  1779. Called when a ping gets sent.
  1780.  
  1781. @type room: Room
  1782. @param room: room where the event occured
  1783. """
  1784. pass
  1785.  
  1786. def onUserCountChange(self, room):
  1787. """
  1788. Called when the user count changes.
  1789.  
  1790. @type room: Room
  1791. @param room: room where the event occured
  1792. """
  1793. pass
  1794.  
  1795. def onBan(self, room, user, target):
  1796. """
  1797. Called when a user gets banned.
  1798.  
  1799. @type room: Room
  1800. @param room: room where the event occured
  1801. @type user: User
  1802. @param user: user that banned someone
  1803. @type target: User
  1804. @param target: user that got banned
  1805. """
  1806. pass
  1807.  
  1808. def onUnban(self, room, user, target):
  1809. """
  1810. Called when a user gets unbanned.
  1811.  
  1812. @type room: Room
  1813. @param room: room where the event occured
  1814. @type user: User
  1815. @param user: user that unbanned someone
  1816. @type target: User
  1817. @param target: user that got unbanned
  1818. """
  1819. pass
  1820.  
  1821. def onBanlistUpdate(self, room):
  1822. """
  1823. Called when a banlist gets updated.
  1824.  
  1825. @type room: Room
  1826. @param room: room where the event occured
  1827. """
  1828. pass
  1829.  
  1830. def onUnBanlistUpdate(self, room):
  1831. """
  1832. Called when a unbanlist gets updated.
  1833.  
  1834. @type room: Room
  1835. @param room: room where the event occured
  1836. """
  1837. pass
  1838.  
  1839. def onPMConnect(self, pm):
  1840. """
  1841. Called when connected to the pm
  1842.  
  1843. @type pm: PM
  1844. @param pm: the pm
  1845. """
  1846. pass
  1847.  
  1848. def onAnonPMDisconnect(self, pm, user):
  1849. """
  1850. Called when disconnected from the pm
  1851.  
  1852. @type pm: PM
  1853. @param pm: the pm
  1854. """
  1855. pass
  1856.  
  1857. def onPMDisconnect(self, pm):
  1858. """
  1859. Called when disconnected from the pm
  1860.  
  1861. @type pm: PM
  1862. @param pm: the pm
  1863. """
  1864. pass
  1865.  
  1866. def onPMPing(self, pm):
  1867. """
  1868. Called when sending a ping to the pm
  1869.  
  1870. @type pm: PM
  1871. @param pm: the pm
  1872. """
  1873. pass
  1874.  
  1875. def onPMMessage(self, pm, user, body):
  1876. """
  1877. Called when a message is received
  1878.  
  1879. @type pm: PM
  1880. @param pm: the pm
  1881. @type user: User
  1882. @param user: owner of message
  1883. @type message: Message
  1884. @param message: received message
  1885. """
  1886. pass
  1887.  
  1888. def onPMOfflineMessage(self, pm, user, body):
  1889. """
  1890. Called when connected if a message is received while offline
  1891.  
  1892. @type pm: PM
  1893. @param pm: the pm
  1894. @type user: User
  1895. @param user: owner of message
  1896. @type message: Message
  1897. @param message: received message
  1898. """
  1899. pass
  1900.  
  1901. def onPMContactlistReceive(self, pm):
  1902. """
  1903. Called when the contact list is received
  1904.  
  1905. @type pm: PM
  1906. @param pm: the pm
  1907. """
  1908. pass
  1909.  
  1910. def onPMBlocklistReceive(self, pm):
  1911. """
  1912. Called when the block list is received
  1913.  
  1914. @type pm: PM
  1915. @param pm: the pm
  1916. """
  1917. pass
  1918.  
  1919. def onPMContactAdd(self, pm, user):
  1920. """
  1921. Called when the contact added message is received
  1922.  
  1923. @type pm: PM
  1924. @param pm: the pm
  1925. @type user: User
  1926. @param user: the user that gotten added
  1927. """
  1928. pass
  1929.  
  1930. def onPMContactRemove(self, pm, user):
  1931. """
  1932. Called when the contact remove message is received
  1933.  
  1934. @type pm: PM
  1935. @param pm: the pm
  1936. @type user: User
  1937. @param user: the user that gotten remove
  1938. """
  1939. pass
  1940.  
  1941. def onPMBlock(self, pm, user):
  1942. """
  1943. Called when successfully block a user
  1944.  
  1945. @type pm: PM
  1946. @param pm: the pm
  1947. @type user: User
  1948. @param user: the user that gotten block
  1949. """
  1950. pass
  1951.  
  1952. def onPMUnblock(self, pm, user):
  1953. """
  1954. Called when successfully unblock a user
  1955.  
  1956. @type pm: PM
  1957. @param pm: the pm
  1958. @type user: User
  1959. @param user: the user that gotten unblock
  1960. """
  1961. pass
  1962.  
  1963. def onPMContactOnline(self, pm, user):
  1964. """
  1965. Called when a user from the contact come online
  1966.  
  1967. @type pm: PM
  1968. @param pm: the pm
  1969. @type user: User
  1970. @param user: the user that came online
  1971. """
  1972. pass
  1973.  
  1974. def onPMContactOffline(self, pm, user):
  1975. """
  1976. Called when a user from the contact go offline
  1977.  
  1978. @type pm: PM
  1979. @param pm: the pm
  1980. @type user: User
  1981. @param user: the user that went offline
  1982. """
  1983. pass
  1984.  
  1985. def onEventCalled(self, room, evt, *args, **kw):
  1986. """
  1987. Called on every room-based event.
  1988.  
  1989. @type room: Room
  1990. @param room: room where the event occured
  1991. @type evt: str
  1992. @param evt: the event
  1993. """
  1994. pass
  1995.  
  1996. ####
  1997. # Deferring
  1998. ####
  1999. def deferToThread(self, callback, func, *args, **kw):
  2000. """
  2001. Defer a function to a thread and callback the return value.
  2002.  
  2003. @type callback: function
  2004. @param callback: function to call on completion
  2005. @type cbargs: tuple or list
  2006. @param cbargs: arguments to get supplied to the callback
  2007. @type func: function
  2008. @param func: function to call
  2009. """
  2010. def f(func, callback, *args, **kw):
  2011. ret = func(*args, **kw)
  2012. self.setTimeout(0, callback, ret)
  2013. threading._start_new_thread(f, (func, callback) + args, kw)
  2014.  
  2015. ####
  2016. # Scheduling
  2017. ####
  2018. class _Task:
  2019. def cancel(self):
  2020. """Sugar for removeTask."""
  2021. self.mgr.removeTask(self)
  2022.  
  2023. def _tick(self):
  2024. now = time.time()
  2025. for task in set(self._tasks):
  2026. if task.target <= now:
  2027. task.func(*task.args, **task.kw)
  2028. if task.isInterval:
  2029. task.target = now + task.timeout
  2030. else:
  2031. self._tasks.remove(task)
  2032.  
  2033. def setTimeout(self, timeout, func, *args, **kw):
  2034. """
  2035. Call a function after at least timeout seconds with specified arguments.
  2036.  
  2037. @type timeout: int
  2038. @param timeout: timeout
  2039. @type func: function
  2040. @param func: function to call
  2041.  
  2042. @rtype: _Task
  2043. @return: object representing the task
  2044. """
  2045. task = self._Task()
  2046. task.mgr = self
  2047. task.target = time.time() + timeout
  2048. task.timeout = timeout
  2049. task.func = func
  2050. task.isInterval = False
  2051. task.args = args
  2052. task.kw = kw
  2053. self._tasks.add(task)
  2054. return task
  2055.  
  2056. def setInterval(self, timeout, func, *args, **kw):
  2057. """
  2058. Call a function at least every timeout seconds with specified arguments.
  2059.  
  2060. @type timeout: int
  2061. @param timeout: timeout
  2062. @type func: function
  2063. @param func: function to call
  2064.  
  2065. @rtype: _Task
  2066. @return: object representing the task
  2067. """
  2068. task = self._Task()
  2069. task.mgr = self
  2070. task.target = time.time() + timeout
  2071. task.timeout = timeout
  2072. task.func = func
  2073. task.isInterval = True
  2074. task.args = args
  2075. task.kw = kw
  2076. self._tasks.add(task)
  2077. return task
  2078.  
  2079. def removeTask(self, task):
  2080. """
  2081. Cancel a task.
  2082.  
  2083. @type task: _Task
  2084. @param task: task to cancel
  2085. """
  2086. self._tasks.remove(task)
  2087.  
  2088. ####
  2089. # Util
  2090. ####
  2091. def _write(self, room, data):
  2092. room._wbuf += data
  2093.  
  2094. def getConnections(self):
  2095. li = list(self._rooms.values())
  2096. if self._pm:
  2097. li.extend(self._pm.getConnections())
  2098. return [c for c in li if c._sock != None]
  2099.  
  2100. ####
  2101. # Main
  2102. ####
  2103. def main(self):
  2104. self.onInit()
  2105. self._running = True
  2106. for l in range(0,Number_of_Threads):
  2107. t = threading.Thread(target=self._joinThread)
  2108. t.daemon = True
  2109. t.start()
  2110. while self._running:
  2111. conns = self.getConnections()
  2112. socks = [x._sock for x in conns]
  2113. wsocks = [x._sock for x in conns if x._wbuf != b""]
  2114. rd, wr, sp = select.select(socks, wsocks, [], self._TimerResolution)
  2115. for sock in rd:
  2116. con = [c for c in conns if c._sock == sock][0]
  2117. try:
  2118. data = sock.recv(1024)
  2119. if(len(data) > 0):
  2120. con._feed(data)
  2121. else:
  2122. con.disconnect()
  2123. except socket.error:
  2124. pass
  2125. for sock in wr:
  2126. con = [c for c in conns if c._sock == sock][0]
  2127. try:
  2128. size = sock.send(con._wbuf)
  2129. con._wbuf = con._wbuf[size:]
  2130. except socket.error:
  2131. pass
  2132. self._tick()
  2133.  
  2134. @classmethod
  2135. def easy_start(cl, rooms = None, name = None, password = None, pm = True):
  2136. """
  2137. Prompts the user for missing info, then starts.
  2138.  
  2139. @type rooms: list
  2140. @param room: rooms to join
  2141. @type name: str
  2142. @param name: name to join as ("" = None, None = unspecified)
  2143. @type password: str
  2144. @param password: password to join with ("" = None, None = unspecified)
  2145. """
  2146. if not rooms: rooms = str(input("Room names separated by semicolons: ")).split(";")
  2147. if len(rooms) == 1 and rooms[0] == "": rooms = []
  2148. if not name: name = str(input("User name: "))
  2149. if name == "": name = None
  2150. if not password: password = str(input("User password: "))
  2151. if password == "": password = None
  2152. self = cl(name, password, pm = pm)
  2153. for room in rooms:
  2154. self.joinRoom(room)
  2155. self.main()
  2156.  
  2157. def stop(self):
  2158. for conn in list(self._rooms.values()):
  2159. conn.disconnect()
  2160. self._running = False
  2161.  
  2162. ####
  2163. # Commands
  2164. ####
  2165. def enableBg(self):
  2166. """Enable background if available."""
  2167. self.user._mbg = True
  2168. for room in self.rooms:
  2169. room.setBgMode(1)
  2170.  
  2171. def disableBg(self):
  2172. """Disable background."""
  2173. self.user._mbg = False
  2174. for room in self.rooms:
  2175. room.setBgMode(0)
  2176.  
  2177. def enableRecording(self):
  2178. """Enable recording if available."""
  2179. self.user._mrec = True
  2180. for room in self.rooms:
  2181. room.setRecordingMode(1)
  2182.  
  2183. def disableRecording(self):
  2184. """Disable recording."""
  2185. self.user._mrec = False
  2186. for room in self.rooms:
  2187. room.setRecordingMode(0)
  2188.  
  2189. def setNameColor(self, color3x):
  2190. """
  2191. Set name color.
  2192.  
  2193. @type color3x: str
  2194. @param color3x: a 3-char RGB hex code for the color
  2195. """
  2196. self.user._nameColor = color3x
  2197.  
  2198. def setFontColor(self, color3x):
  2199. """
  2200. Set font color.
  2201.  
  2202. @type color3x: str
  2203. @param color3x: a 3-char RGB hex code for the color
  2204. """
  2205. self.user._fontColor = color3x
  2206.  
  2207. def setFontFace(self, face):
  2208. """
  2209. Set font face/family.
  2210.  
  2211. @type face: str
  2212. @param face: the font face
  2213. """
  2214. self.user._fontFace = face
  2215.  
  2216. def setFontSize(self, size):
  2217. """
  2218. Set font size.
  2219.  
  2220. @type size: int
  2221. @param size: the font size (limited: 9 to 22)
  2222. """
  2223. if size < 9: size = 9
  2224. if size > 22: size = 22
  2225. self.user._fontSize = size
  2226.  
  2227. ################################################################
  2228. # User class (well, yeah, I lied, it's actually _User)
  2229. ################################################################
  2230. _users = dict()
  2231. def User(name, *args, **kw):
  2232. if name == None: name = ""
  2233. name = name.lower()
  2234. user = _users.get(name)
  2235. if not user:
  2236. user = _User(name = name, *args, **kw)
  2237. _users[name] = user
  2238. return user
  2239.  
  2240. class _User:
  2241. """Class that represents a user."""
  2242. ####
  2243. # Init
  2244. ####
  2245. def __init__(self, name, **kw):
  2246. self._name = name.lower()
  2247. self._puid = ""
  2248. self._sids = dict()
  2249. self._msgs = list()
  2250. self._nameColor = "000"
  2251. self._fontSize = 12
  2252. self._fontFace = "0"
  2253. self._fontColor = "000"
  2254. self._mbg = False
  2255. self._mrec = False
  2256. for attr, val in kw.items():
  2257. if val == None: continue
  2258. setattr(self, "_" + attr, val)
  2259.  
  2260. ####
  2261. # Properties
  2262. ####
  2263. def _getName(self): return self._name
  2264. def _getPuid(self): return self._puid
  2265. def _getSessionIds(self, room = None):
  2266. if room:
  2267. return self._sids.get(room, set())
  2268. else:
  2269. return set.union(*self._sids.values())
  2270. def _getRooms(self): return self._sids.keys()
  2271. def _getRoomNames(self): return [room.name for room in self._getRooms()]
  2272. def _getFontColor(self): return self._fontColor
  2273. def _getFontFace(self): return self._fontFace
  2274. def _getFontSize(self): return self._fontSize
  2275. def _getNameColor(self): return self._nameColor
  2276.  
  2277. name = property(_getName)
  2278. puid = property(_getPuid)
  2279. sessionids = property(_getSessionIds)
  2280. rooms = property(_getRooms)
  2281. roomnames = property(_getRoomNames)
  2282. fontColor = property(_getFontColor)
  2283. fontFace = property(_getFontFace)
  2284. fontSize = property(_getFontSize)
  2285. nameColor = property(_getNameColor)
  2286.  
  2287. ####
  2288. # Util
  2289. ####
  2290. def addSessionId(self, room, sid):
  2291. if room not in self._sids:
  2292. self._sids[room] = set()
  2293. self._sids[room].add(sid)
  2294.  
  2295. def removeSessionId(self, room, sid):
  2296. try:
  2297. self._sids[room].remove(sid)
  2298. if len(self._sids[room]) == 0:
  2299. del self._sids[room]
  2300. except KeyError:
  2301. pass
  2302.  
  2303. def clearSessionIds(self, room):
  2304. try:
  2305. del self._sids[room]
  2306. except KeyError:
  2307. pass
  2308.  
  2309. def hasSessionId(self, room, sid):
  2310. try:
  2311. if sid in self._sids[room]:
  2312. return True
  2313. else:
  2314. return False
  2315. except KeyError:
  2316. return False
  2317.  
  2318. def updatePuid(self, puid):
  2319. self._puid = puid
  2320.  
  2321. ####
  2322. # Repr
  2323. ####
  2324. def __repr__(self):
  2325. return "<User: %s>" %(self.name)
  2326.  
  2327. ################################################################
  2328. # Message class
  2329. ################################################################
  2330. class Message:
  2331. """Class that represents a message."""
  2332. ####
  2333. # Attach/detach
  2334. ####
  2335. def attach(self, room, msgid):
  2336. """
  2337. Attach the Message to a message id.
  2338.  
  2339. @type msgid: str
  2340. @param msgid: message id
  2341. """
  2342. if self._msgid == None:
  2343. self._room = room
  2344. self._msgid = msgid
  2345. self._room._msgs[msgid] = self
  2346.  
  2347. def detach(self):
  2348. """Detach the Message."""
  2349. if self._msgid != None and self._msgid in self._room._msgs:
  2350. del self._room._msgs[self._msgid]
  2351. self._msgid = None
  2352.  
  2353. ####
  2354. # Init
  2355. ####
  2356. def __init__(self, **kw):
  2357. """init, don't overwrite"""
  2358. self._msgid = None
  2359. self._time = None
  2360. self._user = None
  2361. self._body = None
  2362. self._room = None
  2363. self._raw = ""
  2364. self._ip = None
  2365. self._unid = ""
  2366. self._nameColor = "000"
  2367. self._fontSize = 12
  2368. self._fontFace = "0"
  2369. self._fontColor = "000"
  2370. for attr, val in kw.items():
  2371. if val == None: continue
  2372. setattr(self, "_" + attr, val)
  2373.  
  2374. ####
  2375. # Properties
  2376. ####
  2377. def _getId(self): return self._msgid
  2378. def _getTime(self): return self._time
  2379. def _getUser(self): return self._user
  2380. def _getBody(self): return self._body
  2381. def _getIP(self): return self._ip
  2382. def _getFontColor(self): return self._fontColor
  2383. def _getFontFace(self): return self._fontFace
  2384. def _getFontSize(self): return self._fontSize
  2385. def _getNameColor(self): return self._nameColor
  2386. def _getRoom(self): return self._room
  2387. def _getRaw(self): return self._raw
  2388. def _getUnid(self): return self._unid
  2389.  
  2390. msgid = property(_getId)
  2391. time = property(_getTime)
  2392. user = property(_getUser)
  2393. body = property(_getBody)
  2394. room = property(_getRoom)
  2395. ip = property(_getIP)
  2396. fontColor = property(_getFontColor)
  2397. fontFace = property(_getFontFace)
  2398. fontSize = property(_getFontSize)
  2399. raw = property(_getRaw)
  2400. nameColor = property(_getNameColor)
  2401. unid = property(_getUnid)
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement