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.
 
 

113 lines
3.1 KiB

from collections import defaultdict
import gevent
from gevent.event import AsyncResult
class EventEmitter(object):
"""
Implements event emitter using ``gevent`` module.
Other modules can inherit from this object.
.. code:: python
class SomeClass(EventEmitter):
pass
"""
def emit(self, event, *args):
"""
Emit event with some arguments
:type event: any
:param args: any or no arguments
"""
gevent.idle()
if hasattr(self, '_EventEmitter__callbacks'):
if event in self.__callbacks:
for callback in list(self.__callbacks[event]):
if isinstance(callback, AsyncResult):
self.remove_listener(event, callback)
result = args
if len(args) == 1:
result = args[0]
callback.set(result)
else:
gevent.spawn(callback, *args)
gevent.idle()
# every event is also emitted as None
if event is not None:
self.emit(None, event, *args)
def remove_listener(self, event, callback):
"""
Removes callback for the specified event
:param event: event identifier
:param callback: callback reference
:type callback: function, method or :py:class:`gevent.event.AsyncResult`
"""
if not hasattr(self, '_EventEmitter__callbacks'):
return
if event in self.__callbacks:
del self.__callbacks[event][callback]
if not self.__callbacks[event]:
del self.__callbacks[event]
def wait_event(self, event, timeout=None):
"""
Blocks until an event and returns the results
:param event: event identifier
:param timeout: seconds to wait before raising an exception
:type timeout: int
:return: returns event arguments, if any. If there are many, returns tuple.
:rtype: None, any, or tuple
:raises: gevent.Timeout
"""
result = AsyncResult()
self.on(event, result)
return result.get(True, timeout)
def on(self, event, callback=None):
"""
Registers a callback for the specified event
:param event: event name
:param callback: callback function
Can be as function decorator if only event is specified.
.. code:: python
@instaceOfSomeClass.on("some event")
def handle_event():
pass
instaceOfSomeClass.on("some event", handle_event)
To listen for any event, use :py:class:`None` as event identifier.
"""
if not hasattr(self, '_EventEmitter__callbacks'):
self.__callbacks = defaultdict(dict)
# when used function
if callback:
self.__callbacks[event][callback] = None
return
# as decorator
def wrapper(callback):
self.__callbacks[event][callback] = None
return callback
return wrapper