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.
192 lines
8.1 KiB
192 lines
8.1 KiB
11 months ago
|
from threading import Event
|
||
|
from socketio import Client
|
||
|
from socketio.exceptions import SocketIOError, TimeoutError, DisconnectedError
|
||
|
|
||
|
|
||
|
class SimpleClient:
|
||
|
"""A Socket.IO client.
|
||
|
|
||
|
This class implements a simple, yet fully compliant Socket.IO web client
|
||
|
with support for websocket and long-polling transports.
|
||
|
|
||
|
The positional and keyword arguments given in the constructor are passed
|
||
|
to the underlying :func:`socketio.Client` object.
|
||
|
"""
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
self.client_args = args
|
||
|
self.client_kwargs = kwargs
|
||
|
self.client = None
|
||
|
self.namespace = '/'
|
||
|
self.connected_event = Event()
|
||
|
self.connected = False
|
||
|
self.input_event = Event()
|
||
|
self.input_buffer = []
|
||
|
|
||
|
def connect(self, url, headers={}, auth=None, transports=None,
|
||
|
namespace='/', socketio_path='socket.io', wait_timeout=5):
|
||
|
"""Connect to a Socket.IO server.
|
||
|
|
||
|
:param url: The URL of the Socket.IO server. It can include custom
|
||
|
query string parameters if required by the server. If a
|
||
|
function is provided, the client will invoke it to obtain
|
||
|
the URL each time a connection or reconnection is
|
||
|
attempted.
|
||
|
:param headers: A dictionary with custom headers to send with the
|
||
|
connection request. If a function is provided, the
|
||
|
client will invoke it to obtain the headers dictionary
|
||
|
each time a connection or reconnection is attempted.
|
||
|
:param auth: Authentication data passed to the server with the
|
||
|
connection request, normally a dictionary with one or
|
||
|
more string key/value pairs. If a function is provided,
|
||
|
the client will invoke it to obtain the authentication
|
||
|
data each time a connection or reconnection is attempted.
|
||
|
:param transports: The list of allowed transports. Valid transports
|
||
|
are ``'polling'`` and ``'websocket'``. If not
|
||
|
given, the polling transport is connected first,
|
||
|
then an upgrade to websocket is attempted.
|
||
|
:param namespace: The namespace to connect to as a string. If not
|
||
|
given, the default namespace ``/`` is used.
|
||
|
:param socketio_path: The endpoint where the Socket.IO server is
|
||
|
installed. The default value is appropriate for
|
||
|
most cases.
|
||
|
:param wait_timeout: How long the client should wait for the
|
||
|
connection to be established. The default is 5
|
||
|
seconds.
|
||
|
"""
|
||
|
if self.connected:
|
||
|
raise RuntimeError('Already connected')
|
||
|
self.namespace = namespace
|
||
|
self.input_buffer = []
|
||
|
self.input_event.clear()
|
||
|
self.client = Client(*self.client_args, **self.client_kwargs)
|
||
|
|
||
|
@self.client.event(namespace=self.namespace)
|
||
|
def connect(): # pragma: no cover
|
||
|
self.connected = True
|
||
|
self.connected_event.set()
|
||
|
|
||
|
@self.client.event(namespace=self.namespace)
|
||
|
def disconnect(): # pragma: no cover
|
||
|
self.connected_event.clear()
|
||
|
|
||
|
@self.client.event(namespace=self.namespace)
|
||
|
def __disconnect_final(): # pragma: no cover
|
||
|
self.connected = False
|
||
|
self.connected_event.set()
|
||
|
|
||
|
@self.client.on('*', namespace=self.namespace)
|
||
|
def on_event(event, *args): # pragma: no cover
|
||
|
self.input_buffer.append([event, *args])
|
||
|
self.input_event.set()
|
||
|
|
||
|
self.client.connect(url, headers=headers, auth=auth,
|
||
|
transports=transports, namespaces=[namespace],
|
||
|
socketio_path=socketio_path,
|
||
|
wait_timeout=wait_timeout)
|
||
|
|
||
|
@property
|
||
|
def sid(self):
|
||
|
"""The session ID received from the server.
|
||
|
|
||
|
The session ID is not guaranteed to remain constant throughout the life
|
||
|
of the connection, as reconnections can cause it to change.
|
||
|
"""
|
||
|
return self.client.get_sid(self.namespace) if self.client else None
|
||
|
|
||
|
@property
|
||
|
def transport(self):
|
||
|
"""The name of the transport currently in use.
|
||
|
|
||
|
The transport is returned as a string and can be one of ``polling``
|
||
|
and ``websocket``.
|
||
|
"""
|
||
|
return self.client.transport if self.client else ''
|
||
|
|
||
|
def emit(self, event, data=None):
|
||
|
"""Emit an event to the server.
|
||
|
|
||
|
: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 server. Data can be of
|
||
|
type ``str``, ``bytes``, ``list`` or ``dict``. To send
|
||
|
multiple arguments, use a tuple where each element is of
|
||
|
one of the types indicated above.
|
||
|
|
||
|
This method schedules the event to be sent out and returns, without
|
||
|
actually waiting for its delivery. In cases where the client needs to
|
||
|
ensure that the event was received, :func:`socketio.SimpleClient.call`
|
||
|
should be used instead.
|
||
|
"""
|
||
|
while True:
|
||
|
self.connected_event.wait()
|
||
|
if not self.connected:
|
||
|
raise DisconnectedError()
|
||
|
try:
|
||
|
return self.client.emit(event, data, namespace=self.namespace)
|
||
|
except SocketIOError:
|
||
|
pass
|
||
|
|
||
|
def call(self, event, data=None, timeout=60):
|
||
|
"""Emit an event to the server and wait for a response.
|
||
|
|
||
|
This method issues an emit and waits for the server to provide a
|
||
|
response or acknowledgement. If the response does not arrive before the
|
||
|
timeout, then a ``TimeoutError`` exception is raised.
|
||
|
|
||
|
: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 server. Data can be of
|
||
|
type ``str``, ``bytes``, ``list`` or ``dict``. To send
|
||
|
multiple arguments, use a tuple where each element is of
|
||
|
one of the types indicated above.
|
||
|
:param timeout: The waiting timeout. If the timeout is reached before
|
||
|
the server acknowledges the event, then a
|
||
|
``TimeoutError`` exception is raised.
|
||
|
"""
|
||
|
while True:
|
||
|
self.connected_event.wait()
|
||
|
if not self.connected:
|
||
|
raise DisconnectedError()
|
||
|
try:
|
||
|
return self.client.call(event, data, namespace=self.namespace,
|
||
|
timeout=timeout)
|
||
|
except SocketIOError:
|
||
|
pass
|
||
|
|
||
|
def receive(self, timeout=None):
|
||
|
"""Wait for an event from the server.
|
||
|
|
||
|
:param timeout: The waiting timeout. If the timeout is reached before
|
||
|
the server acknowledges the event, then a
|
||
|
``TimeoutError`` exception is raised.
|
||
|
|
||
|
The return value is a list with the event name as the first element. If
|
||
|
the server included arguments with the event, they are returned as
|
||
|
additional list elements.
|
||
|
"""
|
||
|
while not self.input_buffer:
|
||
|
if not self.connected_event.wait(
|
||
|
timeout=timeout): # pragma: no cover
|
||
|
raise TimeoutError()
|
||
|
if not self.connected:
|
||
|
raise DisconnectedError()
|
||
|
if not self.input_event.wait(timeout=timeout):
|
||
|
raise TimeoutError()
|
||
|
self.input_event.clear()
|
||
|
return self.input_buffer.pop(0)
|
||
|
|
||
|
def disconnect(self):
|
||
|
"""Disconnect from the server."""
|
||
|
if self.connected:
|
||
|
self.client.disconnect()
|
||
|
self.client = None
|
||
|
self.connected = False
|
||
|
|
||
|
def __enter__(self):
|
||
|
return self
|
||
|
|
||
|
def __exit__(self, exc_type, exc_val, exc_tb):
|
||
|
self.disconnect()
|