2016-07-05 11:18:47 +00:00
|
|
|
import json, re, traceback, urllib.request, urllib.parse, urllib.error, ssl
|
2018-09-11 07:52:12 +00:00
|
|
|
import string
|
2016-03-29 11:56:58 +00:00
|
|
|
import bs4
|
2018-09-19 11:35:34 +00:00
|
|
|
import ModuleManager
|
2016-03-29 11:56:58 +00:00
|
|
|
|
|
|
|
USER_AGENT = ("Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 "
|
|
|
|
"(KHTML, like Gecko) Chrome/49.0.2623.87 Safari/537.36")
|
|
|
|
REGEX_HTTP = re.compile("https?://", re.I)
|
2018-09-12 12:48:19 +00:00
|
|
|
ASCII_UPPER = string.ascii_uppercase
|
|
|
|
ASCII_LOWER = string.ascii_lowercase
|
|
|
|
STRICT_RFC1459_UPPER = ASCII_UPPER+r'\[]'
|
|
|
|
STRICT_RFC1459_LOWER = ASCII_LOWER+r'|{}'
|
2018-09-12 09:28:06 +00:00
|
|
|
RFC1459_UPPER = STRICT_RFC1459_UPPER+"^"
|
|
|
|
RFC1459_LOWER = STRICT_RFC1459_LOWER+"~"
|
2018-09-11 07:52:12 +00:00
|
|
|
|
2016-03-29 11:56:58 +00:00
|
|
|
def remove_colon(s):
|
|
|
|
if s.startswith(":"):
|
|
|
|
s = s[1:]
|
|
|
|
return s
|
|
|
|
|
|
|
|
def arbitrary(s, n):
|
|
|
|
return remove_colon(" ".join(s[n:]))
|
|
|
|
|
2018-09-12 09:28:06 +00:00
|
|
|
# case mapping lowercase/uppcase logic
|
|
|
|
def _multi_replace(s, chars1, chars2):
|
|
|
|
for char1, char2 in zip(chars1, chars2):
|
2018-09-17 12:09:45 +00:00
|
|
|
s = s.replace(char1, char2)
|
2018-09-12 09:28:06 +00:00
|
|
|
return s
|
2018-09-11 07:52:12 +00:00
|
|
|
def irc_lower(server, s):
|
|
|
|
if server.case_mapping == "ascii":
|
2018-09-12 12:48:19 +00:00
|
|
|
return _multi_replace(s, ASCII_UPPER, ASCII_LOWER)
|
2018-09-11 07:52:12 +00:00
|
|
|
elif server.case_mapping == "rfc1459":
|
2018-09-12 12:48:19 +00:00
|
|
|
return _multi_replace(s, RFC1459_UPPER, RFC1459_LOWER)
|
2018-09-12 09:18:12 +00:00
|
|
|
elif server.case_mapping == "strict-rfc1459":
|
2018-09-12 12:48:19 +00:00
|
|
|
return _multi_replace(s, STRICT_RFC1459_UPPER, STRICT_RFC1459_LOWER)
|
2018-09-11 07:52:12 +00:00
|
|
|
else:
|
|
|
|
raise ValueError("unknown casemapping '%s'" % server.case_mapping)
|
|
|
|
|
2018-09-12 09:28:06 +00:00
|
|
|
# compare a string while respecting case mapping
|
2018-09-11 07:52:12 +00:00
|
|
|
def irc_equals(server, s1, s2):
|
|
|
|
return irc_lower(server, s1) == irc_lower(server, s2)
|
|
|
|
|
2018-09-10 08:31:24 +00:00
|
|
|
class IRCHostmask(object):
|
|
|
|
def __init__(self, nickname, username, hostname, hostmask):
|
|
|
|
self.nickname = nickname
|
|
|
|
self.username = username
|
|
|
|
self.hostname = hostname
|
|
|
|
self.hostmask = hostmask
|
2018-09-17 20:49:32 +00:00
|
|
|
def __repr__(self):
|
|
|
|
return "Utils.IRCHostmask(%s)" % self.__str__()
|
|
|
|
def __str__(self):
|
|
|
|
return self.hostmask
|
|
|
|
|
2016-03-29 11:56:58 +00:00
|
|
|
def seperate_hostmask(hostmask):
|
|
|
|
hostmask = remove_colon(hostmask)
|
|
|
|
first_delim = hostmask.find("!")
|
|
|
|
second_delim = hostmask.find("@")
|
2018-09-13 10:46:42 +00:00
|
|
|
nickname, username, hostname = hostmask, None, None
|
2016-03-29 11:56:58 +00:00
|
|
|
if first_delim > -1 and second_delim > first_delim:
|
|
|
|
nickname, username = hostmask.split("!", 1)
|
2018-09-04 07:05:02 +00:00
|
|
|
username, hostname = username.split("@", 1)
|
2018-09-13 10:50:18 +00:00
|
|
|
elif second_delim > -1:
|
2018-09-13 10:46:42 +00:00
|
|
|
nickname, hostname = hostmask.split("@", 1)
|
2018-09-10 08:31:24 +00:00
|
|
|
return IRCHostmask(nickname, username, hostname, hostmask)
|
2016-03-29 11:56:58 +00:00
|
|
|
|
|
|
|
def get_url(url, **kwargs):
|
2016-07-05 11:18:47 +00:00
|
|
|
if not urllib.parse.urlparse(url).scheme:
|
2016-03-29 11:56:58 +00:00
|
|
|
url = "http://%s" % url
|
2016-07-05 11:18:47 +00:00
|
|
|
url_parsed = urllib.parse.urlparse(url)
|
2016-03-29 11:56:58 +00:00
|
|
|
|
|
|
|
method = kwargs.get("method", "GET")
|
|
|
|
get_params = kwargs.get("get_params", "")
|
|
|
|
post_params = kwargs.get("post_params", None)
|
|
|
|
headers = kwargs.get("headers", {})
|
|
|
|
if get_params:
|
|
|
|
get_params = "?%s" % urllib.parse.urlencode(get_params)
|
|
|
|
if post_params:
|
|
|
|
post_params = urllib.parse.urlencode(post_params).encode("utf8")
|
|
|
|
url = "%s%s" % (url, get_params)
|
2016-03-29 20:05:07 +00:00
|
|
|
try:
|
|
|
|
url.encode("latin-1")
|
|
|
|
except UnicodeEncodeError:
|
|
|
|
if kwargs.get("code"):
|
|
|
|
return 0, False
|
|
|
|
return False
|
2016-03-29 11:56:58 +00:00
|
|
|
|
|
|
|
request = urllib.request.Request(url, post_params)
|
|
|
|
request.add_header("Accept-Language", "en-US")
|
|
|
|
request.add_header("User-Agent", USER_AGENT)
|
|
|
|
for header, value in headers.items():
|
|
|
|
request.add_header(header, value)
|
|
|
|
request.method = method
|
|
|
|
|
|
|
|
try:
|
2017-11-03 19:20:23 +00:00
|
|
|
response = urllib.request.urlopen(request, timeout=5)
|
2016-03-29 15:21:27 +00:00
|
|
|
except urllib.error.HTTPError as e:
|
2016-03-29 11:56:58 +00:00
|
|
|
traceback.print_exc()
|
2016-03-29 15:21:27 +00:00
|
|
|
if kwargs.get("code"):
|
|
|
|
return e.code, False
|
2016-07-05 11:18:47 +00:00
|
|
|
return False
|
2016-06-24 08:17:54 +00:00
|
|
|
except urllib.error.URLError as e:
|
|
|
|
traceback.print_exc()
|
|
|
|
if kwargs.get("code"):
|
|
|
|
return -1, False
|
2016-07-05 11:18:47 +00:00
|
|
|
return False
|
|
|
|
except ssl.CertificateError as e:
|
|
|
|
traceback.print_exc()
|
|
|
|
if kwargs.get("code"):
|
|
|
|
return -1, False,
|
|
|
|
return False
|
2016-03-29 11:56:58 +00:00
|
|
|
|
|
|
|
response_content = response.read()
|
|
|
|
encoding = response.info().get_content_charset()
|
|
|
|
if kwargs.get("soup"):
|
2017-06-07 20:36:11 +00:00
|
|
|
return bs4.BeautifulSoup(response_content, kwargs.get("parser", "lxml"))
|
2016-03-29 11:56:58 +00:00
|
|
|
if not encoding:
|
2017-06-07 20:36:11 +00:00
|
|
|
soup = bs4.BeautifulSoup(response_content, kwargs.get("parser", "lxml"))
|
2016-03-29 11:56:58 +00:00
|
|
|
metas = soup.find_all("meta")
|
|
|
|
for meta in metas:
|
|
|
|
if "charset=" in meta.get("content", ""):
|
|
|
|
encoding = meta.get("content").split("charset=", 1)[1
|
|
|
|
].split(";", 1)[0]
|
|
|
|
elif meta.get("charset", ""):
|
|
|
|
encoding = meta.get("charset")
|
|
|
|
else:
|
|
|
|
continue
|
|
|
|
break
|
|
|
|
if not encoding:
|
|
|
|
for item in soup.contents:
|
|
|
|
if isinstance(item, bs4.Doctype):
|
|
|
|
if item == "html":
|
|
|
|
encoding = "utf8"
|
|
|
|
else:
|
|
|
|
encoding = "latin-1"
|
|
|
|
break
|
|
|
|
response_content = response_content.decode(encoding or "utf8")
|
|
|
|
data = response_content
|
2016-03-30 18:32:47 +00:00
|
|
|
if kwargs.get("json") and data:
|
2016-03-29 11:56:58 +00:00
|
|
|
try:
|
|
|
|
data = json.loads(response_content)
|
|
|
|
except json.decoder.JSONDecodeError:
|
|
|
|
traceback.print_exc()
|
|
|
|
return False
|
|
|
|
if kwargs.get("code"):
|
|
|
|
return response.code, data
|
|
|
|
else:
|
|
|
|
return data
|
|
|
|
|
|
|
|
COLOR_WHITE, COLOR_BLACK, COLOR_BLUE, COLOR_GREEN = 0, 1, 2, 3
|
|
|
|
COLOR_RED, COLOR_BROWN, COLOR_PURPLE, COLOR_ORANGE = 4, 5, 6, 7
|
|
|
|
COLOR_YELLOW, COLOR_LIGHTGREEN, COLOR_CYAN, COLOR_LIGHTCYAN = (8, 9,
|
|
|
|
10, 11)
|
|
|
|
COLOR_LIGHTBLUE, COLOR_PINK, COLOR_GREY, COLOR_LIGHTGREY = (12, 13,
|
|
|
|
14, 15)
|
|
|
|
FONT_BOLD, FONT_ITALIC, FONT_UNDERLINE, FONT_INVERT = ("\x02", "\x1D",
|
|
|
|
"\x1F", "\x16")
|
|
|
|
FONT_COLOR, FONT_RESET = "\x03", "\x0F"
|
|
|
|
|
|
|
|
def color(foreground, background=None):
|
|
|
|
foreground = str(foreground).zfill(2)
|
|
|
|
if background:
|
2016-11-07 11:05:50 +00:00
|
|
|
background = str(background).zfill(2)
|
2016-03-29 11:56:58 +00:00
|
|
|
return "%s%s%s" % (FONT_COLOR, foreground,
|
|
|
|
"" if not background else ",%s" % background)
|
|
|
|
|
2018-09-01 09:34:55 +00:00
|
|
|
def bold(s):
|
|
|
|
return "%s%s%s" % (FONT_BOLD, s, FONT_BOLD)
|
|
|
|
|
|
|
|
def underline(s):
|
|
|
|
return "%s%s%s" % (FONT_UNDERLINE, s, FONT_UNDERLINE)
|
|
|
|
|
2016-03-29 11:56:58 +00:00
|
|
|
TIME_SECOND = 1
|
|
|
|
TIME_MINUTE = TIME_SECOND*60
|
|
|
|
TIME_HOUR = TIME_MINUTE*60
|
|
|
|
TIME_DAY = TIME_HOUR*24
|
|
|
|
TIME_WEEK = TIME_DAY*7
|
|
|
|
|
|
|
|
def time_unit(seconds):
|
|
|
|
since = None
|
|
|
|
unit = None
|
|
|
|
if seconds >= TIME_WEEK:
|
|
|
|
since = seconds/TIME_WEEK
|
|
|
|
unit = "week"
|
|
|
|
elif seconds >= TIME_DAY:
|
|
|
|
since = seconds/TIME_DAY
|
|
|
|
unit = "day"
|
|
|
|
elif seconds >= TIME_HOUR:
|
|
|
|
since = seconds/TIME_HOUR
|
|
|
|
unit = "hour"
|
|
|
|
elif seconds >= TIME_MINUTE:
|
|
|
|
since = seconds/TIME_MINUTE
|
|
|
|
unit = "minute"
|
|
|
|
else:
|
|
|
|
since = seconds
|
|
|
|
unit = "second"
|
|
|
|
since = int(since)
|
|
|
|
if since > 1:
|
|
|
|
unit = "%ss" % unit # pluralise the unit
|
|
|
|
return [since, unit]
|
|
|
|
|
|
|
|
REGEX_PRETTYTIME = re.compile("\d+[wdhms]", re.I)
|
|
|
|
|
|
|
|
SECONDS_MINUTES = 60
|
|
|
|
SECONDS_HOURS = SECONDS_MINUTES*60
|
|
|
|
SECONDS_DAYS = SECONDS_HOURS*24
|
|
|
|
SECONDS_WEEKS = SECONDS_DAYS*7
|
|
|
|
|
|
|
|
def from_pretty_time(pretty_time):
|
|
|
|
seconds = 0
|
|
|
|
for match in re.findall(REGEX_PRETTYTIME, pretty_time):
|
|
|
|
number, unit = int(match[:-1]), match[-1].lower()
|
|
|
|
if unit == "m":
|
|
|
|
number = number*SECONDS_MINUTES
|
|
|
|
elif unit == "h":
|
|
|
|
number = number*SECONDS_HOURS
|
|
|
|
elif unit == "d":
|
|
|
|
number = number*SECONDS_DAYS
|
|
|
|
elif unit == "w":
|
|
|
|
number = number*SECONDS_WEEKS
|
|
|
|
seconds += number
|
|
|
|
if seconds > 0:
|
|
|
|
return seconds
|
2016-04-10 16:40:58 +00:00
|
|
|
|
2018-08-29 14:52:04 +00:00
|
|
|
UNIT_SECOND = 5
|
|
|
|
UNIT_MINUTE = 4
|
|
|
|
UNIT_HOUR = 3
|
|
|
|
UNIT_DAY = 2
|
|
|
|
UNIT_WEEK = 1
|
|
|
|
def to_pretty_time(total_seconds, minimum_unit=UNIT_SECOND, max_units=6):
|
2018-08-09 11:23:54 +00:00
|
|
|
minutes, seconds = divmod(total_seconds, 60)
|
|
|
|
hours, minutes = divmod(minutes, 60)
|
|
|
|
days, hours = divmod(hours, 24)
|
|
|
|
weeks, days = divmod(days, 7)
|
|
|
|
out = ""
|
|
|
|
|
2018-08-29 14:52:04 +00:00
|
|
|
units = 0
|
|
|
|
if weeks and minimum_unit >= UNIT_WEEK and units < max_units:
|
2018-08-09 11:23:54 +00:00
|
|
|
out += "%dw" % weeks
|
2018-08-29 14:52:04 +00:00
|
|
|
units += 1
|
|
|
|
if days and minimum_unit >= UNIT_DAY and units < max_units:
|
2018-08-09 11:23:54 +00:00
|
|
|
out += "%dd" % days
|
2018-08-29 14:52:04 +00:00
|
|
|
units += 1
|
|
|
|
if hours and minimum_unit >= UNIT_HOUR and units < max_units:
|
2018-08-09 11:23:54 +00:00
|
|
|
out += "%dh" % hours
|
2018-08-29 14:52:04 +00:00
|
|
|
units += 1
|
|
|
|
if minutes and minimum_unit >= UNIT_MINUTE and units < max_units:
|
2018-08-09 11:23:54 +00:00
|
|
|
out += "%dm" % minutes
|
2018-08-29 14:52:04 +00:00
|
|
|
units += 1
|
|
|
|
if seconds and minimum_unit >= UNIT_SECOND and units < max_units:
|
2018-08-09 11:23:54 +00:00
|
|
|
out += "%ds" % seconds
|
2018-08-29 14:52:04 +00:00
|
|
|
units += 1
|
2018-08-09 11:23:54 +00:00
|
|
|
return out
|
|
|
|
|
2016-04-10 16:40:58 +00:00
|
|
|
IS_TRUE = ["true", "yes", "on", "y"]
|
|
|
|
IS_FALSE = ["false", "no", "off", "n"]
|
|
|
|
def bool_or_none(s):
|
|
|
|
s = s.lower()
|
|
|
|
if s in IS_TRUE:
|
|
|
|
return True
|
|
|
|
elif s in IS_FALSE:
|
|
|
|
return False
|
2017-12-26 11:50:18 +00:00
|
|
|
def int_or_none(s):
|
|
|
|
stripped_s = s.lstrip("0")
|
|
|
|
if stripped_s.isdigit():
|
|
|
|
return int(stripped_s)
|
2016-05-06 12:36:01 +00:00
|
|
|
|
|
|
|
def get_closest_setting(event, setting, default=None):
|
|
|
|
server = event["server"]
|
|
|
|
if "channel" in event:
|
|
|
|
closest = event["channel"]
|
|
|
|
elif "target" in event and "is_channel" in event and event["is_channel"]:
|
|
|
|
closest = event["target"]
|
|
|
|
else:
|
|
|
|
closest = event["user"]
|
|
|
|
return closest.get_setting(setting, server.get_setting(setting, default))
|
2018-08-10 12:56:45 +00:00
|
|
|
|
|
|
|
def prevent_highlight(nickname):
|
2018-08-20 10:03:01 +00:00
|
|
|
return nickname[0]+"\u200d"+nickname[1:]
|
2018-09-19 11:35:34 +00:00
|
|
|
|
|
|
|
def hook(event, **kwargs):
|
|
|
|
def _hook_func(func):
|
|
|
|
if not hasattr(func, ModuleManager.BITBOT_HOOKS_MAGIC):
|
|
|
|
setattr(func, ModuleManager.BITBOT_HOOKS_MAGIC, [])
|
|
|
|
getattr(func, ModuleManager.BITBOT_HOOKS_MAGIC).append(
|
|
|
|
{"event": event, "kwargs": kwargs})
|
|
|
|
return func
|
|
|
|
return _hook_func
|