@ -1,71 +1,96 @@
import logging
import gevent
from gevent . backdoor import BackdoorServer
from holster . emitter import Emitter
from disco . state import State
from disco . api . client import APIClient
from disco . gateway . client import GatewayClient
from disco . util . logging import LoggingClass
log = logging . getLogger ( __name__ )
class ClientConfig ( LoggingClass ) :
"""
Configuration for the : class : ` Client ` .
class DiscoClient ( object ) :
Attributes
- - - - - - - - - -
token : str
Discord authentication token , ca be validated using the
: func : ` disco . util . token . is_valid_token ` function .
shard_id : int
The shard ID for the current client instance .
shard_count : int
The total count of shards running .
manhole_enable : bool
Whether to enable the manhole ( e . g . console backdoor server ) utility .
manhole_bind : tuple ( str , int )
A ( host , port ) combination which the manhole server will bind to ( if its
enabled using : attr : ` manhole_enable ` ) .
encoding_cls : class
The class to use for encoding / decoding data from websockets .
"""
The DiscoClient represents the base entry point to utilizing the Discord API
through disco . It wraps the functionality of both the REST API , and the realtime
secure websocket gateway .
token = " "
shard_id = 0
shard_count = 1
manhole_enable = True
manhole_bind = ( ' 127.0.0.1 ' , 8484 )
encoding_cls = None
class Client ( object ) :
"""
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
( : class : ` disco . api . client . APIClient ` ) and the realtime gateway API
( : class : ` disco . gateway . client . GatewayClient ` ) .
Parameters
- - - - - - - - - -
token : str
The Discord authentication token which is used for both the : class : ` APIClient `
and the : class : ` GatewayClient ` . This token can be validated before being
passed in , by using the : func : ` disco . util . token . is_valid_token ` function .
sharding : Optional [ dict ( str , int ) ]
A dictionary containing two pairs with information that is used to control
the sharding behavior of the : class : ` GatewayClient ` . By setting the ` number `
key , the current shard ID can be controlled . While when setting the ` total `
key , the total number of running shards can be set .
config : : class : ` ClientConfig `
Configuration for this client instance .
Attributes
- - - - - - - - - -
config : : class : ` ClientConfig `
The runtime configuration for this client .
events : : class : ` Emitter `
An emitter which emits Gateway events
An emitter which emits Gateway events .
packets : : class : ` Emitter `
An emitter which emits Gateway packets
An emitter which emits Gateway packets .
state : : class : ` State `
The state tracking object
The state tracking object .
api : : class : ` APIClient `
The API client
The API client .
gw : : class : ` GatewayClient `
The gateway client
The gateway client .
manhole : Optional [ : class : ` BackdoorServer ` ]
Gevent backdoor server ( if the manhole is enabled ) .
"""
def __init__ ( self , token , sharding = None ) :
self . log = log
self . token = token
self . sharding = sharding or { ' number ' : 0 , ' total ' : 1 }
def __init__ ( self , config ) :
super ( Client , self ) . __init__ ( )
self . config = config
self . events = Emitter ( gevent . spawn )
self . packets = Emitter ( gevent . spawn )
self . state = State ( self )
self . api = APIClient ( self )
self . gw = GatewayClient ( self )
self . gw = GatewayClient ( self , self . config . encoding_cls )
@classmethod
def from_cli ( cls , args ) :
"""
Create a new client from a argparse command line argument object , usually
generated from the : func : ` disco_main ` function .
"""
inst = cls (
token = args . token ,
sharding = {
' number ' : args . shard_id ,
' total ' : args . shard_count ,
} )
return inst
if self . config . manhole_enable :
self . manhole = BackdoorServer ( self . config . manhole_bind ,
banner = ' Disco Manhole ' ,
locals = {
' client ' : self ,
' state ' : self . state ,
' api ' : self . api ,
' gw ' : self . gw ,
} )
self . manhole . start ( )
def run ( self ) :
"""