Experimental Discord bot written in Python
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

cogsetting.py 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. """
  2. A guild configuration setting available for editing via bot commands.
  3. """
  4. from datetime import timedelta
  5. from typing import Any, Optional, Type, Literal, Union
  6. from discord import Interaction, Permissions
  7. from discord.app_commands import Range, Transform, describe
  8. from discord.app_commands.commands import Command, Group, CommandCallback, rename
  9. from discord.ext.commands import Bot
  10. from config import CONFIG
  11. from rocketbot.storage import Storage
  12. from rocketbot.utils import bot_log, TimeDeltaTransformer, MOD_PERMISSIONS, dump_stacktrace, str_from_timedelta
  13. def describe_type(datatype: Type) -> str:
  14. if datatype is int:
  15. return 'integer'
  16. if datatype is float:
  17. return 'float'
  18. if datatype is str:
  19. return 'string'
  20. if datatype is bool:
  21. return 'boolean'
  22. if datatype is timedelta:
  23. return 'timespan'
  24. if getattr(datatype, '__origin__', None) is Union:
  25. return '|'.join([ describe_type(a) for a in datatype.__args__ ])
  26. if getattr(datatype, '__origin__', None) is Literal:
  27. return '"' + ('"|"'.join(datatype.__args__)) + '"'
  28. return datatype.__class__.__name__
  29. class CogSetting:
  30. """
  31. Describes a configuration setting for a guild that can be edited by the
  32. mods of those guilds. BaseCog can generate "/get" and "/set" commands (or
  33. "/enable" and "/disable" commands for boolean values) automatically, reducing
  34. the boilerplate of generating commands manually. Offers simple validation rules.
  35. """
  36. permissions: Permissions = Permissions(Permissions.manage_messages.flag)
  37. def __init__(self,
  38. name: str,
  39. datatype: Optional[Type],
  40. default_value: Any,
  41. brief: Optional[str] = None,
  42. description: Optional[str] = None,
  43. min_value: Optional[Any] = None,
  44. max_value: Optional[Any] = None,
  45. enum_values: Optional[set[Any]] = None):
  46. """
  47. Parameters
  48. ----------
  49. name: str
  50. Setting identifier. Must follow variable naming conventions.
  51. datatype: Optional[Type]
  52. Datatype of the setting. E.g. int, float, str
  53. default_value: Any
  54. Value to use if a guild has not yet configured one.
  55. brief: Optional[str]
  56. Description of the setting, starting with lower case.
  57. Will be inserted into phrases like "Sets <brief>" and
  58. "Gets <brief>".
  59. description: Optional[str]
  60. Long-form description. Min, max, and enum values will be
  61. appended to the end, so does not need to include these.
  62. min_value: Optional[Any]
  63. Smallest allowable value. Must be of the same datatype as
  64. the value. None for no minimum.
  65. max_value: Optional[Any]
  66. Largest allowable value. None for no maximum.
  67. enum_values: Optional[set[Any]]
  68. Set of allowed values. None if unconstrained.
  69. """
  70. self.name: str = name
  71. self.datatype: Type = datatype
  72. self.default_value = default_value
  73. self.brief: Optional[str] = brief
  74. self.description: str = description or '' # Can't be None
  75. self.min_value: Optional[Any] = min_value
  76. self.max_value: Optional[Any] = max_value
  77. self.enum_values: Optional[set[Any]] = enum_values
  78. if self.enum_values:
  79. value_list = '`' + ('`, `'.join(self.enum_values)) + '`'
  80. self.description += f' (Permitted values: {value_list})'
  81. elif self.min_value is not None and self.max_value is not None:
  82. self.description += f' (Value must be between `{self.min_value}` and `{self.max_value}`)'
  83. elif self.min_value is not None:
  84. self.description += f' (Minimum value: {self.min_value})'
  85. elif self.max_value is not None:
  86. self.description += f' (Maximum value: {self.max_value})'
  87. def validate_value(self, new_value: Any) -> None:
  88. """
  89. Checks if a value is legal for this setting. Raises a ValueError if not.
  90. """
  91. if self.min_value is not None and new_value < self.min_value:
  92. raise ValueError(f'`{self.name}` must be >= {self.min_value}')
  93. if self.max_value is not None and new_value > self.max_value:
  94. raise ValueError(f'`{self.name}` must be <= {self.max_value}')
  95. if self.enum_values is not None and new_value not in self.enum_values:
  96. allowed_values = '`' + ('`, `'.join(self.enum_values)) + '`'
  97. raise ValueError(f'`{self.name}` must be one of {allowed_values}')
  98. def set_up(self, cog: 'BaseCog') -> None:
  99. """
  100. Sets up getter and setter commands for this setting. This should
  101. usually only be called by BaseCog.
  102. """
  103. if self.name in ('enabled', 'is_enabled'):
  104. self.__enable_group.add_command(self.__make_enable_command(cog))
  105. self.__disable_group.add_command(self.__make_disable_command(cog))
  106. else:
  107. self.__get_group.add_command(self.__make_getter_command(cog))
  108. self.__set_group.add_command(self.__make_setter_command(cog))
  109. def to_stored_value(self, native_value: Any) -> Any:
  110. """Converts a configuration value to a JSON-compatible datatype."""
  111. if self.datatype is timedelta:
  112. return native_value.total_seconds()
  113. return native_value
  114. def to_native_value(self, stored_value: Any) -> Any:
  115. """Converts the stored JSON-compatible datatype to its actual value."""
  116. if self.datatype is timedelta and isinstance(stored_value, (int, float)):
  117. return timedelta(seconds=stored_value)
  118. return stored_value
  119. @staticmethod
  120. def native_value_to_str(native_value: Any) -> str:
  121. """Formats a native configuration value to a user-presentable string."""
  122. if native_value is None:
  123. return '<no value>'
  124. if isinstance(native_value, timedelta):
  125. return str_from_timedelta(native_value)
  126. if isinstance(native_value, bool):
  127. return 'true' if native_value else 'false'
  128. return f'{native_value}'
  129. def __make_getter_command(self, cog: 'BaseCog') -> Command:
  130. setting: CogSetting = self
  131. setting_name = setting.name
  132. if cog.config_prefix is not None:
  133. setting_name = f'{cog.config_prefix}_{setting_name}'
  134. async def getter(interaction: Interaction) -> None:
  135. key = f'{cog.__class__.__name__}.{setting.name}'
  136. value = setting.to_native_value(Storage.get_config_value(interaction.guild, key))
  137. cog.log(interaction.guild, f'{interaction.user.name} used /get setting_name')
  138. if value is None:
  139. value = setting.to_native_value(cog.get_cog_default(setting.name))
  140. await interaction.response.send_message(
  141. f'{CONFIG["info_emoji"]} `{setting_name}` is using default of `{CogSetting.native_value_to_str(value)}`',
  142. ephemeral=True
  143. )
  144. else:
  145. await interaction.response.send_message(
  146. f'{CONFIG["info_emoji"]} `{setting_name}` is set to `{CogSetting.native_value_to_str(value)}`',
  147. ephemeral=True
  148. )
  149. bot_log(None, cog.__class__, f"Creating command: /get {setting_name}")
  150. command = Command(
  151. name=setting_name,
  152. description=f'Shows {self.brief}.',
  153. callback=getter,
  154. parent=CogSetting.__get_group,
  155. extras={
  156. 'cog': cog,
  157. 'setting': setting,
  158. 'long_description': setting.description,
  159. },
  160. )
  161. return command
  162. def __make_setter_command(self, cog: 'BaseCog') -> Command:
  163. setting: CogSetting = self
  164. setting_name = setting.name
  165. if cog.config_prefix is not None:
  166. setting_name = f'{cog.config_prefix}_{setting_name}'
  167. async def setter_general(interaction: Interaction, new_value) -> None:
  168. cog.log(interaction.guild, f'{interaction.user.name} used /set setting_name {new_value}')
  169. try:
  170. setting.validate_value(new_value)
  171. except ValueError as ve:
  172. await interaction.response.send_message(
  173. f'{CONFIG["failure_emoji"]} {ve}',
  174. ephemeral=True
  175. )
  176. return
  177. key = f'{cog.__class__.__name__}.{setting.name}'
  178. Storage.set_config_value(interaction.guild, key, setting.to_stored_value(new_value))
  179. await interaction.response.send_message(
  180. f'{CONFIG["success_emoji"]} `{setting_name}` is now set to `{setting.to_native_value(new_value)}`',
  181. ephemeral=True
  182. )
  183. await cog.on_setting_updated(interaction.guild, setting)
  184. setter: CommandCallback = setter_general
  185. if self.datatype is int:
  186. if self.min_value is not None or self.max_value is not None:
  187. r_min = self.min_value
  188. r_max = self.max_value
  189. @rename(new_value=self.name)
  190. @describe(new_value=self.brief)
  191. async def setter_range(interaction: Interaction, new_value: Range[int, r_min, r_max]) -> None:
  192. await setter_general(interaction, new_value)
  193. setter = setter_range
  194. else:
  195. @rename(new_value=self.name)
  196. @describe(new_value=self.brief)
  197. async def setter_int(interaction: Interaction, new_value: int) -> None:
  198. await setter_general(interaction, new_value)
  199. setter = setter_int
  200. elif self.datatype is float:
  201. @rename(new_value=self.name)
  202. @describe(new_value=self.brief)
  203. async def setter_float(interaction: Interaction, new_value: float) -> None:
  204. await setter_general(interaction, new_value)
  205. setter = setter_float
  206. elif self.datatype is timedelta:
  207. @rename(new_value=self.name)
  208. @describe(new_value=f'{self.brief} (e.g. 30s, 5m, 1h30s, 7d)')
  209. async def setter_timedelta(interaction: Interaction, new_value: Transform[timedelta, TimeDeltaTransformer]) -> None:
  210. await setter_general(interaction, new_value)
  211. setter = setter_timedelta
  212. elif getattr(self.datatype, '__origin__', None) == Literal:
  213. dt = self.datatype
  214. @rename(new_value=self.name)
  215. @describe(new_value=self.brief)
  216. async def setter_enum(interaction: Interaction, new_value: dt) -> None:
  217. await setter_general(interaction, new_value)
  218. setter = setter_enum
  219. elif self.datatype is str:
  220. if self.enum_values is not None:
  221. raise ValueError('Type for a setting with enum values should be typing.Literal')
  222. else:
  223. @rename(new_value=self.name)
  224. @describe(new_value=self.brief)
  225. async def setter_str(interaction: Interaction, new_value: str) -> None:
  226. await setter_general(interaction, new_value)
  227. setter = setter_str
  228. elif self.datatype is bool:
  229. @rename(new_value=self.name)
  230. @describe(new_value=self.brief)
  231. async def setter_bool(interaction: Interaction, new_value: bool) -> None:
  232. await setter_general(interaction, new_value)
  233. setter = setter_bool
  234. elif self.datatype is not None:
  235. raise ValueError(f'Invalid type {self.datatype}')
  236. bot_log(None, cog.__class__, f"Creating command: /set {setting_name} <{describe_type(self.datatype)}>")
  237. command = Command(
  238. name=setting_name,
  239. description=f'Sets {self.brief}.',
  240. callback=setter,
  241. parent=CogSetting.__set_group,
  242. extras={
  243. 'cog': cog,
  244. 'setting': setting,
  245. 'long_description': setting.description,
  246. },
  247. )
  248. return command
  249. def __make_enable_command(self, cog: 'BaseCog') -> Command:
  250. setting: CogSetting = self
  251. async def enabler(interaction: Interaction) -> None:
  252. key = f'{cog.__class__.__name__}.{setting.name}'
  253. Storage.set_config_value(interaction.guild, key, True)
  254. await interaction.response.send_message(
  255. f'{CONFIG["success_emoji"]} {setting.brief.capitalize()} enabled.',
  256. ephemeral=True
  257. )
  258. await cog.on_setting_updated(interaction.guild, setting)
  259. cog.log(interaction.guild, f'{interaction.user.name} used /enable {cog.__class__.__name__}')
  260. bot_log(None, cog.__class__, f"Creating command: /enable {cog.config_prefix}")
  261. command = Command(
  262. name=cog.config_prefix,
  263. description=f'Enables {cog.qualified_name} functionality.',
  264. callback=enabler,
  265. parent=CogSetting.__enable_group,
  266. extras={
  267. 'cog': cog,
  268. 'setting': setting,
  269. 'long_description': setting.description,
  270. },
  271. )
  272. return command
  273. def __make_disable_command(self, cog: 'BaseCog') -> Command:
  274. setting: CogSetting = self
  275. async def disabler(interaction: Interaction) -> None:
  276. key = f'{cog.__class__.__name__}.{setting.name}'
  277. Storage.set_config_value(interaction.guild, key, False)
  278. await interaction.response.send_message(
  279. f'{CONFIG["success_emoji"]} {setting.brief.capitalize()} disabled.',
  280. ephemeral=True
  281. )
  282. await cog.on_setting_updated(interaction.guild, setting)
  283. cog.log(interaction.guild, f'{interaction.user.name} used /disable {cog.__class__.__name__}')
  284. bot_log(None, cog.__class__, f"Creating command: /disable {cog.config_prefix}")
  285. command = Command(
  286. name=cog.config_prefix,
  287. description=f'Disables {cog.config_prefix} functionality',
  288. callback=disabler,
  289. parent=CogSetting.__disable_group,
  290. extras={
  291. 'cog': cog,
  292. 'setting': setting,
  293. 'long_description': setting.description,
  294. },
  295. )
  296. return command
  297. __set_group: Group
  298. __get_group: Group
  299. __enable_group: Group
  300. __disable_group: Group
  301. @classmethod
  302. def set_up_all(cls, cog: 'BaseCog', bot: Bot, settings: list['CogSetting']) -> None:
  303. """
  304. Sets up editing commands for a list of CogSettings and adds them to a
  305. cog. If the cog has a command Group, commands will be added to it.
  306. Otherwise, they will be added at the top level.
  307. """
  308. cls.__set_up_base_commands(bot)
  309. if len(settings) == 0:
  310. return
  311. for setting in settings:
  312. setting.set_up(cog)
  313. @classmethod
  314. def __set_up_base_commands(cls, bot: Bot) -> None:
  315. if getattr(cls, f'_CogSetting__set_group', None) is not None:
  316. return
  317. cls.__set_group = Group(
  318. name='set',
  319. description='Sets a configuration value for this guild.',
  320. guild_only=True,
  321. default_permissions=MOD_PERMISSIONS,
  322. extras={
  323. 'long_description': 'Settings are guild-specific. If no value is set, a default is used. Use `/get` to '
  324. 'see the current value for this guild.',
  325. },
  326. )
  327. cls.__get_group = Group(
  328. name='get',
  329. description='Shows a configuration value for this guild.',
  330. guild_only=True,
  331. default_permissions=MOD_PERMISSIONS,
  332. extras={
  333. 'long_description': 'Settings are guild-specific. If no value is set, a default is used. Use `/set` to '
  334. 'change the value.',
  335. },
  336. )
  337. cls.__enable_group = Group(
  338. name='enable',
  339. description='Enables a module for this guild.',
  340. guild_only=True,
  341. default_permissions=MOD_PERMISSIONS,
  342. extras={
  343. 'long_description': 'Modules are enabled on a per-guild basis and are off by default. Use `/disable` '
  344. 'to disable an enabled module.',
  345. },
  346. )
  347. cls.__disable_group = Group(
  348. name='disable',
  349. description='Disables a module for this guild.',
  350. guild_only=True,
  351. default_permissions=MOD_PERMISSIONS,
  352. extras={
  353. 'long_description': 'Modules are enabled on a per-guild basis and are off by default. Use `/enable` '
  354. 're-enable a disabled module.',
  355. },
  356. )
  357. bot.tree.add_command(cls.__set_group)
  358. bot.tree.add_command(cls.__get_group)
  359. bot.tree.add_command(cls.__enable_group)
  360. bot.tree.add_command(cls.__disable_group)
  361. from rocketbot.cogs.basecog import BaseCog
  362. async def show_all(interaction: Interaction) -> None:
  363. try:
  364. bot_log(interaction.guild, None, f'{interaction.user.name} used /get all')
  365. guild = interaction.guild
  366. if guild is None:
  367. await interaction.response.send_message(
  368. f'{CONFIG["failure_emoji"]} No guild.',
  369. ephemeral=True,
  370. delete_after=10,
  371. )
  372. return
  373. text = '## :information_source: Configuration'
  374. for cog_name, cog in sorted(bot.cogs.items()):
  375. if not isinstance(cog, BaseCog):
  376. continue
  377. bcog: BaseCog = cog
  378. if len(bcog.settings) == 0:
  379. continue
  380. text += f'\n### {bcog.qualified_name} Module'
  381. for setting in sorted(bcog.settings, key=lambda s: (s.name != 'enabled', s.name)):
  382. key = f'{bcog.__class__.__name__}.{setting.name}'
  383. value = setting.to_native_value(Storage.get_config_value(guild, key))
  384. deflt = setting.to_native_value(bcog.get_cog_default(setting.name))
  385. if setting.name == 'enabled':
  386. text += f'\n- Module is '
  387. if value is not None:
  388. text += '**' + ('enabled' if value else 'disabled') + '**'
  389. else:
  390. text += ('enabled' if deflt else 'disabled') + ' _(default)_'
  391. else:
  392. if value is not None:
  393. text += f'\n- `{bcog.config_prefix}_{setting.name}` = **{CogSetting.native_value_to_str(value)}**'
  394. else:
  395. text += f'\n- `{bcog.config_prefix}_{setting.name}` = {CogSetting.native_value_to_str(deflt)} _(using default)_'
  396. await interaction.response.send_message(
  397. text,
  398. ephemeral=True,
  399. )
  400. except BaseException as e:
  401. dump_stacktrace(e)
  402. show_all_command = Command(
  403. name='all',
  404. description='Shows all configuration for this guild.',
  405. callback=show_all,
  406. )
  407. cls.__get_group.add_command(show_all_command)