diff --git a/.env.sample b/.env.sample index efb7da1..f550ae5 100644 --- a/.env.sample +++ b/.env.sample @@ -5,3 +5,5 @@ DISCORD_UID_MAP="user1=1234,user2=4567,user3=7890" LDAP_USERNAME= LDAP_PASSWORD= + +AGENDA_TEMPLATE_URL="https://md.redbrick.dcu.ie/foo" diff --git a/.github/deploy/production.hcl b/.github/deploy/production.hcl index 67d9e2a..765233a 100644 --- a/.github/deploy/production.hcl +++ b/.github/deploy/production.hcl @@ -34,6 +34,8 @@ LDAP_USERNAME={{ key "blockbot/ldap/username" }} LDAP_PASSWORD={{ key "blockbot/ldap/password" }} DISCORD_UID_MAP={{ key "blockbot/discord/uid_map" }} +AGENDA_TEMPLATE_URL={{ key "blockbot/agenda/template_url" }} + DB_HOST={{ env "NOMAD_ADDR_db" }} # address and port DB_NAME={{ key "blockbot/db/name" }} # database name DB_PASSWORD={{ key "blockbot/db/password" }} diff --git a/.github/deploy/review.hcl b/.github/deploy/review.hcl index b06ce26..14af437 100644 --- a/.github/deploy/review.hcl +++ b/.github/deploy/review.hcl @@ -35,6 +35,8 @@ LDAP_USERNAME={{ key "blockbot-dev/ldap/username" }} LDAP_PASSWORD={{ key "blockbot-dev/ldap/password" }} DISCORD_UID_MAP={{ key "blockbot-dev/discord/uid_map" }} +AGENDA_TEMPLATE_URL={{ key "blockbot-dev/agenda/template_url" }} + DB_HOST={{ env "NOMAD_ADDR_db" }} # address and port DB_NAME={{ key "blockbot-dev/db/name" }} # database name DB_PASSWORD={{ key "blockbot-dev/db/password" }} diff --git a/src/config.py b/src/config.py index e651c60..9e7d37a 100644 --- a/src/config.py +++ b/src/config.py @@ -13,6 +13,8 @@ "bot-private": 853071983452225536, "bots-cmt": 1162038557922312312, "action-items": 1029132014210793513, + "cowboys-and-cowgirls-committee": 578712722330353684, + "committee-announcements": 763113612340363304, } # TODO: query API/LDAP for these @@ -30,3 +32,5 @@ LDAP_USERNAME = os.environ.get("LDAP_USERNAME") LDAP_PASSWORD = os.environ.get("LDAP_PASSWORD") + +AGENDA_TEMPLATE_URL = os.environ.get("AGENDA_TEMPLATE_URL") diff --git a/src/extensions/agenda.py b/src/extensions/agenda.py new file mode 100644 index 0000000..58c4304 --- /dev/null +++ b/src/extensions/agenda.py @@ -0,0 +1,199 @@ +import arc +import hikari +import aiohttp +from urllib.parse import urlparse +import datetime + +from src.utils import role_mention, hedgedoc_login +from src.hooks import restrict_to_channels, restrict_to_roles +from src.config import CHANNEL_IDS, ROLE_IDS, UID_MAPS, AGENDA_TEMPLATE_URL + + +plugin = arc.GatewayPlugin(name="Agenda") + +agenda = plugin.include_slash_group("agenda", "Interact with the agenda.") + + +async def generate_date_choices( + data: arc.AutocompleteData[arc.GatewayClient, str], +) -> list[str]: + """Generate date options for the next 7 days.""" + today = datetime.date.today() + return [ + (today + datetime.timedelta(days=i)).strftime("%A %d/%m/%Y") for i in range(7) + ] + + +def generate_time_choices() -> list[str]: + """Generate time options for every hour.""" + base_time = datetime.time(0, 0) + times: list[str] = [] + + for hour in range(24): + current_time = ( + datetime.datetime.combine(datetime.date.today(), base_time) + + datetime.timedelta(hours=hour) + ).time() + times.append(current_time.strftime("%H:%M")) + + return times + + +@agenda.include +@arc.with_hook( + restrict_to_channels( + channel_ids=[ + CHANNEL_IDS["bots-cmt"], + CHANNEL_IDS["committee-announcements"], + CHANNEL_IDS["cowboys-and-cowgirls-committee"], + ] + ) +) +@arc.with_hook(restrict_to_roles(role_ids=[ROLE_IDS["committee"]])) +@arc.slash_subcommand( + "generate", + "Generate a new agenda for committee meetings.", + autodefer=arc.AutodeferMode.EPHEMERAL, +) +async def gen_agenda( + ctx: arc.GatewayContext, + date: arc.Option[ + str, + arc.StrParams("Select a date", autocomplete_with=generate_date_choices), + ], + time: arc.Option[ + str, + arc.StrParams( + "Enter the time in HH:MM format", choices=generate_time_choices() + ), + ], + room: arc.Option[ + str, + arc.StrParams("Select a Room"), + ], + url: arc.Option[ + str, arc.StrParams("URL of the agenda template from the MD") + ] = AGENDA_TEMPLATE_URL, + aiohttp_client: aiohttp.ClientSession = arc.inject(), +) -> None: + """Generate a new agenda for committee meetings.""" + + parsed_date = datetime.datetime.strptime(date, "%A %d/%m/%Y").date() + parsed_time = datetime.datetime.strptime(time, "%H:%M").time() + + parsed_datetime = datetime.datetime.combine(parsed_date, parsed_time) + + formatted_date = parsed_datetime.strftime("%Y-%m-%d") + formatted_time = parsed_datetime.strftime("%H:%M") + formatted_datetime = parsed_datetime.strftime("%A, %Y-%m-%d %H:%M") + + if "https://md.redbrick.dcu.ie" not in url: + await ctx.respond( + f"❌ `{url}` is not a valid MD URL. Please provide a valid URL.", + flags=hikari.MessageFlag.EPHEMERAL, + ) + return + + await hedgedoc_login(aiohttp_client) + + parsed_url = urlparse(url) + request_url = ( + f"{parsed_url.scheme}://{parsed_url.hostname}{parsed_url.path}/download" + ) + + async with aiohttp_client.get(request_url) as response: + if response.status != 200: + await ctx.respond( + f"❌ Failed to fetch the agenda template. Status code: `{response.status}`", + flags=hikari.MessageFlag.EPHEMERAL, + ) + return + + content = await response.text() + + modified_content = content.format( + DATE=formatted_date, TIME=formatted_time, ROOM=room + ) + + post_url = f"{parsed_url.scheme}://{parsed_url.hostname}/new" + post_headers = {"Content-Type": "text/markdown"} + + async with aiohttp_client.post( + url=post_url, + headers=post_headers, + data=modified_content, + ) as response: + if response.status != 200: + await ctx.respond( + f"❌ Failed to generate the agenda. Status code: `{response.status}`", + flags=hikari.MessageFlag.EPHEMERAL, + ) + return + + new_agenda_url = response.url + announce_text = f""" +## 📣 Agenda for this week's meeting | {formatted_datetime} | {room} <:bigRed:634311607039819776> + + +[{formatted_date} Agenda](<{new_agenda_url}>) + +- Please fill in your sections with anything you would like to discuss. +- Put your Redbrick `username` beside any agenda items you add. +- If you can't attend the meeting, please DM <@{UID_MAPS["kronos"]}> with your reason. +- React with <:bigRed:634311607039819776> if you can make it. + +||{role_mention(ROLE_IDS["committee"])}|| + """ + + announce = await plugin.client.rest.create_message( + CHANNEL_IDS["committee-announcements"], + mentions_everyone=False, + user_mentions=True, + role_mentions=True, + content=announce_text, + ) + + await plugin.client.rest.add_reaction( + channel=announce.channel_id, + message=announce.id, + emoji=hikari.CustomEmoji.parse("<:bigRed:634311607039819776>"), + ) + + # respond with success if it executes successfully + await ctx.respond( + "✅ Agenda generated. Announcement sent successfully!", + flags=hikari.MessageFlag.EPHEMERAL, + ) + return + + +@agenda.include +@arc.with_hook(restrict_to_roles(role_ids=[ROLE_IDS["committee"]])) +@arc.slash_subcommand( + "template", + "View the agenda template", +) +async def view_template( + ctx: arc.GatewayContext, +) -> None: + """View the agenda template.""" + + embed = hikari.Embed( + title="Agenda Template", + url=AGENDA_TEMPLATE_URL, + description="Click the link above to view the agenda template.\n\n**NOTE:** Any edits made to this template will affect the generated agenda.", + colour=0x5865F2, + ) + embed = embed.set_image( + "https://cdn.redbrick.dcu.ie/hedgedoc-uploads/sonic-the-hedgedoc.png" + ) + + await ctx.respond( + embed, + flags=hikari.MessageFlag.EPHEMERAL, + ) + + +@arc.loader +def loader(client: arc.GatewayClient) -> None: + client.add_plugin(plugin)