2019-02-10 14:18:33 +00:00
|
|
|
import datetime, typing
|
2019-02-23 21:33:04 +00:00
|
|
|
from src import IRCObject
|
2019-02-10 18:43:04 +00:00
|
|
|
|
2019-02-10 22:52:23 +00:00
|
|
|
# this should be 510 (RFC1459, 512 with \r\n) but a server BitBot uses is broken
|
2019-02-10 20:54:12 +00:00
|
|
|
LINE_CUTOFF = 470
|
2019-02-10 14:18:33 +00:00
|
|
|
|
2019-02-23 21:33:04 +00:00
|
|
|
class IRCArgs(object):
|
|
|
|
def __init__(self, args: typing.List[str]):
|
|
|
|
self._args = args
|
|
|
|
|
|
|
|
def get(self, index: int) -> typing.Optional[str]:
|
|
|
|
if len(self._args) > index:
|
|
|
|
return self._args[index]
|
|
|
|
return None
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return "IRCArgs(%s)" % self._args
|
|
|
|
def __len__(self) -> int:
|
|
|
|
return len(self._args)
|
2019-02-23 21:39:46 +00:00
|
|
|
def __getitem__(self, index: int) -> str:
|
2019-02-23 21:33:04 +00:00
|
|
|
return self._args[index]
|
2019-02-23 21:39:46 +00:00
|
|
|
def __setitem__(self, index: int, value: str):
|
|
|
|
self._args[index] = value
|
2019-02-23 21:33:04 +00:00
|
|
|
|
|
|
|
class Hostmask(object):
|
|
|
|
def __init__(self, nickname: str, username: str, hostname: str,
|
|
|
|
hostmask: str):
|
|
|
|
self.nickname = nickname
|
|
|
|
self.username = username
|
|
|
|
self.hostname = hostname
|
|
|
|
self.hostmask = hostmask
|
|
|
|
def __repr__(self):
|
|
|
|
return "Hostmask(%s)" % self.__str__()
|
|
|
|
def __str__(self):
|
|
|
|
return self.hostmask
|
|
|
|
|
|
|
|
class ParsedLine(object):
|
|
|
|
def __init__(self, command: str, args: typing.List[str],
|
|
|
|
prefix: Hostmask=None,
|
|
|
|
tags: typing.Dict[str, str]={}):
|
|
|
|
self.command = command
|
|
|
|
self._args = args
|
|
|
|
self.args = IRCArgs(args)
|
|
|
|
self.prefix = prefix
|
|
|
|
self.tags = {} if tags == None else tags
|
|
|
|
|
|
|
|
def _tag_str(self, tags: typing.Dict[str, str]) -> str:
|
2019-02-27 22:14:53 +00:00
|
|
|
tag_pieces = []
|
2019-02-23 21:33:04 +00:00
|
|
|
for tag, value in tags.items():
|
|
|
|
if value:
|
2019-02-27 22:14:53 +00:00
|
|
|
tag_pieces.append("%s=%s" % (tag, value))
|
|
|
|
else:
|
|
|
|
tag_pieces.append(tag)
|
|
|
|
|
|
|
|
if tag_pieces:
|
|
|
|
return "@%s" % ",".join(tag_pieces)
|
|
|
|
return ""
|
2019-02-23 21:33:04 +00:00
|
|
|
|
|
|
|
def format(self) -> str:
|
2019-02-27 22:14:53 +00:00
|
|
|
pieces = []
|
2019-02-23 21:33:04 +00:00
|
|
|
if self.tags:
|
2019-02-27 22:14:53 +00:00
|
|
|
pieces.append(self._tag_str(self.tags))
|
2019-02-23 21:33:04 +00:00
|
|
|
|
|
|
|
if self.prefix:
|
2019-02-27 22:14:53 +00:00
|
|
|
pieces.append(self.prefix)
|
2019-02-10 18:43:04 +00:00
|
|
|
|
2019-02-27 22:14:53 +00:00
|
|
|
pieces.append(self.command.upper())
|
2019-02-10 18:43:04 +00:00
|
|
|
|
2019-02-23 21:33:04 +00:00
|
|
|
if self.args:
|
2019-02-27 22:14:53 +00:00
|
|
|
for i, arg in enumerate(self._args):
|
|
|
|
if i == len(self._args)-1 and (" " in arg or arg[0] == ":"):
|
|
|
|
pieces.append(":%s" % arg)
|
|
|
|
else:
|
|
|
|
pieces.append(arg)
|
2019-02-23 21:33:04 +00:00
|
|
|
|
2019-02-27 22:14:53 +00:00
|
|
|
return " ".join(pieces).split("\n")[0].strip("\r")
|
2019-02-23 21:33:04 +00:00
|
|
|
|
2019-02-24 08:24:58 +00:00
|
|
|
class SentLine(IRCObject.Object):
|
2019-02-23 21:33:04 +00:00
|
|
|
def __init__(self, send_time: datetime.datetime, hostmask: str,
|
|
|
|
line: ParsedLine):
|
|
|
|
self.send_time = send_time
|
|
|
|
self._hostmask = hostmask
|
|
|
|
self.parsed_line = line
|
|
|
|
|
|
|
|
self._on_send: typing.List[typing.Callable[[], None]] = []
|
|
|
|
self.truncate_marker: typing.Optional[str] = None
|
2019-02-10 14:18:33 +00:00
|
|
|
|
|
|
|
def __repr__(self) -> str:
|
2019-02-24 08:24:58 +00:00
|
|
|
return "IRCLine.SentLine(%s)" % self.__str__()
|
2019-02-10 14:18:33 +00:00
|
|
|
def __str__(self) -> str:
|
2019-02-10 23:57:38 +00:00
|
|
|
return self.decoded_data()
|
2019-02-10 14:18:33 +00:00
|
|
|
|
2019-02-10 23:52:25 +00:00
|
|
|
def _char_limit(self) -> int:
|
2019-02-23 21:33:04 +00:00
|
|
|
return LINE_CUTOFF-len(":%s " % self._hostmask)
|
2019-02-10 20:49:59 +00:00
|
|
|
|
2019-02-23 21:33:04 +00:00
|
|
|
def _encode_truncate(self) -> typing.Tuple[bytes, str]:
|
|
|
|
line = self.parsed_line.format()
|
|
|
|
byte_max = self._char_limit()
|
|
|
|
encoded = b""
|
|
|
|
truncated = ""
|
|
|
|
truncate_marker = b""
|
|
|
|
if not self.truncate_marker == None:
|
|
|
|
truncate_marker = typing.cast(str, self.truncate_marker
|
|
|
|
).encode("utf8")
|
|
|
|
|
|
|
|
for i, character in enumerate(line):
|
|
|
|
encoded_character = character.encode("utf8")
|
|
|
|
new_len = len(encoded + encoded_character)
|
|
|
|
if truncate_marker and (byte_max-new_len) < len(truncate_marker):
|
|
|
|
encoded += truncate_marker
|
|
|
|
truncated = line[i:]
|
|
|
|
break
|
|
|
|
elif new_len > byte_max:
|
|
|
|
truncated = line[i:]
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
encoded += encoded_character
|
|
|
|
return (encoded, truncated)
|
|
|
|
|
|
|
|
def _data(self) -> bytes:
|
|
|
|
return self._encode_truncate()[0]
|
2019-02-10 20:49:59 +00:00
|
|
|
def data(self) -> bytes:
|
2019-02-23 21:33:04 +00:00
|
|
|
return b"%s\r\n" % self._data()
|
2019-02-10 23:52:25 +00:00
|
|
|
def decoded_data(self) -> str:
|
2019-02-23 21:33:04 +00:00
|
|
|
return self._data().decode("utf8")
|
2019-02-10 20:49:59 +00:00
|
|
|
def truncated(self) -> str:
|
2019-02-23 21:33:04 +00:00
|
|
|
return self._encode_truncate()[1]
|
2019-02-10 20:49:59 +00:00
|
|
|
|
2019-02-10 14:18:33 +00:00
|
|
|
def on_send(self, func: typing.Callable[[], None]):
|
|
|
|
self._on_send.append(func)
|
|
|
|
def sent(self):
|
|
|
|
for func in self._on_send[:]:
|
|
|
|
func()
|