Browse Source
- s/DiscoClient/Client (was redundant and looked ugly) - move cached_property to functional utils - abstract client configuration out to a ClientConfig - command line utility is now completely isolated from the client - add ETF (using erlpack) support, optionally enabled via command line flags (and only works on 2.x because of erlpack) - Refactor the way gateway events are built a bit - Add documentation on utilitiespull/3/head
23 changed files with 314 additions and 131 deletions
@ -0,0 +1,11 @@ |
|||||
|
from .json import JSONEncoder |
||||
|
|
||||
|
ENCODERS = { |
||||
|
'json': JSONEncoder, |
||||
|
} |
||||
|
|
||||
|
try: |
||||
|
from .etf import ETFEncoder |
||||
|
ENCODERS['etf'] = ETFEncoder |
||||
|
except ImportError: |
||||
|
pass |
@ -0,0 +1,14 @@ |
|||||
|
from websocket import ABNF |
||||
|
|
||||
|
|
||||
|
class BaseEncoder(object): |
||||
|
TYPE = None |
||||
|
OPCODE = ABNF.OPCODE_TEXT |
||||
|
|
||||
|
@staticmethod |
||||
|
def encode(obj): |
||||
|
pass |
||||
|
|
||||
|
@staticmethod |
||||
|
def decode(obj): |
||||
|
pass |
@ -0,0 +1,28 @@ |
|||||
|
import six |
||||
|
|
||||
|
from websocket import ABNF |
||||
|
from erlpack import Atom, unpack, pack |
||||
|
|
||||
|
from disco.gateway.encoding.base import BaseEncoder |
||||
|
|
||||
|
|
||||
|
def make_keys_atom(obj): |
||||
|
res = {} |
||||
|
for k, v in six.iteritems(obj): |
||||
|
if isinstance(v, dict): |
||||
|
v = make_keys_atom(v) |
||||
|
res[Atom(k)] = v |
||||
|
return res |
||||
|
|
||||
|
|
||||
|
class ETFEncoder(BaseEncoder): |
||||
|
TYPE = 'etf' |
||||
|
OPCODE = ABNF.OPCODE_BINARY |
||||
|
|
||||
|
@staticmethod |
||||
|
def encode(obj): |
||||
|
return pack(obj) |
||||
|
|
||||
|
@staticmethod |
||||
|
def decode(obj): |
||||
|
return unpack(obj) |
@ -0,0 +1,22 @@ |
|||||
|
from __future__ import absolute_import, print_function |
||||
|
|
||||
|
import six |
||||
|
|
||||
|
try: |
||||
|
import ujson as json |
||||
|
except ImportError: |
||||
|
import json |
||||
|
|
||||
|
from disco.gateway.encoding.base import BaseEncoder |
||||
|
|
||||
|
|
||||
|
class JSONEncoder(BaseEncoder): |
||||
|
TYPE = 'json' |
||||
|
|
||||
|
@staticmethod |
||||
|
def encode(obj): |
||||
|
return json.dumps(obj) |
||||
|
|
||||
|
@staticmethod |
||||
|
def decode(obj): |
||||
|
return json.loads(obj) |
@ -1,15 +0,0 @@ |
|||||
|
|
||||
|
|
||||
def cached_property(f): |
|
||||
def getf(self, *args, **kwargs): |
|
||||
if not hasattr(self, '__' + f.__name__): |
|
||||
setattr(self, '__' + f.__name__, f(self, *args, **kwargs)) |
|
||||
return getattr(self, '__' + f.__name__) |
|
||||
|
|
||||
def setf(self, value): |
|
||||
setattr(self, '__' + f.__name__, value) |
|
||||
|
|
||||
def delf(self): |
|
||||
setattr(self, '__' + f.__name__, None) |
|
||||
|
|
||||
return property(getf, setf, delf) |
|
@ -0,0 +1,49 @@ |
|||||
|
from gevent.lock import RLock |
||||
|
|
||||
|
|
||||
|
def cached_property(f): |
||||
|
""" |
||||
|
Creates a cached property out of ``f``. When the property is resolved for |
||||
|
the first time, the function will be called and its result will be cached. |
||||
|
Subsequent calls will return the cached value. If this property is set, the |
||||
|
cached value will be replaced (or set initially) with the value provided. If |
||||
|
this property is deleted, the cache will be cleared and the next call will |
||||
|
refill it with a new value. |
||||
|
|
||||
|
Notes |
||||
|
----- |
||||
|
This function is greenlet safe. |
||||
|
|
||||
|
Args |
||||
|
---- |
||||
|
f : function |
||||
|
The function to wrap. |
||||
|
|
||||
|
Returns |
||||
|
------- |
||||
|
property |
||||
|
The cached property created. |
||||
|
""" |
||||
|
lock = RLock() |
||||
|
f._value = None |
||||
|
f._has_value = False |
||||
|
|
||||
|
def getf(*args, **kwargs): |
||||
|
if not f._has_value: |
||||
|
with lock: |
||||
|
if f._has_value: |
||||
|
return f._value |
||||
|
|
||||
|
f._value = f(*args, **kwargs) |
||||
|
f._has_value = True |
||||
|
|
||||
|
return f._value |
||||
|
|
||||
|
def setf(self, value): |
||||
|
f._value = value |
||||
|
|
||||
|
def delf(self): |
||||
|
f._value = None |
||||
|
f._has_value = False |
||||
|
|
||||
|
return property(getf, setf, delf) |
@ -1,11 +0,0 @@ |
|||||
from __future__ import absolute_import, print_function |
|
||||
|
|
||||
from json import dumps |
|
||||
|
|
||||
try: |
|
||||
from rapidjson import loads |
|
||||
except ImportError: |
|
||||
print('[WARNING] rapidjson not installed, falling back to default Python JSON parser') |
|
||||
from json import loads |
|
||||
|
|
||||
__all__ = ['dumps', 'loads'] |
|
@ -0,0 +1,27 @@ |
|||||
|
.. currentmodule:: disco |
||||
|
|
||||
|
Utilities |
||||
|
========= |
||||
|
|
||||
|
This section details information about various utilities provided in the disco |
||||
|
package, which aid the development and runtime management of disco clients/bots. |
||||
|
Generally these utilties are situational, and can be enabled depending on |
||||
|
various scenarious developers and users may find themselves in. |
||||
|
|
||||
|
Manhole |
||||
|
------- |
||||
|
|
||||
|
The manhole utilty is a backdoor server that allows opening a interactive REPL |
||||
|
while the client is running. This can be very useful for attaching and |
||||
|
inspecting runtime state, without distribing the normal client operations. To |
||||
|
enable the backdoor, simply set the |
||||
|
:attr:`disco.client.ClientConfig.manhole_enable` setting, and tweak |
||||
|
:attr:`disco.client.ClientConfig.manhole_bind` settings based on the connection |
||||
|
parameters you'd like. |
||||
|
|
||||
|
It's recommended you connect to the manhole via ``rlwrap`` and ``netcat``, which |
||||
|
will give a proper TTY-like readline experience. For example: |
||||
|
|
||||
|
.. sourcecode:: bash |
||||
|
|
||||
|
rlwrap netcat localhost 8484 |
Loading…
Reference in new issue