committed by
GitHub
6 changed files with 468 additions and 4 deletions
@ -1,9 +1,11 @@ |
|||
from .middleware import Middleware |
|||
from .namespace import Namespace |
|||
from .base_manager import BaseManager |
|||
from .pubsub_manager import PubSubManager |
|||
from .kombu_manager import KombuManager |
|||
from .redis_manager import RedisManager |
|||
from .server import Server |
|||
from .util import apply_middleware |
|||
|
|||
__all__ = [Middleware, Server, BaseManager, PubSubManager, KombuManager, |
|||
RedisManager] |
|||
__all__ = [Middleware, Namespace, Server, BaseManager, PubSubManager, |
|||
KombuManager, RedisManager, apply_middleware] |
|||
|
@ -0,0 +1,95 @@ |
|||
import types |
|||
|
|||
from . import util |
|||
|
|||
|
|||
class Namespace(object): |
|||
"""A container for a set of event handlers for a specific namespace. |
|||
|
|||
A method of this class named ``on_xxx`` is considered as the event handler |
|||
for the event ``'xxx'`` in the namespace this class is registered to. |
|||
|
|||
There are also the following methods available that insert the current |
|||
namespace automatically when none is given before they call their matching |
|||
method of the ``Server`` instance: |
|||
|
|||
``emit``, ``send``, ``enter_room``, ``leave_room``, ``close_room``, |
|||
``rooms``, ``disconnect`` |
|||
|
|||
Example: |
|||
|
|||
from socketio import Namespace, Server |
|||
|
|||
class ChatNamespace(Namespace): |
|||
def on_msg(self, sid, msg): |
|||
# self.server references to the socketio.Server object |
|||
data = "[%s]: %s" \ |
|||
% (self.server.environ[sid].get("REMOTE_ADDR"), msg) |
|||
# Note that we don't pass namespace="/chat" to the emit method. |
|||
# It is done automatically for us. |
|||
self.emit("msg", data, skip_sid=sid) |
|||
return "received your message: %s" % msg |
|||
|
|||
# Here we set the event name explicitly by decorator. |
|||
@Namespace.event_name("event name with spaces") |
|||
def foo(self, sid): |
|||
# ... |
|||
|
|||
sio = socketio.Server() |
|||
ns = sio.register_namespace("/chat", ChatNamespace) |
|||
# ns now holds the instantiated ChatNamespace object |
|||
""" |
|||
|
|||
def __init__(self, name, server): |
|||
self.name = name |
|||
self.server = server |
|||
self.middlewares = [] |
|||
|
|||
# wrap methods of Server object |
|||
def get_wrapped_method(func_name): |
|||
def wrapped_func(self, *args, **kwargs): |
|||
"""If namespace is None, it is automatically set to this |
|||
object's one before the original method is called. |
|||
""" |
|||
if kwargs.get('namespace') is None: |
|||
kwargs['namespace'] = self.name |
|||
return getattr(self.server, func_name)(*args, **kwargs) |
|||
return types.MethodType(wrapped_func, self) |
|||
for func_name in ('emit', 'send', 'enter_room', 'leave_room', |
|||
'close_room', 'rooms', 'disconnect'): |
|||
setattr(self, func_name, get_wrapped_method(func_name)) |
|||
|
|||
def _get_event_handler(self, event_name): |
|||
"""Returns the event handler for given ``event`` in this namespace or |
|||
``None``, if none exists. |
|||
|
|||
:param event: The event name the handler is required for. |
|||
""" |
|||
for attr_name in dir(self): |
|||
attr = getattr(self, attr_name) |
|||
if hasattr(attr, '_sio_event_name'): |
|||
_event_name = getattr(attr, '_sio_event_name') |
|||
elif attr_name.startswith('on_'): |
|||
_event_name = attr_name[3:] |
|||
else: |
|||
continue |
|||
if _event_name == event_name: |
|||
return attr |
|||
|
|||
@staticmethod |
|||
def event_name(name): |
|||
"""Decorator to overwrite event names: |
|||
|
|||
@Namespace.event_name("event name with spaces") |
|||
def foo(self, sid, data): |
|||
return "received: %s" % data |
|||
|
|||
Ensure that you only add well-behaving decorators after this one |
|||
(meaning such that preserve attributes) because you'll loose them |
|||
otherwise. |
|||
""" |
|||
@util._simple_decorator |
|||
def wrapper(handler): |
|||
handler._sio_event_name = name |
|||
return handler |
|||
return wrapper |
@ -0,0 +1,49 @@ |
|||
def _simple_decorator(decorator): |
|||
"""This decorator can be used to turn simple functions |
|||
into well-behaved decorators, so long as the decorators |
|||
are fairly simple. If a decorator expects a function and |
|||
returns a function (no descriptors), and if it doesn't |
|||
modify function attributes or docstring, then it is |
|||
eligible to use this. Simply apply @_simple_decorator to |
|||
your decorator and it will automatically preserve the |
|||
docstring and function attributes of functions to which |
|||
it is applied. |
|||
|
|||
Also preserves all properties starting with ``'_sio'``. |
|||
""" |
|||
def copy_attrs(a, b): |
|||
"""Copies attributes from a to b.""" |
|||
for attr_name in ('__name__', '__doc__'): |
|||
if hasattr(a, attr_name): |
|||
setattr(b, attr_name, getattr(a, attr_name)) |
|||
if hasattr(a, '__dict__') and hasattr(b, '__dict__'): |
|||
b.__dict__.update(a.__dict__) |
|||
|
|||
def new_decorator(f): |
|||
g = decorator(f) |
|||
copy_attrs(f, g) |
|||
return g |
|||
|
|||
# Now a few lines needed to make _simple_decorator itself |
|||
# be a well-behaved decorator. |
|||
copy_attrs(decorator, new_decorator) |
|||
return new_decorator |
|||
|
|||
|
|||
def apply_middleware(middleware): |
|||
"""Returns a decorator for event handlers that adds the given |
|||
middleware to the handler decorated with it. |
|||
|
|||
:param middleware: The middleware to add |
|||
|
|||
Ensure that you only add well-behaving decorators after this one |
|||
(meaning such that preserve attributes) because you'll loose them |
|||
otherwise. |
|||
""" |
|||
@_simple_decorator |
|||
def wrapper(handler): |
|||
if not hasattr(handler, '_sio_middlewares'): |
|||
handler._sio_middlewares = [] |
|||
handler._sio_middlewares.append(middleware) |
|||
return handler |
|||
return wrapper |
Loading…
Reference in new issue