Small fixes (#134)

* Fix cancellation behavior.
Fix some compilation warnings.

* Fix cancelAndWait() and add proper documentation.

* Add completed(Future) and done(Future) calls to check if Future[T] completed without an error.

* Add stepsAsync() and tests.

* Fix comments.

* Fix new primitive comment, to avoid usage for task switches.
This commit is contained in:
Eugene Kabanov 2020-11-13 14:22:58 +02:00 committed by GitHub
parent d3018ae908
commit b5915ecd29
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 128 additions and 41 deletions

View File

@ -146,9 +146,9 @@ proc clean*[T](future: FutureVar[T]) =
Future[T](future).error = nil Future[T](future).error = nil
proc finished*(future: FutureBase | FutureVar): bool {.inline.} = proc finished*(future: FutureBase | FutureVar): bool {.inline.} =
## Determines whether ``future`` has completed. ## Determines whether ``future`` has completed, i.e. ``future`` state changed
## ## from state ``Pending`` to one of the states (``Finished``, ``Cancelled``,
## ``True`` may indicate an error or a value. Use ``failed`` to distinguish. ## ``Failed``).
when future is FutureVar: when future is FutureVar:
result = (FutureBase(future).state != FutureState.Pending) result = (FutureBase(future).state != FutureState.Pending)
else: else:
@ -156,11 +156,19 @@ proc finished*(future: FutureBase | FutureVar): bool {.inline.} =
proc cancelled*(future: FutureBase): bool {.inline.} = proc cancelled*(future: FutureBase): bool {.inline.} =
## Determines whether ``future`` has cancelled. ## Determines whether ``future`` has cancelled.
result = (future.state == FutureState.Cancelled) (future.state == FutureState.Cancelled)
proc failed*(future: FutureBase): bool {.inline.} = proc failed*(future: FutureBase): bool {.inline.} =
## Determines whether ``future`` completed with an error. ## Determines whether ``future`` completed with an error.
result = (future.state == FutureState.Failed) (future.state == FutureState.Failed)
proc completed*(future: FutureBase): bool {.inline.} =
## Determines whether ``future`` completed without an error.
(future.state == FutureState.Finished)
proc done*(future: FutureBase): bool {.inline.} =
## This is an alias for ``completed(future)`` procedure.
completed(future)
when defined(chronosFutureTracking): when defined(chronosFutureTracking):
proc futureDestructor(udata: pointer) {.gcsafe.} = proc futureDestructor(udata: pointer) {.gcsafe.} =
@ -292,24 +300,39 @@ proc cancelAndSchedule(future: FutureBase, loc: ptr SrcLoc) =
template cancelAndSchedule*[T](future: Future[T]) = template cancelAndSchedule*[T](future: Future[T]) =
cancelAndSchedule(FutureBase(future), getSrcLocation()) cancelAndSchedule(FutureBase(future), getSrcLocation())
proc cancel(future: FutureBase, loc: ptr SrcLoc) = proc cancel(future: FutureBase, loc: ptr SrcLoc): bool =
if not(future.finished()): ## Request that Future ``future`` cancel itself.
# Cancel the bottom-most child. When that happens, its parent's `await` call ##
# will raise CancelledError. Some macro will catch that and call ## This arranges for a `CancelledError` to be thrown into procedure which
# `cancelAndSchedule()` on that parent, thus propagating the cancellation ## waits for ``future`` on the next cycle through the event loop.
# up the chain. ## The procedure then has a chance to clean up or even deny the request
if not(isNil(future.child)): ## using `try/except/finally`.
cancel(future.child, getSrcLocation()) ##
future.mustCancel = true ## This call do not guarantee that the ``future`` will be cancelled: the
else: ## exception might be caught and acted upon, delaying cancellation of the
if not(isNil(future.cancelcb)): ## ``future`` or preventing cancellation completely. The ``future`` may also
future.cancelcb(cast[pointer](future)) ## return value or raise different exception.
future.cancelcb = nil ##
cancelAndSchedule(future, getSrcLocation()) ## Immediately after this procedure is called, ``future.cancelled()`` will
## not return ``true`` (unless the Future was already cancelled).
if future.finished():
return false
if not(isNil(future.child)):
if cancel(future.child, getSrcLocation()):
return true
else:
if not(isNil(future.cancelcb)):
future.cancelcb(cast[pointer](future))
future.cancelcb = nil
cancelAndSchedule(future, getSrcLocation())
future.mustCancel = true
return true
template cancel*[T](future: Future[T]) = template cancel*[T](future: Future[T]) =
## Cancel ``future``. ## Cancel ``future``.
cancel(FutureBase(future), getSrcLocation()) discard cancel(FutureBase(future), getSrcLocation())
proc clearCallbacks(future: FutureBase) = proc clearCallbacks(future: FutureBase) =
future.callbacks = default(seq[AsyncCallback]) future.callbacks = default(seq[AsyncCallback])
@ -566,8 +589,10 @@ proc `and`*[T, Y](fut1: Future[T], fut2: Future[Y]): Future[void] {.
proc cancellation(udata: pointer) {.gcsafe.} = proc cancellation(udata: pointer) {.gcsafe.} =
# On cancel we remove all our callbacks only. # On cancel we remove all our callbacks only.
fut1.removeCallback(cb) if not(fut1.finished()):
fut2.removeCallback(cb) fut1.removeCallback(cb)
if not(fut2.finished()):
fut2.removeCallback(cb)
retFuture.cancelCallback = cancellation retFuture.cancelCallback = cancellation
return retFuture return retFuture
@ -600,8 +625,10 @@ proc `or`*[T, Y](fut1: Future[T], fut2: Future[Y]): Future[void] =
proc cancellation(udata: pointer) {.gcsafe.} = proc cancellation(udata: pointer) {.gcsafe.} =
# On cancel we remove all our callbacks only. # On cancel we remove all our callbacks only.
fut1.removeCallback(cb) if not(fut1.finished()):
fut2.removeCallback(cb) fut1.removeCallback(cb)
if not(fut2.finished()):
fut2.removeCallback(cb)
if fut1.finished(): if fut1.finished():
if fut1.failed(): if fut1.failed():
@ -761,22 +788,24 @@ proc oneValue*[T](futs: varargs[Future[T]]): Future[T] {.
return retFuture return retFuture
proc cancelAndWait*[T](fut: Future[T]): Future[void] = proc cancelAndWait*[T](fut: Future[T]): Future[void] =
## Cancel ``fut`` and wait until it completes, in case it already ## Initiate cancellation process for Future ``fut`` and wait until ``fut`` is
## ``await``s on another Future. ## done e.g. changes its state (become completed, failed or cancelled).
##
# When `retFuture` completes, `fut` and all its children have been ## If ``fut`` is already finished (completed, failed or cancelled) result
# cancelled. If `fut` doesn't have any children, the `continuation()` callback ## Future[void] object will be returned complete.
# runs immediately, without control getting back to the dispatcher.
var retFuture = newFuture[void]("chronos.cancelAndWait(T)") var retFuture = newFuture[void]("chronos.cancelAndWait(T)")
proc continuation(udata: pointer) {.gcsafe.} = proc continuation(udata: pointer) {.gcsafe.} =
if not(retFuture.finished()): if not(retFuture.finished()):
retFuture.complete() retFuture.complete()
fut.addCallback(continuation) proc cancellation(udata: pointer) {.gcsafe.} =
if not(fut.finished()):
# Start the cancellation process. If `fut` has children, multiple event loop fut.removeCallback(continuation)
# steps will be needed for it to complete. if fut.finished():
fut.cancel() retFuture.complete()
else:
fut.addCallback(continuation)
# Initiate cancellation process.
fut.cancel()
return retFuture return retFuture
proc allFutures*[T](futs: varargs[Future[T]]): Future[void] = proc allFutures*[T](futs: varargs[Future[T]]): Future[void] =

View File

@ -790,10 +790,12 @@ proc sleepAsync*(duration: Duration): Future[void] =
var timer: TimerCallback var timer: TimerCallback
proc completion(data: pointer) {.gcsafe.} = proc completion(data: pointer) {.gcsafe.} =
retFuture.complete() if not(retFuture.finished()):
retFuture.complete()
proc cancellation(udata: pointer) {.gcsafe.} = proc cancellation(udata: pointer) {.gcsafe.} =
clearTimer(timer) if not(retFuture.finished()):
clearTimer(timer)
retFuture.cancelCallback = cancellation retFuture.cancelCallback = cancellation
timer = setTimer(moment, completion, cast[pointer](retFuture)) timer = setTimer(moment, completion, cast[pointer](retFuture))
@ -803,6 +805,38 @@ proc sleepAsync*(ms: int): Future[void] {.
inline, deprecated: "Use sleepAsync(Duration)".} = inline, deprecated: "Use sleepAsync(Duration)".} =
result = sleepAsync(ms.milliseconds()) result = sleepAsync(ms.milliseconds())
proc stepsAsync*(number: int): Future[void] =
## Suspends the execution of the current async procedure for the next
## ``number`` of asynchronous steps (``poll()`` calls).
##
## This primitive can be useful when you need to create more deterministic
## tests and cases.
##
## WARNING! Do not use this primitive to perform switch between tasks, because
## this can lead to 100% CPU load in the moments when there are no I/O
## events. Usually when there no I/O events CPU consumption should be near 0%.
var retFuture = newFuture[void]("chronos.stepsAsync(int)")
var counter = 0
proc continuation(data: pointer) {.gcsafe.} =
if not(retFuture.finished()):
inc(counter)
if counter < number:
callSoon(continuation, nil)
else:
retFuture.complete()
proc cancellation(udata: pointer) {.gcsafe.} =
discard
if number <= 0:
retFuture.complete()
else:
retFuture.cancelCallback = cancellation
callSoon(continuation, nil)
retFuture
proc withTimeout*[T](fut: Future[T], timeout: Duration): Future[bool] = proc withTimeout*[T](fut: Future[T], timeout: Duration): Future[bool] =
## Returns a future which will complete once ``fut`` completes or after ## Returns a future which will complete once ``fut`` completes or after
## ``timeout`` milliseconds has elapsed. ## ``timeout`` milliseconds has elapsed.

View File

@ -5,7 +5,7 @@
# Licensed under either of # Licensed under either of
# Apache License, version 2.0, (LICENSE-APACHEv2) # Apache License, version 2.0, (LICENSE-APACHEv2)
# MIT license (LICENSE-MIT) # MIT license (LICENSE-MIT)
import strutils, unittest import unittest
import ../chronos, ../chronos/streams/tlsstream import ../chronos, ../chronos/streams/tlsstream
when defined(nimHasUsed): {.used.} when defined(nimHasUsed): {.used.}

View File

@ -5,7 +5,7 @@
# Licensed under either of # Licensed under either of
# Apache License, version 2.0, (LICENSE-APACHEv2) # Apache License, version 2.0, (LICENSE-APACHEv2)
# MIT license (LICENSE-MIT) # MIT license (LICENSE-MIT)
import strutils, unittest import unittest
import ../chronos import ../chronos
when defined(nimHasUsed): {.used.} when defined(nimHasUsed): {.used.}

View File

@ -10,7 +10,7 @@ import ../chronos, ../chronos/timer
when defined(nimHasUsed): {.used.} when defined(nimHasUsed): {.used.}
suite "Asynchronous timers test suite": suite "Asynchronous timers & steps test suite":
const TimersCount = 10 const TimersCount = 10
proc timeWorker(time: Duration): Future[Duration] {.async.} = proc timeWorker(time: Duration): Future[Duration] {.async.} =
@ -61,3 +61,27 @@ suite "Asynchronous timers test suite":
test $TimersCount & " timers with 1000ms timeout": test $TimersCount & " timers with 1000ms timeout":
var res = waitFor(test(1000.milliseconds)) var res = waitFor(test(1000.milliseconds))
check (res >= 1000.milliseconds) and (res <= 5000.milliseconds) check (res >= 1000.milliseconds) and (res <= 5000.milliseconds)
test "Asynchronous steps test":
var futn1 = stepsAsync(-1)
var fut0 = stepsAsync(0)
var fut1 = stepsAsync(1)
var fut2 = stepsAsync(2)
var fut3 = stepsAsync(3)
check:
futn1.completed() == true
fut0.completed() == true
fut1.completed() == false
fut2.completed() == false
fut3.completed() == false
poll()
check:
fut1.completed() == true
fut2.completed() == false
fut3.completed() == false
poll()
check:
fut2.completed() == true
fut3.completed() == false
poll()
check:
fut3.completed() == true