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.
580 lines
21 KiB
580 lines
21 KiB
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
The MIT License (MIT)
|
|
|
|
Copyright (c) 2015-2016 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.
|
|
"""
|
|
|
|
import asyncio
|
|
import re
|
|
|
|
import discord.utils
|
|
import discord.abc
|
|
from .user import User
|
|
from .reaction import Reaction
|
|
from .emoji import Emoji
|
|
from .object import Object
|
|
from .calls import CallMessage
|
|
from .enums import MessageType, try_enum
|
|
from .errors import InvalidArgument
|
|
from .embeds import Embed
|
|
|
|
class Message:
|
|
"""Represents a message from Discord.
|
|
|
|
There should be no need to create one of these manually.
|
|
|
|
Attributes
|
|
-----------
|
|
edited_timestamp: Optional[datetime.datetime]
|
|
A naive UTC datetime object containing the edited time of the message.
|
|
tts: bool
|
|
Specifies if the message was done with text-to-speech.
|
|
type: :class:`MessageType`
|
|
The type of message. In most cases this should not be checked, but it is helpful
|
|
in cases where it might be a system message for :attr:`system_content`.
|
|
author
|
|
A :class:`Member` that sent the message. If :attr:`channel` is a
|
|
private channel, then it is a :class:`User` instead.
|
|
content: str
|
|
The actual contents of the message.
|
|
nonce
|
|
The value used by the discord guild and the client to verify that the message is successfully sent.
|
|
This is typically non-important.
|
|
embeds: List[:class:`Embed`]
|
|
A list embeds the message has.
|
|
channel
|
|
The :class:`Channel` that the message was sent from.
|
|
Could be a :class:`PrivateChannel` if it's a private message.
|
|
In :issue:`very rare cases <21>` this could be a :class:`Object` instead.
|
|
|
|
For the sake of convenience, this :class:`Object` instance has an attribute ``is_private`` set to ``True``.
|
|
call: Optional[:class:`CallMessage`]
|
|
The call that the message refers to. This is only applicable to messages of type
|
|
:attr:`MessageType.call`.
|
|
mention_everyone: bool
|
|
Specifies if the message mentions everyone.
|
|
|
|
.. note::
|
|
|
|
This does not check if the ``@everyone`` text is in the message itself.
|
|
Rather this boolean indicates if the ``@everyone`` text is in the message
|
|
**and** it did end up mentioning everyone.
|
|
|
|
mentions: list
|
|
A list of :class:`Member` that were mentioned. If the message is in a private message
|
|
then the list will be of :class:`User` instead. For messages that are not of type
|
|
:attr:`MessageType.default`\, this array can be used to aid in system messages.
|
|
For more information, see :attr:`system_content`.
|
|
|
|
.. warning::
|
|
|
|
The order of the mentions list is not in any particular order so you should
|
|
not rely on it. This is a discord limitation, not one with the library.
|
|
|
|
channel_mentions: list
|
|
A list of :class:`Channel` that were mentioned. If the message is in a private message
|
|
then the list is always empty.
|
|
role_mentions: list
|
|
A list of :class:`Role` that were mentioned. If the message is in a private message
|
|
then the list is always empty.
|
|
id: int
|
|
The message ID.
|
|
webhook_id: Optional[int]
|
|
If this message was sent by a webhook, then this is the webhook ID's that sent this
|
|
message.
|
|
attachments: list
|
|
A list of attachments given to a message.
|
|
pinned: bool
|
|
Specifies if the message is currently pinned.
|
|
reactions : List[:class:`Reaction`]
|
|
Reactions to a message. Reactions can be either custom emoji or standard unicode emoji.
|
|
"""
|
|
|
|
__slots__ = ( 'edited_timestamp', 'tts', 'content', 'channel', 'webhook_id',
|
|
'mention_everyone', 'embeds', 'id', 'mentions', 'author',
|
|
'_cs_channel_mentions', '_cs_raw_mentions', 'attachments',
|
|
'_cs_clean_content', '_cs_raw_channel_mentions', 'nonce', 'pinned',
|
|
'role_mentions', '_cs_raw_role_mentions', 'type', 'call',
|
|
'_cs_system_content', '_cs_guild', '_state', 'reactions' )
|
|
|
|
def __init__(self, *, state, channel, data):
|
|
self._state = state
|
|
self.id = int(data['id'])
|
|
self.reactions = [Reaction(message=self, data=d) for d in data.get('reactions', [])]
|
|
self._update(channel, data)
|
|
|
|
def __repr__(self):
|
|
return '<Message id={0.id} pinned={0.pinned} author={0.author!r}>'.format(self)
|
|
|
|
def _try_patch(self, data, key, transform):
|
|
try:
|
|
value = data[key]
|
|
except KeyError:
|
|
pass
|
|
else:
|
|
setattr(self, key, transform(value))
|
|
|
|
def _add_reaction(self, data):
|
|
emoji = self._state.reaction_emoji(data['emoji'])
|
|
reaction = discord.utils.find(lambda r: r.emoji == emoji, self.reactions)
|
|
is_me = data['me'] = int(data['user_id']) == self._state.self_id
|
|
|
|
if reaction is None:
|
|
reaction = Reaction(message=self, data=data, emoji=emoji)
|
|
self.reactions.append(reaction)
|
|
else:
|
|
reaction.count += 1
|
|
if is_me:
|
|
reaction.me = is_me
|
|
|
|
return reaction
|
|
|
|
def _remove_reaction(self, data):
|
|
emoji = self._state.reaction_emoji(data['emoji'])
|
|
reaction = discord.utils.find(lambda r: r.emoji == emoji, self.reactions)
|
|
|
|
if reaction is None:
|
|
# already removed?
|
|
raise ValueError('Emoji already removed?')
|
|
|
|
# if reaction isn't in the list, we crash. This means discord
|
|
# sent bad data, or we stored improperly
|
|
reaction.count -= 1
|
|
|
|
if int(data['user_id']) == self._state.self_id:
|
|
reaction.me = False
|
|
if reaction.count == 0:
|
|
# this raises ValueError if something went wrong as well.
|
|
self.reactions.remove(reaction)
|
|
|
|
return reaction
|
|
|
|
def _update(self, channel, data):
|
|
self.channel = channel
|
|
for handler in ('author', 'mentions', 'mention_roles', 'call'):
|
|
try:
|
|
getattr(self, '_handle_%s' % handler)(data[handler])
|
|
except KeyError:
|
|
continue
|
|
|
|
self._try_patch(data, 'edited_timestamp', discord.utils.parse_time)
|
|
self._try_patch(data, 'pinned', bool)
|
|
self._try_patch(data, 'mention_everyone', bool)
|
|
self._try_patch(data, 'tts', bool)
|
|
self._try_patch(data, 'type', lambda x: try_enum(MessageType, x))
|
|
self._try_patch(data, 'content', str)
|
|
self._try_patch(data, 'attachments', lambda x: x)
|
|
self._try_patch(data, 'embeds', lambda x: list(map(Embed.from_data, x)))
|
|
self._try_patch(data, 'nonce', lambda x: x)
|
|
|
|
# clear the cached properties
|
|
cached = filter(lambda attr: attr.startswith('_cs_'), self.__slots__)
|
|
for attr in cached:
|
|
try:
|
|
delattr(self, attr)
|
|
except AttributeError:
|
|
pass
|
|
|
|
def _handle_author(self, author):
|
|
self.author = self._state.store_user(author)
|
|
if self.guild is not None:
|
|
found = self.guild.get_member(self.author.id)
|
|
if found is not None:
|
|
self.author = found
|
|
|
|
def _handle_mentions(self, mentions):
|
|
self.mentions = []
|
|
if self.guild is None:
|
|
self.mentions = [self._state.store_user(m) for m in mentions]
|
|
return
|
|
|
|
for mention in mentions:
|
|
id_search = int(mention['id'])
|
|
member = self.guild.get_member(id_search)
|
|
if member is not None:
|
|
self.mentions.append(member)
|
|
|
|
def _handle_mention_roles(self, role_mentions):
|
|
self.role_mentions = []
|
|
if self.guild is not None:
|
|
for role_id in role_mentions:
|
|
role = discord.utils.get(self.guild.roles, id=role_id)
|
|
if role is not None:
|
|
self.role_mentions.append(role)
|
|
|
|
def _handle_call(self, call):
|
|
if call is None or self.type is not MessageType.call:
|
|
self.call = None
|
|
return
|
|
|
|
# we get the participant source from the mentions array or
|
|
# the author
|
|
|
|
participants = []
|
|
for uid in map(int, call.get('participants', [])):
|
|
if uid == self.author.id:
|
|
participants.append(self.author)
|
|
else:
|
|
user = discord.utils.find(lambda u: u.id == uid, self.mentions)
|
|
if user is not None:
|
|
participants.append(user)
|
|
|
|
call['participants'] = participants
|
|
self.call = CallMessage(message=self, **call)
|
|
|
|
@discord.utils.cached_slot_property('_cs_guild')
|
|
def guild(self):
|
|
"""Optional[:class:`Guild`]: The guild that the message belongs to, if applicable."""
|
|
return getattr(self.channel, 'guild', None)
|
|
|
|
@discord.utils.cached_slot_property('_cs_raw_mentions')
|
|
def raw_mentions(self):
|
|
"""A property that returns an array of user IDs matched with
|
|
the syntax of <@user_id> in the message content.
|
|
|
|
This allows you receive the user IDs of mentioned users
|
|
even in a private message context.
|
|
"""
|
|
return [int(x) for x in re.findall(r'<@!?([0-9]+)>', self.content)]
|
|
|
|
@discord.utils.cached_slot_property('_cs_raw_channel_mentions')
|
|
def raw_channel_mentions(self):
|
|
"""A property that returns an array of channel IDs matched with
|
|
the syntax of <#channel_id> in the message content.
|
|
"""
|
|
return [int(x) for x in re.findall(r'<#([0-9]+)>', self.content)]
|
|
|
|
@discord.utils.cached_slot_property('_cs_raw_role_mentions')
|
|
def raw_role_mentions(self):
|
|
"""A property that returns an array of role IDs matched with
|
|
the syntax of <@&role_id> in the message content.
|
|
"""
|
|
return [int(x) for x in re.findall(r'<@&([0-9]+)>', self.content)]
|
|
|
|
@discord.utils.cached_slot_property('_cs_channel_mentions')
|
|
def channel_mentions(self):
|
|
if self.guild is None:
|
|
return []
|
|
it = filter(None, map(lambda m: self.guild.get_channel(m), self.raw_channel_mentions))
|
|
return discord.utils._unique(it)
|
|
|
|
@discord.utils.cached_slot_property('_cs_clean_content')
|
|
def clean_content(self):
|
|
"""A property that returns the content in a "cleaned up"
|
|
manner. This basically means that mentions are transformed
|
|
into the way the client shows it. e.g. ``<#id>`` will transform
|
|
into ``#name``.
|
|
|
|
This will also transform @everyone and @here mentions into
|
|
non-mentions.
|
|
"""
|
|
|
|
transformations = {
|
|
re.escape('<#{0.id}>'.format(channel)): '#' + channel.name
|
|
for channel in self.channel_mentions
|
|
}
|
|
|
|
mention_transforms = {
|
|
re.escape('<@{0.id}>'.format(member)): '@' + member.display_name
|
|
for member in self.mentions
|
|
}
|
|
|
|
# add the <@!user_id> cases as well..
|
|
second_mention_transforms = {
|
|
re.escape('<@!{0.id}>'.format(member)): '@' + member.display_name
|
|
for member in self.mentions
|
|
}
|
|
|
|
transformations.update(mention_transforms)
|
|
transformations.update(second_mention_transforms)
|
|
|
|
if self.guild is not None:
|
|
role_transforms = {
|
|
re.escape('<@&{0.id}>'.format(role)): '@' + role.name
|
|
for role in self.role_mentions
|
|
}
|
|
transformations.update(role_transforms)
|
|
|
|
def repl(obj):
|
|
return transformations.get(re.escape(obj.group(0)), '')
|
|
|
|
pattern = re.compile('|'.join(transformations.keys()))
|
|
result = pattern.sub(repl, self.content)
|
|
|
|
transformations = {
|
|
'@everyone': '@\u200beveryone',
|
|
'@here': '@\u200bhere'
|
|
}
|
|
|
|
def repl2(obj):
|
|
return transformations.get(obj.group(0), '')
|
|
|
|
pattern = re.compile('|'.join(transformations.keys()))
|
|
return pattern.sub(repl2, result)
|
|
|
|
def _handle_upgrades(self, channel_id):
|
|
self.guild = None
|
|
if isinstance(self.channel, Object):
|
|
return
|
|
|
|
if self.channel is None:
|
|
if channel_id is not None:
|
|
self.channel = Object(id=channel_id)
|
|
self.channel.is_private = True
|
|
return
|
|
|
|
if isinstance(self.channel, discord.abc.GuildChannel):
|
|
self.guild = self.channel.guild
|
|
found = self.guild.get_member(self.author.id)
|
|
if found is not None:
|
|
self.author = found
|
|
|
|
@property
|
|
def created_at(self):
|
|
"""Returns the message's creation time in UTC."""
|
|
return discord.utils.snowflake_time(self.id)
|
|
|
|
@discord.utils.cached_slot_property('_cs_system_content')
|
|
def system_content(self):
|
|
"""A property that returns the content that is rendered
|
|
regardless of the :attr:`Message.type`.
|
|
|
|
In the case of :attr:`MessageType.default`\, this just returns the
|
|
regular :attr:`Message.content`. Otherwise this returns an English
|
|
message denoting the contents of the system message.
|
|
"""
|
|
|
|
if self.type is MessageType.default:
|
|
return self.content
|
|
|
|
if self.type is MessageType.pins_add:
|
|
return '{0.name} pinned a message to this channel.'.format(self.author)
|
|
|
|
if self.type is MessageType.recipient_add:
|
|
return '{0.name} added {1.name} to the group.'.format(self.author, self.mentions[0])
|
|
|
|
if self.type is MessageType.recipient_remove:
|
|
return '{0.name} removed {1.name} from the group.'.format(self.author, self.mentions[0])
|
|
|
|
if self.type is MessageType.channel_name_change:
|
|
return '{0.author.name} changed the channel name: {0.content}'.format(self)
|
|
|
|
if self.type is MessageType.channel_icon_change:
|
|
return '{0.author.name} changed the channel icon.'.format(self)
|
|
|
|
if self.type is MessageType.call:
|
|
# we're at the call message type now, which is a bit more complicated.
|
|
# we can make the assumption that Message.channel is a PrivateChannel
|
|
# with the type ChannelType.group or ChannelType.private
|
|
call_ended = self.call.ended_timestamp is not None
|
|
|
|
if self.channel.me in self.call.participants:
|
|
return '{0.author.name} started a call.'.format(self)
|
|
elif call_ended:
|
|
return 'You missed a call from {0.author.name}'.format(self)
|
|
else:
|
|
return '{0.author.name} started a call \N{EM DASH} Join the call.'.format(self)
|
|
|
|
@asyncio.coroutine
|
|
def delete(self):
|
|
"""|coro|
|
|
|
|
Deletes the message.
|
|
|
|
Your own messages could be deleted without any proper permissions. However to
|
|
delete other people's messages, you need the :attr:`Permissions.manage_messages`
|
|
permission.
|
|
|
|
Raises
|
|
------
|
|
Forbidden
|
|
You do not have proper permissions to delete the message.
|
|
HTTPException
|
|
Deleting the message failed.
|
|
"""
|
|
yield from self._state.http.delete_message(self.channel.id, self.id, getattr(self.guild, 'id', None))
|
|
|
|
@asyncio.coroutine
|
|
def edit(self, *, content: str = None, embed: Embed = None):
|
|
"""|coro|
|
|
|
|
Edits the message.
|
|
|
|
The content must be able to be transformed into a string via ``str(content)``.
|
|
|
|
Parameters
|
|
-----------
|
|
content: str
|
|
The new content to replace the message with.
|
|
embed: :class:`Embed`
|
|
The new embed to replace the original with.
|
|
|
|
Raises
|
|
-------
|
|
HTTPException
|
|
Editing the message failed.
|
|
"""
|
|
|
|
guild_id = getattr(self.guild, 'id', None)
|
|
content = str(content) if content else None
|
|
embed = embed.to_dict() if embed else None
|
|
data = yield from self._state.http.edit_message(self.id, self.channel.id, content, guild_id=guild_id, embed=embed)
|
|
self._update(channel=self.channel, data=data)
|
|
|
|
@asyncio.coroutine
|
|
def pin(self):
|
|
"""|coro|
|
|
|
|
Pins the message. You must have :attr:`Permissions.manage_messages`
|
|
permissions to do this in a non-private channel context.
|
|
|
|
Raises
|
|
-------
|
|
Forbidden
|
|
You do not have permissions to pin the message.
|
|
NotFound
|
|
The message or channel was not found or deleted.
|
|
HTTPException
|
|
Pinning the message failed, probably due to the channel
|
|
having more than 50 pinned messages.
|
|
"""
|
|
|
|
yield from self._state.http.pin_message(self.channel.id, self.id)
|
|
self.pinned = True
|
|
|
|
@asyncio.coroutine
|
|
def unpin(self):
|
|
"""|coro|
|
|
|
|
Unpins the message. You must have :attr:`Permissions.manage_messages`
|
|
permissions to do this in a non-private channel context.
|
|
|
|
Raises
|
|
-------
|
|
Forbidden
|
|
You do not have permissions to unpin the message.
|
|
NotFound
|
|
The message or channel was not found or deleted.
|
|
HTTPException
|
|
Unpinning the message failed.
|
|
"""
|
|
|
|
yield from self._state.http.unpin_message(self.channel.id, self.id)
|
|
self.pinned = False
|
|
|
|
@asyncio.coroutine
|
|
def add_reaction(self, emoji):
|
|
"""|coro|
|
|
|
|
Add a reaction to the message.
|
|
|
|
The emoji may be a unicode emoji or a custom server :class:`Emoji`.
|
|
|
|
You must have the :attr:`Permissions.add_reactions` permission to
|
|
add new reactions to a message.
|
|
|
|
Parameters
|
|
------------
|
|
emoji: :class:`Emoji` or str
|
|
The emoji to react with.
|
|
|
|
Raises
|
|
--------
|
|
HTTPException
|
|
Adding the reaction failed.
|
|
Forbidden
|
|
You do not have the proper permissions to react to the message.
|
|
NotFound
|
|
The emoji you specified was not found.
|
|
InvalidArgument
|
|
The emoji parameter is invalid.
|
|
"""
|
|
|
|
if isinstance(emoji, Emoji):
|
|
emoji = '%s:%s' % (emoji.name, emoji.id)
|
|
elif isinstance(emoji, str):
|
|
pass # this is okay
|
|
else:
|
|
raise InvalidArgument('emoji argument must be a string or discord.Emoji')
|
|
|
|
yield from self._state.http.add_reaction(self.id, self.channel.id, emoji)
|
|
|
|
@asyncio.coroutine
|
|
def remove_reaction(self, emoji, member):
|
|
"""|coro|
|
|
|
|
Remove a reaction by the member from the message.
|
|
|
|
The emoji may be a unicode emoji or a custom server :class:`Emoji`.
|
|
|
|
If the reaction is not your own (i.e. ``member`` parameter is not you) then
|
|
the :attr:`Permissions.manage_messages` permission is needed.
|
|
|
|
The ``member`` parameter must represent a member and meet
|
|
the :class:`abc.Snowflake` abc.
|
|
|
|
Parameters
|
|
------------
|
|
emoji: :class:`Emoji` or str
|
|
The emoji to remove.
|
|
member: :class:`abc.Snowflake`
|
|
The member for which to remove the reaction.
|
|
|
|
Raises
|
|
--------
|
|
HTTPException
|
|
Removing the reaction failed.
|
|
Forbidden
|
|
You do not have the proper permissions to remove the reaction.
|
|
NotFound
|
|
The member or emoji you specified was not found.
|
|
InvalidArgument
|
|
The emoji parameter is invalid.
|
|
"""
|
|
|
|
if isinstance(emoji, Emoji):
|
|
emoji = '%s:%s' % (emoji.name, emoji.id)
|
|
elif isinstance(emoji, str):
|
|
pass # this is okay
|
|
else:
|
|
raise InvalidArgument('emoji argument must be a string or discord.Emoji')
|
|
|
|
yield from self._state.http.remove_reaction(self.id, self.channel.id, emoji, member.id)
|
|
|
|
@asyncio.coroutine
|
|
def clear_reactions(self):
|
|
"""|coro|
|
|
|
|
Removes all the reactions from the message.
|
|
|
|
You need :attr:`Permissions.manage_messages` permission
|
|
to use this.
|
|
|
|
Raises
|
|
--------
|
|
HTTPException
|
|
Removing the reactions failed.
|
|
Forbidden
|
|
You do not have the proper permissions to remove all the reactions.
|
|
"""
|
|
yield from self._state.http.clear_reactions(self.id, self.channel.id)
|
|
|