diff --git a/chronos/asyncfutures2.nim b/chronos/asyncfutures2.nim index eb46e359a..20baca3b5 100644 --- a/chronos/asyncfutures2.nim +++ b/chronos/asyncfutures2.nim @@ -608,7 +608,7 @@ proc `and`*[T, Y](fut1: Future[T], fut2: Future[Y]): Future[void] {. ## Returns a future which will complete once both ``fut1`` and ``fut2`` ## complete. ## - ## If cancelled, ``fut1`` and ``fut2`` futures WILL NOT BE cancelled. + ## On cancellation, ``fut1`` and ``fut2`` futures WILL NOT BE cancelled. var retFuture = newFuture[void]("chronos.`and`") proc cb(data: pointer) = if not(retFuture.finished()): @@ -648,7 +648,7 @@ proc `or`*[T, Y](fut1: Future[T], fut2: Future[Y]): Future[void] = ## is failed, the result future will also be failed, if ``fut1`` future is ## completed, the result future will also be completed. ## - ## If cancelled, ``fut1`` and ``fut2`` futures WILL NOT BE cancelled. + ## On cancellation, ``fut1`` and ``fut2`` futures WILL NOT BE cancelled. var retFuture = newFuture[void]("chronos.or") var cb: proc(udata: pointer) {.gcsafe, raises: [Defect].} cb = proc(udata: pointer) {.gcsafe, raises: [Defect].} = @@ -854,11 +854,11 @@ proc cancelAndWait*[T](fut: Future[T]): Future[void] = proc allFutures*(futs: varargs[FutureBase]): Future[void] = ## Returns a future which will complete only when all futures in ``futs`` - ## will be completed, failed or canceled. + ## are completed, failed or cancelled. ## ## If the argument is empty, the returned future COMPLETES immediately. ## - ## On cancel all the awaited futures ``futs`` WILL NOT BE cancelled. + ## On cancellation, all the awaited futures ``futs`` WILL NOT BE cancelled. var retFuture = newFuture[void]("chronos.allFutures()") let totalFutures = len(futs) var completedFutures = 0 @@ -892,11 +892,11 @@ proc allFutures*(futs: varargs[FutureBase]): Future[void] = proc allFutures*[T](futs: varargs[Future[T]]): Future[void] = ## Returns a future which will complete only when all futures in ``futs`` - ## will be completed, failed or canceled. + ## are completed, failed or cancelled. ## ## If the argument is empty, the returned future COMPLETES immediately. ## - ## On cancel all the awaited futures ``futs`` WILL NOT BE cancelled. + ## On cancellation, all the awaited futures ``futs`` WILL NOT BE cancelled. # Because we can't capture varargs[T] in closures we need to create copy. var nfuts: seq[FutureBase] for future in futs: @@ -905,14 +905,14 @@ proc allFutures*[T](futs: varargs[Future[T]]): Future[void] = proc allFinished*[T](futs: varargs[Future[T]]): Future[seq[Future[T]]] = ## Returns a future which will complete only when all futures in ``futs`` - ## will be completed, failed or canceled. + ## will be completed, failed or cancelled. ## ## Returned sequence will hold all the Future[T] objects passed to ## ``allCompleted`` with the order preserved. ## ## If the argument is empty, the returned future COMPLETES immediately. ## - ## On cancel all the awaited futures ``futs`` WILL NOT BE cancelled. + ## On cancellation, all the awaited futures ``futs`` WILL NOT BE cancelled. var retFuture = newFuture[seq[Future[T]]]("chronos.allFinished()") let totalFutures = len(futs) var completedFutures = 0 @@ -945,13 +945,13 @@ proc allFinished*[T](futs: varargs[Future[T]]): Future[seq[Future[T]]] = proc one*[T](futs: varargs[Future[T]]): Future[Future[T]] = ## Returns a future which will complete and return completed Future[T] inside, - ## when one of the futures in ``futs`` will be completed, failed or canceled. + ## when one of the futures in ``futs`` will be completed, failed or cancelled. ## ## If the argument is empty, the returned future FAILS immediately. ## ## On success returned Future will hold finished Future[T]. ## - ## On cancel futures in ``futs`` WILL NOT BE cancelled. + ## On cancellation, futures in ``futs`` WILL NOT BE cancelled. var retFuture = newFuture[Future[T]]("chronos.one()") # Because we can't capture varargs[T] in closures we need to create copy. @@ -990,15 +990,108 @@ proc one*[T](futs: varargs[Future[T]]): Future[Future[T]] = retFuture.cancelCallback = cancellation return retFuture +proc firstCompletedFuture*(futs: varargs[FutureBase]): Future[FutureBase] = + ## Returns a future which will complete and return completed FutureBase, + ## when one of the futures in ``futs`` is completed. + ## + ## If the argument is empty, the returned future FAILS immediately. + ## + ## On success, the returned Future will hold the completed FutureBase. + ## + ## If all futures fail naturally or due to cancellation, the returned + ## future will be failed as well. + ## + ## On cancellation, futures in ``futs`` WILL NOT BE cancelled. + + var retFuture = newFuture[FutureBase]("chronos.firstCompletedFuture()") + + # Because we can't capture varargs[T] in closures we need to create copy. + var nfuts = @futs + + # If one of the Future[T] already finished we return it as result + for fut in nfuts: + if fut.completed(): + retFuture.complete(fut) + return retFuture + + if len(nfuts) == 0: + retFuture.fail(newException(ValueError, "Empty Future[T] list")) + return + + var failedFutures = 0 + + var cb: proc(udata: pointer) {.gcsafe, raises: [Defect].} + cb = proc(udata: pointer) {.gcsafe, raises: [Defect].} = + if not(retFuture.finished()): + var res: FutureBase + var rfut = cast[FutureBase](udata) + if rfut.completed: + for i in 0..