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.
268 lines
9.5 KiB
268 lines
9.5 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.
|
|
"""
|
|
|
|
from copy import deepcopy
|
|
from . import utils
|
|
from .permissions import Permissions
|
|
from .enums import ChannelType
|
|
from collections import namedtuple
|
|
from .mixins import Hashable
|
|
|
|
Overwrites = namedtuple('Overwrites', 'id allow deny type')
|
|
|
|
class Channel(Hashable):
|
|
"""Represents a Discord server channel.
|
|
|
|
Supported Operations:
|
|
|
|
+-----------+---------------------------------------+
|
|
| Operation | Description |
|
|
+===========+=======================================+
|
|
| x == y | Checks if two channels are equal. |
|
|
+-----------+---------------------------------------+
|
|
| x != y | Checks if two channels are not equal. |
|
|
+-----------+---------------------------------------+
|
|
| hash(x) | Returns the channel's hash. |
|
|
+-----------+---------------------------------------+
|
|
| str(x) | Returns the channel's name. |
|
|
+-----------+---------------------------------------+
|
|
|
|
Attributes
|
|
-----------
|
|
name : str
|
|
The channel name.
|
|
server : :class:`Server`
|
|
The server the channel belongs to.
|
|
id : str
|
|
The channel ID.
|
|
topic : Optional[str]
|
|
The channel's topic. None if it doesn't exist.
|
|
is_private : bool
|
|
``True`` if the channel is a private channel (i.e. PM). ``False`` in this case.
|
|
position : int
|
|
The position in the channel list.
|
|
type : :class:`ChannelType`
|
|
The channel type. There is a chance that the type will be ``str`` if
|
|
the channel type is not within the ones recognised by the enumerator.
|
|
changed_roles
|
|
A list of :class:`Roles` that have been overridden from their default
|
|
values in the :attr:`Server.roles` attribute.
|
|
voice_members
|
|
A list of :class:`Members` that are currently inside this voice channel.
|
|
If :attr:`type` is not :attr:`ChannelType.voice` then this is always an empty array.
|
|
"""
|
|
|
|
def __init__(self, **kwargs):
|
|
self.update(**kwargs)
|
|
self.voice_members = []
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
def update(self, **kwargs):
|
|
self.name = kwargs.get('name')
|
|
self.server = kwargs.get('server')
|
|
self.id = kwargs.get('id')
|
|
self.topic = kwargs.get('topic')
|
|
self.is_private = False
|
|
self.position = kwargs.get('position')
|
|
self.type = kwargs.get('type')
|
|
try:
|
|
self.type = ChannelType(self.type)
|
|
except:
|
|
pass
|
|
|
|
self.changed_roles = []
|
|
self._permission_overwrites = []
|
|
for overridden in kwargs.get('permission_overwrites', []):
|
|
self._permission_overwrites.append(Overwrites(**overridden))
|
|
|
|
if overridden.get('type') == 'member':
|
|
continue
|
|
|
|
# this is pretty inefficient due to the deep nested loops unfortunately
|
|
role = utils.find(lambda r: r.id == overridden['id'], self.server.roles)
|
|
if role is None:
|
|
continue
|
|
|
|
denied = overridden.get('deny', 0)
|
|
allowed = overridden.get('allow', 0)
|
|
override = deepcopy(role)
|
|
override.permissions.handle_overwrite(allowed, denied)
|
|
self.changed_roles.append(override)
|
|
|
|
@property
|
|
def is_default(self):
|
|
"""bool : Indicates if this is the default channel for the :class:`Server` it belongs to."""
|
|
return self.server.id == self.id
|
|
|
|
@property
|
|
def mention(self):
|
|
"""str : The string that allows you to mention the channel."""
|
|
return '<#{0.id}>'.format(self)
|
|
|
|
def permissions_for(self, member):
|
|
"""Handles permission resolution for the current :class:`Member`.
|
|
|
|
This function takes into consideration the following cases:
|
|
|
|
- Server owner
|
|
- Server roles
|
|
- Channel overrides
|
|
- Member overrides
|
|
- Whether the channel is the default channel.
|
|
|
|
Parameters
|
|
----------
|
|
member : :class:`Member`
|
|
The member to resolve permissions for.
|
|
|
|
Returns
|
|
-------
|
|
:class:`Permissions`
|
|
The resolved permissions for the member.
|
|
"""
|
|
|
|
# The current cases can be explained as:
|
|
# Server owner get all permissions -- no questions asked. Otherwise...
|
|
# The @everyone role gets the first application.
|
|
# After that, the applied roles that the user has in the channel
|
|
# (or otherwise) are then OR'd together.
|
|
# After the role permissions are resolved, the member permissions
|
|
# have to take into effect.
|
|
# After all that is done.. you have to do the following:
|
|
|
|
# If manage permissions is True, then all permissions are set to
|
|
# True. If the channel is the default channel then everyone gets
|
|
# read permissions regardless.
|
|
|
|
# The operation first takes into consideration the denied
|
|
# and then the allowed.
|
|
|
|
if member.id == self.server.owner.id:
|
|
return Permissions.all()
|
|
|
|
default = self.server.default_role
|
|
base = deepcopy(default.permissions)
|
|
|
|
# Apply server roles that the member has.
|
|
for role in member.roles:
|
|
base.value |= role.permissions.value
|
|
|
|
# Server-wide Manage Roles -> True for everything
|
|
if base.manage_roles:
|
|
base = Permissions.all()
|
|
|
|
member_role_ids = set(map(lambda r: r.id, member.roles))
|
|
|
|
# Apply channel specific role permission overwrites
|
|
for overwrite in self._permission_overwrites:
|
|
if overwrite.type == 'role':
|
|
if overwrite.id in member_role_ids:
|
|
base.handle_overwrite(allow=overwrite.allow, deny=overwrite.deny)
|
|
|
|
# Apply member specific permission overwrites
|
|
for overwrite in self._permission_overwrites:
|
|
if overwrite.type == 'member' and overwrite.id == member.id:
|
|
base.handle_overwrite(allow=overwrite.allow, deny=overwrite.deny)
|
|
|
|
if base.manage_roles:
|
|
# This point is essentially Channel-specific Manage Roles.
|
|
tmp = Permissions.all_channel()
|
|
base.value |= tmp.value
|
|
|
|
if self.is_default:
|
|
base.read_messages = True
|
|
|
|
return base
|
|
|
|
class PrivateChannel(Hashable):
|
|
"""Represents a Discord private channel.
|
|
|
|
Supported Operations:
|
|
|
|
+-----------+-------------------------------------------------+
|
|
| Operation | Description |
|
|
+===========+=================================================+
|
|
| x == y | Checks if two channels are equal. |
|
|
+-----------+-------------------------------------------------+
|
|
| x != y | Checks if two channels are not equal. |
|
|
+-----------+-------------------------------------------------+
|
|
| hash(x) | Returns the channel's hash. |
|
|
+-----------+-------------------------------------------------+
|
|
| str(x) | Returns the string "Direct Message with <User>" |
|
|
+-----------+-------------------------------------------------+
|
|
|
|
Attributes
|
|
----------
|
|
user : :class:`User`
|
|
The user you are participating with in the private channel.
|
|
id : str
|
|
The private channel ID.
|
|
is_private : bool
|
|
``True`` if the channel is a private channel (i.e. PM). ``True`` in this case.
|
|
"""
|
|
|
|
__slots__ = ['user', 'id', 'is_private']
|
|
|
|
def __init__(self, user, id, **kwargs):
|
|
self.user = user
|
|
self.id = id
|
|
self.is_private = True
|
|
|
|
def __str__(self):
|
|
return 'Direct Message with {0.name}'.format(self.user)
|
|
|
|
def permissions_for(user):
|
|
"""Handles permission resolution for a :class:`User`.
|
|
|
|
This function is there for compatibility with :class:`Channel`.
|
|
|
|
Actual private messages do not really have the concept of permissions.
|
|
|
|
This returns all the Text related permissions set to true except:
|
|
|
|
- send_tts_messages: You cannot send TTS messages in a PM.
|
|
- manage_messages: You cannot delete others messages in a PM.
|
|
- mention_everyone: There is no one to mention in a PM.
|
|
|
|
Parameters
|
|
-----------
|
|
user : :class:`User`
|
|
The user to check permissions for.
|
|
|
|
Returns
|
|
--------
|
|
:class:`Permission`
|
|
The resolved permissions for the user.
|
|
"""
|
|
|
|
base = Permissions.text()
|
|
base.send_tts_messages = False
|
|
base.manage_messages = False
|
|
base.mention_everyone = False
|
|
return base
|
|
|
|
|
|
|