bitbot-3.11-fork/modules/badges.py
2019-02-13 16:33:28 +00:00

205 lines
7 KiB
Python

import datetime, re
from src import ModuleManager, utils
RE_HUMAN_FORMAT = re.compile(r"(\d\d\d\d)-(\d?\d)-(\d?\d)")
HUMAN_FORMAT_HELP = "year-month-day (e.g. 2018-12-29)"
DATETIME_FORMAT = "%Y-%m-%dT%H:%M:%SZ"
DATE_FORMAT = "%Y-%m-%d"
class Module(ModuleManager.BaseModule):
def _now(self):
return datetime.datetime.utcnow()
def _format_datetime(self, dt: datetime.datetime):
return datetime.datetime.strftime(dt, DATETIME_FORMAT)
def _parse_datetime(self, dt: str):
return datetime.datetime.strptime(dt, DATETIME_FORMAT)
def _parse_date(self, dt: str):
if dt.lower() == "today":
return self._now()
else:
match = RE_HUMAN_FORMAT.match(dt)
if not match:
raise utils.EventError("Invalid date format, please use %s" %
HUMAN_FORMAT_HELP)
return datetime.datetime(
year=int(match.group(1)),
month=int(match.group(2)),
day=int(match.group(3))
)
def _date_str(self, dt: datetime.datetime):
return datetime.datetime.strftime(dt, DATE_FORMAT)
def _round_up_day(self, dt: datetime.datetime):
return dt.date()+datetime.timedelta(days=1)
def _days_since(self, now: datetime.date, dt: datetime.datetime):
return (now-dt.date()).days
def _find_badge(self, badges, badge):
badge_lower = badge.lower()
for badge_name in badges.keys():
if badge_name.lower() == badge_lower:
return badge_name
def _get_badges(self, user):
return user.get_setting("badges", {})
def _set_badges(self, user, badges):
user.set_setting("badges", badges)
def _del_badges(self, user):
user.del_setting("badges")
@utils.hook("received.command.badge", min_args=1)
def badge(self, event):
"""
:help: Show a badge
:usage: <badge>
"""
badge = event["args"]
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
now = self._round_up_day(self._now())
found_badge = self._find_badge(badges, badge)
if found_badge:
dt = self._parse_datetime(badges[found_badge])
days_since = self._days_since(now, dt)
event["stdout"].write("(%s) %s on day %s (%s)" % (
event["user"].nickname, found_badge, days_since,
self._date_str(dt)))
else:
event["stderr"].write("You have no '%s' badge" % badge)
@utils.hook("received.command.badges")
def badges(self, event):
"""
:help: Show all badges for you or a given user
"""
user = event["user"]
if event["args"]:
user = event["server"].get_user(event["args_split"][0])
now = self._round_up_day(self._now())
badges = []
for badge, date in self._get_badges(user).items():
days_since = self._days_since(now, self._parse_datetime(date))
badges.append("%s on day %s" % (
badge, days_since))
event["stdout"].write("Badges for %s: %s" % (
user.nickname, ", ".join(badges)))
@utils.hook("received.command.addbadge", min_args=1)
def add_badge(self, event):
"""
:help: Add a badge with today's date
:usage: <badge>
"""
badge = event["args"]
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
for badge_name in badges.keys():
if badge_name.lower() == badge_lower:
raise utils.EventError("You already have a '%s' badge" % badge)
badges[badge] = self._format_datetime(self._now())
self._set_badges(event["user"], badges)
event["stdout"].write("Added '%s' badge" % badge)
@utils.hook("received.command.removebadge", min_args=1)
def remove_badge(self, event):
"""
:help: Remove a badge
:usage: <badge>
"""
badge = event["args"]
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
found_badge = self._find_badge(badges, badge)
if found_badge:
del badges[found_badge]
self._set_badges(event["user"], badges)
event["stdout"].write("Removed '%s' badge" % found_badge)
else:
event["stderr"].write("You have no '%s' badge" % badge)
@utils.hook("received.command.resetbadge", min_args=1)
def reset_badge(self, event):
"""
:help: Reset a badge to today's date
:usage: <badge>
"""
badge = event["args"]
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
found_badge = self._find_badge(badges, badge)
if found_badge:
badges[found_badge] = self._format_datetime(self._now())
self._set_badges(event["user"], badges)
event["stdout"].write("Reset badge '%s'" % found_badge)
else:
event["stderr"].write("You have no '%s' badge" % badge)
@utils.hook("received.command.updatebadge", min_args=2)
def update_badge(self, event):
"""
:help: Change the date of a badge
:usage: <badge> today|<yyyy-mm-dd>
"""
badge = " ".join(event["args_split"][:-1])
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
found_badge = self._find_badge(badges, badge)
if not found_badge:
raise utils.EventError("You have no '%s' badge" % badge)
dt = self._parse_date(event["args_split"][-1])
badges[found_badge] = self._format_datetime(dt)
self._set_badges(event["user"], badges)
event["stdout"].write("Updated '%s' badge to %s" % (
found_badge, self._date_str(dt)))
@utils.hook("received.command.upsertbadge", min_args=2)
def upsert_badge(self, event):
"""
:help: Add or update a badge
:usage: <badge> today|<yyyy-mm-dd>
"""
badge = " ".join(event["args_split"][:-1])
badge_lower = badge.lower()
badges = self._get_badges(event["user"])
found_badge = self._find_badge(badges, badge)
dt = self._parse_date(event["args_split"][-1])
badges[found_badge or badge] = self._format_datetime(dt)
self._set_badges(event["user"], badges)
add_or_update = "Added" if not found_badge else "Updated"
event["stdout"].write("%s '%s' badge to %s" % (
add_or_update, badge, self._date_str(dt)))
@utils.hook("received.command.clearbadges", min_args=1)
def clear_badges(self, event):
"""
:help: Clear another user's badges
:usage: <nickname>
:permission: clearbadges
"""
user = event["server"].get_user(event["args_split"][0])
badges = self._get_badges(user)
if badges:
self._del_badges(user)
event["stdout"].write("Cleared badges for %s" % user.nickname)
else:
event["stderr"].write("%s has no badges" % user.nickname)