2018-05-16 11:22:34 +03:00
|
|
|
#
|
2019-02-06 15:49:11 +01:00
|
|
|
# Chronos synchronization primitives
|
2018-05-16 11:22:34 +03:00
|
|
|
#
|
2019-02-06 15:49:11 +01:00
|
|
|
# (c) Copyright 2018-Present Eugene Kabanov
|
|
|
|
# (c) Copyright 2018-Present Status Research & Development GmbH
|
2018-05-16 11:22:34 +03:00
|
|
|
#
|
|
|
|
# Licensed under either of
|
|
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
|
|
# MIT license (LICENSE-MIT)
|
|
|
|
|
2021-09-15 16:55:15 +03:00
|
|
|
## This module implements some core synchronization primitives.
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
2021-09-15 16:55:15 +03:00
|
|
|
import std/[sequtils, deques, tables, typetraits]
|
2020-09-10 23:28:20 +03:00
|
|
|
import ./asyncloop
|
2021-08-26 14:22:29 +03:00
|
|
|
export asyncloop
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
type
|
|
|
|
AsyncLock* = ref object of RootRef
|
|
|
|
## A primitive lock is a synchronization primitive that is not owned by
|
|
|
|
## a particular coroutine when locked. A primitive lock is in one of two
|
|
|
|
## states, ``locked`` or ``unlocked``.
|
|
|
|
##
|
|
|
|
## When more than one coroutine is blocked in ``acquire()`` waiting for
|
|
|
|
## the state to turn to unlocked, only one coroutine proceeds when a
|
|
|
|
## ``release()`` call resets the state to unlocked; first coroutine which
|
|
|
|
## is blocked in ``acquire()`` is being processed.
|
|
|
|
locked: bool
|
2020-09-10 23:28:20 +03:00
|
|
|
acquired: bool
|
2019-07-17 16:12:31 +03:00
|
|
|
waiters: seq[Future[void]]
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
AsyncEvent* = ref object of RootRef
|
|
|
|
## A primitive event object.
|
|
|
|
##
|
|
|
|
## An event manages a flag that can be set to `true` with the ``fire()``
|
|
|
|
## procedure and reset to `false` with the ``clear()`` procedure.
|
|
|
|
## The ``wait()`` coroutine blocks until the flag is `false`.
|
|
|
|
##
|
|
|
|
## If more than one coroutine blocked in ``wait()`` waiting for event
|
|
|
|
## state to be signaled, when event get fired, then all coroutines
|
|
|
|
## continue proceeds in order, they have entered waiting state.
|
|
|
|
flag: bool
|
2021-01-22 15:02:13 +02:00
|
|
|
waiters: seq[Future[void]]
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
AsyncQueue*[T] = ref object of RootRef
|
|
|
|
## A queue, useful for coordinating producer and consumer coroutines.
|
|
|
|
##
|
|
|
|
## If ``maxsize`` is less than or equal to zero, the queue size is
|
|
|
|
## infinite. If it is an integer greater than ``0``, then "await put()"
|
|
|
|
## will block when the queue reaches ``maxsize``, until an item is
|
|
|
|
## removed by "await get()".
|
2019-07-17 16:12:31 +03:00
|
|
|
getters: seq[Future[void]]
|
|
|
|
putters: seq[Future[void]]
|
2018-05-16 11:22:34 +03:00
|
|
|
queue: Deque[T]
|
|
|
|
maxsize: int
|
|
|
|
|
2019-11-22 18:59:02 +02:00
|
|
|
AsyncQueueEmptyError* = object of CatchableError
|
2018-05-16 11:22:34 +03:00
|
|
|
## ``AsyncQueue`` is empty.
|
2019-11-22 18:59:02 +02:00
|
|
|
AsyncQueueFullError* = object of CatchableError
|
2018-05-16 11:22:34 +03:00
|
|
|
## ``AsyncQueue`` is full.
|
2019-11-22 18:59:02 +02:00
|
|
|
AsyncLockError* = object of CatchableError
|
2018-05-16 11:22:34 +03:00
|
|
|
## ``AsyncLock`` is either locked or unlocked.
|
|
|
|
|
2021-09-15 16:55:15 +03:00
|
|
|
EventBusSubscription*[T] = proc(bus: AsyncEventBus,
|
|
|
|
payload: EventPayload[T]): Future[void] {.
|
|
|
|
gcsafe, raises: [Defect].}
|
|
|
|
## EventBus subscription callback type.
|
|
|
|
|
|
|
|
EventBusAllSubscription* = proc(bus: AsyncEventBus,
|
|
|
|
event: AwaitableEvent): Future[void] {.
|
|
|
|
gcsafe, raises: [Defect].}
|
|
|
|
## EventBus subscription callback type.
|
|
|
|
|
|
|
|
EventBusCallback = proc(bus: AsyncEventBus, event: string, key: EventBusKey,
|
|
|
|
data: EventPayloadBase) {.
|
|
|
|
gcsafe, raises: [Defect].}
|
|
|
|
|
|
|
|
EventBusKey* = object
|
|
|
|
## Unique subscription key.
|
|
|
|
eventName: string
|
|
|
|
typeName: string
|
|
|
|
unique: uint64
|
|
|
|
cb: EventBusCallback
|
|
|
|
|
|
|
|
EventItem = object
|
|
|
|
waiters: seq[FutureBase]
|
|
|
|
subscribers: seq[EventBusKey]
|
|
|
|
|
|
|
|
AsyncEventBus* = ref object of RootObj
|
|
|
|
## An eventbus object.
|
|
|
|
counter: uint64
|
|
|
|
events: Table[string, EventItem]
|
|
|
|
subscribers: seq[EventBusKey]
|
|
|
|
waiters: seq[Future[AwaitableEvent]]
|
|
|
|
|
|
|
|
EventPayloadBase* = ref object of RootObj
|
|
|
|
loc: ptr SrcLoc
|
|
|
|
|
|
|
|
EventPayload*[T] = ref object of EventPayloadBase
|
|
|
|
## Eventbus' event payload object
|
|
|
|
value: T
|
|
|
|
|
|
|
|
AwaitableEvent* = object
|
|
|
|
## Eventbus' event payload object
|
|
|
|
eventName: string
|
|
|
|
payload: EventPayloadBase
|
|
|
|
|
2018-05-16 11:22:34 +03:00
|
|
|
proc newAsyncLock*(): AsyncLock =
|
|
|
|
## Creates new asynchronous lock ``AsyncLock``.
|
|
|
|
##
|
|
|
|
## Lock is created in the unlocked state. When the state is unlocked,
|
|
|
|
## ``acquire()`` changes the state to locked and returns immediately.
|
|
|
|
## When the state is locked, ``acquire()`` blocks until a call to
|
|
|
|
## ``release()`` in another coroutine changes it to unlocked.
|
|
|
|
##
|
|
|
|
## The ``release()`` procedure changes the state to unlocked and returns
|
|
|
|
## immediately.
|
|
|
|
|
|
|
|
# Workaround for callSoon() not worked correctly before
|
2021-01-11 19:15:23 +02:00
|
|
|
# getThreadDispatcher() call.
|
|
|
|
discard getThreadDispatcher()
|
2020-09-10 23:28:20 +03:00
|
|
|
AsyncLock(waiters: newSeq[Future[void]](), locked: false, acquired: false)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
2020-09-10 23:28:20 +03:00
|
|
|
proc wakeUpFirst(lock: AsyncLock): bool {.inline.} =
|
2019-07-17 16:12:31 +03:00
|
|
|
## Wake up the first waiter if it isn't done.
|
2020-09-10 23:28:20 +03:00
|
|
|
var i = 0
|
|
|
|
var res = false
|
|
|
|
while i < len(lock.waiters):
|
|
|
|
var waiter = lock.waiters[i]
|
|
|
|
inc(i)
|
|
|
|
if not(waiter.finished()):
|
|
|
|
waiter.complete()
|
|
|
|
res = true
|
2019-07-17 16:12:31 +03:00
|
|
|
break
|
2020-09-10 23:28:20 +03:00
|
|
|
if i > 0:
|
|
|
|
lock.waiters.delete(0, i - 1)
|
|
|
|
res
|
2019-07-17 16:12:31 +03:00
|
|
|
|
|
|
|
proc checkAll(lock: AsyncLock): bool {.inline.} =
|
|
|
|
## Returns ``true`` if waiters array is empty or full of cancelled futures.
|
|
|
|
for fut in lock.waiters.mitems():
|
|
|
|
if not(fut.cancelled()):
|
2020-09-10 23:28:20 +03:00
|
|
|
return false
|
|
|
|
return true
|
2019-07-17 16:12:31 +03:00
|
|
|
|
2018-05-16 11:22:34 +03:00
|
|
|
proc acquire*(lock: AsyncLock) {.async.} =
|
|
|
|
## Acquire a lock ``lock``.
|
|
|
|
##
|
|
|
|
## This procedure blocks until the lock ``lock`` is unlocked, then sets it
|
|
|
|
## to locked and returns.
|
2019-07-17 16:12:31 +03:00
|
|
|
if not(lock.locked) and lock.checkAll():
|
2020-09-10 23:28:20 +03:00
|
|
|
lock.acquired = true
|
2018-05-16 11:22:34 +03:00
|
|
|
lock.locked = true
|
|
|
|
else:
|
2019-03-31 18:33:01 +03:00
|
|
|
var w = newFuture[void]("AsyncLock.acquire")
|
2019-07-17 16:12:31 +03:00
|
|
|
lock.waiters.add(w)
|
2020-09-10 23:28:20 +03:00
|
|
|
await w
|
|
|
|
lock.acquired = true
|
2018-05-16 11:22:34 +03:00
|
|
|
lock.locked = true
|
|
|
|
|
|
|
|
proc locked*(lock: AsyncLock): bool =
|
|
|
|
## Return `true` if the lock ``lock`` is acquired, `false` otherwise.
|
2020-09-10 23:28:20 +03:00
|
|
|
lock.locked
|
2018-05-16 11:22:34 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc release*(lock: AsyncLock) {.raises: [Defect, AsyncLockError].} =
|
2018-05-16 11:22:34 +03:00
|
|
|
## Release a lock ``lock``.
|
|
|
|
##
|
|
|
|
## When the ``lock`` is locked, reset it to unlocked, and return. If any
|
|
|
|
## other coroutines are blocked waiting for the lock to become unlocked,
|
|
|
|
## allow exactly one of them to proceed.
|
|
|
|
if lock.locked:
|
2020-09-10 23:28:20 +03:00
|
|
|
# We set ``lock.locked`` to ``false`` only when there no active waiters.
|
|
|
|
# If active waiters are present, then ``lock.locked`` will be set to `true`
|
|
|
|
# in ``acquire()`` procedure's continuation.
|
|
|
|
if not(lock.acquired):
|
|
|
|
raise newException(AsyncLockError, "AsyncLock was already released!")
|
|
|
|
else:
|
|
|
|
lock.acquired = false
|
|
|
|
if not(lock.wakeUpFirst()):
|
|
|
|
lock.locked = false
|
2018-05-16 11:22:34 +03:00
|
|
|
else:
|
|
|
|
raise newException(AsyncLockError, "AsyncLock is not acquired!")
|
|
|
|
|
|
|
|
proc newAsyncEvent*(): AsyncEvent =
|
|
|
|
## Creates new asyncronous event ``AsyncEvent``.
|
|
|
|
##
|
|
|
|
## An event manages a flag that can be set to `true` with the `fire()`
|
|
|
|
## procedure and reset to `false` with the `clear()` procedure.
|
|
|
|
## The `wait()` procedure blocks until the flag is `true`. The flag is
|
|
|
|
## initially `false`.
|
|
|
|
|
|
|
|
# Workaround for callSoon() not worked correctly before
|
2021-01-11 19:15:23 +02:00
|
|
|
# getThreadDispatcher() call.
|
|
|
|
discard getThreadDispatcher()
|
2020-09-10 10:39:10 +02:00
|
|
|
AsyncEvent(waiters: newSeq[Future[void]](), flag: false)
|
2019-07-17 16:12:31 +03:00
|
|
|
|
2020-09-10 10:39:10 +02:00
|
|
|
proc wait*(event: AsyncEvent): Future[void] =
|
2018-05-16 11:22:34 +03:00
|
|
|
## Block until the internal flag of ``event`` is `true`.
|
|
|
|
## If the internal flag is `true` on entry, return immediately. Otherwise,
|
|
|
|
## block until another task calls `fire()` to set the flag to `true`,
|
|
|
|
## then return.
|
2020-09-10 10:39:10 +02:00
|
|
|
var w = newFuture[void]("AsyncEvent.wait")
|
2019-07-17 16:12:31 +03:00
|
|
|
if not(event.flag):
|
|
|
|
event.waiters.add(w)
|
2020-09-10 10:39:10 +02:00
|
|
|
else:
|
|
|
|
w.complete()
|
|
|
|
w
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc fire*(event: AsyncEvent) =
|
|
|
|
## Set the internal flag of ``event`` to `true`. All tasks waiting for it
|
|
|
|
## to become `true` are awakened. Task that call `wait()` once the flag is
|
|
|
|
## `true` will not block at all.
|
2019-06-20 23:30:41 +03:00
|
|
|
if not(event.flag):
|
2018-05-16 11:22:34 +03:00
|
|
|
event.flag = true
|
2019-07-17 16:12:31 +03:00
|
|
|
for fut in event.waiters:
|
2020-09-10 10:39:10 +02:00
|
|
|
if not(fut.finished()): # Could have been cancelled
|
2019-07-17 16:12:31 +03:00
|
|
|
fut.complete()
|
2020-09-10 10:39:10 +02:00
|
|
|
event.waiters.setLen(0)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc clear*(event: AsyncEvent) =
|
|
|
|
## Reset the internal flag of ``event`` to `false`. Subsequently, tasks
|
|
|
|
## calling `wait()` will block until `fire()` is called to set the internal
|
|
|
|
## flag to `true` again.
|
|
|
|
event.flag = false
|
|
|
|
|
|
|
|
proc isSet*(event: AsyncEvent): bool =
|
|
|
|
## Return `true` if and only if the internal flag of ``event`` is `true`.
|
2020-09-10 10:39:10 +02:00
|
|
|
event.flag
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc newAsyncQueue*[T](maxsize: int = 0): AsyncQueue[T] =
|
|
|
|
## Creates a new asynchronous queue ``AsyncQueue``.
|
|
|
|
|
|
|
|
# Workaround for callSoon() not worked correctly before
|
2021-01-11 19:15:23 +02:00
|
|
|
# getThreadDispatcher() call.
|
|
|
|
discard getThreadDispatcher()
|
2020-09-10 10:39:10 +02:00
|
|
|
AsyncQueue[T](
|
|
|
|
getters: newSeq[Future[void]](),
|
|
|
|
putters: newSeq[Future[void]](),
|
|
|
|
queue: initDeque[T](),
|
|
|
|
maxsize: maxsize
|
|
|
|
)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
2019-07-17 16:12:31 +03:00
|
|
|
proc wakeupNext(waiters: var seq[Future[void]]) {.inline.} =
|
|
|
|
var i = 0
|
|
|
|
while i < len(waiters):
|
|
|
|
var waiter = waiters[i]
|
2020-09-10 10:39:10 +02:00
|
|
|
inc(i)
|
|
|
|
|
2019-07-17 16:12:31 +03:00
|
|
|
if not(waiter.finished()):
|
|
|
|
waiter.complete()
|
|
|
|
break
|
|
|
|
|
2020-09-10 10:39:10 +02:00
|
|
|
if i > 0:
|
|
|
|
waiters.delete(0, i - 1)
|
2019-07-17 16:12:31 +03:00
|
|
|
|
2018-05-16 11:22:34 +03:00
|
|
|
proc full*[T](aq: AsyncQueue[T]): bool {.inline.} =
|
|
|
|
## Return ``true`` if there are ``maxsize`` items in the queue.
|
|
|
|
##
|
|
|
|
## Note: If the ``aq`` was initialized with ``maxsize = 0`` (default),
|
|
|
|
## then ``full()`` is never ``true``.
|
|
|
|
if aq.maxsize <= 0:
|
2020-09-10 10:39:10 +02:00
|
|
|
false
|
2018-05-16 11:22:34 +03:00
|
|
|
else:
|
2020-09-10 10:39:10 +02:00
|
|
|
(len(aq.queue) >= aq.maxsize)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc empty*[T](aq: AsyncQueue[T]): bool {.inline.} =
|
|
|
|
## Return ``true`` if the queue is empty, ``false`` otherwise.
|
2020-09-10 10:39:10 +02:00
|
|
|
(len(aq.queue) == 0)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc addFirstNoWait*[T](aq: AsyncQueue[T], item: T) {.
|
|
|
|
raises: [Defect, AsyncQueueFullError].}=
|
2018-07-31 12:50:22 +03:00
|
|
|
## Put an item ``item`` to the beginning of the queue ``aq`` immediately.
|
2018-05-16 11:22:34 +03:00
|
|
|
##
|
2018-07-31 12:50:22 +03:00
|
|
|
## If queue ``aq`` is full, then ``AsyncQueueFullError`` exception raised.
|
2018-05-16 11:22:34 +03:00
|
|
|
if aq.full():
|
|
|
|
raise newException(AsyncQueueFullError, "AsyncQueue is full!")
|
2018-07-31 12:50:22 +03:00
|
|
|
aq.queue.addFirst(item)
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.getters.wakeupNext()
|
2018-05-16 11:22:34 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc addLastNoWait*[T](aq: AsyncQueue[T], item: T) {.
|
|
|
|
raises: [Defect, AsyncQueueFullError].}=
|
2018-07-31 12:50:22 +03:00
|
|
|
## Put an item ``item`` at the end of the queue ``aq`` immediately.
|
2018-05-16 11:22:34 +03:00
|
|
|
##
|
2018-07-31 12:50:22 +03:00
|
|
|
## If queue ``aq`` is full, then ``AsyncQueueFullError`` exception raised.
|
|
|
|
if aq.full():
|
|
|
|
raise newException(AsyncQueueFullError, "AsyncQueue is full!")
|
|
|
|
aq.queue.addLast(item)
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.getters.wakeupNext()
|
2018-07-31 12:50:22 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc popFirstNoWait*[T](aq: AsyncQueue[T]): T {.
|
|
|
|
raises: [Defect, AsyncQueueEmptyError].} =
|
2018-07-31 12:50:22 +03:00
|
|
|
## Get an item from the beginning of the queue ``aq`` immediately.
|
2019-02-06 15:49:11 +01:00
|
|
|
##
|
2018-05-16 11:22:34 +03:00
|
|
|
## If queue ``aq`` is empty, then ``AsyncQueueEmptyError`` exception raised.
|
|
|
|
if aq.empty():
|
|
|
|
raise newException(AsyncQueueEmptyError, "AsyncQueue is empty!")
|
2020-09-10 10:39:10 +02:00
|
|
|
let res = aq.queue.popFirst()
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
res
|
2018-05-16 11:22:34 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc popLastNoWait*[T](aq: AsyncQueue[T]): T {.
|
|
|
|
raises: [Defect, AsyncQueueEmptyError].} =
|
2018-07-31 12:50:22 +03:00
|
|
|
## Get an item from the end of the queue ``aq`` immediately.
|
2019-02-06 15:49:11 +01:00
|
|
|
##
|
2018-07-31 12:50:22 +03:00
|
|
|
## If queue ``aq`` is empty, then ``AsyncQueueEmptyError`` exception raised.
|
|
|
|
if aq.empty():
|
|
|
|
raise newException(AsyncQueueEmptyError, "AsyncQueue is empty!")
|
2020-09-10 10:39:10 +02:00
|
|
|
let res = aq.queue.popLast()
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
res
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc addFirst*[T](aq: AsyncQueue[T], item: T) {.async.} =
|
|
|
|
## Put an ``item`` to the beginning of the queue ``aq``. If the queue is full,
|
|
|
|
## wait until a free slot is available before adding item.
|
2018-05-16 11:22:34 +03:00
|
|
|
while aq.full():
|
2018-07-31 12:50:22 +03:00
|
|
|
var putter = newFuture[void]("AsyncQueue.addFirst")
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.add(putter)
|
|
|
|
try:
|
|
|
|
await putter
|
2020-09-10 10:39:10 +02:00
|
|
|
except CatchableError as exc:
|
|
|
|
if not(aq.full()) and not(putter.cancelled()):
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
raise exc
|
2018-07-31 12:50:22 +03:00
|
|
|
aq.addFirstNoWait(item)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
2018-07-31 12:50:22 +03:00
|
|
|
proc addLast*[T](aq: AsyncQueue[T], item: T) {.async.} =
|
|
|
|
## Put an ``item`` to the end of the queue ``aq``. If the queue is full,
|
|
|
|
## wait until a free slot is available before adding item.
|
|
|
|
while aq.full():
|
|
|
|
var putter = newFuture[void]("AsyncQueue.addLast")
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.add(putter)
|
|
|
|
try:
|
|
|
|
await putter
|
2020-09-10 10:39:10 +02:00
|
|
|
except CatchableError as exc:
|
|
|
|
if not(aq.full()) and not(putter.cancelled()):
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.putters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
raise exc
|
2018-07-31 12:50:22 +03:00
|
|
|
aq.addLastNoWait(item)
|
|
|
|
|
|
|
|
proc popFirst*[T](aq: AsyncQueue[T]): Future[T] {.async.} =
|
|
|
|
## Remove and return an ``item`` from the beginning of the queue ``aq``.
|
|
|
|
## If the queue is empty, wait until an item is available.
|
|
|
|
while aq.empty():
|
|
|
|
var getter = newFuture[void]("AsyncQueue.popFirst")
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.getters.add(getter)
|
|
|
|
try:
|
|
|
|
await getter
|
2020-09-10 10:39:10 +02:00
|
|
|
except CatchableError as exc:
|
2019-07-17 16:12:31 +03:00
|
|
|
if not(aq.empty()) and not(getter.cancelled()):
|
|
|
|
aq.getters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
raise exc
|
|
|
|
return aq.popFirstNoWait()
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc popLast*[T](aq: AsyncQueue[T]): Future[T] {.async.} =
|
|
|
|
## Remove and return an ``item`` from the end of the queue ``aq``.
|
|
|
|
## If the queue is empty, wait until an item is available.
|
2018-05-16 11:22:34 +03:00
|
|
|
while aq.empty():
|
2018-07-31 12:50:22 +03:00
|
|
|
var getter = newFuture[void]("AsyncQueue.popLast")
|
2019-07-17 16:12:31 +03:00
|
|
|
aq.getters.add(getter)
|
|
|
|
try:
|
|
|
|
await getter
|
2020-09-10 10:39:10 +02:00
|
|
|
except CatchableError as exc:
|
2019-07-17 16:12:31 +03:00
|
|
|
if not(aq.empty()) and not(getter.cancelled()):
|
|
|
|
aq.getters.wakeupNext()
|
2020-09-10 10:39:10 +02:00
|
|
|
raise exc
|
|
|
|
return aq.popLastNoWait()
|
2018-07-31 12:50:22 +03:00
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc putNoWait*[T](aq: AsyncQueue[T], item: T) {.
|
|
|
|
raises: [Defect, AsyncQueueFullError].} =
|
2018-07-31 12:50:22 +03:00
|
|
|
## Alias of ``addLastNoWait()``.
|
|
|
|
aq.addLastNoWait(item)
|
|
|
|
|
exception tracking (#166)
* exception tracking
This PR adds minimal exception tracking to chronos, moving the goalpost
one step further.
In particular, it becomes invalid to raise exceptions from `callSoon`
callbacks: this is critical for writing correct error handling because
there's no reasonable way that a user of chronos can possibly _reason_
about exceptions coming out of there: the event loop will be in an
indeterminite state when the loop is executing an _random_ callback.
As expected, there are several issues in the error handling of chronos:
in particular, it will end up in an inconsistent internal state whenever
the selector loop operations fail, because the internal state update
functions are not written in an exception-safe way. This PR turns this
into a Defect, which probably is not the optimal way of handling things
- expect more work to be done here.
Some API have no way of reporting back errors to callers - for example,
when something fails in the accept loop, there's not much it can do, and
no way to report it back to the user of the API - this has been fixed
with the new accept flow - the old one should be deprecated.
Finally, there is information loss in the API: in composite operations
like `poll` and `waitFor` there's no way to differentiate internal
errors from user-level errors originating from callbacks.
* store `CatchableError` in future
* annotate proc's with correct raises information
* `selectors2` to avoid non-CatchableError IOSelectorsException
* `$` should never raise
* remove unnecessary gcsafe annotations
* fix exceptions leaking out of timer waits
* fix some imports
* functions must signal raising the union of all exceptions across all
platforms to enable cross-platform code
* switch to unittest2
* add `selectors2` which supercedes the std library version and fixes
several exception handling issues in there
* fixes
* docs, platform-independent eh specifiers for some functions
* add feature flag for strict exception mode
also bump version to 3.0.0 - _most_ existing code should be compatible
with this version of exception handling but some things might need
fixing - callbacks, existing raises specifications etc.
* fix AsyncCheck for non-void T
2021-03-24 10:08:33 +01:00
|
|
|
proc getNoWait*[T](aq: AsyncQueue[T]): T {.
|
|
|
|
raises: [Defect, AsyncQueueEmptyError].} =
|
2018-07-31 12:50:22 +03:00
|
|
|
## Alias of ``popFirstNoWait()``.
|
2020-09-10 10:39:10 +02:00
|
|
|
aq.popFirstNoWait()
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc put*[T](aq: AsyncQueue[T], item: T): Future[void] {.inline.} =
|
|
|
|
## Alias of ``addLast()``.
|
2020-09-10 10:39:10 +02:00
|
|
|
aq.addLast(item)
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc get*[T](aq: AsyncQueue[T]): Future[T] {.inline.} =
|
|
|
|
## Alias of ``popFirst()``.
|
2020-09-10 10:39:10 +02:00
|
|
|
aq.popFirst()
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc clear*[T](aq: AsyncQueue[T]) {.inline.} =
|
|
|
|
## Clears all elements of queue ``aq``.
|
|
|
|
aq.queue.clear()
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc len*[T](aq: AsyncQueue[T]): int {.inline.} =
|
|
|
|
## Return the number of elements in ``aq``.
|
2020-09-10 10:39:10 +02:00
|
|
|
len(aq.queue)
|
2018-05-16 11:22:34 +03:00
|
|
|
|
|
|
|
proc size*[T](aq: AsyncQueue[T]): int {.inline.} =
|
|
|
|
## Return the maximum number of elements in ``aq``.
|
2020-09-10 10:39:10 +02:00
|
|
|
len(aq.maxsize)
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc `[]`*[T](aq: AsyncQueue[T], i: Natural) : T {.inline.} =
|
|
|
|
## Access the i-th element of ``aq`` by order from first to last.
|
|
|
|
## ``aq[0]`` is the first element, ``aq[^1]`` is the last element.
|
2020-09-10 10:39:10 +02:00
|
|
|
aq.queue[i]
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc `[]`*[T](aq: AsyncQueue[T], i: BackwardsIndex) : T {.inline.} =
|
|
|
|
## Access the i-th element of ``aq`` by order from first to last.
|
|
|
|
## ``aq[0]`` is the first element, ``aq[^1]`` is the last element.
|
2020-09-10 10:39:10 +02:00
|
|
|
aq.queue[len(aq.queue) - int(i)]
|
2018-07-31 12:50:22 +03:00
|
|
|
|
|
|
|
proc `[]=`* [T](aq: AsyncQueue[T], i: Natural, item: T) {.inline.} =
|
|
|
|
## Change the i-th element of ``aq``.
|
|
|
|
aq.queue[i] = item
|
|
|
|
|
|
|
|
proc `[]=`* [T](aq: AsyncQueue[T], i: BackwardsIndex, item: T) {.inline.} =
|
|
|
|
## Change the i-th element of ``aq``.
|
|
|
|
aq.queue[len(aq.queue) - int(i)] = item
|
|
|
|
|
|
|
|
iterator items*[T](aq: AsyncQueue[T]): T {.inline.} =
|
|
|
|
## Yield every element of ``aq``.
|
|
|
|
for item in aq.queue.items():
|
|
|
|
yield item
|
|
|
|
|
|
|
|
iterator mitems*[T](aq: AsyncQueue[T]): var T {.inline.} =
|
|
|
|
## Yield every element of ``aq``.
|
|
|
|
for mitem in aq.queue.mitems():
|
|
|
|
yield mitem
|
|
|
|
|
|
|
|
iterator pairs*[T](aq: AsyncQueue[T]): tuple[key: int, val: T] {.inline.} =
|
|
|
|
## Yield every (position, value) of ``aq``.
|
|
|
|
for pair in aq.queue.pairs():
|
|
|
|
yield pair
|
|
|
|
|
|
|
|
proc contains*[T](aq: AsyncQueue[T], item: T): bool {.inline.} =
|
|
|
|
## Return true if ``item`` is in ``aq`` or false if not found. Usually used
|
|
|
|
## via the ``in`` operator.
|
|
|
|
for e in aq.queue.items():
|
|
|
|
if e == item: return true
|
|
|
|
return false
|
|
|
|
|
|
|
|
proc `$`*[T](aq: AsyncQueue[T]): string =
|
|
|
|
## Turn an async queue ``aq`` into its string representation.
|
2020-09-10 10:39:10 +02:00
|
|
|
var res = "["
|
2018-07-31 12:50:22 +03:00
|
|
|
for item in aq.queue.items():
|
2020-09-10 10:39:10 +02:00
|
|
|
if len(res) > 1: res.add(", ")
|
|
|
|
res.addQuoted(item)
|
|
|
|
res.add("]")
|
|
|
|
res
|
2021-09-15 16:55:15 +03:00
|
|
|
|
|
|
|
template generateKey(typeName, eventName: string): string =
|
|
|
|
"type[" & typeName & "]-key[" & eventName & "]"
|
|
|
|
|
|
|
|
proc newAsyncEventBus*(): AsyncEventBus =
|
|
|
|
## Creates new ``AsyncEventBus``.
|
|
|
|
AsyncEventBus(counter: 0'u64, events: initTable[string, EventItem]())
|
|
|
|
|
|
|
|
template get*[T](payload: EventPayload[T]): T =
|
|
|
|
## Returns event payload data.
|
|
|
|
payload.value
|
|
|
|
|
|
|
|
template location*(payload: EventPayloadBase): SrcLoc =
|
|
|
|
## Returns source location address of event emitter.
|
|
|
|
payload.loc[]
|
|
|
|
|
|
|
|
proc get*(event: AwaitableEvent, T: typedesc): T =
|
|
|
|
## Returns event's payload of type ``T`` from event ``event``.
|
|
|
|
cast[EventPayload[T]](event.payload).value
|
|
|
|
|
|
|
|
template event*(event: AwaitableEvent): string =
|
|
|
|
## Returns event's name from event ``event``.
|
|
|
|
event.eventName
|
|
|
|
|
|
|
|
template location*(event: AwaitableEvent): SrcLoc =
|
|
|
|
## Returns source location address of event emitter.
|
|
|
|
event.payload.loc[]
|
|
|
|
|
|
|
|
proc waitEvent*(bus: AsyncEventBus, T: typedesc, event: string): Future[T] =
|
|
|
|
## Wait for the event from AsyncEventBus ``bus`` with name ``event``.
|
|
|
|
##
|
|
|
|
## Returned ``Future[T]`` will hold event's payload of type ``T``.
|
|
|
|
var default: EventItem
|
|
|
|
var retFuture = newFuture[T]("AsyncEventBus.waitEvent")
|
|
|
|
let eventKey = generateKey(T.name, event)
|
|
|
|
proc cancellation(udata: pointer) {.gcsafe, raises: [Defect].} =
|
|
|
|
if not(retFuture.finished()):
|
|
|
|
bus.events.withValue(eventKey, item):
|
|
|
|
item.waiters.keepItIf(it != cast[FutureBase](retFuture))
|
|
|
|
retFuture.cancelCallback = cancellation
|
|
|
|
let baseFuture = cast[FutureBase](retFuture)
|
|
|
|
bus.events.mgetOrPut(eventKey, default).waiters.add(baseFuture)
|
|
|
|
retFuture
|
|
|
|
|
|
|
|
proc waitAllEvents*(bus: AsyncEventBus): Future[AwaitableEvent] =
|
|
|
|
## Wait for any event from AsyncEventBus ``bus``.
|
|
|
|
##
|
|
|
|
## Returns ``Future`` which holds helper object. Using this object you can
|
|
|
|
## retrieve event's name and payload.
|
|
|
|
var retFuture = newFuture[AwaitableEvent]("AsyncEventBus.waitAllEvents")
|
|
|
|
proc cancellation(udata: pointer) {.gcsafe, raises: [Defect].} =
|
|
|
|
if not(retFuture.finished()):
|
|
|
|
bus.waiters.keepItIf(it != retFuture)
|
|
|
|
retFuture.cancelCallback = cancellation
|
|
|
|
bus.waiters.add(retFuture)
|
|
|
|
retFuture
|
|
|
|
|
|
|
|
proc subscribe*[T](bus: AsyncEventBus, event: string,
|
|
|
|
callback: EventBusSubscription[T]): EventBusKey =
|
|
|
|
## Subscribe to the event ``event`` passed through eventbus ``bus`` with
|
|
|
|
## callback ``callback``.
|
|
|
|
##
|
|
|
|
## Returns key that can be used to unsubscribe.
|
|
|
|
proc trampoline(tbus: AsyncEventBus, event: string, key: EventBusKey,
|
|
|
|
data: EventPayloadBase) {.gcsafe, raises: [Defect].} =
|
|
|
|
let payload = cast[EventPayload[T]](data)
|
|
|
|
asyncSpawn callback(bus, payload)
|
|
|
|
|
|
|
|
let subkey =
|
|
|
|
block:
|
|
|
|
inc(bus.counter)
|
|
|
|
EventBusKey(eventName: event, typeName: T.name, unique: bus.counter,
|
|
|
|
cb: trampoline)
|
|
|
|
|
|
|
|
var default: EventItem
|
|
|
|
let eventKey = generateKey(T.name, event)
|
|
|
|
bus.events.mgetOrPut(eventKey, default).subscribers.add(subkey)
|
|
|
|
subkey
|
|
|
|
|
|
|
|
proc subscribeAll*(bus: AsyncEventBus,
|
|
|
|
callback: EventBusAllSubscription): EventBusKey =
|
|
|
|
## Subscribe to all events passed through eventbus ``bus`` with callback
|
|
|
|
## ``callback``.
|
|
|
|
##
|
|
|
|
## Returns key that can be used to unsubscribe.
|
|
|
|
proc trampoline(tbus: AsyncEventBus, event: string, key: EventBusKey,
|
|
|
|
data: EventPayloadBase) {.gcsafe, raises: [Defect].} =
|
|
|
|
let event = AwaitableEvent(eventName: event, payload: data)
|
|
|
|
asyncSpawn callback(bus, event)
|
|
|
|
|
|
|
|
let subkey =
|
|
|
|
block:
|
|
|
|
inc(bus.counter)
|
|
|
|
EventBusKey(eventName: "", typeName: "", unique: bus.counter,
|
|
|
|
cb: trampoline)
|
|
|
|
bus.subscribers.add(subkey)
|
|
|
|
subkey
|
|
|
|
|
|
|
|
proc unsubscribe*(bus: AsyncEventBus, key: EventBusKey) =
|
|
|
|
## Cancel subscription of subscriber with key ``key`` from eventbus ``bus``.
|
|
|
|
let eventKey = generateKey(key.typeName, key.eventName)
|
|
|
|
|
|
|
|
# Clean event's subscribers.
|
|
|
|
bus.events.withValue(eventKey, item):
|
|
|
|
item.subscribers.keepItIf(it.unique != key.unique)
|
|
|
|
|
|
|
|
# Clean subscribers subscribed to all events.
|
|
|
|
bus.subscribers.keepItIf(it.unique != key.unique)
|
|
|
|
|
|
|
|
proc emit[T](bus: AsyncEventBus, event: string, data: T, loc: ptr SrcLoc) =
|
|
|
|
let
|
|
|
|
eventKey = generateKey(T.name, event)
|
|
|
|
payload =
|
|
|
|
block:
|
|
|
|
var data = EventPayload[T](value: data, loc: loc)
|
|
|
|
cast[EventPayloadBase](data)
|
|
|
|
|
|
|
|
bus.events.withValue(eventKey, item):
|
|
|
|
# Schedule waiters which are waiting for the event ``event``.
|
|
|
|
for waiter in item.waiters:
|
|
|
|
var fut = cast[Future[T]](waiter)
|
|
|
|
fut.complete(data)
|
|
|
|
# Clear all the waiters.
|
|
|
|
item.waiters.setLen(0)
|
|
|
|
|
|
|
|
# Schedule subscriber's callbacks, which are subscribed to the event.
|
|
|
|
for subscriber in item.subscribers:
|
|
|
|
callSoon(proc(udata: pointer) =
|
|
|
|
subscriber.cb(bus, event, subscriber, payload)
|
|
|
|
)
|
|
|
|
|
|
|
|
# Schedule waiters which are waiting all events
|
|
|
|
for waiter in bus.waiters:
|
|
|
|
waiter.complete(AwaitableEvent(eventName: event, payload: payload))
|
|
|
|
# Clear all the waiters.
|
|
|
|
bus.waiters.setLen(0)
|
|
|
|
|
|
|
|
# Schedule subscriber's callbacks which are subscribed to all events.
|
|
|
|
for subscriber in bus.subscribers:
|
|
|
|
callSoon(proc(udata: pointer) =
|
|
|
|
subscriber.cb(bus, event, subscriber, payload)
|
|
|
|
)
|
|
|
|
|
|
|
|
template emit*[T](bus: AsyncEventBus, event: string, data: T) =
|
|
|
|
## Emit new event ``event`` to the eventbus ``bus`` with payload ``data``.
|
|
|
|
emit(bus, event, data, getSrcLocation())
|
|
|
|
|
|
|
|
proc emitWait[T](bus: AsyncEventBus, event: string, data: T,
|
|
|
|
loc: ptr SrcLoc): Future[void] =
|
|
|
|
var retFuture = newFuture[void]("AsyncEventBus.emitWait")
|
|
|
|
proc continuation(udata: pointer) {.gcsafe.} =
|
|
|
|
if not(retFuture.finished()):
|
|
|
|
retFuture.complete()
|
|
|
|
emit(bus, event, data, loc)
|
|
|
|
callSoon(continuation)
|
|
|
|
return retFuture
|
|
|
|
|
|
|
|
template emitWait*[T](bus: AsyncEventBus, event: string,
|
|
|
|
data: T): Future[void] =
|
|
|
|
## Emit new event ``event`` to the eventbus ``bus`` with payload ``data`` and
|
|
|
|
## wait until all the subscribers/waiters will receive notification about
|
|
|
|
## event.
|
|
|
|
emitWait(bus, event, data, getSrcLocation())
|