aboutsummaryrefslogtreecommitdiffstats
path: root/bot
diff options
context:
space:
mode:
authorGravatar Janine vN <[email protected]>2022-11-28 19:04:24 -0500
committerGravatar Janine vN <[email protected]>2022-11-28 19:04:24 -0500
commitc4379ce83e4b6ca9f08895807eb023d0e15cbd84 (patch)
tree346c5df5c3a3809bec88f6987c22b85f11a1fce5 /bot
parentMerge pull request #1154 from python-discord/dependabot/pip/flake8-6.0.0 (diff)
Remove Advent of Code Cog
and related json. This cog was ported to Sir Robin.
Diffstat (limited to 'bot')
-rw-r--r--bot/exts/events/advent_of_code/__init__.py10
-rw-r--r--bot/exts/events/advent_of_code/_caches.py5
-rw-r--r--bot/exts/events/advent_of_code/_cog.py488
-rw-r--r--bot/exts/events/advent_of_code/_helpers.py642
-rw-r--r--bot/exts/events/advent_of_code/views/dayandstarview.py82
-rw-r--r--bot/resources/events/advent_of_code/about.json27
6 files changed, 0 insertions, 1254 deletions
diff --git a/bot/exts/events/advent_of_code/__init__.py b/bot/exts/events/advent_of_code/__init__.py
deleted file mode 100644
index 33c3971a..00000000
--- a/bot/exts/events/advent_of_code/__init__.py
+++ /dev/null
@@ -1,10 +0,0 @@
-from bot.bot import Bot
-
-
-async def setup(bot: Bot) -> None:
- """Set up the Advent of Code extension."""
- # Import the Cog at runtime to prevent side effects like defining
- # RedisCache instances too early.
- from ._cog import AdventOfCode
-
- await bot.add_cog(AdventOfCode(bot))
diff --git a/bot/exts/events/advent_of_code/_caches.py b/bot/exts/events/advent_of_code/_caches.py
deleted file mode 100644
index 32d5394f..00000000
--- a/bot/exts/events/advent_of_code/_caches.py
+++ /dev/null
@@ -1,5 +0,0 @@
-import async_rediscache
-
-leaderboard_counts = async_rediscache.RedisCache(namespace="AOC_leaderboard_counts")
-leaderboard_cache = async_rediscache.RedisCache(namespace="AOC_leaderboard_cache")
-assigned_leaderboard = async_rediscache.RedisCache(namespace="AOC_assigned_leaderboard")
diff --git a/bot/exts/events/advent_of_code/_cog.py b/bot/exts/events/advent_of_code/_cog.py
deleted file mode 100644
index 49140a3f..00000000
--- a/bot/exts/events/advent_of_code/_cog.py
+++ /dev/null
@@ -1,488 +0,0 @@
-import json
-import logging
-from datetime import datetime, timedelta
-from pathlib import Path
-from typing import Optional
-
-import arrow
-import discord
-from async_rediscache import RedisCache
-from discord.ext import commands, tasks
-
-from bot.bot import Bot
-from bot.constants import (
- AdventOfCode as AocConfig, Channels, Client, Colours, Emojis, Month, PYTHON_PREFIX, Roles, WHITELISTED_CHANNELS
-)
-from bot.exts.events.advent_of_code import _helpers
-from bot.exts.events.advent_of_code.views.dayandstarview import AoCDropdownView
-from bot.utils import members
-from bot.utils.decorators import InChannelCheckFailure, in_month, whitelist_override, with_role
-from bot.utils.exceptions import MovedCommandError
-
-log = logging.getLogger(__name__)
-
-AOC_REQUEST_HEADER = {"user-agent": "PythonDiscord AoC Event Bot"}
-
-AOC_WHITELIST_RESTRICTED = WHITELISTED_CHANNELS + (Channels.advent_of_code_commands,)
-
-# Some commands can be run in the regular advent of code channel
-# They aren't spammy and foster discussion
-AOC_WHITELIST = AOC_WHITELIST_RESTRICTED + (Channels.advent_of_code,)
-
-
-class AdventOfCode(commands.Cog):
- """Advent of Code festivities! Ho Ho Ho!"""
-
- # Redis Cache for linking Discord IDs to Advent of Code usernames
- # RedisCache[member_id: aoc_username_string]
- account_links = RedisCache()
-
- # A dict with keys of member_ids to block from getting the role
- # RedisCache[member_id: None]
- completionist_block_list = RedisCache()
-
- def __init__(self, bot: Bot):
- self.bot = bot
-
- self._base_url = f"https://adventofcode.com/{AocConfig.year}"
- self.global_leaderboard_url = f"https://adventofcode.com/{AocConfig.year}/leaderboard"
-
- self.about_aoc_filepath = Path("./bot/resources/events/advent_of_code/about.json")
- self.cached_about_aoc = self._build_about_embed()
-
- notification_coro = _helpers.new_puzzle_notification(self.bot)
- self.notification_task = self.bot.loop.create_task(notification_coro)
- self.notification_task.set_name("Daily AoC Notification")
- self.notification_task.add_done_callback(_helpers.background_task_callback)
-
- status_coro = _helpers.countdown_status(self.bot)
- self.status_task = self.bot.loop.create_task(status_coro)
- self.status_task.set_name("AoC Status Countdown")
- self.status_task.add_done_callback(_helpers.background_task_callback)
-
- # Don't start task while event isn't running
- # self.completionist_task.start()
-
- @tasks.loop(minutes=10.0)
- async def completionist_task(self) -> None:
- """
- Give members who have completed all 50 AoC stars the completionist role.
-
- Runs on a schedule, as defined in the task.loop decorator.
- """
- guild = self.bot.get_guild(Client.guild)
- completionist_role = guild.get_role(Roles.aoc_completionist)
- if completionist_role is None:
- log.warning("Could not find the AoC completionist role; cancelling completionist task.")
- self.completionist_task.cancel()
- return
-
- aoc_name_to_member_id = {
- aoc_name: member_id
- for member_id, aoc_name in await self.account_links.items()
- }
-
- try:
- leaderboard = await _helpers.fetch_leaderboard()
- except _helpers.FetchingLeaderboardFailedError:
- await self.bot.log_to_dev_log("Unable to fetch AoC leaderboard during role sync.")
- return
-
- placement_leaderboard = json.loads(leaderboard["placement_leaderboard"])
-
- for member_aoc_info in placement_leaderboard.values():
- if not member_aoc_info["stars"] == 50:
- # Only give the role to people who have completed all 50 stars
- continue
-
- aoc_name = member_aoc_info["name"] or f"Anonymous #{member_aoc_info['id']}"
-
- member_id = aoc_name_to_member_id.get(aoc_name)
- if not member_id:
- log.debug(f"Could not find member_id for {member_aoc_info['name']}, not giving role.")
- continue
-
- member = await members.get_or_fetch_member(guild, member_id)
- if member is None:
- log.debug(f"Could not find {member_id}, not giving role.")
- continue
-
- if completionist_role in member.roles:
- log.debug(f"{member.name} ({member.mention}) already has the completionist role.")
- continue
-
- if not await self.completionist_block_list.contains(member_id):
- log.debug(f"Giving completionist role to {member.name} ({member.mention}).")
- await members.handle_role_change(member, member.add_roles, completionist_role)
-
- @commands.group(name="adventofcode", aliases=("aoc",))
- @whitelist_override(channels=AOC_WHITELIST)
- async def adventofcode_group(self, ctx: commands.Context) -> None:
- """All of the Advent of Code commands."""
- if not ctx.invoked_subcommand:
- await self.bot.invoke_help_command(ctx)
-
- @with_role(Roles.admins)
- @adventofcode_group.command(
- name="block",
- brief="Block a user from getting the completionist role.",
- )
- async def block_from_role(self, ctx: commands.Context, member: discord.Member) -> None:
- """Block the given member from receiving the AoC completionist role, removing it from them if needed."""
- completionist_role = ctx.guild.get_role(Roles.aoc_completionist)
- if completionist_role in member.roles:
- await member.remove_roles(completionist_role)
-
- await self.completionist_block_list.set(member.id, "sentinel")
- await ctx.send(f":+1: Blocked {member.mention} from getting the AoC completionist role.")
-
- @commands.guild_only()
- @adventofcode_group.command(
- name="subscribe",
- aliases=("sub", "notifications", "notify", "notifs", "unsubscribe", "unsub"),
- help=f"NOTE: This command has been moved to {PYTHON_PREFIX}subscribe",
- )
- @whitelist_override(channels=AOC_WHITELIST)
- async def aoc_subscribe(self, ctx: commands.Context) -> None:
- """
- Deprecated role command.
-
- This command has been moved to bot, and will be removed in the future.
- """
- raise MovedCommandError(f"{PYTHON_PREFIX}subscribe")
-
- @adventofcode_group.command(name="countdown", aliases=("count", "c"), brief="Return time left until next day")
- @whitelist_override(channels=AOC_WHITELIST)
- async def aoc_countdown(self, ctx: commands.Context) -> None:
- """Return time left until next day."""
- if _helpers.is_in_advent():
- tomorrow, _ = _helpers.time_left_to_est_midnight()
- next_day_timestamp = int(tomorrow.timestamp())
-
- await ctx.send(f"Day {tomorrow.day} starts <t:{next_day_timestamp}:R>.")
- return
-
- datetime_now = arrow.now(_helpers.EST)
- # Calculate the delta to this & next year's December 1st to see which one is closest and not in the past
- this_year = arrow.get(datetime(datetime_now.year, 12, 1), _helpers.EST)
- next_year = arrow.get(datetime(datetime_now.year + 1, 12, 1), _helpers.EST)
- deltas = (dec_first - datetime_now for dec_first in (this_year, next_year))
- delta = min(delta for delta in deltas if delta >= timedelta()) # timedelta() gives 0 duration delta
-
- next_aoc_timestamp = int((datetime_now + delta).timestamp())
-
- await ctx.send(
- "The Advent of Code event is not currently running. "
- f"The next event will start <t:{next_aoc_timestamp}:R>."
- )
-
- @adventofcode_group.command(name="about", aliases=("ab", "info"), brief="Learn about Advent of Code")
- @whitelist_override(channels=AOC_WHITELIST)
- async def about_aoc(self, ctx: commands.Context) -> None:
- """Respond with an explanation of all things Advent of Code."""
- await ctx.send(embed=self.cached_about_aoc)
-
- @commands.guild_only()
- @adventofcode_group.command(name="join", aliases=("j",), brief="Learn how to join the leaderboard (via DM)")
- @whitelist_override(channels=AOC_WHITELIST)
- async def join_leaderboard(self, ctx: commands.Context) -> None:
- """DM the user the information for joining the Python Discord leaderboard."""
- current_date = datetime.now()
- allowed_months = (Month.NOVEMBER.value, Month.DECEMBER.value)
- if not (
- current_date.month in allowed_months and current_date.year == AocConfig.year or
- current_date.month == Month.JANUARY.value and current_date.year == AocConfig.year + 1
- ):
- # Only allow joining the leaderboard in the run up to AOC and the January following.
- await ctx.send(f"The Python Discord leaderboard for {current_date.year} is not yet available!")
- return
-
- author = ctx.author
- log.info(f"{author.name} ({author.id}) has requested a PyDis AoC leaderboard code")
-
- if AocConfig.staff_leaderboard_id and any(r.id == Roles.helpers for r in author.roles):
- join_code = AocConfig.leaderboards[AocConfig.staff_leaderboard_id].join_code
- else:
- try:
- join_code = await _helpers.get_public_join_code(author)
- except _helpers.FetchingLeaderboardFailedError:
- await ctx.send(":x: Failed to get join code! Notified maintainers.")
- return
-
- if not join_code:
- log.error(f"Failed to get a join code for user {author} ({author.id})")
- error_embed = discord.Embed(
- title="Unable to get join code",
- description="Failed to get a join code to one of our boards. Please notify staff.",
- colour=discord.Colour.red(),
- )
- await ctx.send(embed=error_embed)
- return
-
- info_str = [
- "To join our leaderboard, follow these steps:",
- "• Log in on https://adventofcode.com",
- "• Head over to https://adventofcode.com/leaderboard/private",
- f"• Use this code `{join_code}` to join the Python Discord leaderboard!",
- ]
- try:
- await author.send("\n".join(info_str))
- except discord.errors.Forbidden:
- log.debug(f"{author.name} ({author.id}) has disabled DMs from server members")
- await ctx.send(f":x: {author.mention}, please (temporarily) enable DMs to receive the join code")
- else:
- await ctx.message.add_reaction(Emojis.envelope)
-
- @in_month(Month.NOVEMBER, Month.DECEMBER, Month.JANUARY)
- @adventofcode_group.command(
- name="link",
- aliases=("connect",),
- brief="Tie your Discord account with your Advent of Code name."
- )
- @whitelist_override(channels=AOC_WHITELIST)
- async def aoc_link_account(self, ctx: commands.Context, *, aoc_name: str = None) -> None:
- """
- Link your Discord Account to your Advent of Code name.
-
- Stored in a Redis Cache with the format of `Discord ID: Advent of Code Name`
- """
- cache_items = await self.account_links.items()
- cache_aoc_names = [value for _, value in cache_items]
-
- if aoc_name:
- # Let's check the current values in the cache to make sure it isn't already tied to a different account
- if aoc_name == await self.account_links.get(ctx.author.id):
- await ctx.reply(f"{aoc_name} is already tied to your account.")
- return
- elif aoc_name in cache_aoc_names:
- log.info(
- f"{ctx.author} ({ctx.author.id}) tried to connect their account to {aoc_name},"
- " but it's already connected to another user."
- )
- await ctx.reply(
- f"{aoc_name} is already tied to another account."
- " Please contact an admin if you believe this is an error."
- )
- return
-
- # Update an existing link
- if old_aoc_name := await self.account_links.get(ctx.author.id):
- log.info(f"Changing link for {ctx.author} ({ctx.author.id}) from {old_aoc_name} to {aoc_name}.")
- await self.account_links.set(ctx.author.id, aoc_name)
- await ctx.reply(f"Your linked account has been changed to {aoc_name}.")
- else:
- # Create a new link
- log.info(f"Linking {ctx.author} ({ctx.author.id}) to account {aoc_name}.")
- await self.account_links.set(ctx.author.id, aoc_name)
- await ctx.reply(f"You have linked your Discord ID to {aoc_name}.")
- else:
- # User has not supplied a name, let's check if they're in the cache or not
- if cache_name := await self.account_links.get(ctx.author.id):
- await ctx.reply(f"You have already linked an Advent of Code account: {cache_name}.")
- else:
- await ctx.reply(
- "You have not linked an Advent of Code account."
- " Please re-run the command with one specified."
- )
-
- @in_month(Month.NOVEMBER, Month.DECEMBER, Month.JANUARY)
- @adventofcode_group.command(
- name="unlink",
- aliases=("disconnect",),
- brief="Tie your Discord account with your Advent of Code name."
- )
- @whitelist_override(channels=AOC_WHITELIST)
- async def aoc_unlink_account(self, ctx: commands.Context) -> None:
- """
- Unlink your Discord ID with your Advent of Code leaderboard name.
-
- Deletes the entry that was Stored in the Redis cache.
- """
- if aoc_cache_name := await self.account_links.get(ctx.author.id):
- log.info(f"Unlinking {ctx.author} ({ctx.author.id}) from Advent of Code account {aoc_cache_name}")
- await self.account_links.delete(ctx.author.id)
- await ctx.reply(f"We have removed the link between your Discord ID and {aoc_cache_name}.")
- else:
- log.info(f"Attempted to unlink {ctx.author} ({ctx.author.id}), but no link was found.")
- await ctx.reply("You don't have an Advent of Code account linked.")
-
- @in_month(Month.DECEMBER, Month.JANUARY)
- @adventofcode_group.command(
- name="dayandstar",
- aliases=("daynstar", "daystar"),
- brief="Get a view that lets you filter the leaderboard by day and star",
- )
- @whitelist_override(channels=AOC_WHITELIST_RESTRICTED)
- async def aoc_day_and_star_leaderboard(
- self,
- ctx: commands.Context,
- maximum_scorers_day_and_star: Optional[int] = 10
- ) -> None:
- """Have the bot send a View that will let you filter the leaderboard by day and star."""
- if maximum_scorers_day_and_star > AocConfig.max_day_and_star_results or maximum_scorers_day_and_star <= 0:
- raise commands.BadArgument(
- f"The maximum number of results you can query is {AocConfig.max_day_and_star_results}"
- )
- async with ctx.typing():
- try:
- leaderboard = await _helpers.fetch_leaderboard()
- except _helpers.FetchingLeaderboardFailedError:
- await ctx.send(":x: Unable to fetch leaderboard!")
- return
- # This is a dictionary that contains solvers in respect of day, and star.
- # e.g. 1-1 means the solvers of the first star of the first day and their completion time
- per_day_and_star = json.loads(leaderboard['leaderboard_per_day_and_star'])
- view = AoCDropdownView(
- day_and_star_data=per_day_and_star,
- maximum_scorers=maximum_scorers_day_and_star,
- original_author=ctx.author
- )
- message = await ctx.send(
- content="Please select a day and a star to filter by!",
- view=view
- )
- await view.wait()
- await message.edit(view=None)
-
- @in_month(Month.DECEMBER, Month.JANUARY)
- @adventofcode_group.command(
- name="leaderboard",
- aliases=("board", "lb"),
- brief="Get a snapshot of the PyDis private AoC leaderboard",
- )
- @whitelist_override(channels=AOC_WHITELIST_RESTRICTED)
- async def aoc_leaderboard(self, ctx: commands.Context, *, aoc_name: Optional[str] = None) -> None:
- """
- Get the current top scorers of the Python Discord Leaderboard.
-
- Additionally you can specify an `aoc_name` that will append the
- specified profile's personal stats to the top of the leaderboard
- """
- # Strip quotes from the AoC username if needed (e.g. "My Name" -> My Name)
- # This is to keep compatibility with those already used to wrapping the AoC name in quotes
- # Note: only strips one layer of quotes to allow names with quotes at the start and end
- # e.g. ""My Name"" -> "My Name"
- if aoc_name and aoc_name.startswith('"') and aoc_name.endswith('"'):
- aoc_name = aoc_name[1:-1]
-
- # Check if an advent of code account is linked in the Redis Cache if aoc_name is not given
- if (aoc_cache_name := await self.account_links.get(ctx.author.id)) and aoc_name is None:
- aoc_name = aoc_cache_name
-
- async with ctx.typing():
- try:
- leaderboard = await _helpers.fetch_leaderboard(self_placement_name=aoc_name)
- except _helpers.FetchingLeaderboardFailedError:
- await ctx.send(":x: Unable to fetch leaderboard!")
- return
-
- number_of_participants = leaderboard["number_of_participants"]
-
- top_count = min(AocConfig.leaderboard_displayed_members, number_of_participants)
- self_placement_header = " (and your personal stats compared to the top 10)" if aoc_name else ""
- header = f"Here's our current top {top_count}{self_placement_header}! {Emojis.christmas_tree * 3}"
- table = "```\n" \
- f"{leaderboard['placement_leaderboard'] if aoc_name else leaderboard['top_leaderboard']}" \
- "\n```"
- info_embed = _helpers.get_summary_embed(leaderboard)
-
- await ctx.send(content=f"{header}\n\n{table}", embed=info_embed)
- return
-
- @in_month(Month.DECEMBER, Month.JANUARY)
- @adventofcode_group.command(
- name="global",
- aliases=("globalboard", "gb"),
- brief="Get a link to the global leaderboard",
- )
- @whitelist_override(channels=AOC_WHITELIST_RESTRICTED)
- async def aoc_global_leaderboard(self, ctx: commands.Context) -> None:
- """Get a link to the global Advent of Code leaderboard."""
- url = self.global_leaderboard_url
- global_leaderboard = discord.Embed(
- title="Advent of Code — Global Leaderboard",
- description=f"You can find the global leaderboard [here]({url})."
- )
- global_leaderboard.set_thumbnail(url=_helpers.AOC_EMBED_THUMBNAIL)
- await ctx.send(embed=global_leaderboard)
-
- @adventofcode_group.command(
- name="stats",
- aliases=("dailystats", "ds"),
- brief="Get daily statistics for the Python Discord leaderboard"
- )
- @whitelist_override(channels=AOC_WHITELIST_RESTRICTED)
- async def private_leaderboard_daily_stats(self, ctx: commands.Context) -> None:
- """Send an embed with daily completion statistics for the Python Discord leaderboard."""
- try:
- leaderboard = await _helpers.fetch_leaderboard()
- except _helpers.FetchingLeaderboardFailedError:
- await ctx.send(":x: Can't fetch leaderboard for stats right now!")
- return
-
- # The daily stats are serialized as JSON as they have to be cached in Redis
- daily_stats = json.loads(leaderboard["daily_stats"])
- async with ctx.typing():
- lines = ["Day ⭐ ⭐⭐ | %⭐ %⭐⭐\n================================"]
- for day, stars in daily_stats.items():
- star_one = stars["star_one"]
- star_two = stars["star_two"]
- p_star_one = star_one / leaderboard["number_of_participants"]
- p_star_two = star_two / leaderboard["number_of_participants"]
- lines.append(
- f"{day:>2}) {star_one:>4} {star_two:>4} | {p_star_one:>7.2%} {p_star_two:>7.2%}"
- )
- table = "\n".join(lines)
- info_embed = _helpers.get_summary_embed(leaderboard)
- await ctx.send(f"```\n{table}\n```", embed=info_embed)
-
- @with_role(Roles.admins)
- @adventofcode_group.command(
- name="refresh",
- aliases=("fetch",),
- brief="Force a refresh of the leaderboard cache.",
- )
- async def refresh_leaderboard(self, ctx: commands.Context) -> None:
- """
- Force a refresh of the leaderboard cache.
-
- Note: This should be used sparingly, as we want to prevent sending too
- many requests to the Advent of Code server.
- """
- async with ctx.typing():
- try:
- await _helpers.fetch_leaderboard(invalidate_cache=True)
- except _helpers.FetchingLeaderboardFailedError:
- await ctx.send(":x: Something went wrong while trying to refresh the cache!")
- else:
- await ctx.send("\N{OK Hand Sign} Refreshed leaderboard cache!")
-
- def cog_unload(self) -> None:
- """Cancel season-related tasks on cog unload."""
- log.debug("Unloading the cog and canceling the background task.")
- self.notification_task.cancel()
- self.status_task.cancel()
- self.completionist_task.cancel()
-
- def _build_about_embed(self) -> discord.Embed:
- """Build and return the informational "About AoC" embed from the resources file."""
- embed_fields = json.loads(self.about_aoc_filepath.read_text("utf8"))
-
- about_embed = discord.Embed(
- title=self._base_url,
- colour=Colours.soft_green,
- url=self._base_url,
- timestamp=datetime.utcnow()
- )
- about_embed.set_author(name="Advent of Code", url=self._base_url)
- for field in embed_fields:
- about_embed.add_field(**field)
-
- about_embed.set_footer(text="Last Updated")
- return about_embed
-
- async def cog_command_error(self, ctx: commands.Context, error: Exception) -> None:
- """Custom error handler if an advent of code command was posted in the wrong channel."""
- if isinstance(error, InChannelCheckFailure):
- await ctx.send(f":x: Please use <#{Channels.advent_of_code_commands}> for aoc commands instead.")
- error.handled = True
diff --git a/bot/exts/events/advent_of_code/_helpers.py b/bot/exts/events/advent_of_code/_helpers.py
deleted file mode 100644
index abd80b77..00000000
--- a/bot/exts/events/advent_of_code/_helpers.py
+++ /dev/null
@@ -1,642 +0,0 @@
-import asyncio
-import collections
-import datetime
-import json
-import logging
-import math
-import operator
-from typing import Any, Optional
-
-import aiohttp
-import arrow
-import discord
-from discord.ext import commands
-
-from bot.bot import Bot
-from bot.constants import AdventOfCode, Channels, Colours
-from bot.exts.events.advent_of_code import _caches
-
-log = logging.getLogger(__name__)
-
-PASTE_URL = "https://paste.pythondiscord.com/documents"
-RAW_PASTE_URL_TEMPLATE = "https://paste.pythondiscord.com/raw/{key}"
-
-# Base API URL for Advent of Code Private Leaderboards
-AOC_API_URL = "https://adventofcode.com/{year}/leaderboard/private/view/{leaderboard_id}.json"
-AOC_REQUEST_HEADER = {"user-agent": "PythonDiscord AoC Event Bot"}
-
-# Leaderboard Line Template
-AOC_TABLE_TEMPLATE = "{rank: >4} | {name:25.25} | {score: >5} | {stars}"
-HEADER = AOC_TABLE_TEMPLATE.format(rank="", name="Name", score="Score", stars="⭐, ⭐⭐")
-HEADER = f"{HEADER}\n{'-' * (len(HEADER) + 2)}"
-HEADER_LINES = len(HEADER.splitlines())
-TOP_LEADERBOARD_LINES = HEADER_LINES + AdventOfCode.leaderboard_displayed_members
-
-# Keys that need to be set for a cached leaderboard
-REQUIRED_CACHE_KEYS = (
- "full_leaderboard",
- "top_leaderboard",
- "full_leaderboard_url",
- "leaderboard_fetched_at",
- "number_of_participants",
- "daily_stats",
-)
-
-AOC_EMBED_THUMBNAIL = (
- "https://raw.githubusercontent.com/python-discord"
- "/branding/main/seasonal/christmas/server_icons/festive_256.gif"
-)
-
-# Create an easy constant for the EST timezone
-EST = "America/New_York"
-
-# Step size for the challenge countdown status
-COUNTDOWN_STEP = 60 * 5
-
-# Create namedtuple that combines a participant's name and their completion
-# time for a specific star. We're going to use this later to order the results
-# for each star to compute the rank score.
-StarResult = collections.namedtuple("StarResult", "member_id completion_time")
-
-
-class UnexpectedRedirect(aiohttp.ClientError):
- """Raised when an unexpected redirect was detected."""
-
-
-class UnexpectedResponseStatus(aiohttp.ClientError):
- """Raised when an unexpected redirect was detected."""
-
-
-class FetchingLeaderboardFailedError(Exception):
- """Raised when one or more leaderboards could not be fetched at all."""
-
-
-def _format_leaderboard_line(rank: int, data: dict[str, Any], *, is_author: bool) -> str:
- """
- Build a string representing a line of the leaderboard.
-
- Parameters:
- rank:
- Rank in the leaderboard of this entry.
-
- data:
- Mapping with entry information.
-
- Keyword arguments:
- is_author:
- Whether to address the name displayed in the returned line
- personally.
-
- Returns:
- A formatted line for the leaderboard.
- """
- return AOC_TABLE_TEMPLATE.format(
- rank=rank,
- name=data['name'] if not is_author else f"(You) {data['name']}",
- score=str(data['score']),
- stars=f"({data['star_1']}, {data['star_2']})"
- )
-
-
-def leaderboard_sorting_function(entry: tuple[str, dict]) -> tuple[int, int]:
- """
- Provide a sorting value for our leaderboard.
-
- The leaderboard is sorted primarily on the score someone has received and
- secondary on the number of stars someone has completed.
- """
- result = entry[1]
- return result["score"], result["star_2"] + result["star_1"]
-
-
-def _parse_raw_leaderboard_data(raw_leaderboard_data: dict) -> dict:
- """
- Parse the leaderboard data received from the AoC website.
-
- The data we receive from AoC is structured by member, not by day/star. This
- means that we need to "transpose" the data to a per star structure in order
- to calculate the rank scores each individual should get.
-
- As we need our data both "per participant" as well as "per day", we return
- the parsed and analyzed data in both formats.
- """
- # We need to get an aggregate of completion times for each star of each day,
- # instead of per participant to compute the rank scores. This dictionary will
- # provide such a transposed dataset.
- star_results = collections.defaultdict(list)
-
- # As we're already iterating over the participants, we can record the number of
- # first stars and second stars they've achieved right here and now. This means
- # we won't have to iterate over the participants again later.
- leaderboard = {}
-
- # The data we get from the AoC website is structured by member, not by day/star,
- # which means we need to iterate over the members to transpose the data to a per
- # star view. We need that per star view to compute rank scores per star.
- per_day_star_stats = collections.defaultdict(list)
- for member in raw_leaderboard_data.values():
- name = member["name"] if member["name"] else f"Anonymous #{member['id']}"
- member_id = member["id"]
- leaderboard[member_id] = {"name": name, "score": 0, "star_1": 0, "star_2": 0}
-
- # Iterate over all days for this participant
- for day, stars in member["completion_day_level"].items():
- # Iterate over the complete stars for this day for this participant
- for star, data in stars.items():
- # Record completion of this star for this individual
- leaderboard[member_id][f"star_{star}"] += 1
-
- # Record completion datetime for this participant for this day/star
- completion_time = datetime.datetime.fromtimestamp(int(data["get_star_ts"]))
- star_results[(day, star)].append(
- StarResult(member_id=member_id, completion_time=completion_time)
- )
- per_day_star_stats[f"{day}-{star}"].append(
- {'completion_time': int(data["get_star_ts"]), 'member_name': name}
- )
- for key in per_day_star_stats:
- per_day_star_stats[key] = sorted(per_day_star_stats[key], key=operator.itemgetter('completion_time'))
-
- # Now that we have a transposed dataset that holds the completion time of all
- # participants per star, we can compute the rank-based scores each participant
- # should get for that star.
- max_score = len(leaderboard)
- for (day, _star), results in star_results.items():
- # If this day should not count in the ranking, skip it.
- if day in AdventOfCode.ignored_days:
- continue
-
- sorted_result = sorted(results, key=operator.attrgetter("completion_time"))
- for rank, star_result in enumerate(sorted_result):
- leaderboard[star_result.member_id]["score"] += max_score - rank
-
- # Since dictionaries now retain insertion order, let's use that
- sorted_leaderboard = dict(
- sorted(leaderboard.items(), key=leaderboard_sorting_function, reverse=True)
- )
-
- # Create summary stats for the stars completed for each day of the event.
- daily_stats = {}
- for day in range(1, 26):
- day = str(day)
- star_one = len(star_results.get((day, "1"), []))
- star_two = len(star_results.get((day, "2"), []))
- # By using a dictionary instead of namedtuple here, we can serialize
- # this data to JSON in order to cache it in Redis.
- daily_stats[day] = {"star_one": star_one, "star_two": star_two}
-
- return {"daily_stats": daily_stats, "leaderboard": sorted_leaderboard, 'per_day_and_star': per_day_star_stats}
-
-
-def _format_leaderboard(leaderboard: dict[str, dict], self_placement_name: str = None) -> str:
- """Format the leaderboard using the AOC_TABLE_TEMPLATE."""
- leaderboard_lines = [HEADER]
- self_placement_exists = False
- for rank, data in enumerate(leaderboard.values(), start=1):
- if self_placement_name and data["name"].lower() == self_placement_name.lower():
- leaderboard_lines.insert(
- 1,
- AOC_TABLE_TEMPLATE.format(
- rank=rank,
- name=f"(You) {data['name']}",
- score=str(data["score"]),
- stars=f"({data['star_1']}, {data['star_2']})"
- )
- )
- self_placement_exists = True
- continue
- leaderboard_lines.append(
- AOC_TABLE_TEMPLATE.format(
- rank=rank,
- name=data["name"],
- score=str(data["score"]),
- stars=f"({data['star_1']}, {data['star_2']})"
- )
- )
- if self_placement_name and not self_placement_exists:
- raise commands.BadArgument(
- "Sorry, your profile does not exist in this leaderboard."
- "\n\n"
- "To join our leaderboard, run the command `.aoc join`."
- " If you've joined recently, please wait up to 30 minutes for our leaderboard to refresh."
- )
- return "\n".join(leaderboard_lines)
-
-
-async def _leaderboard_request(url: str, board: str, cookies: dict) -> dict[str, Any]:
- """Make a leaderboard request using the specified session cookie."""
- async with aiohttp.request("GET", url, headers=AOC_REQUEST_HEADER, cookies=cookies) as resp:
- # The Advent of Code website redirects silently with a 200 response if a
- # session cookie has expired, is invalid, or was not provided.
- if str(resp.url) != url:
- log.error(f"Fetching leaderboard `{board}` failed! Check the session cookie.")
- raise UnexpectedRedirect(f"redirected unexpectedly to {resp.url} for board `{board}`")
-
- # Every status other than `200` is unexpected, not only 400+
- if not resp.status == 200:
- log.error(f"Unexpected response `{resp.status}` while fetching leaderboard `{board}`")
- raise UnexpectedResponseStatus(f"status `{resp.status}`")
-
- return await resp.json()
-
-
-async def _fetch_leaderboard_data() -> dict[str, Any]:
- """Fetch data for all leaderboards and return a pooled result."""
- year = AdventOfCode.year
-
- # We'll make our requests one at a time to not flood the AoC website with
- # up to six simultaneous requests. This may take a little longer, but it
- # does avoid putting unnecessary stress on the Advent of Code website.
-
- # Container to store the raw data of each leaderboard
- participants = {}
- for leaderboard in AdventOfCode.leaderboards.values():
- leaderboard_url = AOC_API_URL.format(year=year, leaderboard_id=leaderboard.id)
-
- # Two attempts, one with the original session cookie and one with the fallback session
- for attempt in range(1, 3):
- log.debug(f"Attempting to fetch leaderboard `{leaderboard.id}` ({attempt}/2)")
- cookies = {"session": leaderboard.session}
- try:
- raw_data = await _leaderboard_request(leaderboard_url, leaderboard.id, cookies)
- except UnexpectedRedirect:
- if cookies["session"] == AdventOfCode.fallback_session:
- log.error("It seems like the fallback cookie has expired!")
- raise FetchingLeaderboardFailedError from None
-
- # If we're here, it means that the original session did not
- # work. Let's fall back to the fallback session.
- leaderboard.use_fallback_session = True
- continue
- except aiohttp.ClientError:
- # Don't retry, something unexpected is wrong and it may not be the session.
- raise FetchingLeaderboardFailedError from None
- else:
- # Get the participants and store their current count.
- board_participants = raw_data["members"]
- await _caches.leaderboard_counts.set(leaderboard.id, len(board_participants))
- participants.update(board_participants)
- break
- else:
- log.error(f"reached 'unreachable' state while fetching board `{leaderboard.id}`.")
- raise FetchingLeaderboardFailedError
-
- log.info(f"Fetched leaderboard information for {len(participants)} participants")
- return participants
-
-
-async def _upload_leaderboard(leaderboard: str) -> str:
- """Upload the full leaderboard to our paste service and return the URL."""
- async with aiohttp.request("POST", PASTE_URL, data=leaderboard) as resp:
- try:
- resp_json = await resp.json()
- except Exception:
- log.exception("Failed to upload full leaderboard to paste service")
- return ""
-
- if "key" in resp_json:
- return RAW_PASTE_URL_TEMPLATE.format(key=resp_json["key"])
-
- log.error(f"Unexpected response from paste service while uploading leaderboard {resp_json}")
- return ""
-
-
-def _get_top_leaderboard(full_leaderboard: str) -> str:
- """Get the leaderboard up to the maximum specified entries."""
- return "\n".join(full_leaderboard.splitlines()[:TOP_LEADERBOARD_LINES])
-
-
-@_caches.leaderboard_cache.atomic_transaction
-async def fetch_leaderboard(invalidate_cache: bool = False, self_placement_name: str = None) -> dict:
- """
- Get the current Python Discord combined leaderboard.
-
- The leaderboard is cached and only fetched from the API if the current data
- is older than the lifetime set in the constants. To prevent multiple calls
- to this function fetching new leaderboard information in case of a cache
- miss, this function is locked to one call at a time using a decorator.
- """
- cached_leaderboard = await _caches.leaderboard_cache.to_dict()
- # Check if the cached leaderboard contains everything we expect it to. If it
- # does not, this probably means the cache has not been created yet or has
- # expired in Redis. This check also accounts for a malformed cache.
- if invalidate_cache or any(key not in cached_leaderboard for key in REQUIRED_CACHE_KEYS):
- log.info("No leaderboard cache available, fetching leaderboards...")
- # Fetch the raw data
- raw_leaderboard_data = await _fetch_leaderboard_data()
-
- # Parse it to extract "per star, per day" data and participant scores
- parsed_leaderboard_data = _parse_raw_leaderboard_data(raw_leaderboard_data)
-
- leaderboard = parsed_leaderboard_data["leaderboard"]
- number_of_participants = len(leaderboard)
- formatted_leaderboard = _format_leaderboard(leaderboard)
- full_leaderboard_url = await _upload_leaderboard(formatted_leaderboard)
- leaderboard_fetched_at = datetime.datetime.now(datetime.timezone.utc).isoformat()
-
- cached_leaderboard = {
- "placement_leaderboard": json.dumps(raw_leaderboard_data),
- "full_leaderboard": formatted_leaderboard,
- "top_leaderboard": _get_top_leaderboard(formatted_leaderboard),
- "full_leaderboard_url": full_leaderboard_url,
- "leaderboard_fetched_at": leaderboard_fetched_at,
- "number_of_participants": number_of_participants,
- "daily_stats": json.dumps(parsed_leaderboard_data["daily_stats"]),
- "leaderboard_per_day_and_star": json.dumps(parsed_leaderboard_data["per_day_and_star"])
- }
-
- # Store the new values in Redis
- await _caches.leaderboard_cache.update(cached_leaderboard)
-
- # Set an expiry on the leaderboard RedisCache
- with await _caches.leaderboard_cache._get_pool_connection() as connection:
- await connection.expire(
- _caches.leaderboard_cache.namespace,
- AdventOfCode.leaderboard_cache_expiry_seconds
- )
- if self_placement_name:
- formatted_placement_leaderboard = _parse_raw_leaderboard_data(
- json.loads(cached_leaderboard["placement_leaderboard"])
- )["leaderboard"]
- cached_leaderboard["placement_leaderboard"] = _get_top_leaderboard(
- _format_leaderboard(formatted_placement_leaderboard, self_placement_name=self_placement_name)
- )
- return cached_leaderboard
-
-
-def get_summary_embed(leaderboard: dict) -> discord.Embed:
- """Get an embed with the current summary stats of the leaderboard."""
- leaderboard_url = leaderboard["full_leaderboard_url"]
- refresh_minutes = AdventOfCode.leaderboard_cache_expiry_seconds // 60
- refreshed_unix = int(datetime.datetime.fromisoformat(leaderboard["leaderboard_fetched_at"]).timestamp())
-
- aoc_embed = discord.Embed(colour=Colours.soft_green)
-
- aoc_embed.description = (
- f"The leaderboard is refreshed every {refresh_minutes} minutes.\n"
- f"Last Updated: <t:{refreshed_unix}:t>"
- )
- aoc_embed.add_field(
- name="Number of Participants",
- value=leaderboard["number_of_participants"],
- inline=True,
- )
- if leaderboard_url:
- aoc_embed.add_field(
- name="Full Leaderboard",
- value=f"[Python Discord Leaderboard]({leaderboard_url})",
- inline=True,
- )
- aoc_embed.set_author(name="Advent of Code", url=leaderboard_url)
- aoc_embed.set_thumbnail(url=AOC_EMBED_THUMBNAIL)
-
- return aoc_embed
-
-
-async def get_public_join_code(author: discord.Member) -> Optional[str]:
- """
- Get the join code for one of the non-staff leaderboards.
-
- If a user has previously requested a join code and their assigned board
- hasn't filled up yet, we'll return the same join code to prevent them from
- getting join codes for multiple boards.
- """
- # Make sure to fetch new leaderboard information if the cache is older than
- # 30 minutes. While this still means that there could be a discrepancy
- # between the current leaderboard state and the numbers we have here, this
- # should work fairly well given the buffer of slots that we have.
- await fetch_leaderboard()
- previously_assigned_board = await _caches.assigned_leaderboard.get(author.id)
- current_board_counts = await _caches.leaderboard_counts.to_dict()
-
- # Remove the staff board from the current board counts as it should be ignored.
- current_board_counts.pop(AdventOfCode.staff_leaderboard_id, None)
-
- # If this user has already received a join code, we'll give them the
- # exact same one to prevent them from joining multiple boards and taking
- # up multiple slots.
- if previously_assigned_board:
- # Check if their previously assigned board still has room for them
- if current_board_counts.get(previously_assigned_board, 0) < 200:
- log.info(f"{author} ({author.id}) was already assigned to a board with open slots.")
- return AdventOfCode.leaderboards[previously_assigned_board].join_code
-
- log.info(
- f"User {author} ({author.id}) previously received the join code for "
- f"board `{previously_assigned_board}`, but that board's now full. "
- "Assigning another board to this user."
- )
-
- # If we don't have the current board counts cached, let's force fetching a new cache
- if not current_board_counts:
- log.warning("Leaderboard counts were missing from the cache unexpectedly!")
- await fetch_leaderboard(invalidate_cache=True)
- current_board_counts = await _caches.leaderboard_counts.to_dict()
-
- # Find the board with the current lowest participant count. As we can't
- best_board, _count = min(current_board_counts.items(), key=operator.itemgetter(1))
-
- if current_board_counts.get(best_board, 0) >= 200:
- log.warning(f"User {author} `{author.id}` requested a join code, but all boards are full!")
- return
-
- log.info(f"Assigning user {author} ({author.id}) to board `{best_board}`")
- await _caches.assigned_leaderboard.set(author.id, best_board)
-
- # Return the join code for this board
- return AdventOfCode.leaderboards[best_board].join_code
-
-
-def is_in_advent() -> bool:
- """
- Check if we're currently on an Advent of Code day, excluding 25 December.
-
- This helper function is used to check whether or not a feature that prepares
- something for the next Advent of Code challenge should run. As the puzzle
- published on the 25th is the last puzzle, this check excludes that date.
- """
- return arrow.now(EST).day in range(1, 25) and arrow.now(EST).month == 12
-
-
-def time_left_to_est_midnight() -> tuple[datetime.datetime, datetime.timedelta]:
- """Calculate the amount of time left until midnight EST/UTC-5."""
- # Change all time properties back to 00:00
- todays_midnight = arrow.now(EST).replace(
- microsecond=0,
- second=0,
- minute=0,
- hour=0
- )
-
- # We want tomorrow so add a day on
- tomorrow = todays_midnight + datetime.timedelta(days=1)
-
- # Calculate the timedelta between the current time and midnight
- return tomorrow, tomorrow - arrow.now(EST)
-
-
-async def wait_for_advent_of_code(*, hours_before: int = 1) -> None:
- """
- Wait for the Advent of Code event to start.
-
- This function returns `hours_before` (default: 1) the Advent of Code
- actually starts. This allows functions to schedule and execute code that
- needs to run before the event starts.
-
- If the event has already started, this function returns immediately.
-
- Note: The "next Advent of Code" is determined based on the current value
- of the `AOC_YEAR` environment variable. This allows callers to exit early
- if we're already past the Advent of Code edition the bot is currently
- configured for.
- """
- start = arrow.get(datetime.datetime(AdventOfCode.year, 12, 1), EST)
- target = start - datetime.timedelta(hours=hours_before)
- now = arrow.now(EST)
-
- # If we've already reached or passed to target, we
- # simply return immediately.
- if now >= target:
- return
-
- delta = target - now
- await asyncio.sleep(delta.total_seconds())
-
-
-async def countdown_status(bot: Bot) -> None:
- """
- Add the time until the next challenge is published to the bot's status.
-
- This function sleeps until 2 hours before the event and exists one hour
- after the last challenge has been published. It will not start up again
- automatically for next year's event, as it will wait for the environment
- variable AOC_YEAR to be updated.
-
- This ensures that the task will only start sleeping again once the next
- event approaches and we're making preparations for that event.
- """
- log.debug("Initializing status countdown task.")
- # We wait until 2 hours before the event starts. Then we
- # set our first countdown status.
- await wait_for_advent_of_code(hours_before=2)
-
- # Log that we're going to start with the countdown status.
- log.info("The Advent of Code has started or will start soon, starting countdown status.")
-
- # Calculate when the task needs to stop running. To prevent the task from
- # sleeping for the entire year, it will only wait in the currently
- # configured year. This means that the task will only start hibernating once
- # we start preparing the next event by changing environment variables.
- last_challenge = arrow.get(datetime.datetime(AdventOfCode.year, 12, 25), EST)
- end = last_challenge + datetime.timedelta(hours=1)
-
- while arrow.now(EST) < end:
- _, time_left = time_left_to_est_midnight()
-
- aligned_seconds = int(math.ceil(time_left.seconds / COUNTDOWN_STEP)) * COUNTDOWN_STEP
- hours, minutes = aligned_seconds // 3600, aligned_seconds // 60 % 60
-
- if aligned_seconds == 0:
- playing = "right now!"
- elif aligned_seconds == COUNTDOWN_STEP:
- playing = f"in less than {minutes} minutes"
- elif hours == 0:
- playing = f"in {minutes} minutes"
- elif hours == 23:
- playing = f"since {60 - minutes} minutes ago"
- else:
- playing = f"in {hours} hours and {minutes} minutes"
-
- log.trace(f"Changing presence to {playing!r}")
- # Status will look like "Playing in 5 hours and 30 minutes"
- await bot.change_presence(activity=discord.Game(playing))
-
- # Sleep until next aligned time or a full step if already aligned
- delay = time_left.seconds % COUNTDOWN_STEP or COUNTDOWN_STEP
- log.trace(f"The countdown status task will sleep for {delay} seconds.")
- await asyncio.sleep(delay)
-
-
-async def new_puzzle_notification(bot: Bot) -> None:
- """
- Announce the release of a new Advent of Code puzzle.
-
- This background task hibernates until just before the Advent of Code starts
- and will then start announcing puzzles as they are published. After the
- event has finished, this task will terminate.
- """
- # We wake up one hour before the event starts to prepare the announcement
- # of the release of the first puzzle.
- await wait_for_advent_of_code(hours_before=1)
-
- log.info("The Advent of Code has started or will start soon, waking up notification task.")
-
- aoc_channel = bot.get_channel(Channels.advent_of_code)
- aoc_role = aoc_channel.guild.get_role(AdventOfCode.role_id)
-
- if not aoc_channel:
- log.error("Could not find the AoC channel to send notification in")
- return
-
- if not aoc_role:
- log.error("Could not find the AoC role to announce the daily puzzle")
- return
-
- # The last event day is 25 December, so we only have to schedule
- # a reminder if the current day is before 25 December.
- end = arrow.get(datetime.datetime(AdventOfCode.year, 12, 25), EST)
- while arrow.now(EST) < end:
- log.trace("Started puzzle notification loop.")
- tomorrow, time_left = time_left_to_est_midnight()
-
- # Use `total_seconds` to get the time left in fractional seconds This
- # should wake us up very close to the target. As a safe guard, the sleep
- # duration is padded with 0.1 second to make sure we wake up after
- # midnight.
- sleep_seconds = time_left.total_seconds() + 0.1
- log.trace(f"The puzzle notification task will sleep for {sleep_seconds} seconds")
- await asyncio.sleep(sleep_seconds)
-
- puzzle_url = f"https://adventofcode.com/{AdventOfCode.year}/day/{tomorrow.day}"
-
- # Check if the puzzle is already available to prevent our members from spamming
- # the puzzle page before it's available by making a small HEAD request.
- for retry in range(1, 5):
- log.debug(f"Checking if the puzzle is already available (attempt {retry}/4)")
- async with bot.http_session.head(puzzle_url, raise_for_status=False) as resp:
- if resp.status == 200:
- log.debug("Puzzle is available; let's send an announcement message.")
- break
- log.debug(f"The puzzle is not yet available (status={resp.status})")
- await asyncio.sleep(10)
- else:
- log.error(
- "The puzzle does does not appear to be available "
- "at this time, canceling announcement"
- )
- break
-
- await aoc_channel.send(
- f"{aoc_role.mention} Good morning! Day {tomorrow.day} is ready to be attempted. "
- f"View it online now at {puzzle_url}. Good luck!",
- allowed_mentions=discord.AllowedMentions(
- everyone=False,
- users=False,
- roles=[aoc_role],
- )
- )
-
- # Ensure that we don't send duplicate announcements by sleeping to well
- # over midnight. This means we're certain to calculate the time to the
- # next midnight at the top of the loop.
- await asyncio.sleep(120)
-
-
-def background_task_callback(task: asyncio.Task) -> None:
- """Check if the finished background task failed to make sure we log errors."""
- if task.cancelled():
- log.info(f"Background task `{task.get_name()}` was cancelled.")
- elif exception := task.exception():
- log.error(f"Background task `{task.get_name()}` failed:", exc_info=exception)
- else:
- log.info(f"Background task `{task.get_name()}` exited normally.")
diff --git a/bot/exts/events/advent_of_code/views/dayandstarview.py b/bot/exts/events/advent_of_code/views/dayandstarview.py
deleted file mode 100644
index f0ebc803..00000000
--- a/bot/exts/events/advent_of_code/views/dayandstarview.py
+++ /dev/null
@@ -1,82 +0,0 @@
-from datetime import datetime
-
-import discord
-
-AOC_DAY_AND_STAR_TEMPLATE = "{rank: >4} | {name:25.25} | {completion_time: >10}"
-
-
-class AoCDropdownView(discord.ui.View):
- """Interactive view to filter AoC stats by Day and Star."""
-
- def __init__(self, original_author: discord.Member, day_and_star_data: dict[str: dict], maximum_scorers: int):
- super().__init__()
- self.day = 0
- self.star = 0
- self.data = day_and_star_data
- self.maximum_scorers = maximum_scorers
- self.original_author = original_author
-
- def generate_output(self) -> str:
- """
- Generates a formatted codeblock with AoC statistics based on the currently selected day and star.
-
- Optionally, when the requested day and star data does not exist yet it returns an error message.
- """
- header = AOC_DAY_AND_STAR_TEMPLATE.format(
- rank="Rank",
- name="Name", completion_time="Completion time (UTC)"
- )
- lines = [f"{header}\n{'-' * (len(header) + 2)}"]
- if not (day_and_star_data := self.data.get(f"{self.day}-{self.star}")):
- return ":x: The requested data for the specified day and star does not exist yet."
- for rank, scorer in enumerate(day_and_star_data[:self.maximum_scorers]):
- time_data = datetime.fromtimestamp(scorer['completion_time']).strftime("%I:%M:%S %p")
- lines.append(AOC_DAY_AND_STAR_TEMPLATE.format(
- datastamp="",
- rank=rank + 1,
- name=scorer['member_name'],
- completion_time=time_data)
- )
- joined_lines = "\n".join(lines)
- return f"Statistics for Day: {self.day}, Star: {self.star}.\n ```\n{joined_lines}\n```"
-
- async def interaction_check(self, interaction: discord.Interaction) -> bool:
- """Global check to ensure that the interacting user is the user who invoked the command originally."""
- if interaction.user != self.original_author:
- await interaction.response.send_message(
- ":x: You can't interact with someone else's response. Please run the command yourself!",
- ephemeral=True
- )
- return False
- return True
-
- @discord.ui.select(
- placeholder="Day",
- options=[discord.SelectOption(label=str(i)) for i in range(1, 26)],
- custom_id="day_select"
- )
- async def day_select(self, _: discord.Interaction, select: discord.ui.Select) -> None:
- """Dropdown to choose a Day of the AoC."""
- self.day = select.values[0]
-
- @discord.ui.select(
- placeholder="Star",
- options=[discord.SelectOption(label=str(i)) for i in range(1, 3)],
- custom_id="star_select"
- )
- async def star_select(self, _: discord.Interaction, select: discord.ui.Select) -> None:
- """Dropdown to choose either the first or the second star."""
- self.star = select.values[0]
-
- @discord.ui.button(label="Fetch", style=discord.ButtonStyle.blurple)
- async def fetch(self, interaction: discord.Interaction, _: discord.ui.Button) -> None:
- """Button that fetches the statistics based on the dropdown values."""
- if self.day == 0 or self.star == 0:
- await interaction.response.send_message(
- "You have to select a value from both of the dropdowns!",
- ephemeral=True
- )
- else:
- await interaction.response.edit_message(content=self.generate_output())
- self.day = 0
- self.star = 0
diff --git a/bot/resources/events/advent_of_code/about.json b/bot/resources/events/advent_of_code/about.json
deleted file mode 100644
index dd0fe59a..00000000
--- a/bot/resources/events/advent_of_code/about.json
+++ /dev/null
@@ -1,27 +0,0 @@
-[
- {
- "name": "What is Advent of Code?",
- "value": "Advent of Code (AoC) is a series of small programming puzzles for a variety of skill levels, run every year during the month of December.\n\nThey are self-contained and are just as appropriate for an expert who wants to stay sharp as they are for a beginner who is just learning to code. Each puzzle calls upon different skills and has two parts that build on a theme.",
- "inline": false
- },
- {
- "name": "How do I sign up?",
- "value": "Sign up with one of these services:",
- "inline": true
- },
- {
- "name": "Auth Services",
- "value": "GitHub\nGoogle\nTwitter\nReddit",
- "inline": true
- },
- {
- "name": "How does scoring work?",
- "value": "For the [global leaderboard](https://adventofcode.com/leaderboard), the first person to get a star first gets 100 points, the second person gets 99 points, and so on down to 1 point at 100th place.\n\nFor private leaderboards, the first person to get a star gets N points, where N is the number of people on the leaderboard. The second person to get the star gets N-1 points and so on and so forth.",
- "inline": false
- },
- {
- "name": "Join our private leaderboard!",
- "value": "Come join the Python Discord private leaderboard and compete against other people in the community! Get the join code using `.aoc join` and visit the [private leaderboard page](https://adventofcode.com/leaderboard/private) to join our leaderboard.",
- "inline": false
- }
-]