mirror of https://github.com/stijndcl/didier
				
				
				
			Editing of custom commands, add posix flags
							parent
							
								
									257eae6fa7
								
							
						
					
					
						commit
						d6a560851b
					
				|  | @ -65,3 +65,23 @@ async def get_command_by_alias(session: AsyncSession, message: str) -> Optional[ | |||
|         return None | ||||
| 
 | ||||
|     return alias.command | ||||
| 
 | ||||
| 
 | ||||
| async def edit_command( | ||||
|     session: AsyncSession, original_name: str, new_name: Optional[str] = None, new_response: Optional[str] = None | ||||
| ) -> CustomCommand: | ||||
|     """Edit an existing command""" | ||||
|     # Check if the command exists | ||||
|     command = await get_command(session, original_name) | ||||
|     if command is None: | ||||
|         raise NoResultFoundException | ||||
| 
 | ||||
|     if new_name is not None: | ||||
|         command.name = new_name | ||||
|     if new_response is not None: | ||||
|         command.response = new_response | ||||
| 
 | ||||
|     session.add(command) | ||||
|     await session.commit() | ||||
| 
 | ||||
|     return command | ||||
|  |  | |||
|  | @ -8,7 +8,8 @@ from database.crud import custom_commands | |||
| from database.exceptions.constraints import DuplicateInsertException | ||||
| from database.exceptions.not_found import NoResultFoundException | ||||
| from didier import Didier | ||||
| from didier.data.modals.custom_commands import CreateCustomCommand | ||||
| from didier.data.modals.custom_commands import CreateCustomCommand, EditCustomCommand | ||||
| from didier.data.flags.owner import EditCustomFlags | ||||
| 
 | ||||
| 
 | ||||
| class Owner(commands.Cog): | ||||
|  | @ -18,6 +19,7 @@ class Owner(commands.Cog): | |||
| 
 | ||||
|     # Slash groups | ||||
|     add_slash = app_commands.Group(name="add", description="Add something new to the database") | ||||
|     edit_slash = app_commands.Group(name="edit", description="Edit an existing database entry") | ||||
| 
 | ||||
|     def __init__(self, client: Didier): | ||||
|         self.client = client | ||||
|  | @ -29,6 +31,11 @@ class Owner(commands.Cog): | |||
|         # pylint: disable=W0236 # Pylint thinks this can't be async, but it can | ||||
|         return await self.client.is_owner(ctx.author) | ||||
| 
 | ||||
|     @commands.command(name="Error") | ||||
|     async def _error(self, ctx: commands.Context): | ||||
|         """Raise an exception for debugging purposes""" | ||||
|         raise Exception("Debug") | ||||
| 
 | ||||
|     @commands.command(name="Sync") | ||||
|     async def sync(self, ctx: commands.Context, guild: Optional[discord.Guild] = None): | ||||
|         """Sync all application-commands in Discord""" | ||||
|  | @ -77,14 +84,43 @@ class Owner(commands.Cog): | |||
|                 "Je hebt geen toestemming om dit commando uit te voeren.", ephemeral=True | ||||
|             ) | ||||
| 
 | ||||
|         # await interaction.response.defer(ephemeral=True) | ||||
|         modal = CreateCustomCommand() | ||||
|         modal = CreateCustomCommand(self.client) | ||||
|         await interaction.response.send_modal(modal) | ||||
| 
 | ||||
|     @commands.group(name="Edit") | ||||
|     async def edit(self, ctx: commands.Context): | ||||
|     @commands.group(name="Edit", case_insensitive=True, invoke_without_command=False) | ||||
|     async def edit_msg(self, ctx: commands.Context): | ||||
|         """Command group for [edit X] commands""" | ||||
| 
 | ||||
|     @edit_msg.command(name="Custom") | ||||
|     async def edit_custom_msg(self, ctx: commands.Context, command: str, *, flags: EditCustomFlags): | ||||
|         """Edit an existing custom command""" | ||||
|         async with self.client.db_session as session: | ||||
|             try: | ||||
|                 await custom_commands.edit_command(session, command, flags.name, flags.response) | ||||
|                 return await self.client.confirm_message(ctx.message) | ||||
|             except NoResultFoundException: | ||||
|                 await ctx.reply(f"Geen commando gevonden voor ``{command}``.") | ||||
|                 return await self.client.reject_message(ctx.message) | ||||
| 
 | ||||
|     @edit_slash.command(name="custom", description="Edit a custom command") | ||||
|     @app_commands.describe(command="The name of the command to edit") | ||||
|     async def edit_custom_slash(self, interaction: discord.Interaction, command: str): | ||||
|         """Slash command to edit a custom command""" | ||||
|         if not await self.client.is_owner(interaction.user): | ||||
|             return interaction.response.send_message( | ||||
|                 "Je hebt geen toestemming om dit commando uit te voeren.", ephemeral=True | ||||
|             ) | ||||
| 
 | ||||
|         async with self.client.db_session as session: | ||||
|             _command = await custom_commands.get_command(session, command) | ||||
|             if _command is None: | ||||
|                 return await interaction.response.send_message( | ||||
|                     f"Geen commando gevonden voor ``{command}``.", ephemeral=True | ||||
|                 ) | ||||
| 
 | ||||
|             modal = EditCustomCommand(self.client, _command.name, _command.response) | ||||
|             await interaction.response.send_modal(modal) | ||||
| 
 | ||||
| 
 | ||||
| async def setup(client: Didier): | ||||
|     """Load the cog""" | ||||
|  |  | |||
|  | @ -0,0 +1 @@ | |||
| from .posix import PosixFlags | ||||
|  | @ -0,0 +1,10 @@ | |||
| from typing import Optional | ||||
| 
 | ||||
| from didier.data.flags import PosixFlags | ||||
| 
 | ||||
| 
 | ||||
| class EditCustomFlags(PosixFlags): | ||||
|     """Flags for the edit custom command""" | ||||
| 
 | ||||
|     name: Optional[str] = None | ||||
|     response: Optional[str] = None | ||||
|  | @ -0,0 +1,14 @@ | |||
| from discord.ext import commands | ||||
| 
 | ||||
| 
 | ||||
| class PosixFlags(commands.FlagConverter, delimiter=" ", prefix="--"): | ||||
|     """Base class to add POSIX-like flags to commands | ||||
| 
 | ||||
|     Example usage: | ||||
|         >>> class Flags(PosixFlags): | ||||
|         >>>     name: str | ||||
|         >>> async def command(ctx, *, flags: Flags): | ||||
|         >>>     ... | ||||
|     This can now be called in Discord as | ||||
|     command --name here-be-name | ||||
|     """ | ||||
|  | @ -2,19 +2,66 @@ import traceback | |||
| 
 | ||||
| import discord | ||||
| 
 | ||||
| from database.crud.custom_commands import create_command, edit_command | ||||
| from didier import Didier | ||||
| 
 | ||||
| class CreateCustomCommand(discord.ui.Modal, title="Custom Command"): | ||||
|     """Modal shown to visually create custom commands""" | ||||
| 
 | ||||
|     name: discord.ui.TextInput = discord.ui.TextInput(label="Name", placeholder="Name of the command") | ||||
| class CreateCustomCommand(discord.ui.Modal, title="Create Custom Command"): | ||||
|     """Modal to create new custom commands""" | ||||
| 
 | ||||
|     name: discord.ui.TextInput = discord.ui.TextInput(label="Name", placeholder="Didier") | ||||
| 
 | ||||
|     response: discord.ui.TextInput = discord.ui.TextInput( | ||||
|         label="Response", style=discord.TextStyle.long, placeholder="Response of the command", max_length=2000 | ||||
|         label="Response", style=discord.TextStyle.long, placeholder="Hmm?", max_length=2000 | ||||
|     ) | ||||
| 
 | ||||
|     async def on_submit(self, interaction: discord.Interaction) -> None: | ||||
|         await interaction.response.send_message("Submitted", ephemeral=True) | ||||
|     client: Didier | ||||
| 
 | ||||
|     async def on_error(self, interaction: discord.Interaction, error: Exception) -> None:  # type: ignore | ||||
|         await interaction.response.send_message("Errored", ephemeral=True) | ||||
|     def __init__(self, client: Didier, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.client = client | ||||
| 
 | ||||
|     async def on_submit(self, interaction: discord.Interaction): | ||||
|         async with self.client.db_session as session: | ||||
|             command = await create_command(session, self.name.value, self.response.value) | ||||
| 
 | ||||
|         await interaction.response.send_message(f"Successfully created ``{command.name}``.", ephemeral=True) | ||||
| 
 | ||||
|     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__) | ||||
| 
 | ||||
| 
 | ||||
| class EditCustomCommand(discord.ui.Modal, title="Edit Custom Command"): | ||||
|     """Modal to edit an existing custom command | ||||
|     Fills in the current values as defaults | ||||
|     """ | ||||
| 
 | ||||
|     name: discord.ui.TextInput | ||||
|     response: discord.ui.TextInput | ||||
| 
 | ||||
|     original_name: str | ||||
| 
 | ||||
|     client: Didier | ||||
| 
 | ||||
|     def __init__(self, client: Didier, name: str, response: str, *args, **kwargs): | ||||
|         super().__init__(*args, **kwargs) | ||||
|         self.original_name = name | ||||
|         self.client = client | ||||
| 
 | ||||
|         self.name = self.add_item(discord.ui.TextInput(label="Name", placeholder="Didier", default=name)) | ||||
|         self.response = self.add_item( | ||||
|             discord.ui.TextInput( | ||||
|                 label="Response", placeholder="Hmm?", default=response, style=discord.TextStyle.long, max_length=2000 | ||||
|             ) | ||||
|         ) | ||||
| 
 | ||||
|     async def on_submit(self, interaction: discord.Interaction): | ||||
|         async with self.client.db_session as session: | ||||
|             await edit_command(session, self.original_name, self.name.value, self.response.value) | ||||
| 
 | ||||
|         await interaction.response.send_message(f"Successfully edited ``{self.original_name}``.", ephemeral=True) | ||||
| 
 | ||||
|     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__) | ||||
|  |  | |||
|  | @ -129,8 +129,9 @@ class Didier(commands.Bot): | |||
| 
 | ||||
|     async def on_command_error(self, context: commands.Context, exception: commands.CommandError, /) -> None: | ||||
|         """Event triggered when a regular command errors""" | ||||
|         # If developing, print everything to stdout so you don't have to | ||||
|         # check the logs all the time | ||||
|         # Print everything to the logs/stderr | ||||
|         await super().on_command_error(context, exception) | ||||
| 
 | ||||
|         # If developing, do nothing special | ||||
|         if settings.SANDBOX: | ||||
|             print(traceback.format_exc(), file=sys.stderr) | ||||
|             return | ||||
|  |  | |||
							
								
								
									
										20
									
								
								main.py
								
								
								
								
							
							
						
						
									
										20
									
								
								main.py
								
								
								
								
							|  | @ -1,4 +1,5 @@ | |||
| import logging | ||||
| import sys | ||||
| from logging.handlers import RotatingFileHandler | ||||
| 
 | ||||
| import asyncio | ||||
|  | @ -18,15 +19,26 @@ def setup_logging(): | |||
|     """Configure custom loggers""" | ||||
|     max_log_size = 32 * 1024 * 1024 | ||||
| 
 | ||||
|     # Configure Didier handler | ||||
|     didier_log = logging.getLogger("didier") | ||||
| 
 | ||||
|     handler = RotatingFileHandler(settings.LOGFILE, mode="a", maxBytes=max_log_size, backupCount=5) | ||||
|     handler.setFormatter(logging.Formatter("[%(asctime)s] [%(levelname)s]: %(message)s")) | ||||
|     didier_handler = RotatingFileHandler(settings.LOGFILE, mode="a", maxBytes=max_log_size, backupCount=5) | ||||
|     didier_handler.setFormatter(logging.Formatter("[%(asctime)s] [%(levelname)s]: %(message)s")) | ||||
| 
 | ||||
|     didier_log.addHandler(handler) | ||||
|     didier_log.addHandler(didier_handler) | ||||
|     didier_log.setLevel(logging.INFO) | ||||
| 
 | ||||
|     logging.getLogger("discord").setLevel(logging.ERROR) | ||||
|     # Configure discord handler | ||||
|     discord_log = logging.getLogger("discord") | ||||
| 
 | ||||
|     # Make dev print to stderr instead, so you don't have to watch the file | ||||
|     if settings.SANDBOX: | ||||
|         discord_handler = logging.StreamHandler(sys.stderr) | ||||
|     else: | ||||
|         discord_handler = RotatingFileHandler("discord.log", mode="a", maxBytes=max_log_size, backupCount=5) | ||||
| 
 | ||||
|     discord_handler.setFormatter(logging.Formatter("[%(asctime)s] [%(levelname)s] %(name)s: %(message)s")) | ||||
|     discord_log.addHandler(discord_handler) | ||||
| 
 | ||||
| 
 | ||||
| async def main(): | ||||
|  |  | |||
|  | @ -4,7 +4,8 @@ from sqlalchemy.ext.asyncio import AsyncSession | |||
| 
 | ||||
| from database.crud import custom_commands as crud | ||||
| from database.exceptions.constraints import DuplicateInsertException | ||||
| from database.models import CustomCommand, CustomCommandAlias | ||||
| from database.exceptions.not_found import NoResultFoundException | ||||
| from database.models import CustomCommand | ||||
| 
 | ||||
| 
 | ||||
| async def test_create_command_non_existing(database_session: AsyncSession): | ||||
|  | @ -33,7 +34,7 @@ async def test_create_command_name_is_alias(database_session: AsyncSession): | |||
|         await crud.create_command(database_session, "n", "other response") | ||||
| 
 | ||||
| 
 | ||||
| async def test_create_alias_non_existing(database_session: AsyncSession): | ||||
| async def test_create_alias(database_session: AsyncSession): | ||||
|     """Test creating an alias when the name is still free""" | ||||
|     command = await crud.create_command(database_session, "name", "response") | ||||
|     await crud.create_alias(database_session, command.name, "n") | ||||
|  | @ -43,6 +44,12 @@ async def test_create_alias_non_existing(database_session: AsyncSession): | |||
|     assert command.aliases[0].alias == "n" | ||||
| 
 | ||||
| 
 | ||||
| async def test_create_alias_non_existing(database_session: AsyncSession): | ||||
|     """Test creating an alias when the command doesn't exist""" | ||||
|     with pytest.raises(NoResultFoundException): | ||||
|         await crud.create_alias(database_session, "name", "alias") | ||||
| 
 | ||||
| 
 | ||||
| async def test_create_alias_duplicate(database_session: AsyncSession): | ||||
|     """Test creating an alias when another alias already has this name""" | ||||
|     command = await crud.create_command(database_session, "name", "response") | ||||
|  | @ -96,3 +103,17 @@ async def test_get_command_by_alias(database_session: AsyncSession): | |||
| async def test_get_command_non_existing(database_session: AsyncSession): | ||||
|     """Test getting a command when it doesn't exist""" | ||||
|     assert await crud.get_command(database_session, "name") is None | ||||
| 
 | ||||
| 
 | ||||
| async def test_edit_command(database_session: AsyncSession): | ||||
|     """Test editing an existing command""" | ||||
|     command = await crud.create_command(database_session, "name", "response") | ||||
|     await crud.edit_command(database_session, command.name, "new name", "new response") | ||||
|     assert command.name == "new name" | ||||
|     assert command.response == "new response" | ||||
| 
 | ||||
| 
 | ||||
| async def test_edit_command_non_existing(database_session: AsyncSession): | ||||
|     """Test editing a command that doesn't exist""" | ||||
|     with pytest.raises(NoResultFoundException): | ||||
|         await crud.edit_command(database_session, "name", "n", "r") | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue