205 lines
7 KiB
Python
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)
|