Browse Source

Fix smashing PresenceUpdate.user value in state module

feature/storage
Andrei 8 years ago
parent
commit
bad953e58d
  1. 24
      disco/client.py
  2. 45
      disco/state.py

24
disco/client.py

@ -15,13 +15,13 @@ from disco.util.backdoor import DiscoBackdoorServer
class ClientConfig(Config): class ClientConfig(Config):
""" """
Configuration for the :class:`Client`. Configuration for the `Client`.
Attributes Attributes
---------- ----------
token : str token : str
Discord authentication token, can be validated using the Discord authentication token, can be validated using the
:func:`disco.util.token.is_valid_token` function. `disco.util.token.is_valid_token` function.
shard_id : int shard_id : int
The shard ID for the current client instance. The shard ID for the current client instance.
shard_count : int shard_count : int
@ -53,32 +53,32 @@ class Client(LoggingClass):
""" """
Class representing the base entry point that should be used in almost all Class representing the base entry point that should be used in almost all
implementation cases. This class wraps the functionality of both the REST API implementation cases. This class wraps the functionality of both the REST API
(:class:`disco.api.client.APIClient`) and the realtime gateway API (`disco.api.client.APIClient`) and the realtime gateway API
(:class:`disco.gateway.client.GatewayClient`). (`disco.gateway.client.GatewayClient`).
Parameters Parameters
---------- ----------
config : :class:`ClientConfig` config : `ClientConfig`
Configuration for this client instance. Configuration for this client instance.
Attributes Attributes
---------- ----------
config : :class:`ClientConfig` config : `ClientConfig`
The runtime configuration for this client. The runtime configuration for this client.
events : :class:`Emitter` events : `Emitter`
An emitter which emits Gateway events. An emitter which emits Gateway events.
packets : :class:`Emitter` packets : `Emitter`
An emitter which emits Gateway packets. An emitter which emits Gateway packets.
state : :class:`State` state : `State`
The state tracking object. The state tracking object.
api : :class:`APIClient` api : `APIClient`
The API client. The API client.
gw : :class:`GatewayClient` gw : `GatewayClient`
The gateway client. The gateway client.
manhole_locals : dict manhole_locals : dict
Dictionary of local variables for each manhole connection. This can be Dictionary of local variables for each manhole connection. This can be
modified to add/modify local variables. modified to add/modify local variables.
manhole : Optional[:class:`BackdoorServer`] manhole : Optional[`BackdoorServer`]
Gevent backdoor server (if the manhole is enabled). Gevent backdoor server (if the manhole is enabled).
""" """
def __init__(self, config): def __init__(self, config):

45
disco/state.py

@ -1,4 +1,5 @@
import six import six
import copy
import weakref import weakref
import inflection import inflection
@ -42,10 +43,10 @@ class StateConfig(Config):
find they do not need and may be experiencing memory pressure can disable find they do not need and may be experiencing memory pressure can disable
this feature entirely using this attribute. this feature entirely using this attribute.
track_messages_size : int track_messages_size : int
The size of the deque for each channel. Using this you can calculate the The size of the messages deque for each channel. This value can be used
total number of possible :class:`StackMessage` objects kept in memory, to calculate the total number of possible `StackMessage` objects kept in
using: `total_mesages_size * total_channels`. This can be tweaked based memory, simply: `total_messages_size * total_channels`. This value can
on usage to help prevent memory pressure. be tweaked based on usage and to help prevent memory pressure.
sync_guild_members : bool sync_guild_members : bool
If true, guilds will be automatically synced when they are initially loaded If true, guilds will be automatically synced when they are initially loaded
or joined. Generally this setting is OK for smaller bots, however bots in over or joined. Generally this setting is OK for smaller bots, however bots in over
@ -60,31 +61,31 @@ class StateConfig(Config):
class State(object): class State(object):
""" """
The State class is used to track global state based on events emitted from The State class is used to track global state based on events emitted from
the :class:`GatewayClient`. State tracking is a core component of the Disco the `GatewayClient`. State tracking is a core component of the Disco client,
client, providing the mechanism for most of the higher-level utility functions. providing the mechanism for most of the higher-level utility functions.
Attributes Attributes
---------- ----------
EVENTS : list(str) EVENTS : list(str)
A list of all events the State object binds to A list of all events the State object binds to
client : :class:`disco.client.Client` client : `disco.client.Client`
The Client instance this state is attached to The Client instance this state is attached to
config : :class:`StateConfig` config : `StateConfig`
The configuration for this state instance The configuration for this state instance
me : :class:`disco.types.user.User` me : `User`
The currently logged in user The currently logged in user
dms : dict(snowflake, :class:`disco.types.channel.Channel`) dms : dict(snowflake, `Channel`)
Mapping of all known DM Channels Mapping of all known DM Channels
guilds : dict(snowflake, :class:`disco.types.guild.Guild`) guilds : dict(snowflake, `Guild`)
Mapping of all known/loaded Guilds Mapping of all known/loaded Guilds
channels : dict(snowflake, :class:`disco.types.channel.Channel`) channels : dict(snowflake, `Channel`)
Weak mapping of all known/loaded Channels Weak mapping of all known/loaded Channels
users : dict(snowflake, :class:`disco.types.user.User`) users : dict(snowflake, `User`)
Weak mapping of all known/loaded Users Weak mapping of all known/loaded Users
voice_states : dict(str, :class:`disco.types.voice.VoiceState`) voice_states : dict(str, `VoiceState`)
Weak mapping of all known/active Voice States Weak mapping of all known/active Voice States
messages : Optional[dict(snowflake, :class:`deque`)] messages : Optional[dict(snowflake, deque)]
Mapping of channel ids to deques containing :class:`StackMessage` objects Mapping of channel ids to deques containing `StackMessage` objects
""" """
EVENTS = [ EVENTS = [
'Ready', 'GuildCreate', 'GuildUpdate', 'GuildDelete', 'GuildMemberAdd', 'GuildMemberRemove', 'Ready', 'GuildCreate', 'GuildUpdate', 'GuildDelete', 'GuildMemberAdd', 'GuildMemberRemove',
@ -315,12 +316,14 @@ class State(object):
self.guilds[event.guild_id].emojis = HashMap({i.id: i for i in event.emojis}) self.guilds[event.guild_id].emojis = HashMap({i.id: i for i in event.emojis})
def on_presence_update(self, event): def on_presence_update(self, event):
# Grab a copy of the user, and clear it out. All the operations below # Grab a copy of the entire presence object, and clear the user out. All
# do not want the nested user object, as the normaly structure is # the operations below do not require or want a nested user object, and
# user -> presence, and without clearing this becomes recursive # having it set just complicates the general structure. We copy so as
# to avoid interferring with other events that may require the nested
# user object.
user = event.presence.user user = event.presence.user
user.presence = event.presence user.presence = copy.deepcopy(event.presence)
event.presence.user = None user.presence.user = None
# if we have the user tracked locally, we can just use the presence # if we have the user tracked locally, we can just use the presence
# update to update both their presence and the cached user object. # update to update both their presence and the cached user object.

Loading…
Cancel
Save