### # Copyright (c) 2020, mogad0n # All rights reserved. # # Redistribution and use in source and binary forms, with or wthout # modification, are permitted provided that the following conditions are met: # # * Redistributions of source code must retain the above copyright notice, # this list of conditions, and the following disclaimer. # * Redistributions in binary form must reproduce the above copyright notice, # this list of conditions, and the following disclaimer in the # documentation and/or other materials provided with the distribution. # * Neither the name of the author of this software nor the name of # contributors to this software may be used to endorse or promote products # derived from this software without specific prior written consent. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. ### from supybot import utils, plugins, ircutils, callbacks, world, conf, log from supybot.commands import * # HTTP Imports from supybot import httpserver # Misc from num2words import num2words import pickle import datetime import pytz try: from supybot.i18n import PluginInternationalization _ = PluginInternationalization('Tripsit') except ImportError: # Placeholder that allows to run the plugin on a bot # without the i18n module _ = lambda x: x filename = conf.supybot.directories.data.dirize("Tripsit.db") insufflated = ["Insufflation", "Insufflation-IR", "Insufflation-XR"] METHODS = { "iv": ["IV"], "shot": ["IV"], "im": ["IM"], "oral": ["Oral", "Oral-IR", "Oral-XR"], "insufflated": insufflated, "snorted": insufflated, "smoked": ["Smoked"] } class TripsitServerCallback(httpserver.SupyHTTPServerCallback): name = 'Tripsit' defaultResponse = """ This plugin handles only GET request, please don't use other requests.""" def __init__(self, plugin): self.plugin = plugin # to access db def doGet(self, handler, path): if path == '/doses': # Collect all dose logs from self.db dose_logs = [] for nick, data in self.plugin.db.items(): for dose in data.get('doses', []): dose_logs.append({ 'nick': nick, 'time': dose['time'], 'dose': dose['dose'], 'drug': dose['drug'], 'method': dose['method'], }) # Create HTML response html_response = """ Dose Logs

Dose Logs

""" # Add rows for each dose log for log in dose_logs: html_response += f""" """ html_response += """
Nick Time Dose Drug Method
{log['nick']} {log['time']} {log['dose']} {log['drug']} {log['method']}
""" # Send the HTML response handler.send_response(200) handler.send_header('Content-type', 'text/html') handler.end_headers() handler.wfile.write(html_response.encode('utf-8')) else: # 404 response for unknown paths handler.send_response(404) handler.send_header('Content-type', 'text/html') handler.end_headers() handler.wfile.write(b"

404 Not Found

") class Tripsit(callbacks.Plugin): """Harm-Reduction tools from tripsit's tripbot and the tripsitwiki""" threaded = True def __init__(self, irc): self.__parent = super(Tripsit, self) self.__parent.__init__(irc) self.db = {} self._loadDb() world.flushers.append(self._flushDb) httpserver.hook('tripsit', TripsitServerCallback(self)) # register the callback at `/tripsit` def _loadDb(self): """Loads the (flatfile) database mapping nicks to doses.""" try: with open(filename, "rb") as f: self.db = pickle.load(f) except Exception as e: self.log.debug("Tripsit: Unable to load pickled database: %s", e) def _flushDb(self): """Flushes the (flatfile) database mapping nicks to doses.""" try: with open(filename, "wb") as f: pickle.dump(self.db, f, 2) except Exception as e: self.log.warning("Tripsit: Unable to write pickled database: %s", e) def die(self): self._flushDb() httpserver.unhook('tripsit') world.flushers.remove(self._flushDb) self.__parent.die() def set(self, irc, msg, args, timezone): """ Sets location for your current nick to for eg. America/Chicago """ nick = msg.nick try: timezone = pytz.timezone(timezone) if nick in self.db: self.db[nick]['timezone'] = timezone else: self.db[nick] = {'timezone': timezone } irc.replySuccess() except pytz.UnknownTimeZoneError: irc.error(_('Unknown timezone'), Raise=True) set = wrap(set, ["something"]) @wrap([getopts({'ago': 'something'}), "something", "something", optional("something")]) def idose(self, irc, msg, args, opts, dose, name, method): """[--ago ] [] logs a dose for your , eg. @idose --ago 0100 20mg mph oral would log that dose as if it was taken an hour ago [--ago] and [ROA] fields are optional """ opts = dict(opts) found_method = False onset = None methods = [] if method: methods = [method.lower()] methods = METHODS.get(methods[0], methods) drug_and_method = name if method: if not found_method: method = method drug_and_method = "%s via %s" % (drug_and_method, method) else: method = 'Undefined' nick = msg.nick if nick in self.db: timezone = self.db[nick].get('timezone', 'UTC') tz = pytz.timezone(str(timezone)) time = datetime.datetime.now(tz=tz) dose_td = 0 if 'ago' in opts and len(opts['ago']) == 4: ago = opts['ago'] dose_td = datetime.timedelta(hours=int(ago[0:2]), minutes=int(ago[2:4])) dose_td_s = dose_td.total_seconds() time = time - dose_td doseLog = {'time': time, 'dose': dose, 'drug': name, 'method': method } doses = self.db[nick].get('doses') if doses: doses.append(doseLog) else: doses = [doseLog] self.db[nick]['doses'] = doses else: timezone = 'UTC' tz = pytz.timezone(timezone) time = datetime.datetime.now(tz=tz) dose_td = 0 if 'ago' in opts and len(opts['ago']) == 4: ago = opts['ago'] dose_td = datetime.timedelta(hours=int(ago[0:2]), minutes=int(ago[2:4])) dose_td_s = dose_td.total_seconds() time = time - dose_td doseLog = {'time': time, 'dose': dose, 'drug': name, 'method': method } doses = [doseLog] self.db[nick] = {'timezone': timezone, 'doses': doses} if dose_td == 0: re = utils.str.format("You dosed %s of %s at %s, %s", dose, drug_and_method, time.strftime("%c"), timezone) if onset is not None: re += utils.str.format(". You should start feeling effects %s from now", onset) else: re = utils.str.format("You dosed %s of %s at %s, %s ; %T ago", dose, drug_and_method, time.strftime("%c"), timezone, dose_td.total_seconds()) if onset is not None: re += utils.str.format(". You should have/will start feeling effects %s from/after dosing", onset) irc.reply(re) @wrap([optional('positiveInt')]) def undose(self, irc, msg, args, entry): """ removes your last dose entry, if is provided then deletes the nth last dose """ nick = msg.nick if nick in self.db: nick_dose_log = self.db[nick]['doses'] if entry: try: del nick_dose_log[-int(entry)] entry = num2words(entry, to='ordinal') irc.replySuccess(f"Deleted the {entry} last dose logged for {nick} ") except IndexError: irc.error("The dose entry doesn't exist") return else: del nick_dose_log[-1] irc.replySuccess(f"Deleted the last dose logged for {nick} ") else: irc.error(f'No doses saved for {nick}') def doseslogged(self, irc, msg, args): """ This command takes no arguments. Retrieves the number of doses logged for a given nick """ def doGet(self, handler, path): if path == '/doses': # Collect all dose logs from self.db dose_logs = [] for nick, data in self.plugin.db.items(): for dose in data.get('doses', []): dose_logs.append({ 'nick': nick, 'time': dose['time'], 'dose': dose['dose'], 'drug': dose['drug'], 'method': dose['method'], }) # Create HTML response html_response = """ Dose Logs

Dose Logs

""" # Add rows for each dose log for log in dose_logs: html_response += f""" """ html_response += """
Nick Time Dose Drug Method
{log['nick']} {log['time']} {log['dose']} {log['drug']} {log['method']}
""" # Send the HTML response handler.send_response(200) handler.send_header('Content-type', 'text/html') handler.end_headers() handler.wfile.write(html_response.encode('utf-8')) else: # 404 response for unknown paths handler.send_response(404) handler.send_header('Content-type', 'text/html') handler.end_headers() handler.wfile.write(b"

404 Not Found

") nick = msg.nick if nick in self.db: try: nick_dose_log_count = len(self.db[nick]['doses']) nick_dose_log_since = self.db[nick]['doses'][0]["time"] nick_dose_log_since_string = nick_dose_log_since.strftime("%c") irc.reply(f"{nick} has logged {nick_dose_log_count} doses since {nick_dose_log_since_string}") except IndexError: irc.error("Can't seem to do math, check logs") else: irc.error(f"No doses saved for {nick}") doseslogged = wrap(doseslogged) @wrap([optional('positiveInt')]) def lastdose(self, irc, msg, args, history): """ retrieves your th last logged dose """ nick = msg.nick if nick in self.db: if history: try: lastdose = self.db[nick]['doses'][-int(history)] except IndexError: irc.error("You haven't logged that many doses") return else: lastdose = self.db[nick]['doses'][-1] dose = lastdose['dose'] drug = lastdose['drug'] method = lastdose['method'] dose_time = lastdose['time'] timezone = self.db[nick]['timezone'] tz = pytz.timezone(str(timezone)) time = datetime.datetime.now(tz=tz) since_dose = time - dose_time since_dose_seconds = since_dose.total_seconds() if history: history = num2words(history, to='ordinal') re = utils.str.format("Your %i last dose was %s of %s via %s at %s %s, %T ago", history, dose, drug, method, dose_time.strftime("%c"), timezone, since_dose_seconds) else: re = utils.str.format("You last dosed %s of %s via %s at %s %s, %T ago", dose, drug, method, dose_time.strftime("%c"), timezone, since_dose_seconds) irc.reply(re) else: irc.error(f'No doses saved for {nick}') @wrap([getopts({'drug': 'something'}), 'positiveInt']) def listdose(self, irc, msg, args, opts, history): """[--drug ] Retrieves your last logged doses, optionally filtered by drug. """ if history > 20: irc.error("You can't retrieve more than 20 doses.") return opts = dict(opts) drug_filter = opts.get('drug') nick = msg.nick if nick in self.db: doses = self.db[nick]['doses'] if drug_filter: doses = [dose for dose in doses if dose['drug'].lower() == drug_filter.lower()] if len(doses) == 0: irc.error(f"No doses found for drug '{drug_filter}'.") return try: irc.reply(f"Your last {history} dose(s) are:", private=True) for number in range(history, 0, -1): lastdose = doses[-number] dose = lastdose['dose'] drug = lastdose['drug'] method = lastdose['method'] dose_time = lastdose['time'] timezone = self.db[nick]['timezone'] tz = pytz.timezone(str(timezone)) time = datetime.datetime.now(tz=tz) since_dose = time - dose_time since_dose_seconds = since_dose.total_seconds() if number == 1: number = "The" else: number = num2words(number, to='ordinal') re = utils.str.format( "::> %s last dose: Amount: %s of \x02%s\x0F via %s | datetime: %s %s | timedelta %T", number, dose, drug, method, dose_time.strftime("%c"), timezone, since_dose_seconds ) irc.reply(re, private=True) except IndexError: irc.error("You haven't logged that many doses.") return else: irc.error(f"No doses saved for {nick}.") @wrap(["something"]) def grepdose(self, irc, msg, args, drug): """ pulls most recent dose for drug """ nick = msg.nick if nick in self.db: doselogs = self.db[nick]['doses'] found = False for doselog in reversed(doselogs): if doselog['drug'] == drug: timezone = self.db[nick]['timezone'] tz = pytz.timezone(str(timezone)) now = datetime.datetime.now(tz=tz) since_dose = now - doselog['time'] re = utils.str.format("You last dosed %s of %s via %s at %s %s, %T ago", doselog["dose"], doselog["drug"], doselog["method"], doselog["time"].strftime("%c"), timezone, since_dose.total_seconds()) irc.reply(re) found = True break if not found: irc.error(f"No doses saved for {drug}") else: irc.error(f"No doses saved for {nick}") @wrap(["something"]) def amountdosed(self, irc, msg, args, drug): """ shows Aggregate amount in "mg" for ever logged """ num = 0 unit = "" nick = msg.nick if nick in self.db: doselogs = self.db[nick]['doses'] for doselog in doselogs: if doselog["drug"] == drug: for i,c in enumerate(doselog["dose"]): if not c.isdigit(): break num += int(doselog["dose"][:i]) unit = doselog["dose"][i:].lstrip() irc.reply(f"You have dosed a total of {num}{unit} amount of {drug}") else: irc.error(f"No doses saved for {nick}") Class = Tripsit # vim:set shiftwidth=4 softtabstop=4 expandtab textwidth=79: