You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
527 lines
25 KiB
527 lines
25 KiB
5 years ago
|
import asyncio
|
||
|
|
||
|
import engineio
|
||
|
import six
|
||
|
|
||
|
from . import asyncio_manager
|
||
|
from . import exceptions
|
||
|
from . import packet
|
||
|
from . import server
|
||
|
|
||
|
|
||
|
class AsyncServer(server.Server):
|
||
|
"""A Socket.IO server for asyncio.
|
||
|
|
||
|
This class implements a fully compliant Socket.IO web server with support
|
||
|
for websocket and long-polling transports, compatible with the asyncio
|
||
|
framework on Python 3.5 or newer.
|
||
|
|
||
|
:param client_manager: The client manager instance that will manage the
|
||
|
client list. When this is omitted, the client list
|
||
|
is stored in an in-memory structure, so the use of
|
||
|
multiple connected servers is not possible.
|
||
|
:param logger: To enable logging set to ``True`` or pass a logger object to
|
||
|
use. To disable logging set to ``False``.
|
||
|
:param json: An alternative json module to use for encoding and decoding
|
||
|
packets. Custom json modules must have ``dumps`` and ``loads``
|
||
|
functions that are compatible with the standard library
|
||
|
versions.
|
||
|
:param async_handlers: If set to ``True``, event handlers are executed in
|
||
|
separate threads. To run handlers synchronously,
|
||
|
set to ``False``. The default is ``True``.
|
||
|
:param kwargs: Connection parameters for the underlying Engine.IO server.
|
||
|
|
||
|
The Engine.IO configuration supports the following settings:
|
||
|
|
||
|
:param async_mode: The asynchronous model to use. See the Deployment
|
||
|
section in the documentation for a description of the
|
||
|
available options. Valid async modes are "aiohttp". If
|
||
|
this argument is not given, an async mode is chosen
|
||
|
based on the installed packages.
|
||
|
:param ping_timeout: The time in seconds that the client waits for the
|
||
|
server to respond before disconnecting.
|
||
|
:param ping_interval: The interval in seconds at which the client pings
|
||
|
the server.
|
||
|
:param max_http_buffer_size: The maximum size of a message when using the
|
||
|
polling transport.
|
||
|
:param allow_upgrades: Whether to allow transport upgrades or not.
|
||
|
:param http_compression: Whether to compress packages when using the
|
||
|
polling transport.
|
||
|
:param compression_threshold: Only compress messages when their byte size
|
||
|
is greater than this value.
|
||
|
:param cookie: Name of the HTTP cookie that contains the client session
|
||
|
id. If set to ``None``, a cookie is not sent to the client.
|
||
|
:param cors_allowed_origins: Origin or list of origins that are allowed to
|
||
|
connect to this server. Only the same origin
|
||
|
is allowed by default. Set this argument to
|
||
|
``'*'`` to allow all origins, or to ``[]`` to
|
||
|
disable CORS handling.
|
||
|
:param cors_credentials: Whether credentials (cookies, authentication) are
|
||
|
allowed in requests to this server.
|
||
|
:param monitor_clients: If set to ``True``, a background task will ensure
|
||
|
inactive clients are closed. Set to ``False`` to
|
||
|
disable the monitoring task (not recommended). The
|
||
|
default is ``True``.
|
||
|
:param engineio_logger: To enable Engine.IO logging set to ``True`` or pass
|
||
|
a logger object to use. To disable logging set to
|
||
|
``False``.
|
||
|
"""
|
||
|
def __init__(self, client_manager=None, logger=False, json=None,
|
||
|
async_handlers=True, **kwargs):
|
||
|
if client_manager is None:
|
||
|
client_manager = asyncio_manager.AsyncManager()
|
||
|
super().__init__(client_manager=client_manager, logger=logger,
|
||
|
binary=False, json=json,
|
||
|
async_handlers=async_handlers, **kwargs)
|
||
|
|
||
|
def is_asyncio_based(self):
|
||
|
return True
|
||
|
|
||
|
def attach(self, app, socketio_path='socket.io'):
|
||
|
"""Attach the Socket.IO server to an application."""
|
||
|
self.eio.attach(app, socketio_path)
|
||
|
|
||
|
async def emit(self, event, data=None, to=None, room=None, skip_sid=None,
|
||
|
namespace=None, callback=None, **kwargs):
|
||
|
"""Emit a custom event to one or more connected clients.
|
||
|
|
||
|
:param event: The event name. It can be any string. The event names
|
||
|
``'connect'``, ``'message'`` and ``'disconnect'`` are
|
||
|
reserved and should not be used.
|
||
|
:param data: The data to send to the client or clients. Data can be of
|
||
|
type ``str``, ``bytes``, ``list`` or ``dict``. If a
|
||
|
``list`` or ``dict``, the data will be serialized as JSON.
|
||
|
:param to: The recipient of the message. This can be set to the
|
||
|
session ID of a client to address only that client, or to
|
||
|
to any custom room created by the application to address all
|
||
|
the clients in that room, If this argument is omitted the
|
||
|
event is broadcasted to all connected clients.
|
||
|
:param room: Alias for the ``to`` parameter.
|
||
|
:param skip_sid: The session ID of a client to skip when broadcasting
|
||
|
to a room or to all clients. This can be used to
|
||
|
prevent a message from being sent to the sender.
|
||
|
:param namespace: The Socket.IO namespace for the event. If this
|
||
|
argument is omitted the event is emitted to the
|
||
|
default namespace.
|
||
|
:param callback: If given, this function will be called to acknowledge
|
||
|
the the client has received the message. The arguments
|
||
|
that will be passed to the function are those provided
|
||
|
by the client. Callback functions can only be used
|
||
|
when addressing an individual client.
|
||
|
:param ignore_queue: Only used when a message queue is configured. If
|
||
|
set to ``True``, the event is emitted to the
|
||
|
clients directly, without going through the queue.
|
||
|
This is more efficient, but only works when a
|
||
|
single server process is used. It is recommended
|
||
|
to always leave this parameter with its default
|
||
|
value of ``False``.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
namespace = namespace or '/'
|
||
|
room = to or room
|
||
|
self.logger.info('emitting event "%s" to %s [%s]', event,
|
||
|
room or 'all', namespace)
|
||
|
await self.manager.emit(event, data, namespace, room=room,
|
||
|
skip_sid=skip_sid, callback=callback,
|
||
|
**kwargs)
|
||
|
|
||
|
async def send(self, data, to=None, room=None, skip_sid=None,
|
||
|
namespace=None, callback=None, **kwargs):
|
||
|
"""Send a message to one or more connected clients.
|
||
|
|
||
|
This function emits an event with the name ``'message'``. Use
|
||
|
:func:`emit` to issue custom event names.
|
||
|
|
||
|
:param data: The data to send to the client or clients. Data can be of
|
||
|
type ``str``, ``bytes``, ``list`` or ``dict``. If a
|
||
|
``list`` or ``dict``, the data will be serialized as JSON.
|
||
|
:param to: The recipient of the message. This can be set to the
|
||
|
session ID of a client to address only that client, or to
|
||
|
to any custom room created by the application to address all
|
||
|
the clients in that room, If this argument is omitted the
|
||
|
event is broadcasted to all connected clients.
|
||
|
:param room: Alias for the ``to`` parameter.
|
||
|
:param skip_sid: The session ID of a client to skip when broadcasting
|
||
|
to a room or to all clients. This can be used to
|
||
|
prevent a message from being sent to the sender.
|
||
|
:param namespace: The Socket.IO namespace for the event. If this
|
||
|
argument is omitted the event is emitted to the
|
||
|
default namespace.
|
||
|
:param callback: If given, this function will be called to acknowledge
|
||
|
the the client has received the message. The arguments
|
||
|
that will be passed to the function are those provided
|
||
|
by the client. Callback functions can only be used
|
||
|
when addressing an individual client.
|
||
|
:param ignore_queue: Only used when a message queue is configured. If
|
||
|
set to ``True``, the event is emitted to the
|
||
|
clients directly, without going through the queue.
|
||
|
This is more efficient, but only works when a
|
||
|
single server process is used. It is recommended
|
||
|
to always leave this parameter with its default
|
||
|
value of ``False``.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
await self.emit('message', data=data, to=to, room=room,
|
||
|
skip_sid=skip_sid, namespace=namespace,
|
||
|
callback=callback, **kwargs)
|
||
|
|
||
|
async def call(self, event, data=None, to=None, sid=None, namespace=None,
|
||
|
timeout=60, **kwargs):
|
||
|
"""Emit a custom event to a client and wait for the response.
|
||
|
|
||
|
:param event: The event name. It can be any string. The event names
|
||
|
``'connect'``, ``'message'`` and ``'disconnect'`` are
|
||
|
reserved and should not be used.
|
||
|
:param data: The data to send to the client or clients. Data can be of
|
||
|
type ``str``, ``bytes``, ``list`` or ``dict``. If a
|
||
|
``list`` or ``dict``, the data will be serialized as JSON.
|
||
|
:param to: The session ID of the recipient client.
|
||
|
:param sid: Alias for the ``to`` parameter.
|
||
|
:param namespace: The Socket.IO namespace for the event. If this
|
||
|
argument is omitted the event is emitted to the
|
||
|
default namespace.
|
||
|
:param timeout: The waiting timeout. If the timeout is reached before
|
||
|
the client acknowledges the event, then a
|
||
|
``TimeoutError`` exception is raised.
|
||
|
:param ignore_queue: Only used when a message queue is configured. If
|
||
|
set to ``True``, the event is emitted to the
|
||
|
client directly, without going through the queue.
|
||
|
This is more efficient, but only works when a
|
||
|
single server process is used. It is recommended
|
||
|
to always leave this parameter with its default
|
||
|
value of ``False``.
|
||
|
"""
|
||
|
if not self.async_handlers:
|
||
|
raise RuntimeError(
|
||
|
'Cannot use call() when async_handlers is False.')
|
||
|
callback_event = self.eio.create_event()
|
||
|
callback_args = []
|
||
|
|
||
|
def event_callback(*args):
|
||
|
callback_args.append(args)
|
||
|
callback_event.set()
|
||
|
|
||
|
await self.emit(event, data=data, room=to or sid, namespace=namespace,
|
||
|
callback=event_callback, **kwargs)
|
||
|
try:
|
||
|
await asyncio.wait_for(callback_event.wait(), timeout)
|
||
|
except asyncio.TimeoutError:
|
||
|
six.raise_from(exceptions.TimeoutError(), None)
|
||
|
return callback_args[0] if len(callback_args[0]) > 1 \
|
||
|
else callback_args[0][0] if len(callback_args[0]) == 1 \
|
||
|
else None
|
||
|
|
||
|
async def close_room(self, room, namespace=None):
|
||
|
"""Close a room.
|
||
|
|
||
|
This function removes all the clients from the given room.
|
||
|
|
||
|
:param room: Room name.
|
||
|
:param namespace: The Socket.IO namespace for the event. If this
|
||
|
argument is omitted the default namespace is used.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
namespace = namespace or '/'
|
||
|
self.logger.info('room %s is closing [%s]', room, namespace)
|
||
|
await self.manager.close_room(room, namespace)
|
||
|
|
||
|
async def get_session(self, sid, namespace=None):
|
||
|
"""Return the user session for a client.
|
||
|
|
||
|
:param sid: The session id of the client.
|
||
|
:param namespace: The Socket.IO namespace. If this argument is omitted
|
||
|
the default namespace is used.
|
||
|
|
||
|
The return value is a dictionary. Modifications made to this
|
||
|
dictionary are not guaranteed to be preserved. If you want to modify
|
||
|
the user session, use the ``session`` context manager instead.
|
||
|
"""
|
||
|
namespace = namespace or '/'
|
||
|
eio_session = await self.eio.get_session(sid)
|
||
|
return eio_session.setdefault(namespace, {})
|
||
|
|
||
|
async def save_session(self, sid, session, namespace=None):
|
||
|
"""Store the user session for a client.
|
||
|
|
||
|
:param sid: The session id of the client.
|
||
|
:param session: The session dictionary.
|
||
|
:param namespace: The Socket.IO namespace. If this argument is omitted
|
||
|
the default namespace is used.
|
||
|
"""
|
||
|
namespace = namespace or '/'
|
||
|
eio_session = await self.eio.get_session(sid)
|
||
|
eio_session[namespace] = session
|
||
|
|
||
|
def session(self, sid, namespace=None):
|
||
|
"""Return the user session for a client with context manager syntax.
|
||
|
|
||
|
:param sid: The session id of the client.
|
||
|
|
||
|
This is a context manager that returns the user session dictionary for
|
||
|
the client. Any changes that are made to this dictionary inside the
|
||
|
context manager block are saved back to the session. Example usage::
|
||
|
|
||
|
@eio.on('connect')
|
||
|
def on_connect(sid, environ):
|
||
|
username = authenticate_user(environ)
|
||
|
if not username:
|
||
|
return False
|
||
|
with eio.session(sid) as session:
|
||
|
session['username'] = username
|
||
|
|
||
|
@eio.on('message')
|
||
|
def on_message(sid, msg):
|
||
|
async with eio.session(sid) as session:
|
||
|
print('received message from ', session['username'])
|
||
|
"""
|
||
|
class _session_context_manager(object):
|
||
|
def __init__(self, server, sid, namespace):
|
||
|
self.server = server
|
||
|
self.sid = sid
|
||
|
self.namespace = namespace
|
||
|
self.session = None
|
||
|
|
||
|
async def __aenter__(self):
|
||
|
self.session = await self.server.get_session(
|
||
|
sid, namespace=self.namespace)
|
||
|
return self.session
|
||
|
|
||
|
async def __aexit__(self, *args):
|
||
|
await self.server.save_session(sid, self.session,
|
||
|
namespace=self.namespace)
|
||
|
|
||
|
return _session_context_manager(self, sid, namespace)
|
||
|
|
||
|
async def disconnect(self, sid, namespace=None):
|
||
|
"""Disconnect a client.
|
||
|
|
||
|
:param sid: Session ID of the client.
|
||
|
:param namespace: The Socket.IO namespace to disconnect. If this
|
||
|
argument is omitted the default namespace is used.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
namespace = namespace or '/'
|
||
|
if self.manager.is_connected(sid, namespace=namespace):
|
||
|
self.logger.info('Disconnecting %s [%s]', sid, namespace)
|
||
|
self.manager.pre_disconnect(sid, namespace=namespace)
|
||
|
await self._send_packet(sid, packet.Packet(packet.DISCONNECT,
|
||
|
namespace=namespace))
|
||
|
await self._trigger_event('disconnect', namespace, sid)
|
||
|
self.manager.disconnect(sid, namespace=namespace)
|
||
|
if namespace == '/':
|
||
|
await self.eio.disconnect(sid)
|
||
|
|
||
|
async def handle_request(self, *args, **kwargs):
|
||
|
"""Handle an HTTP request from the client.
|
||
|
|
||
|
This is the entry point of the Socket.IO application. This function
|
||
|
returns the HTTP response body to deliver to the client.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
return await self.eio.handle_request(*args, **kwargs)
|
||
|
|
||
|
def start_background_task(self, target, *args, **kwargs):
|
||
|
"""Start a background task using the appropriate async model.
|
||
|
|
||
|
This is a utility function that applications can use to start a
|
||
|
background task using the method that is compatible with the
|
||
|
selected async mode.
|
||
|
|
||
|
:param target: the target function to execute. Must be a coroutine.
|
||
|
:param args: arguments to pass to the function.
|
||
|
:param kwargs: keyword arguments to pass to the function.
|
||
|
|
||
|
The return value is a ``asyncio.Task`` object.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
return self.eio.start_background_task(target, *args, **kwargs)
|
||
|
|
||
|
async def sleep(self, seconds=0):
|
||
|
"""Sleep for the requested amount of time using the appropriate async
|
||
|
model.
|
||
|
|
||
|
This is a utility function that applications can use to put a task to
|
||
|
sleep without having to worry about using the correct call for the
|
||
|
selected async mode.
|
||
|
|
||
|
Note: this method is a coroutine.
|
||
|
"""
|
||
|
return await self.eio.sleep(seconds)
|
||
|
|
||
|
async def _emit_internal(self, sid, event, data, namespace=None, id=None):
|
||
|
"""Send a message to a client."""
|
||
|
# tuples are expanded to multiple arguments, everything else is sent
|
||
|
# as a single argument
|
||
|
if isinstance(data, tuple):
|
||
|
data = list(data)
|
||
|
else:
|
||
|
data = [data]
|
||
|
await self._send_packet(sid, packet.Packet(
|
||
|
packet.EVENT, namespace=namespace, data=[event] + data, id=id,
|
||
|
binary=None))
|
||
|
|
||
|
async def _send_packet(self, sid, pkt):
|
||
|
"""Send a Socket.IO packet to a client."""
|
||
|
encoded_packet = pkt.encode()
|
||
|
if isinstance(encoded_packet, list):
|
||
|
binary = False
|
||
|
for ep in encoded_packet:
|
||
|
await self.eio.send(sid, ep, binary=binary)
|
||
|
binary = True
|
||
|
else:
|
||
|
await self.eio.send(sid, encoded_packet, binary=False)
|
||
|
|
||
|
async def _handle_connect(self, sid, namespace):
|
||
|
"""Handle a client connection request."""
|
||
|
namespace = namespace or '/'
|
||
|
self.manager.connect(sid, namespace)
|
||
|
if self.always_connect:
|
||
|
await self._send_packet(sid, packet.Packet(packet.CONNECT,
|
||
|
namespace=namespace))
|
||
|
fail_reason = None
|
||
|
try:
|
||
|
success = await self._trigger_event('connect', namespace, sid,
|
||
|
self.environ[sid])
|
||
|
except exceptions.ConnectionRefusedError as exc:
|
||
|
fail_reason = exc.error_args
|
||
|
success = False
|
||
|
|
||
|
if success is False:
|
||
|
if self.always_connect:
|
||
|
self.manager.pre_disconnect(sid, namespace)
|
||
|
await self._send_packet(sid, packet.Packet(
|
||
|
packet.DISCONNECT, data=fail_reason, namespace=namespace))
|
||
|
self.manager.disconnect(sid, namespace)
|
||
|
if not self.always_connect:
|
||
|
await self._send_packet(sid, packet.Packet(
|
||
|
packet.ERROR, data=fail_reason, namespace=namespace))
|
||
|
if sid in self.environ: # pragma: no cover
|
||
|
del self.environ[sid]
|
||
|
elif not self.always_connect:
|
||
|
await self._send_packet(sid, packet.Packet(packet.CONNECT,
|
||
|
namespace=namespace))
|
||
|
|
||
|
async def _handle_disconnect(self, sid, namespace):
|
||
|
"""Handle a client disconnect."""
|
||
|
namespace = namespace or '/'
|
||
|
if namespace == '/':
|
||
|
namespace_list = list(self.manager.get_namespaces())
|
||
|
else:
|
||
|
namespace_list = [namespace]
|
||
|
for n in namespace_list:
|
||
|
if n != '/' and self.manager.is_connected(sid, n):
|
||
|
await self._trigger_event('disconnect', n, sid)
|
||
|
self.manager.disconnect(sid, n)
|
||
|
if namespace == '/' and self.manager.is_connected(sid, namespace):
|
||
|
await self._trigger_event('disconnect', '/', sid)
|
||
|
self.manager.disconnect(sid, '/')
|
||
|
|
||
|
async def _handle_event(self, sid, namespace, id, data):
|
||
|
"""Handle an incoming client event."""
|
||
|
namespace = namespace or '/'
|
||
|
self.logger.info('received event "%s" from %s [%s]', data[0], sid,
|
||
|
namespace)
|
||
|
if not self.manager.is_connected(sid, namespace):
|
||
|
self.logger.warning('%s is not connected to namespace %s',
|
||
|
sid, namespace)
|
||
|
return
|
||
|
if self.async_handlers:
|
||
|
self.start_background_task(self._handle_event_internal, self, sid,
|
||
|
data, namespace, id)
|
||
|
else:
|
||
|
await self._handle_event_internal(self, sid, data, namespace, id)
|
||
|
|
||
|
async def _handle_event_internal(self, server, sid, data, namespace, id):
|
||
|
r = await server._trigger_event(data[0], namespace, sid, *data[1:])
|
||
|
if id is not None:
|
||
|
# send ACK packet with the response returned by the handler
|
||
|
# tuples are expanded as multiple arguments
|
||
|
if r is None:
|
||
|
data = []
|
||
|
elif isinstance(r, tuple):
|
||
|
data = list(r)
|
||
|
else:
|
||
|
data = [r]
|
||
|
await server._send_packet(sid, packet.Packet(packet.ACK,
|
||
|
namespace=namespace,
|
||
|
id=id, data=data,
|
||
|
binary=None))
|
||
|
|
||
|
async def _handle_ack(self, sid, namespace, id, data):
|
||
|
"""Handle ACK packets from the client."""
|
||
|
namespace = namespace or '/'
|
||
|
self.logger.info('received ack from %s [%s]', sid, namespace)
|
||
|
await self.manager.trigger_callback(sid, namespace, id, data)
|
||
|
|
||
|
async def _trigger_event(self, event, namespace, *args):
|
||
|
"""Invoke an application event handler."""
|
||
|
# first see if we have an explicit handler for the event
|
||
|
if namespace in self.handlers and event in self.handlers[namespace]:
|
||
|
if asyncio.iscoroutinefunction(self.handlers[namespace][event]) \
|
||
|
is True:
|
||
|
try:
|
||
|
ret = await self.handlers[namespace][event](*args)
|
||
|
except asyncio.CancelledError: # pragma: no cover
|
||
|
ret = None
|
||
|
else:
|
||
|
ret = self.handlers[namespace][event](*args)
|
||
|
return ret
|
||
|
|
||
|
# or else, forward the event to a namepsace handler if one exists
|
||
|
elif namespace in self.namespace_handlers:
|
||
|
return await self.namespace_handlers[namespace].trigger_event(
|
||
|
event, *args)
|
||
|
|
||
|
async def _handle_eio_connect(self, sid, environ):
|
||
|
"""Handle the Engine.IO connection event."""
|
||
|
if not self.manager_initialized:
|
||
|
self.manager_initialized = True
|
||
|
self.manager.initialize()
|
||
|
self.environ[sid] = environ
|
||
|
return await self._handle_connect(sid, '/')
|
||
|
|
||
|
async def _handle_eio_message(self, sid, data):
|
||
|
"""Dispatch Engine.IO messages."""
|
||
|
if sid in self._binary_packet:
|
||
|
pkt = self._binary_packet[sid]
|
||
|
if pkt.add_attachment(data):
|
||
|
del self._binary_packet[sid]
|
||
|
if pkt.packet_type == packet.BINARY_EVENT:
|
||
|
await self._handle_event(sid, pkt.namespace, pkt.id,
|
||
|
pkt.data)
|
||
|
else:
|
||
|
await self._handle_ack(sid, pkt.namespace, pkt.id,
|
||
|
pkt.data)
|
||
|
else:
|
||
|
pkt = packet.Packet(encoded_packet=data)
|
||
|
if pkt.packet_type == packet.CONNECT:
|
||
|
await self._handle_connect(sid, pkt.namespace)
|
||
|
elif pkt.packet_type == packet.DISCONNECT:
|
||
|
await self._handle_disconnect(sid, pkt.namespace)
|
||
|
elif pkt.packet_type == packet.EVENT:
|
||
|
await self._handle_event(sid, pkt.namespace, pkt.id, pkt.data)
|
||
|
elif pkt.packet_type == packet.ACK:
|
||
|
await self._handle_ack(sid, pkt.namespace, pkt.id, pkt.data)
|
||
|
elif pkt.packet_type == packet.BINARY_EVENT or \
|
||
|
pkt.packet_type == packet.BINARY_ACK:
|
||
|
self._binary_packet[sid] = pkt
|
||
|
elif pkt.packet_type == packet.ERROR:
|
||
|
raise ValueError('Unexpected ERROR packet.')
|
||
|
else:
|
||
|
raise ValueError('Unknown packet type.')
|
||
|
|
||
|
async def _handle_eio_disconnect(self, sid):
|
||
|
"""Handle Engine.IO disconnect event."""
|
||
|
await self._handle_disconnect(sid, '/')
|
||
|
if sid in self.environ:
|
||
|
del self.environ[sid]
|
||
|
|
||
|
def _engineio_server_class(self):
|
||
|
return engineio.AsyncServer
|