From ac04f999cb63aba4c8f5abb4f38497ad9fb2ef3a Mon Sep 17 00:00:00 2001 From: Andrei Date: Wed, 1 May 2019 13:15:33 -0700 Subject: [PATCH 01/12] Add API Client support for kicking guild members --- disco/api/client.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/disco/api/client.py b/disco/api/client.py index 2437ff4..0993ea3 100644 --- a/disco/api/client.py +++ b/disco/api/client.py @@ -348,7 +348,7 @@ class APIClient(LoggingClass): self.http( Routes.GUILDS_MEMBERS_MODIFY, dict(guild=guild, member=member), - json=optional(**kwargs), + json=kwargs, headers=_reason_header(reason)) def guilds_members_roles_add(self, guild, member, role, reason=None): From 06a5e4921a266f479d6c56dd270cfde16c190744 Mon Sep 17 00:00:00 2001 From: Andrei Date: Wed, 1 May 2019 13:17:43 -0700 Subject: [PATCH 02/12] Add GuildMember.disconnect to disconnect a member from voice --- disco/types/guild.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/disco/types/guild.py b/disco/types/guild.py index b02091d..80149f2 100644 --- a/disco/types/guild.py +++ b/disco/types/guild.py @@ -222,6 +222,12 @@ class GuildMember(SlottedModel): else: self.client.api.guilds_members_modify(self.guild.id, self.user.id, nick=nickname or '', **kwargs) + def disconnect(self): + """ + Disconnects the member from voice (if they are connected). + """ + self.modify(channel_id=None) + def modify(self, **kwargs): self.client.api.guilds_members_modify(self.guild.id, self.user.id, **kwargs) From 9bcc9185e7faf1119155caaf0d524440b9748825 Mon Sep 17 00:00:00 2001 From: Justin <14909116+ThatGuyJustin@users.noreply.github.com> Date: Sat, 4 May 2019 13:12:14 -0400 Subject: [PATCH 03/12] Updating emoji cdn (#137) Apparently the /api endpoint stopped working, so updating to use the cdn link. --- disco/types/guild.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/disco/types/guild.py b/disco/types/guild.py index 80149f2..929ee9f 100644 --- a/disco/types/guild.py +++ b/disco/types/guild.py @@ -75,7 +75,7 @@ class GuildEmoji(Emoji): @property def url(self): - return 'https://discordapp.com/api/emojis/{}.{}'.format(self.id, 'gif' if self.animated else 'png') + return 'https://cdn.discordapp.com/emojis/{}.{}'.format(self.id, 'gif' if self.animated else 'png') @cached_property def guild(self): From cec1afd17855685569e306ac7b83d98c6c86530a Mon Sep 17 00:00:00 2001 From: Justin <14909116+ThatGuyJustin@users.noreply.github.com> Date: Thu, 27 Jun 2019 13:07:49 -0400 Subject: [PATCH 04/12] Adding missing permissions (#144) - Add Reactions - View Audit Log - Priority Speaker (cherry picked from commit 4fb136ee085b9219ae57d844bfe0c4999ea33219) --- disco/types/permissions.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/disco/types/permissions.py b/disco/types/permissions.py index d11b08f..37f7e2a 100644 --- a/disco/types/permissions.py +++ b/disco/types/permissions.py @@ -7,6 +7,9 @@ Permissions = Enum( ADMINISTRATOR=1 << 3, MANAGE_CHANNELS=1 << 4, MANAGE_GUILD=1 << 5, + ADD_REACTIONS=1 << 6, + VIEW_AUDIT_LOG=1 << 7, + PRIORITY_SPEAKER=1 << 8, READ_MESSAGES=1 << 10, SEND_MESSAGES=1 << 11, SEND_TSS_MESSAGES=1 << 12, From 02c0051f755edac47f1e5c8d4e72bab55490bc26 Mon Sep 17 00:00:00 2001 From: One-Nub <38899321+One-Nub@users.noreply.github.com> Date: Sun, 6 Oct 2019 18:02:15 -0500 Subject: [PATCH 05/12] Defining a specific YAML loader (#155) As per using one of the "'sugar' methods" mentioned in https://github.com/yaml/pyyaml/wiki/PyYAML-yaml.load(input)-Deprecation#how-to-disable-the-warning --- disco/util/serializer.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/disco/util/serializer.py b/disco/util/serializer.py index de6264a..f842ecc 100644 --- a/disco/util/serializer.py +++ b/disco/util/serializer.py @@ -21,8 +21,8 @@ class Serializer(object): @staticmethod def yaml(): - from yaml import load, dump - return (load, dump) + from yaml import full_load, dump + return (full_load, dump) @staticmethod def pickle(): From a98cd1aac99e73940adce6b2ea9384e924f3868c Mon Sep 17 00:00:00 2001 From: Andrei Date: Sat, 2 Nov 2019 15:49:48 -0700 Subject: [PATCH 06/12] readme: remove invite url --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 6d1413f..b71d8ac 100644 --- a/README.md +++ b/README.md @@ -4,7 +4,7 @@ [![PyPI](https://img.shields.io/pypi/v/disco-py.svg)](https://pypi.python.org/pypi/disco-py/) [![TravisCI](https://img.shields.io/travis/b1naryth1ef/disco.svg)](https://travis-ci.org/b1naryth1ef/disco/) -Disco is an extensive and extendable Python 2.x/3.x library for the [Discord API](https://discordapp.com/developers/docs/intro). Join the Official channel and chat [here](https://discord.gg/WMzzPec). Disco boasts the following major features: +Disco is an extensive and extendable Python 2.x/3.x library for the [Discord API](https://discordapp.com/developers/docs/intro). Disco boasts the following major features: - Expressive, functional interface that gets out of the way - Built for high-performance and efficiency From 31b688ab89d8b9558c2d2b1a4bba2a2119d3aa94 Mon Sep 17 00:00:00 2001 From: Oleg Butuzov Date: Wed, 20 Nov 2019 21:39:25 +0200 Subject: [PATCH 07/12] Added Long Description type (#166) In order to display long description properly at the pypi.org page we need to specify mime type `text/markdown` for the long desccription if readme is in markdown. Current https://pypi.org/project/disco-py/, doesn't looks nice atm. --- setup.py | 1 + 1 file changed, 1 insertion(+) diff --git a/setup.py b/setup.py index eae8156..8d5224c 100644 --- a/setup.py +++ b/setup.py @@ -32,6 +32,7 @@ setup( license='MIT', description='A Python library for Discord', long_description=readme, + long_description_content_type="text/markdown", include_package_data=True, install_requires=requirements, extras_require=extras_require, From 9ffac51b1979bc74c0f004527e82256df664a9d9 Mon Sep 17 00:00:00 2001 From: Matt Carey Date: Sat, 7 Dec 2019 22:51:38 -0500 Subject: [PATCH 08/12] Fix typo (#169) `xsalsa20_poly1305_suffx` -> `xsalsa20_poly1305_suffix` --- disco/voice/udp.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/disco/voice/udp.py b/disco/voice/udp.py index c3fc05c..8b32080 100644 --- a/disco/voice/udp.py +++ b/disco/voice/udp.py @@ -234,7 +234,7 @@ class UDPVoiceClient(LoggingClass): if self.vc.mode == 'xsalsa20_poly1305_lite': nonce[:4] = data[-4:] data = data[:-4] - elif self.vc.mode == 'xsalsa20_poly1305_suffx': + elif self.vc.mode == 'xsalsa20_poly1305_suffix': nonce[:24] = data[-24:] data = data[:-24] elif self.vc.mode == 'xsalsa20_poly1305': From fd63334c82c7aeb338f61d39d1308a25929569ae Mon Sep 17 00:00:00 2001 From: "slicedlime (Mikael Hedberg)" Date: Fri, 20 Dec 2019 20:37:36 +0100 Subject: [PATCH 09/12] Support 'User (mention)' style mentions used by recent client updates (#171) * Support 'User (mention)' style mentions. * Remove conditional that should no longer be needed. (from PR feedback) --- disco/bot/bot.py | 5 ++--- disco/types/user.py | 4 ++++ 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/disco/bot/bot.py b/disco/bot/bot.py index 0679381..d8a3c1d 100644 --- a/disco/bot/bot.py +++ b/disco/bot/bot.py @@ -308,10 +308,9 @@ class Bot(LoggingClass): if msg.guild: member = msg.guild.get_member(self.client.state.me) if member: - # If nickname is set, filter both the normal and nick mentions - if member.nick: - content = content.replace(member.mention, '', 1) + # Filter both the normal and nick mentions content = content.replace(member.user.mention, '', 1) + content = content.replace(member.user.mention_nickname, '', 1) else: content = content.replace(self.client.state.me.mention, '', 1) elif mention_everyone: diff --git a/disco/types/user.py b/disco/types/user.py index 63e9542..974233c 100644 --- a/disco/types/user.py +++ b/disco/types/user.py @@ -44,6 +44,10 @@ class User(SlottedModel, with_equality('id'), with_hash('id')): def mention(self): return '<@{}>'.format(self.id) + @property + def mention_nickname(self): + return '<@!{}>'.format(self.id) + def open_dm(self): return self.client.api.users_me_dms_create(self.id) From bbddbc4956c6ee3046a38438db3b42b032f1921d Mon Sep 17 00:00:00 2001 From: Luke Date: Fri, 27 Dec 2019 23:30:47 +0000 Subject: [PATCH 10/12] member_count tracking (#165) * member_count tracking * Remove unset checks. --- disco/state.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/disco/state.py b/disco/state.py index bb6b60d..0aa8888 100644 --- a/disco/state.py +++ b/disco/state.py @@ -285,6 +285,10 @@ class State(object): if event.member.guild_id not in self.guilds: return + # Avoid adding duplicate events to member_count. + if event.member.id not in self.guilds[event.member.guild_id].members: + self.guilds[event.member.guild_id].member_count += 1 + self.guilds[event.member.guild_id].members[event.member.id] = event.member def on_guild_member_update(self, event): @@ -303,6 +307,8 @@ class State(object): if event.user.id not in self.guilds[event.guild_id].members: return + self.guilds[event.guild_id].member_count -= 1 + del self.guilds[event.guild_id].members[event.user.id] def on_guild_members_chunk(self, event): From 8dbe0421f6545cdfc3b6e67bc4454ce130543478 Mon Sep 17 00:00:00 2001 From: Luke Date: Fri, 27 Dec 2019 23:33:58 +0000 Subject: [PATCH 11/12] Add user_id_or_ids and presences to guild member chunk and improve cache behaviour. (#162) * request_guild_members update, logging on emitter event exceptions and improved state caching. * Update the GuildMembersChunk GatewayEvent * Match presence update's presence handling in guild members chunk * Remove unnecessary string conversion * Remove member count tracking. * Split function * Remove optional report --- disco/api/client.py | 12 +----------- disco/gateway/client.py | 15 ++++++++++++++- disco/gateway/events.py | 6 ++++++ disco/state.py | 14 ++++++++++++++ disco/types/guild.py | 7 +++++-- disco/util/emitter.py | 23 ++++++++++++++++++----- disco/util/functional.py | 15 ++++++++++++--- 7 files changed, 70 insertions(+), 22 deletions(-) diff --git a/disco/api/client.py b/disco/api/client.py index ae75622..313a4bb 100644 --- a/disco/api/client.py +++ b/disco/api/client.py @@ -1,4 +1,3 @@ -import six import json import warnings @@ -7,6 +6,7 @@ from gevent.local import local from six.moves.urllib.parse import quote from disco.api.http import Routes, HTTPClient, to_bytes +from disco.util.functional import optional from disco.util.logging import LoggingClass from disco.util.sanitize import S from disco.types.user import User @@ -22,16 +22,6 @@ from disco.types.voice import VoiceRegion from disco.types.webhook import Webhook -def optional(**kwargs): - """ - Takes a set of keyword arguments, creating a dictionary with only the non- - null values. - - :returns: dict - """ - return {k: v for k, v in six.iteritems(kwargs) if v is not None} - - def _reason_header(value): return optional(**{'X-Audit-Log-Reason': quote(to_bytes(value)) if value else None}) diff --git a/disco/gateway/client.py b/disco/gateway/client.py index b778201..ce9fd79 100644 --- a/disco/gateway/client.py +++ b/disco/gateway/client.py @@ -273,7 +273,7 @@ class GatewayClient(LoggingClass): gevent.spawn(self.connect_and_run) self.ws_event.wait() - def request_guild_members(self, guild_id_or_ids, query=None, limit=0): + def request_guild_members(self, guild_id_or_ids, query=None, limit=0, presences=False): """ Request a batch of Guild members from Discord. Generally this function can be called when initially loading Guilds to fill the local member state. @@ -281,6 +281,19 @@ class GatewayClient(LoggingClass): self.send(OPCode.REQUEST_GUILD_MEMBERS, { # This is simply unfortunate naming on the part of Discord... 'guild_id': guild_id_or_ids, + 'limit': limit, + 'presences': presences, 'query': query or '', + }) + + def request_guild_members_by_id(self, guild_id_or_ids, user_id_or_ids, limit=0, presences=False): + """ + Request a batch of Guild members from Discord by their snowflake(s). + """ + self.send(OPCode.REQUEST_GUILD_MEMBERS, { + 'guild_id': guild_id_or_ids, 'limit': limit, + 'presences': presences, + # This is simply even more unfortunate naming from Discord... + 'user_ids': user_id_or_ids, }) diff --git a/disco/gateway/events.py b/disco/gateway/events.py index 61809fd..2d33a52 100644 --- a/disco/gateway/events.py +++ b/disco/gateway/events.py @@ -338,9 +338,15 @@ class GuildMembersChunk(GatewayEvent): The ID of the guild this member chunk is for. members : list[:class:`disco.types.guild.GuildMember`] The chunk of members. + not_found : list[snowflake] + An array of invalid requested guild members. + presences : list[:class:`disco.types.user.Presence`] + An array of requested member presence states. """ guild_id = Field(snowflake) members = ListField(GuildMember) + not_found = ListField(snowflake) + presences = ListField(Presence) @property def guild(self): diff --git a/disco/state.py b/disco/state.py index 0aa8888..2084be3 100644 --- a/disco/state.py +++ b/disco/state.py @@ -229,6 +229,11 @@ class State(object): elif event.channel.is_dm: self.dms[event.channel.id] = event.channel self.channels[event.channel.id] = event.channel + for user in six.itervalues(event.channel.recipients): + if user.id not in self.users: + self.users[user.id] = user + else: + event.channel.recipients[user.id] = self.users[user.id] def on_channel_update(self, event): if event.channel.id in self.channels: @@ -325,6 +330,15 @@ class State(object): else: member.user = self.users[member.id] + if not event.presences: + return + + for presence in event.presences: + # TODO: this matches the recursive, hackfix method found in on_presence_update + user = presence.user + user.presence = presence + self.users[user.id].inplace_update(user) + def on_guild_role_create(self, event): if event.guild_id not in self.guilds: return diff --git a/disco/types/guild.py b/disco/types/guild.py index 9a55e87..c220cba 100644 --- a/disco/types/guild.py +++ b/disco/types/guild.py @@ -455,8 +455,11 @@ class Guild(SlottedModel, Permissible): return self.client.api.guilds_roles_modify(self.id, to_snowflake(role), **kwargs) - def request_guild_members(self, query=None, limit=0): - self.client.gw.request_guild_members(self.id, query, limit) + def request_guild_members(self, query=None, limit=0, presences=False): + self.client.gw.request_guild_members(self.id, query, limit, presences) + + def request_guild_members_by_id(self, user_id_or_ids, limit=0, presences=False): + self.client.gw.request_guild_members_by_id(self.id, user_id_or_ids, limit, presences) def sync(self): warnings.warn( diff --git a/disco/util/emitter.py b/disco/util/emitter.py index 4382d4f..e5122e8 100644 --- a/disco/util/emitter.py +++ b/disco/util/emitter.py @@ -5,6 +5,9 @@ from gevent.event import AsyncResult from gevent.queue import Queue, Full +from disco.util.logging import LoggingClass + + class Priority(object): # BEFORE is the most dangerous priority level. Every event that flows through # the given emitter instance will be dispatched _sequentially_ to all BEFORE @@ -106,7 +109,7 @@ class EmitterSubscription(object): self.detach(emitter) -class Emitter(object): +class Emitter(LoggingClass): def __init__(self): self.event_handlers = { k: defaultdict(list) for k in Priority.ALL @@ -117,15 +120,25 @@ class Emitter(object): for listener in self.event_handlers[Priority.BEFORE].get(name, []): try: listener(*args, **kwargs) - except Exception: - pass + except Exception as e: + self.log.warning('BEFORE {} event handler `{}` raised {}: {}'.format( + name, + listener.callback.__name__, + e.__class__.__name__, + e, + )) # Next execute all AFTER handlers sequentially for listener in self.event_handlers[Priority.AFTER].get(name, []): try: listener(*args, **kwargs) - except Exception: - pass + except Exception as e: + self.log.warning('AFTER {} event handler `{}` raised {}: {}'.format( + name, + listener.callback.__name__, + e.__class__.__name__, + e, + )) # Next enqueue all sequential handlers. This just puts stuff into a queue # without blocking, so we don't have to worry too much diff --git a/disco/util/functional.py b/disco/util/functional.py index c8f084d..3efa4af 100644 --- a/disco/util/functional.py +++ b/disco/util/functional.py @@ -1,4 +1,4 @@ -from six.moves import range +import six NO_MORE_SENTINEL = object() @@ -14,7 +14,7 @@ def take(seq, count): count : int The number of elements to take. """ - for _ in range(count): + for _ in six.moves.range(count): i = next(seq, NO_MORE_SENTINEL) if i is NO_MORE_SENTINEL: return @@ -32,7 +32,7 @@ def chunks(obj, size): size : int Size of chunks to split list into. """ - for i in range(0, len(obj), size): + for i in six.moves.range(0, len(obj), size): yield obj[i:i + size] @@ -66,3 +66,12 @@ def simple_cached_property(method): delattr(inst, key) return property(_getattr, _setattr, _delattr) + + +def optional(**kwargs): + """ + Takes a set of keyword arguments, creating a dictionary with only the non- + null values. + :returns: dict + """ + return {k: v for k, v in six.iteritems(kwargs) if v is not None} From 76a4b8df94371ca8e9dce4caabf6851560716141 Mon Sep 17 00:00:00 2001 From: Luke Date: Fri, 27 Dec 2019 23:37:41 +0000 Subject: [PATCH 12/12] Add new invite fields for go live and guild counts. (#159) * Go live invite support. * url params to invites get rather than json. --- disco/api/client.py | 4 ++-- disco/types/invite.py | 36 ++++++++++++++++++++++++++---------- 2 files changed, 28 insertions(+), 12 deletions(-) diff --git a/disco/api/client.py b/disco/api/client.py index 313a4bb..45367ff 100644 --- a/disco/api/client.py +++ b/disco/api/client.py @@ -634,8 +634,8 @@ class APIClient(LoggingClass): r = self.http(Routes.USERS_ME_CONNECTIONS_LIST) return Connection.create_map(self.client, r.json()) - def invites_get(self, invite): - r = self.http(Routes.INVITES_GET, dict(invite=invite)) + def invites_get(self, invite, with_counts=None): + r = self.http(Routes.INVITES_GET, dict(invite=invite), params=optional(with_counts=with_counts)) return Invite.create(self.client, r.json()) def invites_delete(self, invite, reason=None): diff --git a/disco/types/invite.py b/disco/types/invite.py index 1d44b3e..4df6939 100644 --- a/disco/types/invite.py +++ b/disco/types/invite.py @@ -1,9 +1,13 @@ -from disco.types.base import SlottedModel, Field, datetime +from disco.types.base import SlottedModel, Field, datetime, enum from disco.types.user import User from disco.types.guild import Guild from disco.types.channel import Channel +class InviteTargetUserType(object): + STREAM = 1 + + class Invite(SlottedModel): """ An invite object. @@ -12,30 +16,42 @@ class Invite(SlottedModel): ---------- code : str The invite code. - inviter : :class:`disco.types.user.User` - The user who created this invite. guild : :class:`disco.types.guild.Guild` The guild this invite is for. channel : :class:`disco.types.channel.Channel` The channel this invite is for. - max_age : int - The time after this invite's creation at which it expires. - max_uses : int - The maximum number of uses. + target_user : :class:`disco.types.user.User` + The user this invite targets. + target_user_type : int + The type of user target for this invite. + approximate_presence_count : int + The approximate count of online members. + approximate_member_count : int + The approximate count of total members. + inviter : :class:`disco.types.user.User` + The user who created this invite. uses : int The current number of times the invite was used. + max_uses : int + The maximum number of uses. + max_age : int + The time after this invite's creation at which it expires. temporary : bool Whether this invite only grants temporary membership. created_at : datetime When this invite was created. """ code = Field(str) - inviter = Field(User) guild = Field(Guild) channel = Field(Channel) - max_age = Field(int) - max_uses = Field(int) + target_user = Field(User) + target_user_type = Field(enum(InviteTargetUserType)) + approximate_presence_count = Field(int) + approximate_member_count = Field(int) + inviter = Field(User) uses = Field(int) + max_uses = Field(int) + max_age = Field(int) temporary = Field(bool) created_at = Field(datetime)