Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add /agenda command #58

Open
wants to merge 10 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 8 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions .env.sample
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,6 @@ DISCORD_UID_MAP="user1=1234,user2=4567,user3=7890"

LDAP_USERNAME=
LDAP_PASSWORD=


AGENDA_TEMPLATE_URL="https://md.redbrick.dcu.ie/foo"
1 change: 1 addition & 0 deletions .github/deploy/production.hcl
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ TOKEN={{ key "blockbot/discord/token" }}
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" }}
EOF
destination = "local/.env"
env = true
Expand Down
1 change: 1 addition & 0 deletions .github/deploy/review.hcl
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ DEBUG=true
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" }}
EOF
destination = "local/.env"
env = true
Expand Down
4 changes: 4 additions & 0 deletions src/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand All @@ -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")
195 changes: 195 additions & 0 deletions src/extensions/agenda.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
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.")


def generate_date_choices():
"""Generate date options for the next 7 days."""
today = datetime.date.today()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These generated day options will never update, so today will always be the day blockbot was started on. The only way around this is using day names instead of dates, or generating options with autocomplete (such as something I've done here).

return [(today + datetime.timedelta(days=i)).strftime("%Y-%m-%d") for i in range(7)]


def generate_time_choices():
"""Generate time options for every hour"""
base_time = datetime.time(0, 0)
times = []
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", choices=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, "%Y-%m-%d").date()
parsed_time = datetime.datetime.strptime(time, "%H:%M").time()

parsed_datetime = datetime.datetime.combine(parsed_date, parsed_time)

DATE = parsed_datetime.strftime("%Y-%m-%d")
TIME = parsed_datetime.strftime("%H:%M")
full_datetime = parsed_datetime.strftime("%A, %Y-%m-%d %H:%M")

ROOM = room
wizzdom marked this conversation as resolved.
Show resolved Hide resolved

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 minutes. Status code: `{response.status}`",
wizzdom marked this conversation as resolved.
Show resolved Hide resolved
flags=hikari.MessageFlag.EPHEMERAL,
)
return

content = await response.text()

modified_content = content.format(DATE=DATE, TIME=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 | {full_datetime} | {ROOM} <:bigRed:634311607039819776>


[{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(
id=634311607039819776, name="bigRed", is_animated=False
),
)

# 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"""
url = AGENDA_TEMPLATE_URL
image = "https://cdn.redbrick.dcu.ie/hedgedoc-uploads/sonic-the-hedgedoc.png"

embed = hikari.Embed(
title="Agenda Template",
url=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(image)

await ctx.respond(
embed,
flags=hikari.MessageFlag.EPHEMERAL,
)


@arc.loader
def loader(client: arc.GatewayClient) -> None:
client.add_plugin(plugin)