Experimental Discord bot written in Python
選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

usernamecog.py 7.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. """
  2. Cog for detecting username patterns.
  3. """
  4. from typing import Optional
  5. from discord import Guild, Interaction, Member
  6. from discord.app_commands import Group
  7. from discord.ext.commands import Cog
  8. from config import CONFIG
  9. from rocketbot.cogs.basecog import BaseCog, BotMessage, BotMessageReaction, CogSetting
  10. from rocketbot.storage import Storage
  11. from rocketbot.utils import MOD_PERMISSIONS
  12. class UsernamePatternContext:
  13. """
  14. BotMessage context for a flagged username
  15. """
  16. def __init__(self, member: Member) -> None:
  17. self.member: Member = member
  18. self.kicked_by: Optional[Member] = None
  19. self.banned_by: Optional[Member] = None
  20. self.ignored_by: Optional[Member] = None
  21. def reactions(self) -> list[BotMessageReaction]:
  22. """
  23. Generates updated BotMessageReactions based on context state.
  24. """
  25. r: list[BotMessageReaction] = []
  26. if self.ignored_by:
  27. r.append(BotMessageReaction(CONFIG['ignore_emoji'], False, f'Ignored by {self.ignored_by.name}'))
  28. elif self.banned_by:
  29. r.append(BotMessageReaction(CONFIG['ban_emoji'], False, f'Banned by {self.banned_by.name}'))
  30. elif self.kicked_by:
  31. r.append(BotMessageReaction(CONFIG['kick_emoji'], False, f'Kicked by {self.kicked_by.name}'))
  32. r.append(BotMessageReaction(CONFIG['ban_emoji'], True, 'Ban user'))
  33. else:
  34. r.append(BotMessageReaction(CONFIG['kick_emoji'], True, 'Kick user'))
  35. r.append(BotMessageReaction(CONFIG['ban_emoji'], True, 'Ban user'))
  36. r.append(BotMessageReaction(CONFIG['ignore_emoji'], True, 'Ignore warning'))
  37. return r
  38. class UsernamePatternCog(BaseCog, name='Username Pattern'):
  39. """
  40. Detects usernames that match certain flagged patterns. Posts a mod warning
  41. message on a match.
  42. """
  43. SETTING_ENABLED = CogSetting(
  44. 'enabled',
  45. bool,
  46. default_value=False,
  47. brief='username pattern detection',
  48. description='Whether new users are checked for username patterns.',
  49. )
  50. SETTING_PATTERNS = CogSetting('patterns', None, default_value=None)
  51. def __init__(self, bot):
  52. super().__init__(
  53. bot,
  54. config_prefix='username',
  55. short_description='Manages username pattern detection.',
  56. long_description='When new users join, if their username matches '
  57. 'a configured pattern the mods will be alerted.'
  58. )
  59. self.add_setting(UsernamePatternCog.SETTING_ENABLED)
  60. def __get_patterns(self, guild: Guild) -> list[str]:
  61. """
  62. Returns an array of username patterns.
  63. """
  64. patterns: list[str] = self.get_guild_setting(guild, self.SETTING_PATTERNS)
  65. if patterns is None:
  66. patterns = []
  67. Storage.set_config_value(guild, 'UsernamePatternCog.patterns', patterns)
  68. return patterns
  69. @classmethod
  70. def __save_patterns(cls,
  71. guild: Guild,
  72. patterns: list[str]) -> None:
  73. """
  74. Saves username pattern array.
  75. """
  76. cls.set_guild_setting(guild, cls.SETTING_PATTERNS, patterns)
  77. username = Group(
  78. name='username',
  79. description='Manages username pattern detection.',
  80. guild_only=True,
  81. default_permissions=MOD_PERMISSIONS,
  82. extras={
  83. 'long_description': 'When new users join, if their username matches '
  84. 'a configured pattern the mods will be alerted.',
  85. },
  86. )
  87. @username.command(
  88. description='Adds a username pattern.',
  89. extras={
  90. 'long_description': 'When a user joins the server, if their username '
  91. 'matches a configured pattern the mods will be alerted. '
  92. 'Matching is currently a simple substring test.',
  93. 'usage': '<pattern>',
  94. },
  95. )
  96. async def add(self, interaction: Interaction, pattern: str) -> None:
  97. """
  98. Adds a username pattern to match against new members.
  99. Parameters
  100. ----------
  101. interaction : Interaction
  102. pattern : str
  103. a substring to look for in usernames
  104. """
  105. norm_pattern = pattern.lower()
  106. patterns: list[str] = self.__get_patterns(interaction.guild)
  107. if norm_pattern in patterns:
  108. await interaction.response.send_message(
  109. f'{CONFIG["warning_emoji"]} Pattern `{norm_pattern}` already added.',
  110. ephemeral=True
  111. )
  112. return
  113. patterns.append(norm_pattern)
  114. self.__save_patterns(interaction.guild, patterns)
  115. await interaction.response.send_message(
  116. f'{CONFIG["success_emoji"]} Pattern `{norm_pattern}` added.',
  117. ephemeral=True
  118. )
  119. @username.command(
  120. description='Removes a username pattern.',
  121. extras={
  122. 'usage': '<pattern>',
  123. },
  124. )
  125. async def remove(self, interaction: Interaction, pattern: str) -> None:
  126. """
  127. Removes a username pattern.
  128. Parameters
  129. ----------
  130. interaction : Interaction
  131. pattern : str
  132. the existing username pattern to remove
  133. """
  134. norm_pattern = pattern.lower()
  135. guild: Guild = interaction.guild
  136. patterns: list[str] = self.__get_patterns(guild)
  137. len_before = len(patterns)
  138. patterns = list(filter(lambda p: p != norm_pattern, patterns))
  139. if len(patterns) == len_before:
  140. await interaction.response.send_message(
  141. f'{CONFIG["warning_emoji"]} Pattern `{norm_pattern}` not found.',
  142. ephemeral=True,
  143. )
  144. return
  145. self.__save_patterns(guild, patterns)
  146. await interaction.response.send_message(
  147. f'{CONFIG["success_emoji"]} Pattern `{norm_pattern}` removed.',
  148. ephemeral=True,
  149. )
  150. @username.command(
  151. description='Lists existing username patterns.'
  152. )
  153. async def list(self, interaction: Interaction) -> None:
  154. """Command handler"""
  155. guild: Guild = interaction.guild
  156. patterns: list[str] = self.__get_patterns(guild)
  157. if len(patterns) == 0:
  158. await interaction.response.send_message(
  159. f'{CONFIG["success_emoji"]} No patterns defined.',
  160. ephemeral=True,
  161. )
  162. else:
  163. msg = f'{CONFIG["info_emoji"]} Patterns:\n\n> `' + '`\n> `'.join(patterns) + '`'
  164. await interaction.response.send_message(
  165. msg,
  166. ephemeral=True,
  167. )
  168. @Cog.listener()
  169. async def on_member_join(self, member: Member) -> None:
  170. """Event handler"""
  171. for pattern in self.__get_patterns(member.guild):
  172. if self.matches(pattern, member.name):
  173. await self.handle_match(member, pattern)
  174. elif self.matches(pattern, member.display_name):
  175. await self.handle_match(member, pattern)
  176. def matches(self, pattern: str, subject: str) -> bool:
  177. """Checks if a username matches a given pattern"""
  178. if pattern is None:
  179. return False
  180. if subject is None:
  181. return False
  182. return pattern.lower() in subject.lower()
  183. async def handle_match(self, member: Member, pattern: str) -> None:
  184. """
  185. Handles a username match.
  186. """
  187. # TODO: Prevent double handling?
  188. self.log(member.guild, f'User {member.id} {member.display_name} matches pattern "{pattern}"')
  189. context = UsernamePatternContext(member)
  190. bm = BotMessage(
  191. member.guild,
  192. f'User {member.mention} ({str(member.id)}, {member.display_name}) has ' +
  193. f'username matching pattern `{pattern}`.',
  194. BotMessage.TYPE_INFO if self.was_warned_recently(member) else BotMessage.TYPE_MOD_WARNING,
  195. context)
  196. self.record_warning(member)
  197. await bm.set_reactions(context.reactions())
  198. await self.post_message(bm)
  199. async def on_mod_react(self,
  200. bot_message: BotMessage,
  201. reaction: BotMessageReaction,
  202. reacted_by: Member) -> None:
  203. context: UsernamePatternContext = bot_message.context
  204. if reaction.emoji == CONFIG['kick_emoji']:
  205. await context.member.kick(
  206. reason=f'Rocketbot: Flagged username pattern. Kicked by {reacted_by.name}.')
  207. context.kicked_by = reacted_by
  208. self.log(context.member.guild, f'User {context.member.name} kicked by {reacted_by.name}')
  209. await bot_message.set_reactions(context.reactions())
  210. elif reaction.emoji == CONFIG['ban_emoji']:
  211. await context.member.ban(
  212. reason=f'Rocketbot: Flagged username pattern. Banned by {reacted_by.name}.',
  213. delete_message_days=0)
  214. context.banned_by = reacted_by
  215. self.log(context.member.guild, f'User {context.member.name} banned by {reacted_by.name}')
  216. await bot_message.set_reactions(context.reactions())
  217. elif reaction.emoji == CONFIG['ignore_emoji']:
  218. context.ignored_by = reacted_by
  219. self.log(context.member.guild, f'Warning ignored by {reacted_by.name}')
  220. await bot_message.set_reactions(context.reactions())