pythonasyncioeventletgeventlong-pollinglow-latencysocket-iosocketiosocketio-serverweb-serverwebsocket
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.
77 lines
3.2 KiB
77 lines
3.2 KiB
from .base_manager import BaseManager
|
|
|
|
|
|
class PubSubManager(BaseManager):
|
|
"""Manage a client list attached to a pub/sub backend.
|
|
|
|
This is a base class that enables multiple servers to share the list of
|
|
clients, with the servers communicating events through a pub/sub backend.
|
|
The use of a pub/sub backend also allows any client connected to the
|
|
backend to emit events addressed to Socket.IO clients.
|
|
|
|
The actual backends must be implemented by subclasses, this class only
|
|
provides a pub/sub generic framework.
|
|
|
|
:param channel: The channel name on which the server sends and receives
|
|
notifications.
|
|
"""
|
|
def __init__(self, channel='socketio'):
|
|
super(PubSubManager, self).__init__()
|
|
self.channel = channel
|
|
|
|
def initialize(self, server):
|
|
super(PubSubManager, self).initialize(server)
|
|
self.thread = self.server.start_background_task(self._thread)
|
|
self.server.logger.info(self.name + ' backend initialized.')
|
|
|
|
def emit(self, event, data, namespace=None, room=None, skip_sid=None,
|
|
callback=None):
|
|
"""Emit a message to a single client, a room, or all the clients
|
|
connected to the namespace.
|
|
|
|
This method takes care or propagating the message to all the servers
|
|
that are connected through the message queue.
|
|
|
|
The parameters are the same as in :meth:`.Server.emit`.
|
|
"""
|
|
self._publish({'method': 'emit', 'event': event, 'data': data,
|
|
'namespace': namespace or '/', 'room': room,
|
|
'skip_sid': skip_sid, 'callback': callback})
|
|
|
|
def close_room(self, room, namespace=None):
|
|
self._publish({'method': 'close_room', 'room': room,
|
|
'namespace': namespace or '/'})
|
|
|
|
def _publish(self, data):
|
|
"""Publish a message on the Socket.IO channel.
|
|
|
|
This method needs to be implemented by the different subclasses that
|
|
support pub/sub backends.
|
|
"""
|
|
raise NotImplementedError('This method must be implemented in a '
|
|
'subclass.')
|
|
|
|
def _listen(self):
|
|
"""Return the next message published on the Socket.IO channel,
|
|
blocking until a message is available.
|
|
|
|
This method needs to be implemented by the different subclasses that
|
|
support pub/sub backends.
|
|
"""
|
|
raise NotImplementedError('This method must be implemented in a '
|
|
'subclass.')
|
|
|
|
def _thread(self):
|
|
for message in self._listen():
|
|
if 'method' in message:
|
|
if message['method'] == 'emit':
|
|
super(PubSubManager, self).emit(
|
|
message['event'], message['data'],
|
|
namespace=message.get('namespace'),
|
|
room=message.get('room'),
|
|
skip_sid=message.get('skip_sid'),
|
|
callback=message.get('callback'))
|
|
elif message['method'] == 'close_room':
|
|
super(PubSubManager, self).close_room(
|
|
room=message.get('room'),
|
|
namespace=message.get('namespace'))
|
|
|