Creating bookmarks + message command

pull/130/head
stijndcl 2022-08-30 01:32:46 +02:00
parent 8308b4ad9a
commit 12d2017cbe
8 changed files with 234 additions and 11 deletions

View File

@ -0,0 +1,40 @@
"""Bookmarks
Revision ID: f5da771a155d
Revises: 38b7c29f10ee
Create Date: 2022-08-30 01:08:54.323883
"""
import sqlalchemy as sa
from alembic import op
# revision identifiers, used by Alembic.
revision = "f5da771a155d"
down_revision = "38b7c29f10ee"
branch_labels = None
depends_on = None
def upgrade() -> None:
# ### commands auto generated by Alembic - please adjust! ###
op.create_table(
"bookmarks",
sa.Column("bookmark_id", sa.Integer(), nullable=False),
sa.Column("label", sa.Text(), nullable=False),
sa.Column("jump_url", sa.Text(), nullable=False),
sa.Column("user_id", sa.BigInteger(), nullable=True),
sa.ForeignKeyConstraint(
["user_id"],
["users.user_id"],
),
sa.PrimaryKeyConstraint("bookmark_id"),
sa.UniqueConstraint("user_id", "label"),
)
# ### end Alembic commands ###
def downgrade() -> None:
# ### commands auto generated by Alembic - please adjust! ###
op.drop_table("bookmarks")
# ### end Alembic commands ###

View File

@ -0,0 +1,46 @@
from typing import Optional
import sqlalchemy.exc
from sqlalchemy import func, select
from sqlalchemy.ext.asyncio import AsyncSession
from database.crud.users import get_or_add_user
from database.exceptions import DuplicateInsertException, ForbiddenNameException
from database.schemas import Bookmark
__all__ = ["create_bookmark", "get_bookmarks", "get_bookmark_by_name"]
async def create_bookmark(session: AsyncSession, user_id: int, label: str, jump_url: str) -> Bookmark:
"""Create a new bookmark to a message"""
# Don't allow bookmarks with names of subcommands
if label.lower() in ["create", "ls", "list", "search"]:
raise ForbiddenNameException
await get_or_add_user(session, user_id)
try:
bookmark = Bookmark(label=label, jump_url=jump_url, user_id=user_id)
session.add(bookmark)
await session.commit()
await session.refresh(bookmark)
except sqlalchemy.exc.IntegrityError as e:
raise DuplicateInsertException from e
return bookmark
async def get_bookmarks(session: AsyncSession, user_id: int, *, query: Optional[str] = None) -> list[Bookmark]:
"""Get all a user's bookmarks"""
statement = select(Bookmark).where(Bookmark.user_id == user_id)
if query is not None:
statement = statement.where(Bookmark.label.ilike(f"%{query.lower()}%"))
return (await session.execute(statement)).scalars().all()
async def get_bookmark_by_name(session: AsyncSession, user_id: int, query: str) -> Optional[Bookmark]:
"""Try to find a bookmark by its name"""
statement = select(Bookmark).where(Bookmark.user_id == user_id).where(func.lower(Bookmark.label) == query.lower())
return (await session.execute(statement)).scalar_one_or_none()

View File

@ -1,5 +1,11 @@
from .constraints import DuplicateInsertException from .constraints import DuplicateInsertException, ForbiddenNameException
from .currency import DoubleNightly, NotEnoughDinks from .currency import DoubleNightly, NotEnoughDinks
from .not_found import NoResultFoundException from .not_found import NoResultFoundException
__all__ = ["DuplicateInsertException", "DoubleNightly", "NotEnoughDinks", "NoResultFoundException"] __all__ = [
"DuplicateInsertException",
"ForbiddenNameException",
"DoubleNightly",
"NotEnoughDinks",
"NoResultFoundException",
]

View File

@ -1,5 +1,9 @@
__all__ = ["DuplicateInsertException"] __all__ = ["DuplicateInsertException", "ForbiddenNameException"]
class DuplicateInsertException(Exception): class DuplicateInsertException(Exception):
"""Exception raised when a value already exists""" """Exception raised when a value already exists"""
class ForbiddenNameException(Exception):
"""Exception raised when trying to insert something with a name that isn't allowed"""

View File

@ -13,6 +13,7 @@ from sqlalchemy import (
ForeignKey, ForeignKey,
Integer, Integer,
Text, Text,
UniqueConstraint,
) )
from sqlalchemy.orm import declarative_base, relationship from sqlalchemy.orm import declarative_base, relationship
@ -25,6 +26,7 @@ __all__ = [
"Base", "Base",
"Bank", "Bank",
"Birthday", "Birthday",
"Bookmark",
"CustomCommand", "CustomCommand",
"CustomCommandAlias", "CustomCommandAlias",
"DadJoke", "DadJoke",
@ -78,6 +80,20 @@ class Birthday(Base):
user: User = relationship("User", uselist=False, back_populates="birthday", lazy="selectin") user: User = relationship("User", uselist=False, back_populates="birthday", lazy="selectin")
class Bookmark(Base):
"""A bookmark to a given message"""
__tablename__ = "bookmarks"
__table_args__ = (UniqueConstraint("user_id", "label"),)
bookmark_id: int = Column(Integer, primary_key=True)
label: str = Column(Text, nullable=False)
jump_url: str = Column(Text, nullable=False)
user_id: int = Column(BigInteger, ForeignKey("users.user_id"))
user: User = relationship("User", back_populates="bookmarks", uselist=False, lazy="selectin")
class CustomCommand(Base): class CustomCommand(Base):
"""Custom commands to fill the hole Dyno couldn't""" """Custom commands to fill the hole Dyno couldn't"""
@ -231,6 +247,9 @@ class User(Base):
birthday: Optional[Birthday] = relationship( birthday: Optional[Birthday] = relationship(
"Birthday", back_populates="user", uselist=False, lazy="selectin", cascade="all, delete-orphan" "Birthday", back_populates="user", uselist=False, lazy="selectin", cascade="all, delete-orphan"
) )
bookmarks: list[Bookmark] = relationship(
"Bookmark", back_populates="user", uselist=True, lazy="selectin", cascade="all, delete-orphan"
)
nightly_data: NightlyData = relationship( nightly_data: NightlyData = relationship(
"NightlyData", back_populates="user", uselist=False, lazy="selectin", cascade="all, delete-orphan" "NightlyData", back_populates="user", uselist=False, lazy="selectin", cascade="all, delete-orphan"
) )

View File

@ -4,10 +4,13 @@ import discord
from discord import app_commands from discord import app_commands
from discord.ext import commands from discord.ext import commands
from database.crud import birthdays from database.crud import birthdays, bookmarks
from database.exceptions import DuplicateInsertException, ForbiddenNameException
from didier import Didier from didier import Didier
from didier.exceptions import expect
from didier.utils.types.datetime import str_to_date from didier.utils.types.datetime import str_to_date
from didier.utils.types.string import leading from didier.utils.types.string import leading
from didier.views.modals import CreateBookmark
class Discord(commands.Cog): class Discord(commands.Cog):
@ -16,16 +19,20 @@ class Discord(commands.Cog):
client: Didier client: Didier
# Context-menu references # Context-menu references
_bookmark_ctx_menu: app_commands.ContextMenu
_pin_ctx_menu: app_commands.ContextMenu _pin_ctx_menu: app_commands.ContextMenu
def __init__(self, client: Didier): def __init__(self, client: Didier):
self.client = client self.client = client
self._pin_ctx_menu = app_commands.ContextMenu(name="Pin", callback=self.pin_ctx) self._bookmark_ctx_menu = app_commands.ContextMenu(name="Bookmark", callback=self._bookmark_ctx)
self._pin_ctx_menu = app_commands.ContextMenu(name="Pin", callback=self._pin_ctx)
self.client.tree.add_command(self._bookmark_ctx_menu)
self.client.tree.add_command(self._pin_ctx_menu) self.client.tree.add_command(self._pin_ctx_menu)
async def cog_unload(self) -> None: async def cog_unload(self) -> None:
"""Remove the commands when the cog is unloaded""" """Remove the commands when the cog is unloaded"""
self.client.tree.remove_command(self._bookmark_ctx_menu.name, type=self._bookmark_ctx_menu.type)
self.client.tree.remove_command(self._pin_ctx_menu.name, type=self._pin_ctx_menu.type) self.client.tree.remove_command(self._pin_ctx_menu.name, type=self._pin_ctx_menu.type)
@commands.group(name="Birthday", aliases=["Bd", "Birthdays"], case_insensitive=True, invoke_without_command=True) @commands.group(name="Birthday", aliases=["Bd", "Birthdays"], case_insensitive=True, invoke_without_command=True)
@ -35,14 +42,14 @@ class Discord(commands.Cog):
async with self.client.postgres_session as session: async with self.client.postgres_session as session:
birthday = await birthdays.get_birthday_for_user(session, user_id) birthday = await birthdays.get_birthday_for_user(session, user_id)
name = "Jouw" if user is None else f"{user.display_name}'s" name = "Your" if user is None else f"{user.display_name}'s"
if birthday is None: if birthday is None:
return await ctx.reply(f"{name} verjaardag zit niet in de database.", mention_author=False) return await ctx.reply(f"I don't know {name} birthday.", mention_author=False)
day, month = leading("0", str(birthday.birthday.day)), leading("0", str(birthday.birthday.month)) day, month = leading("0", str(birthday.birthday.day)), leading("0", str(birthday.birthday.month))
return await ctx.reply(f"{name} verjaardag staat ingesteld op **{day}/{month}**.", mention_author=False) return await ctx.reply(f"{name} birthday is set to **{day}/{month}**.", mention_author=False)
@birthday.command(name="Set", aliases=["Config"]) @birthday.command(name="Set", aliases=["Config"])
async def birthday_set(self, ctx: commands.Context, date_str: str): async def birthday_set(self, ctx: commands.Context, date_str: str):
@ -56,12 +63,56 @@ class Discord(commands.Cog):
date.replace(year=default_year) date.replace(year=default_year)
except ValueError: except ValueError:
return await ctx.reply(f"`{date_str}` is geen geldige datum.", mention_author=False) return await ctx.reply(f"`{date_str}` is not a valid date.", mention_author=False)
async with self.client.postgres_session as session: async with self.client.postgres_session as session:
await birthdays.add_birthday(session, ctx.author.id, date) await birthdays.add_birthday(session, ctx.author.id, date)
await self.client.confirm_message(ctx.message) await self.client.confirm_message(ctx.message)
@commands.group(name="Bookmark", aliases=["Bm", "Bookmarks"], case_insensitive=True, invoke_without_command=True)
async def bookmark(self, ctx: commands.Context, label: str):
"""Post a bookmarked message"""
async with self.client.postgres_session as session:
result = expect(
await bookmarks.get_bookmark_by_name(session, ctx.author.id, label),
entity_type="bookmark",
argument="label",
)
await ctx.reply(result.jump_url, mention_author=False)
@bookmark.command(name="Create", aliases=["New"])
async def bookmark_create(self, ctx: commands.Context, label: str, message: Optional[discord.Message]):
"""Create a new bookmark"""
# If no message was passed, allow replying to the message that should be bookmarked
if message is None and ctx.message.reference is not None:
message = await self.client.resolve_message(ctx.message.reference)
# Didn't fix it, so no message was found
if message is None:
return await ctx.reply("Found no message to bookmark.", delete_after=10)
# Create new bookmark
try:
async with self.client.postgres_session as session:
bm = await bookmarks.create_bookmark(session, ctx.author.id, label, message.jump_url)
await ctx.reply(f"Bookmark `{label}` successfully created (`#{bm.bookmark_id}`).", mention_author=False)
except DuplicateInsertException:
# Label is already in use
return await ctx.reply(f"You already have a bookmark named `{label}`.", mention_author=False)
except ForbiddenNameException:
# Label isn't allowed
return await ctx.reply(f"Bookmarks cannot be named `{label}`.", mention_author=False)
@bookmark.command(name="Search", aliases=["List", "Ls"])
async def bookmark_search(self, ctx: commands.Context, *, query: Optional[str] = None):
"""Search through the list of bookmarks"""
async def _bookmark_ctx(self, interaction: discord.Interaction, message: discord.Message):
"""Create a bookmark out of this message"""
modal = CreateBookmark(self.client, message.jump_url)
await interaction.response.send_modal(modal)
@commands.command(name="Join", usage="[Thread]") @commands.command(name="Join", usage="[Thread]")
async def join(self, ctx: commands.Context, thread: discord.Thread): async def join(self, ctx: commands.Context, thread: discord.Thread):
"""Make Didier join a thread""" """Make Didier join a thread"""
@ -88,7 +139,7 @@ class Discord(commands.Cog):
await message.pin(reason=f"Didier Pin by {ctx.author.display_name}") await message.pin(reason=f"Didier Pin by {ctx.author.display_name}")
await message.add_reaction("📌") await message.add_reaction("📌")
async def pin_ctx(self, interaction: discord.Interaction, message: discord.Message): async def _pin_ctx(self, interaction: discord.Interaction, message: discord.Message):
"""Pin a message in the current channel""" """Pin a message in the current channel"""
# Is already pinned # Is already pinned
if message.pinned: if message.pinned:

View File

@ -1,7 +1,16 @@
from .bookmarks import CreateBookmark
from .custom_commands import CreateCustomCommand, EditCustomCommand from .custom_commands import CreateCustomCommand, EditCustomCommand
from .dad_jokes import AddDadJoke from .dad_jokes import AddDadJoke
from .deadlines import AddDeadline from .deadlines import AddDeadline
from .links import AddLink from .links import AddLink
from .memes import GenerateMeme from .memes import GenerateMeme
__all__ = ["AddDadJoke", "AddDeadline", "CreateCustomCommand", "EditCustomCommand", "AddLink", "GenerateMeme"] __all__ = [
"CreateBookmark",
"AddDadJoke",
"AddDeadline",
"CreateCustomCommand",
"EditCustomCommand",
"AddLink",
"GenerateMeme",
]

View File

@ -0,0 +1,48 @@
import traceback
import discord.ui
from overrides import overrides
from database.crud.bookmarks import create_bookmark
from database.exceptions import DuplicateInsertException, ForbiddenNameException
from didier import Didier
__all__ = ["CreateBookmark"]
class CreateBookmark(discord.ui.Modal, title="Create Bookmark"):
"""Modal to create a bookmark"""
client: Didier
jump_url: str
name: discord.ui.TextInput = discord.ui.TextInput(label="Name", style=discord.TextStyle.short, required=True)
def __init__(self, client: Didier, jump_url: str, *args, **kwargs):
super().__init__(*args, **kwargs)
self.client = client
self.jump_url = jump_url
@overrides
async def on_submit(self, interaction: discord.Interaction):
label = self.name.value.strip()
try:
async with self.client.postgres_session as session:
bm = await create_bookmark(session, interaction.user.id, label, self.jump_url)
return await interaction.response.send_message(
f"Bookmark `{label}` successfully created (`#{bm.bookmark_id}`).", ephemeral=True
)
except DuplicateInsertException:
# Label is already in use
return await interaction.response.send_message(
f"You already have a bookmark named `{label}`.", ephemeral=True
)
except ForbiddenNameException:
# Label isn't allowed
return await interaction.response.send_message(f"Bookmarks cannot be named `{label}`.", ephemeral=True)
@overrides
async def on_error(self, interaction: discord.Interaction, error: Exception): # type: ignore
await interaction.response.send_message("Something went wrong.", ephemeral=True)
traceback.print_tb(error.__traceback__)