Browse Source

Catch-all event handlers

pull/789/head
Miguel Grinberg 4 years ago
parent
commit
28569d48ad
  1. 22
      docs/client.rst
  2. 22
      docs/server.rst
  3. 25
      src/socketio/asyncio_client.py
  4. 26
      src/socketio/asyncio_server.py
  5. 7
      src/socketio/client.py
  6. 7
      src/socketio/server.py
  7. 8
      tests/asyncio/test_asyncio_client.py
  8. 18
      tests/asyncio/test_asyncio_server.py
  9. 8
      tests/common/test_client.py
  10. 18
      tests/common/test_server.py

22
docs/client.rst

@ -65,6 +65,28 @@ or can also be coroutines::
async def message(data):
print('I received a message!')
Catch-All Event Handlers
------------------------
A "catch-all" event handler is invoked for any events that do not have an
event handler. You can define a catch-all handler using ``'*'`` as event name::
@sio.on('*')
def catch_all(event, sid, data):
pass
Asyncio clients can also use a coroutine::
@sio.on('*')
async def catch_all(event, sid, data):
pass
A catch-all event handler receives the event name as a first argument. The
remaining arguments are the same as for a regular event handler.
Connect, Connect Error and Disconnect Event Handlers
----------------------------------------------------
The ``connect``, ``connect_error`` and ``disconnect`` events are special; they
are invoked automatically when a client connects or disconnects from the
server::

22
docs/server.rst

@ -178,6 +178,28 @@ The ``sid`` argument is the Socket.IO session id, a unique identifier of each
client connection. All the events sent by a given client will have the same
``sid`` value.
Catch-All Event Handlers
------------------------
A "catch-all" event handler is invoked for any events that do not have an
event handler. You can define a catch-all handler using ``'*'`` as event name::
@sio.on('*')
def catch_all(event, sid, data):
pass
Asyncio servers can also use a coroutine::
@sio.on('*')
async def catch_all(event, sid, data):
pass
A catch-all event handler receives the event name as a first argument. The
remaining arguments are the same as for a regular event handler.
Connect and Disconnect Event Handlers
-------------------------------------
The ``connect`` and ``disconnect`` events are special; they are invoked
automatically when a client connects or disconnects from the server::

25
src/socketio/asyncio_client.py

@ -418,15 +418,22 @@ class AsyncClient(client.Client):
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]):
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
if namespace in self.handlers:
handler = None
if event in self.handlers[namespace]:
handler = self.handlers[namespace][event]
elif '*' in self.handlers[namespace]:
handler = self.handlers[namespace]['*']
args = (event, *args)
if handler:
if asyncio.iscoroutinefunction(handler):
try:
ret = await handler(*args)
except asyncio.CancelledError: # pragma: no cover
ret = None
else:
ret = handler(*args)
return ret
# or else, forward the event to a namepsace handler if one exists
elif namespace in self.namespace_handlers:

26
src/socketio/asyncio_server.py

@ -524,16 +524,22 @@ class AsyncServer(server.Server):
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
if namespace in self.handlers:
handler = None
if event in self.handlers[namespace]:
handler = self.handlers[namespace][event]
elif '*' in self.handlers[namespace]:
handler = self.handlers[namespace]['*']
args = (event, *args)
if handler:
if asyncio.iscoroutinefunction(handler):
try:
ret = await handler(*args)
except asyncio.CancelledError: # pragma: no cover
ret = None
else:
ret = handler(*args)
return ret
# or else, forward the event to a namepsace handler if one exists
elif namespace in self.namespace_handlers:

7
src/socketio/client.py

@ -609,8 +609,11 @@ class Client(object):
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]:
return self.handlers[namespace][event](*args)
if namespace in self.handlers:
if event in self.handlers[namespace]:
return self.handlers[namespace][event](*args)
elif '*' in self.handlers[namespace]:
return self.handlers[namespace]['*'](event, *args)
# or else, forward the event to a namespace handler if one exists
elif namespace in self.namespace_handlers:

7
src/socketio/server.py

@ -732,8 +732,11 @@ class Server(object):
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]:
return self.handlers[namespace][event](*args)
if namespace in self.handlers:
if event in self.handlers[namespace]:
return self.handlers[namespace][event](*args)
elif '*' in self.handlers[namespace]:
return self.handlers[namespace]['*'](event, *args)
# or else, forward the event to a namespace handler if one exists
elif namespace in self.namespace_handlers:

8
tests/asyncio/test_asyncio_client.py

@ -833,16 +833,24 @@ class TestAsyncClient(unittest.TestCase):
def test_trigger_event(self):
c = asyncio_client.AsyncClient()
handler = mock.MagicMock()
catchall_handler = mock.MagicMock()
c.on('foo', handler)
c.on('*', catchall_handler)
_run(c._trigger_event('foo', '/', 1, '2'))
_run(c._trigger_event('bar', '/', 1, '2', 3))
handler.assert_called_once_with(1, '2')
catchall_handler.assert_called_once_with('bar', 1, '2', 3)
def test_trigger_event_namespace(self):
c = asyncio_client.AsyncClient()
handler = AsyncMock()
catchall_handler = AsyncMock()
c.on('foo', handler, namespace='/bar')
c.on('*', catchall_handler, namespace='/bar')
_run(c._trigger_event('foo', '/bar', 1, '2'))
_run(c._trigger_event('bar', '/bar', 1, '2', 3))
handler.mock.assert_called_once_with(1, '2')
catchall_handler.mock.assert_called_once_with('bar', 1, '2', 3)
def test_trigger_event_class_namespace(self):
c = asyncio_client.AsyncClient()

18
tests/asyncio/test_asyncio_server.py

@ -618,18 +618,28 @@ class TestAsyncServer(unittest.TestCase):
s = asyncio_server.AsyncServer(async_handlers=False)
sid = s.manager.connect('123', '/')
handler = AsyncMock()
s.on('my message', handler)
catchall_handler = AsyncMock()
s.on('msg', handler)
s.on('*', catchall_handler)
_run(s._handle_eio_message('123', '2["msg","a","b"]'))
_run(s._handle_eio_message('123', '2["my message","a","b","c"]'))
handler.mock.assert_called_once_with(sid, 'a', 'b', 'c')
handler.mock.assert_called_once_with(sid, 'a', 'b')
catchall_handler.mock.assert_called_once_with(
'my message', sid, 'a', 'b', 'c')
def test_handle_event_with_namespace(self, eio):
eio.return_value.send = AsyncMock()
s = asyncio_server.AsyncServer(async_handlers=False)
sid = s.manager.connect('123', '/foo')
handler = mock.MagicMock()
s.on('my message', handler, namespace='/foo')
catchall_handler = mock.MagicMock()
s.on('msg', handler, namespace='/foo')
s.on('*', catchall_handler, namespace='/foo')
_run(s._handle_eio_message('123', '2/foo,["msg","a","b"]'))
_run(s._handle_eio_message('123', '2/foo,["my message","a","b","c"]'))
handler.assert_called_once_with(sid, 'a', 'b', 'c')
handler.assert_called_once_with(sid, 'a', 'b')
catchall_handler.assert_called_once_with(
'my message', sid, 'a', 'b', 'c')
def test_handle_event_with_disconnected_namespace(self, eio):
eio.return_value.send = AsyncMock()

8
tests/common/test_client.py

@ -934,16 +934,24 @@ class TestClient(unittest.TestCase):
def test_trigger_event(self):
c = client.Client()
handler = mock.MagicMock()
catchall_handler = mock.MagicMock()
c.on('foo', handler)
c.on('*', catchall_handler)
c._trigger_event('foo', '/', 1, '2')
c._trigger_event('bar', '/', 1, '2', 3)
handler.assert_called_once_with(1, '2')
catchall_handler.assert_called_once_with('bar', 1, '2', 3)
def test_trigger_event_namespace(self):
c = client.Client()
handler = mock.MagicMock()
catchall_handler = mock.MagicMock()
c.on('foo', handler, namespace='/bar')
c.on('*', catchall_handler, namespace='/bar')
c._trigger_event('foo', '/bar', 1, '2')
c._trigger_event('bar', '/bar', 1, '2', 3)
handler.assert_called_once_with(1, '2')
catchall_handler.assert_called_once_with('bar', 1, '2', 3)
def test_trigger_event_class_namespace(self):
c = client.Client()

18
tests/common/test_server.py

@ -546,17 +546,27 @@ class TestServer(unittest.TestCase):
s = server.Server(async_handlers=False)
s.manager.connect('123', '/')
handler = mock.MagicMock()
s.on('my message', handler)
catchall_handler = mock.MagicMock()
s.on('msg', handler)
s.on('*', catchall_handler)
s._handle_eio_message('123', '2["msg","a","b"]')
s._handle_eio_message('123', '2["my message","a","b","c"]')
handler.assert_called_once_with('1', 'a', 'b', 'c')
handler.assert_called_once_with('1', 'a', 'b')
catchall_handler.assert_called_once_with(
'my message', '1', 'a', 'b', 'c')
def test_handle_event_with_namespace(self, eio):
s = server.Server(async_handlers=False)
s.manager.connect('123', '/foo')
handler = mock.MagicMock()
s.on('my message', handler, namespace='/foo')
catchall_handler = mock.MagicMock()
s.on('msg', handler, namespace='/foo')
s.on('*', catchall_handler, namespace='/foo')
s._handle_eio_message('123', '2/foo,["msg","a","b"]')
s._handle_eio_message('123', '2/foo,["my message","a","b","c"]')
handler.assert_called_once_with('1', 'a', 'b', 'c')
handler.assert_called_once_with('1', 'a', 'b')
catchall_handler.assert_called_once_with(
'my message', '1', 'a', 'b', 'c')
def test_handle_event_with_disconnected_namespace(self, eio):
s = server.Server(async_handlers=False)

Loading…
Cancel
Save