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.
540 lines
21 KiB
540 lines
21 KiB
"""
|
|
The MIT License (MIT)
|
|
|
|
Copyright (c) 2015-present Rapptz
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a
|
|
copy of this software and associated documentation files (the "Software"),
|
|
to deal in the Software without restriction, including without limitation
|
|
the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
and/or sell copies of the Software, and to permit persons to whom the
|
|
Software is furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
DEALINGS IN THE SOFTWARE.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
import datetime
|
|
|
|
from typing import TYPE_CHECKING, Any, Dict, Optional, List, Sequence, Set, Union, Sequence
|
|
|
|
|
|
from .enums import AutoModRuleTriggerType, AutoModRuleActionType, AutoModRuleEventType, try_enum
|
|
from .flags import AutoModPresets
|
|
from . import utils
|
|
from .utils import MISSING, cached_slot_property
|
|
|
|
if TYPE_CHECKING:
|
|
from typing_extensions import Self
|
|
from .abc import Snowflake, GuildChannel
|
|
from .threads import Thread
|
|
from .guild import Guild
|
|
from .member import Member
|
|
from .state import ConnectionState
|
|
from .types.automod import (
|
|
AutoModerationRule as AutoModerationRulePayload,
|
|
AutoModerationTriggerMetadata as AutoModerationTriggerMetadataPayload,
|
|
AutoModerationAction as AutoModerationActionPayload,
|
|
AutoModerationActionExecution as AutoModerationActionExecutionPayload,
|
|
)
|
|
from .role import Role
|
|
|
|
__all__ = (
|
|
'AutoModRuleAction',
|
|
'AutoModTrigger',
|
|
'AutoModRule',
|
|
'AutoModAction',
|
|
)
|
|
|
|
|
|
class AutoModRuleAction:
|
|
"""Represents an auto moderation's rule action.
|
|
|
|
.. versionadded:: 2.0
|
|
|
|
Attributes
|
|
-----------
|
|
type: :class:`AutoModRuleActionType`
|
|
The type of action to take.
|
|
Defaults to :attr:`~AutoModRuleActionType.block_message`.
|
|
channel_id: Optional[:class:`int`]
|
|
The ID of the channel or thread to send the alert message to, if any.
|
|
Passing this sets :attr:`type` to :attr:`~AutoModRuleActionType.send_alert_message`.
|
|
duration: Optional[:class:`datetime.timedelta`]
|
|
The duration of the timeout to apply, if any.
|
|
Has a maximum of 28 days.
|
|
Passing this sets :attr:`type` to :attr:`~AutoModRuleActionType.timeout`.
|
|
"""
|
|
|
|
__slots__ = ('type', 'channel_id', 'duration')
|
|
|
|
def __init__(self, *, channel_id: Optional[int] = None, duration: Optional[datetime.timedelta] = None) -> None:
|
|
self.channel_id: Optional[int] = channel_id
|
|
self.duration: Optional[datetime.timedelta] = duration
|
|
if channel_id and duration:
|
|
raise ValueError('Please provide only one of ``channel`` or ``duration``')
|
|
|
|
if channel_id:
|
|
self.type = AutoModRuleActionType.send_alert_message
|
|
elif duration:
|
|
self.type = AutoModRuleActionType.timeout
|
|
else:
|
|
self.type = AutoModRuleActionType.block_message
|
|
|
|
def __repr__(self) -> str:
|
|
return f'<AutoModRuleAction type={self.type.value} channel={self.channel_id} duration={self.duration}>'
|
|
|
|
@classmethod
|
|
def from_data(cls, data: AutoModerationActionPayload) -> Self:
|
|
type_ = try_enum(AutoModRuleActionType, data['type'])
|
|
if data['type'] == AutoModRuleActionType.timeout.value:
|
|
duration_seconds = data['metadata']['duration_seconds']
|
|
return cls(duration=datetime.timedelta(seconds=duration_seconds))
|
|
elif data['type'] == AutoModRuleActionType.send_alert_message.value:
|
|
channel_id = int(data['metadata']['channel_id'])
|
|
return cls(channel_id=channel_id)
|
|
return cls()
|
|
|
|
def to_dict(self) -> Dict[str, Any]:
|
|
ret = {'type': self.type.value, 'metadata': {}}
|
|
if self.type is AutoModRuleActionType.timeout:
|
|
ret['metadata'] = {'duration_seconds': int(self.duration.total_seconds())} # type: ignore # duration cannot be None here
|
|
elif self.type is AutoModRuleActionType.send_alert_message:
|
|
ret['metadata'] = {'channel_id': str(self.channel_id)}
|
|
return ret
|
|
|
|
|
|
class AutoModTrigger:
|
|
r"""Represents a trigger for an auto moderation rule.
|
|
|
|
The following table illustrates relevant attributes for each :class:`AutoModRuleTriggerType`:
|
|
|
|
+-----------------------------------------------+--------------------------------------+
|
|
| Type | Attributes |
|
|
+===============================================+======================================+
|
|
| :attr:`AutoModRuleTriggerType.keyword` | :attr:`keyword_filter` |
|
|
+-----------------------------------------------+--------------------------------------+
|
|
| :attr:`AutoModRuleTriggerType.spam` | |
|
|
+-----------------------------------------------+--------------------------------------+
|
|
| :attr:`AutoModRuleTriggerType.keyword_preset` | :attr:`presets`\, :attr:`allow_list` |
|
|
+-----------------------------------------------+--------------------------------------+
|
|
| :attr:`AutoModRuleTriggerType.mention_spam` | :attr:`mention_limit` |
|
|
+-----------------------------------------------+--------------------------------------+
|
|
|
|
.. versionadded:: 2.0
|
|
|
|
Attributes
|
|
-----------
|
|
type: :class:`AutoModRuleTriggerType`
|
|
The type of trigger.
|
|
keyword_filter: List[:class:`str`]
|
|
The list of strings that will trigger the keyword filter.
|
|
presets: :class:`AutoModPresets`
|
|
The presets used with the preset keyword filter.
|
|
allow_list: List[:class:`str`]
|
|
The list of words that are exempt from the commonly flagged words.
|
|
mention_limit: :class:`int`
|
|
The total number of user and role mentions a message can contain.
|
|
Has a maximum of 50.
|
|
"""
|
|
|
|
__slots__ = (
|
|
'type',
|
|
'keyword_filter',
|
|
'presets',
|
|
'allow_list',
|
|
'mention_limit',
|
|
)
|
|
|
|
def __init__(
|
|
self,
|
|
*,
|
|
type: Optional[AutoModRuleTriggerType] = None,
|
|
keyword_filter: Optional[List[str]] = None,
|
|
presets: Optional[AutoModPresets] = None,
|
|
allow_list: Optional[List[str]] = None,
|
|
mention_limit: Optional[int] = None,
|
|
) -> None:
|
|
if type is None and sum(arg is not None for arg in (keyword_filter, presets, mention_limit)) > 1:
|
|
raise ValueError('Please pass only one of keyword_filter, presets, or mention_limit.')
|
|
|
|
if type is not None:
|
|
self.type = type
|
|
elif keyword_filter is not None:
|
|
self.type = AutoModRuleTriggerType.keyword
|
|
elif presets is not None:
|
|
self.type = AutoModRuleTriggerType.keyword_preset
|
|
elif mention_limit is not None:
|
|
self.type = AutoModRuleTriggerType.mention_spam
|
|
else:
|
|
raise ValueError(
|
|
'Please pass the trigger type explicitly if not using keyword_filter, presets, or mention_limit.'
|
|
)
|
|
|
|
self.keyword_filter: List[str] = keyword_filter if keyword_filter is not None else []
|
|
self.presets: AutoModPresets = presets if presets is not None else AutoModPresets()
|
|
self.allow_list: List[str] = allow_list if allow_list is not None else []
|
|
self.mention_limit: int = mention_limit if mention_limit is not None else 0
|
|
|
|
@classmethod
|
|
def from_data(cls, type: int, data: Optional[AutoModerationTriggerMetadataPayload]) -> Self:
|
|
type_ = try_enum(AutoModRuleTriggerType, type)
|
|
if data is None:
|
|
return cls(type=type_)
|
|
elif type_ is AutoModRuleTriggerType.keyword:
|
|
return cls(type=type_, keyword_filter=data.get('keyword_filter'))
|
|
elif type_ is AutoModRuleTriggerType.keyword_preset:
|
|
return cls(
|
|
type=type_, presets=AutoModPresets._from_value(data.get('presets', [])), allow_list=data.get('allow_list')
|
|
)
|
|
elif type_ is AutoModRuleTriggerType.mention_spam:
|
|
return cls(type=type_, mention_limit=data.get('mention_total_limit'))
|
|
else:
|
|
return cls(type=type_)
|
|
|
|
def to_metadata_dict(self) -> Optional[Dict[str, Any]]:
|
|
if self.type is AutoModRuleTriggerType.keyword:
|
|
return {'keyword_filter': self.keyword_filter}
|
|
elif self.type is AutoModRuleTriggerType.keyword_preset:
|
|
return {'presets': self.presets.to_array(), 'allow_list': self.allow_list}
|
|
elif self.type is AutoModRuleTriggerType.mention_spam:
|
|
return {'mention_total_limit': self.mention_limit}
|
|
|
|
|
|
class AutoModRule:
|
|
"""Represents an auto moderation rule.
|
|
|
|
.. versionadded:: 2.0
|
|
|
|
Attributes
|
|
-----------
|
|
id: :class:`int`
|
|
The ID of the rule.
|
|
guild: :class:`Guild`
|
|
The guild the rule is for.
|
|
name: :class:`str`
|
|
The name of the rule.
|
|
creator_id: :class:`int`
|
|
The ID of the user that created the rule.
|
|
trigger: :class:`AutoModTrigger`
|
|
The rule's trigger.
|
|
enabled: :class:`bool`
|
|
Whether the rule is enabled.
|
|
exempt_role_ids: Set[:class:`int`]
|
|
The IDs of the roles that are exempt from the rule.
|
|
exempt_channel_ids: Set[:class:`int`]
|
|
The IDs of the channels that are exempt from the rule.
|
|
"""
|
|
|
|
__slots__ = (
|
|
'_state',
|
|
'_cs_exempt_roles',
|
|
'_cs_exempt_channels',
|
|
'_cs_actions',
|
|
'id',
|
|
'guild',
|
|
'name',
|
|
'creator_id',
|
|
'event_type',
|
|
'trigger',
|
|
'enabled',
|
|
'exempt_role_ids',
|
|
'exempt_channel_ids',
|
|
'_actions',
|
|
)
|
|
|
|
def __init__(self, *, data: AutoModerationRulePayload, guild: Guild, state: ConnectionState) -> None:
|
|
self._state: ConnectionState = state
|
|
self.guild: Guild = guild
|
|
self.id: int = int(data['id'])
|
|
self.name: str = data['name']
|
|
self.creator_id = int(data['creator_id'])
|
|
self.event_type: AutoModRuleEventType = try_enum(AutoModRuleEventType, data['event_type'])
|
|
self.trigger: AutoModTrigger = AutoModTrigger.from_data(data['trigger_type'], data=data.get('trigger_metadata'))
|
|
self.enabled: bool = data['enabled']
|
|
self.exempt_role_ids: Set[int] = {int(role_id) for role_id in data['exempt_roles']}
|
|
self.exempt_channel_ids: Set[int] = {int(channel_id) for channel_id in data['exempt_channels']}
|
|
self._actions: List[AutoModerationActionPayload] = data['actions']
|
|
|
|
def __repr__(self) -> str:
|
|
return f'<AutoModRule id={self.id} name={self.name!r} guild={self.guild!r}>'
|
|
|
|
def to_dict(self) -> AutoModerationRulePayload:
|
|
ret: AutoModerationRulePayload = {
|
|
'id': str(self.id),
|
|
'guild_id': str(self.guild.id),
|
|
'name': self.name,
|
|
'creator_id': str(self.creator_id),
|
|
'event_type': self.event_type.value,
|
|
'trigger_type': self.trigger.type.value,
|
|
'trigger_metadata': self.trigger.to_metadata_dict(),
|
|
'actions': [action.to_dict() for action in self.actions],
|
|
'enabled': self.enabled,
|
|
'exempt_roles': [str(role_id) for role_id in self.exempt_role_ids],
|
|
'exempt_channels': [str(channel_id) for channel_id in self.exempt_channel_ids],
|
|
} # type: ignore # trigger types break the flow here.
|
|
|
|
return ret
|
|
|
|
@property
|
|
def creator(self) -> Optional[Member]:
|
|
"""Optional[:class:`Member`]: The member that created this rule."""
|
|
return self.guild.get_member(self.creator_id)
|
|
|
|
@cached_slot_property('_cs_exempt_roles')
|
|
def exempt_roles(self) -> List[Role]:
|
|
"""List[:class:`Role`]: The roles that are exempt from this rule."""
|
|
result = []
|
|
get_role = self.guild.get_role
|
|
for role_id in self.exempt_role_ids:
|
|
role = get_role(role_id)
|
|
if role is not None:
|
|
result.append(role)
|
|
|
|
return utils._unique(result)
|
|
|
|
@cached_slot_property('_cs_exempt_channels')
|
|
def exempt_channels(self) -> List[Union[GuildChannel, Thread]]:
|
|
"""List[Union[:class:`abc.GuildChannel`, :class:`Thread`]]: The channels that are exempt from this rule."""
|
|
it = filter(None, map(self.guild._resolve_channel, self.exempt_channel_ids))
|
|
return utils._unique(it)
|
|
|
|
@cached_slot_property('_cs_actions')
|
|
def actions(self) -> List[AutoModRuleAction]:
|
|
"""List[:class:`AutoModRuleAction`]: The actions that are taken when this rule is triggered."""
|
|
return [AutoModRuleAction.from_data(action) for action in self._actions]
|
|
|
|
def is_exempt(self, obj: Snowflake, /) -> bool:
|
|
"""Check if an object is exempt from the automod rule.
|
|
|
|
Parameters
|
|
-----------
|
|
obj: :class:`abc.Snowflake`
|
|
The role, channel, or thread to check.
|
|
|
|
Returns
|
|
--------
|
|
:class:`bool`
|
|
Whether the object is exempt from the automod rule.
|
|
"""
|
|
return obj.id in self.exempt_channel_ids or obj.id in self.exempt_role_ids
|
|
|
|
async def edit(
|
|
self,
|
|
*,
|
|
name: str = MISSING,
|
|
event_type: AutoModRuleEventType = MISSING,
|
|
actions: List[AutoModRuleAction] = MISSING,
|
|
trigger: AutoModTrigger = MISSING,
|
|
enabled: bool = MISSING,
|
|
exempt_roles: Sequence[Snowflake] = MISSING,
|
|
exempt_channels: Sequence[Snowflake] = MISSING,
|
|
reason: str = MISSING,
|
|
) -> Self:
|
|
"""|coro|
|
|
|
|
Edits this auto moderation rule.
|
|
|
|
You must have :attr:`Permissions.manage_guild` to edit rules.
|
|
|
|
Parameters
|
|
-----------
|
|
name: :class:`str`
|
|
The new name to change to.
|
|
event_type: :class:`AutoModRuleEventType`
|
|
The new event type to change to.
|
|
actions: List[:class:`AutoModRuleAction`]
|
|
The new rule actions to update.
|
|
trigger: :class:`AutoModTrigger`
|
|
The new trigger to update.
|
|
You can only change the trigger metadata, not the type.
|
|
enabled: :class:`bool`
|
|
Whether the rule should be enabled or not.
|
|
exempt_roles: Sequence[:class:`abc.Snowflake`]
|
|
The new roles to exempt from the rule.
|
|
exempt_channels: Sequence[:class:`abc.Snowflake`]
|
|
The new channels to exempt from the rule.
|
|
reason: :class:`str`
|
|
The reason for updating this rule. Shows up on the audit log.
|
|
|
|
Raises
|
|
-------
|
|
Forbidden
|
|
You do not have permission to edit this rule.
|
|
HTTPException
|
|
Editing the rule failed.
|
|
|
|
Returns
|
|
--------
|
|
:class:`AutoModRule`
|
|
The updated auto moderation rule.
|
|
"""
|
|
payload = {}
|
|
if actions is not MISSING:
|
|
payload['actions'] = [action.to_dict() for action in actions]
|
|
|
|
if name is not MISSING:
|
|
payload['name'] = name
|
|
|
|
if event_type is not MISSING:
|
|
payload['event_type'] = event_type
|
|
|
|
if trigger is not MISSING:
|
|
trigger_metadata = trigger.to_metadata_dict()
|
|
if trigger_metadata is not None:
|
|
payload['trigger_metadata'] = trigger_metadata
|
|
|
|
if enabled is not MISSING:
|
|
payload['enabled'] = enabled
|
|
|
|
if exempt_roles is not MISSING:
|
|
payload['exempt_roles'] = exempt_roles
|
|
|
|
if exempt_channels is not MISSING:
|
|
payload['exempt_channels'] = exempt_channels
|
|
|
|
data = await self._state.http.edit_auto_moderation_rule(
|
|
self.guild.id,
|
|
self.id,
|
|
reason=reason,
|
|
**payload,
|
|
)
|
|
|
|
return AutoModRule(data=data, guild=self.guild, state=self._state)
|
|
|
|
async def delete(self, *, reason: str = MISSING) -> None:
|
|
"""|coro|
|
|
|
|
Deletes the auto moderation rule.
|
|
|
|
You must have :attr:`Permissions.manage_guild` to delete rules.
|
|
|
|
Parameters
|
|
-----------
|
|
reason: :class:`str`
|
|
The reason for deleting this rule. Shows up on the audit log.
|
|
|
|
Raises
|
|
-------
|
|
Forbidden
|
|
You do not have permissions to delete the rule.
|
|
HTTPException
|
|
Deleting the rule failed.
|
|
"""
|
|
await self._state.http.delete_auto_moderation_rule(self.guild.id, self.id, reason=reason)
|
|
|
|
|
|
class AutoModAction:
|
|
"""Represents an action that was taken as the result of a moderation rule.
|
|
|
|
.. versionadded:: 2.0
|
|
|
|
Attributes
|
|
-----------
|
|
action: :class:`AutoModRuleAction`
|
|
The action that was taken.
|
|
message_id: Optional[:class:`int`]
|
|
The message ID that triggered the action. This is only available if the
|
|
action is done on an edited message.
|
|
rule_id: :class:`int`
|
|
The ID of the rule that was triggered.
|
|
rule_trigger_type: :class:`AutoModRuleTriggerType`
|
|
The trigger type of the rule that was triggered.
|
|
guild_id: :class:`int`
|
|
The ID of the guild where the rule was triggered.
|
|
user_id: :class:`int`
|
|
The ID of the user that triggered the rule.
|
|
channel_id: :class:`int`
|
|
The ID of the channel where the rule was triggered.
|
|
alert_system_message_id: Optional[:class:`int`]
|
|
The ID of the system message that was sent to the predefined alert channel.
|
|
content: :class:`str`
|
|
The content of the message that triggered the rule.
|
|
Requires the :attr:`Intents.message_content` or it will always return an empty string.
|
|
matched_keyword: Optional[:class:`str`]
|
|
The matched keyword from the triggering message.
|
|
matched_content: Optional[:class:`str`]
|
|
The matched content from the triggering message.
|
|
Requires the :attr:`Intents.message_content` or it will always return ``None``.
|
|
"""
|
|
|
|
__slots__ = (
|
|
'_state',
|
|
'action',
|
|
'rule_id',
|
|
'rule_trigger_type',
|
|
'guild_id',
|
|
'user_id',
|
|
'channel_id',
|
|
'message_id',
|
|
'alert_system_message_id',
|
|
'content',
|
|
'matched_keyword',
|
|
'matched_content',
|
|
)
|
|
|
|
def __init__(self, *, data: AutoModerationActionExecutionPayload, state: ConnectionState) -> None:
|
|
self._state: ConnectionState = state
|
|
self.message_id: Optional[int] = utils._get_as_snowflake(data, 'message_id')
|
|
self.action: AutoModRuleAction = AutoModRuleAction.from_data(data['action'])
|
|
self.rule_id: int = int(data['rule_id'])
|
|
self.rule_trigger_type: AutoModRuleTriggerType = try_enum(AutoModRuleTriggerType, data['rule_trigger_type'])
|
|
self.guild_id: int = int(data['guild_id'])
|
|
self.channel_id: Optional[int] = utils._get_as_snowflake(data, 'channel_id')
|
|
self.user_id: int = int(data['user_id'])
|
|
self.alert_system_message_id: Optional[int] = utils._get_as_snowflake(data, 'alert_system_message_id')
|
|
self.content: str = data.get('content', '')
|
|
self.matched_keyword: Optional[str] = data['matched_keyword']
|
|
self.matched_content: Optional[str] = data.get('matched_content')
|
|
|
|
def __repr__(self) -> str:
|
|
return f'<AutoModRuleExecution rule_id={self.rule_id} action={self.action!r}>'
|
|
|
|
@property
|
|
def guild(self) -> Guild:
|
|
""":class:`Guild`: The guild this action was taken in."""
|
|
return self._state._get_or_create_unavailable_guild(self.guild_id)
|
|
|
|
@property
|
|
def channel(self) -> Optional[Union[GuildChannel, Thread]]:
|
|
"""Optional[Union[:class:`abc.GuildChannel`, :class:`Thread`]]: The channel this action was taken in."""
|
|
if self.channel_id:
|
|
return self.guild.get_channel_or_thread(self.channel_id)
|
|
return None
|
|
|
|
@property
|
|
def member(self) -> Optional[Member]:
|
|
"""Optional[:class:`Member`]: The member this action was taken against /who triggered this rule."""
|
|
return self.guild.get_member(self.user_id)
|
|
|
|
async def fetch_rule(self) -> AutoModRule:
|
|
"""|coro|
|
|
|
|
Fetch the rule whose action was taken.
|
|
|
|
You must have :attr:`Permissions.manage_guild` to do this.
|
|
|
|
Raises
|
|
-------
|
|
Forbidden
|
|
You do not have permissions to view the rule.
|
|
HTTPException
|
|
Fetching the rule failed.
|
|
|
|
Returns
|
|
--------
|
|
:class:`AutoModRule`
|
|
The rule that was executed.
|
|
"""
|
|
|
|
data = await self._state.http.get_auto_moderation_rule(self.guild.id, self.rule_id)
|
|
return AutoModRule(data=data, guild=self.guild, state=self._state)
|
|
|