12 changed files with 822 additions and 430 deletions
@ -0,0 +1,290 @@ |
|||
The Socket.IO Client |
|||
==================== |
|||
|
|||
This package contains two Socket.IO clients: |
|||
|
|||
- The :func:`socketio.Client` class creates a client compatible with the |
|||
standard Python library. |
|||
- The :func:`socketio.AsyncClient` class creates a client compatible with |
|||
the ``asyncio`` package. |
|||
|
|||
The methods in the two clients are the same, with the only difference that in |
|||
the ``asyncio`` client most methods are implemented as coroutines. |
|||
|
|||
Installation |
|||
------------ |
|||
|
|||
To install the standard Python client along with its dependencies, use the |
|||
following command:: |
|||
|
|||
pip install "python-socketio[client]" |
|||
|
|||
If instead you plan on using the ``asyncio`` client, then use this:: |
|||
|
|||
pip install "python-socketio[asyncio_client]" |
|||
|
|||
Creating a Client Instance |
|||
-------------------------- |
|||
|
|||
To instantiate an Socket.IO client, simply create an instance of the |
|||
appropriate client class:: |
|||
|
|||
import socketio |
|||
|
|||
# standard Python |
|||
sio = socketio.Client() |
|||
|
|||
# asyncio |
|||
sio = socketio.AsyncClient() |
|||
|
|||
Defining Event Handlers |
|||
----------------------- |
|||
|
|||
To responds to events triggered by the connection or the server, event Handler |
|||
functions must be defined using the ``on`` decorator:: |
|||
|
|||
@sio.on('connect') |
|||
def on_connect(): |
|||
print('I'm connected!') |
|||
|
|||
@sio.on('message') |
|||
def on_message(data): |
|||
print('I received a message!') |
|||
|
|||
@sio.on('my message') |
|||
def on_message(data): |
|||
print('I received a custom message!') |
|||
|
|||
@sio.on('disconnect') |
|||
def on_disconnect(): |
|||
print('I'm disconnected!') |
|||
|
|||
For the ``asyncio`` server, event handlers can be regular functions as above, |
|||
or can also be coroutines:: |
|||
|
|||
@sio.on('message') |
|||
async def on_message(data): |
|||
print('I received a message!') |
|||
|
|||
The argument given to the ``on`` decorator is the event name. The predefined |
|||
events that are supported are ``connect``, ``message`` and ``disconnect``. The |
|||
application can define any other desired event names. |
|||
|
|||
Note that the ``disconnect`` handler is invoked for application initiated |
|||
disconnects, server initiated disconnects, or accidental disconnects, for |
|||
example due to networking failures. In the case of an accidental disconnection, |
|||
the client is going to attempt to reconnect immediately after invoking the |
|||
disconnect handler. As soon as the connection is re-established the connect |
|||
handler will be invoked once again. |
|||
|
|||
The ``data`` argument passed to the ``'message'`` and custom event Handlers |
|||
contains application-specific data provided by the server. |
|||
|
|||
Connecting to a Server |
|||
---------------------- |
|||
|
|||
The connection to a server is established by calling the ``connect()`` |
|||
method:: |
|||
|
|||
sio.connect('http://localhost:5000') |
|||
|
|||
In the case of the ``asyncio`` client, the method is a coroutine:: |
|||
|
|||
await sio.connect('http://localhost:5000') |
|||
|
|||
Emitting Events |
|||
--------------- |
|||
|
|||
The client can emit an event to the server using the ``emit()`` method:: |
|||
|
|||
sio.emit('my message', {'foo': 'bar'}) |
|||
|
|||
Or in the case of ``asyncio``, as a coroutine:: |
|||
|
|||
await sio.emit('my message', {'foo': 'bar'}) |
|||
|
|||
The single argument provided to the method is the data that is passed on |
|||
to the server. The data can be of type ``str``, ``bytes``, ``dict`` or |
|||
``list``. The data included inside dictionaries and lists is also |
|||
constrained to these types. |
|||
|
|||
The ``emit()`` method can be invoked inside an event handler as a response |
|||
to a server event, or in any other part of the application, including in |
|||
background tasks. |
|||
|
|||
For convenience, a ``send()`` method is also provided. This method accepts |
|||
a data element as its only argument, and emits the standard ``message`` |
|||
event with it:: |
|||
|
|||
sio.send('some data') |
|||
|
|||
In the case of ``asyncio``, ``send()`` is a coroutine:: |
|||
|
|||
await sio.send('some data') |
|||
|
|||
Event Callbacks |
|||
--------------- |
|||
|
|||
When a server emits an event to a client, it can optionally provide a |
|||
callback function, to be invoked as a way of acknowledgment that the server |
|||
has processed the event. While this is entirely managed by the server, the |
|||
client can provide a list of return values that are to be passed on to the |
|||
callback function set up by the server. This is achieves simply by returning |
|||
the desired values from the handler function:: |
|||
|
|||
@sio.on('my event', namespace='/chat') |
|||
def my_event_handler(sid, data): |
|||
# handle the message |
|||
return "OK", 123 |
|||
|
|||
Likewise, the client can request a callback function to be invoked after the |
|||
server has processed an event. The :func:`socketio.Server.emit` method has an |
|||
optional ``callback`` argument that can be set to a callable. If this |
|||
argument is given, the callable will be invoked after the server has processed |
|||
the event, and any values returned by the server handler will be passed as |
|||
arguments to this function. |
|||
|
|||
Namespaces |
|||
---------- |
|||
|
|||
The Socket.IO protocol supports multiple logical connections, all multiplexed |
|||
on the same physical connection. Clients can open multiple connections by |
|||
specifying a different *namespace* on each. Namespaces use a path syntax |
|||
starting with a forward slash. A list of namespaces can be given by the client |
|||
in the ``connect()`` call. For example, this example creates two logical |
|||
connections, the default one plus a second connection under the ``/chat`` |
|||
namespace:: |
|||
|
|||
sio.connect('http://localhost:5000', namespaces=['/chat']) |
|||
|
|||
To define event handlers on a namespace, the ``namespace`` argument must be |
|||
added to the ``on`` decorator:: |
|||
|
|||
@sio.on('connect', namespace='/chat') |
|||
def on_connect(): |
|||
print('I'm connected to the /chat namespace!') |
|||
|
|||
Likewise, the client can emit an event to the server on a namespace by |
|||
providing its in the ``emit()`` call:: |
|||
|
|||
sio.emit('my message', {'foo': 'bar'}, namespace='/chat') |
|||
|
|||
If the ``namespaces`` argument of the ``connect()`` call isn't given, any |
|||
namespaces used in event handlers are automatically connected. |
|||
|
|||
Class-Based Namespaces |
|||
---------------------- |
|||
|
|||
As an alternative to the decorator-based event handlers, the event handlers |
|||
that belong to a namespace can be created as methods of a subclass of |
|||
:class:`socketio.ClientNamespace`:: |
|||
|
|||
class MyCustomNamespace(socketio.ClientNamespace): |
|||
def on_connect(self): |
|||
pass |
|||
|
|||
def on_disconnect(self): |
|||
pass |
|||
|
|||
def on_my_event(self, data): |
|||
self.emit('my_response', data) |
|||
|
|||
sio.register_namespace(MyCustomNamespace('/chat')) |
|||
|
|||
For asyncio based severs, namespaces must inherit from |
|||
:class:`socketio.AsyncClientNamespace`, and can define event handlers as |
|||
coroutines if desired:: |
|||
|
|||
class MyCustomNamespace(socketio.AsyncClientNamespace): |
|||
def on_connect(self): |
|||
pass |
|||
|
|||
def on_disconnect(self): |
|||
pass |
|||
|
|||
async def on_my_event(self, data): |
|||
await self.emit('my_response', data) |
|||
|
|||
sio.register_namespace(MyCustomNamespace('/chat')) |
|||
|
|||
When class-based namespaces are used, any events received by the client are |
|||
dispatched to a method named as the event name with the ``on_`` prefix. For |
|||
example, event ``my_event`` will be handled by a method named ``on_my_event``. |
|||
If an event is received for which there is no corresponding method defined in |
|||
the namespace class, then the event is ignored. All event names used in |
|||
class-based namespaces must use characters that are legal in method names. |
|||
|
|||
As a convenience to methods defined in a class-based namespace, the namespace |
|||
instance includes versions of several of the methods in the |
|||
:class:`socketio.Client` and :class:`socketio.AsyncClient` classes that |
|||
default to the proper namespace when the ``namespace`` argument is not given. |
|||
|
|||
In the case that an event has a handler in a class-based namespace, and also a |
|||
decorator-based function handler, only the standalone function handler is |
|||
invoked. |
|||
|
|||
Disconnecting from the Server |
|||
----------------------------- |
|||
|
|||
At any time the client can request to be disconnected from the server by |
|||
invoking the ``disconnect()`` method:: |
|||
|
|||
sio.disconnect() |
|||
|
|||
For the ``asyncio`` client this is a coroutine:: |
|||
|
|||
await sio.disconnect() |
|||
|
|||
Managing Background Tasks |
|||
------------------------- |
|||
|
|||
When a client connection to the server is established, a few background |
|||
tasks will be spawned to keep the connection alive and handle incoming |
|||
events. The application running on the main thread is free to do any |
|||
work, as this is not going to prevent the functioning of the Socket.IO |
|||
client. |
|||
|
|||
If the application does not have anything to do in the main thread and |
|||
just wants to wait until the connection with the server ends, it can call |
|||
the ``wait()`` method:: |
|||
|
|||
sio.wait() |
|||
|
|||
Or in the ``asyncio`` version:: |
|||
|
|||
await sio.wait() |
|||
|
|||
For the convenience of the application, a helper function is provided to |
|||
start a custom background task:: |
|||
|
|||
def my_background_task(my_argument) |
|||
# do some background work here! |
|||
pass |
|||
|
|||
sio.start_background_task(my_background_task, 123) |
|||
|
|||
The arguments passed to this method are the background function and any |
|||
positional or keyword arguments to invoke the function with. |
|||
|
|||
Here is the ``asyncio`` version:: |
|||
|
|||
async def my_background_task(my_argument) |
|||
# do some background work here! |
|||
pass |
|||
|
|||
sio.start_background_task(my_background_task, 123) |
|||
|
|||
Note that this function is not a coroutine, since it does not wait for the |
|||
background function to end. The background function must be a coroutine. |
|||
|
|||
The ``sleep()`` method is a second convenince function that is provided for |
|||
the benefit of applications working with background tasks of their own:: |
|||
|
|||
sio.sleep(2) |
|||
|
|||
Or for ``asyncio``:: |
|||
|
|||
await sio.sleep(2) |
|||
|
|||
The single argument passed to the method is the number of seconds to sleep |
|||
for. |
@ -1,277 +0,0 @@ |
|||
Deployment |
|||
========== |
|||
|
|||
The following sections describe a variety of deployment strategies for |
|||
Socket.IO servers. |
|||
|
|||
aiohttp |
|||
------- |
|||
|
|||
`Aiohttp <http://aiohttp.readthedocs.io/>`_ is a framework with support for HTTP |
|||
and WebSocket, based on asyncio. Support for this framework is limited to Python |
|||
3.5 and newer. |
|||
|
|||
Instances of class ``socketio.AsyncServer`` will automatically use aiohttp |
|||
for asynchronous operations if the library is installed. To request its use |
|||
explicitly, the ``async_mode`` option can be given in the constructor:: |
|||
|
|||
sio = socketio.AsyncServer(async_mode='aiohttp') |
|||
|
|||
A server configured for aiohttp must be attached to an existing application:: |
|||
|
|||
app = web.Application() |
|||
sio.attach(app) |
|||
|
|||
The aiohttp application can define regular routes that will coexist with the |
|||
Socket.IO server. A typical pattern is to add routes that serve a client |
|||
application and any associated static files. |
|||
|
|||
The aiohttp application is then executed in the usual manner:: |
|||
|
|||
if __name__ == '__main__': |
|||
web.run_app(app) |
|||
|
|||
Tornado |
|||
------- |
|||
|
|||
`Tornado <http://www.tornadoweb.org//>`_ is a web framework with support |
|||
for HTTP and WebSocket. Support for this framework requires Python 3.5 and |
|||
newer. Only Tornado version 5 and newer are supported, thanks to its tight |
|||
integration with asyncio. |
|||
|
|||
Instances of class ``socketio.AsyncServer`` will automatically use tornado |
|||
for asynchronous operations if the library is installed. To request its use |
|||
explicitly, the ``async_mode`` option can be given in the constructor:: |
|||
|
|||
sio = socketio.AsyncServer(async_mode='tornado') |
|||
|
|||
A server configured for tornado must include a request handler for |
|||
Engine.IO:: |
|||
|
|||
app = tornado.web.Application( |
|||
[ |
|||
(r"/socketio.io/", socketio.get_tornado_handler(sio)), |
|||
], |
|||
# ... other application options |
|||
) |
|||
|
|||
The tornado application can define other routes that will coexist with the |
|||
Socket.IO server. A typical pattern is to add routes that serve a client |
|||
application and any associated static files. |
|||
|
|||
The tornado application is then executed in the usual manner:: |
|||
|
|||
app.listen(port) |
|||
tornado.ioloop.IOLoop.current().start() |
|||
|
|||
Sanic |
|||
----- |
|||
|
|||
`Sanic <http://sanic.readthedocs.io/>`_ is a very efficient asynchronous web |
|||
server for Python 3.5 and newer. |
|||
|
|||
Instances of class ``socketio.AsyncServer`` will automatically use Sanic for |
|||
asynchronous operations if the framework is installed. To request its use |
|||
explicitly, the ``async_mode`` option can be given in the constructor:: |
|||
|
|||
sio = socketio.AsyncServer(async_mode='sanic') |
|||
|
|||
A server configured for aiohttp must be attached to an existing application:: |
|||
|
|||
app = web.Application() |
|||
sio.attach(app) |
|||
|
|||
The Sanic application can define regular routes that will coexist with the |
|||
Socket.IO server. A typical pattern is to add routes that serve a client |
|||
application and any associated static files. |
|||
|
|||
The Sanic application is then executed in the usual manner:: |
|||
|
|||
if __name__ == '__main__': |
|||
app.run() |
|||
|
|||
Uvicorn, Daphne, and other ASGI servers |
|||
--------------------------------------- |
|||
|
|||
The ``socketio.ASGIApp`` class is an ASGI compatible application that can |
|||
forward Socket.IO traffic to an ``socketio.AsyncServer`` instance:: |
|||
|
|||
sio = socketio.AsyncServer(async_mode='asgi') |
|||
app = socketio.ASGIApp(sio) |
|||
|
|||
The application can then be deployed with any ASGI compatible web server. |
|||
|
|||
Eventlet |
|||
-------- |
|||
|
|||
`Eventlet <http://eventlet.net/>`_ is a high performance concurrent networking |
|||
library for Python 2 and 3 that uses coroutines, enabling code to be written in |
|||
the same style used with the blocking standard library functions. An Socket.IO |
|||
server deployed with eventlet has access to the long-polling and WebSocket |
|||
transports. |
|||
|
|||
Instances of class ``socketio.Server`` will automatically use eventlet for |
|||
asynchronous operations if the library is installed. To request its use |
|||
explicitly, the ``async_mode`` option can be given in the constructor:: |
|||
|
|||
sio = socketio.Server(async_mode='eventlet') |
|||
|
|||
A server configured for eventlet is deployed as a regular WSGI application, |
|||
using the provided ``socketio.Middleware``:: |
|||
|
|||
app = socketio.Middleware(sio) |
|||
import eventlet |
|||
eventlet.wsgi.server(eventlet.listen(('', 8000)), app) |
|||
|
|||
Using Gunicorn with Eventlet |
|||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
|||
|
|||
An alternative to running the eventlet WSGI server as above is to use |
|||
`gunicorn <gunicorn.org>`_, a fully featured pure Python web server. The |
|||
command to launch the application under gunicorn is shown below:: |
|||
|
|||
$ gunicorn -k eventlet -w 1 module:app |
|||
|
|||
Due to limitations in its load balancing algorithm, gunicorn can only be used |
|||
with one worker process, so the ``-w`` option cannot be set to a value higher |
|||
than 1. A single eventlet worker can handle a large number of concurrent |
|||
clients, each handled by a greenlet. |
|||
|
|||
Eventlet provides a ``monkey_patch()`` function that replaces all the blocking |
|||
functions in the standard library with equivalent asynchronous versions. While |
|||
python-socketio does not require monkey patching, other libraries such as |
|||
database drivers are likely to require it. |
|||
|
|||
Gevent |
|||
------ |
|||
|
|||
`Gevent <http://gevent.org/>`_ is another asynchronous framework based on |
|||
coroutines, very similar to eventlet. An Socket.IO server deployed with |
|||
gevent has access to the long-polling transport. If project |
|||
`gevent-websocket <https://bitbucket.org/Jeffrey/gevent-websocket/>`_ is |
|||
installed, the WebSocket transport is also available. |
|||
|
|||
Instances of class ``socketio.Server`` will automatically use gevent for |
|||
asynchronous operations if the library is installed and eventlet is not |
|||
installed. To request gevent to be selected explicitly, the ``async_mode`` |
|||
option can be given in the constructor:: |
|||
|
|||
sio = socketio.Server(async_mode='gevent') |
|||
|
|||
A server configured for gevent is deployed as a regular WSGI application, |
|||
using the provided ``socketio.Middleware``:: |
|||
|
|||
app = socketio.Middleware(sio) |
|||
from gevent import pywsgi |
|||
pywsgi.WSGIServer(('', 8000), app).serve_forever() |
|||
|
|||
If the WebSocket transport is installed, then the server must be started as |
|||
follows:: |
|||
|
|||
from gevent import pywsgi |
|||
from geventwebsocket.handler import WebSocketHandler |
|||
app = socketio.Middleware(sio) |
|||
pywsgi.WSGIServer(('', 8000), app, |
|||
handler_class=WebSocketHandler).serve_forever() |
|||
|
|||
Using Gunicorn with Gevent |
|||
~~~~~~~~~~~~~~~~~~~~~~~~~~ |
|||
|
|||
An alternative to running the gevent WSGI server as above is to use |
|||
`gunicorn <gunicorn.org>`_, a fully featured pure Python web server. The |
|||
command to launch the application under gunicorn is shown below:: |
|||
|
|||
$ gunicorn -k gevent -w 1 module:app |
|||
|
|||
Or to include WebSocket:: |
|||
|
|||
$ gunicorn -k geventwebsocket.gunicorn.workers.GeventWebSocketWorker -w 1 module: app |
|||
|
|||
Same as with eventlet, due to limitations in its load balancing algorithm, |
|||
gunicorn can only be used with one worker process, so the ``-w`` option cannot |
|||
be higher than 1. A single gevent worker can handle a large number of |
|||
concurrent clients through the use of greenlets. |
|||
|
|||
Gevent provides a ``monkey_patch()`` function that replaces all the blocking |
|||
functions in the standard library with equivalent asynchronous versions. While |
|||
python-socketio does not require monkey patching, other libraries such as |
|||
database drivers are likely to require it. |
|||
|
|||
uWSGI |
|||
----- |
|||
|
|||
When using the uWSGI server in combination with gevent, the Socket.IO server |
|||
can take advantage of uWSGI's native WebSocket support. |
|||
|
|||
Instances of class ``socketio.Server`` will automatically use this option for |
|||
asynchronous operations if both gevent and uWSGI are installed and eventlet is |
|||
not installed. To request this asynchronous mode explicitly, the |
|||
``async_mode`` option can be given in the constructor:: |
|||
|
|||
# gevent with uWSGI |
|||
sio = socketio.Server(async_mode='gevent_uwsgi') |
|||
|
|||
A complete explanation of the configuration and usage of the uWSGI server is |
|||
beyond the scope of this documentation. The uWSGI server is a fairly complex |
|||
package that provides a large and comprehensive set of options. It must be |
|||
compiled with WebSocket and SSL support for the WebSocket transport to be |
|||
available. As way of an introduction, the following command starts a uWSGI |
|||
server for the ``latency.py`` example on port 5000:: |
|||
|
|||
$ uwsgi --http :5000 --gevent 1000 --http-websockets --master --wsgi-file latency.py --callable app |
|||
|
|||
Standard Threads |
|||
---------------- |
|||
|
|||
While not comparable to eventlet and gevent in terms of performance, |
|||
the Socket.IO server can also be configured to work with multi-threaded web |
|||
servers that use standard Python threads. This is an ideal setup to use with |
|||
development servers such as `Werkzeug <http://werkzeug.pocoo.org>`_. Only the |
|||
long-polling transport is currently available when using standard threads. |
|||
|
|||
Instances of class ``socketio.Server`` will automatically use the threading |
|||
mode if neither eventlet nor gevent are not installed. To request the |
|||
threading mode explicitly, the ``async_mode`` option can be given in the |
|||
constructor:: |
|||
|
|||
sio = socketio.Server(async_mode='threading') |
|||
|
|||
A server configured for threading is deployed as a regular web application, |
|||
using any WSGI complaint multi-threaded server. The example below deploys an |
|||
Socket.IO application combined with a Flask web application, using Flask's |
|||
development web server based on Werkzeug:: |
|||
|
|||
sio = socketio.Server(async_mode='threading') |
|||
app = Flask(__name__) |
|||
app.wsgi_app = socketio.Middleware(sio, app.wsgi_app) |
|||
|
|||
# ... Socket.IO and Flask handler functions ... |
|||
|
|||
if __name__ == '__main__': |
|||
app.run(threaded=True) |
|||
|
|||
When using the threading mode, it is important to ensure that the WSGI server |
|||
can handle multiple concurrent requests using threads, since a client can have |
|||
up to two outstanding requests at any given time. The Werkzeug server is |
|||
single-threaded by default, so the ``threaded=True`` option is required. |
|||
|
|||
Note that servers that use worker processes instead of threads, such as |
|||
gunicorn, do not support a Socket.IO server configured in threading mode. |
|||
|
|||
Scalability Notes |
|||
----------------- |
|||
|
|||
Socket.IO is a stateful protocol, which makes horizontal scaling more |
|||
difficult. To deploy a cluster of Socket.IO processes hosted on one or |
|||
multiple servers, the following conditions must be met: |
|||
|
|||
- Each Socket.IO process must be able to handle multiple requests |
|||
concurrently. This is required because long-polling clients send two |
|||
requests in parallel. Worker processes that can only handle one request at a |
|||
time are not supported. |
|||
- The load balancer must be configured to always forward requests from a |
|||
client to the same worker process. Load balancers call this *sticky |
|||
sessions*, or *session affinity*. |
|||
- The worker processes need to communicate with each other to coordinate |
|||
complex operations such as broadcasts. This is done through a configured |
|||
message queue. See the section on using message queues for details. |
Loading…
Reference in new issue