Add anyCompleted proc (#853)

This commit is contained in:
diegomrsantos 2023-02-07 18:50:42 +01:00 committed by GitHub
parent 0e28d3b828
commit 266c7b117a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 108 additions and 1 deletions

35
libp2p/utils/future.nim Normal file
View File

@ -0,0 +1,35 @@
# Nim-Libp2p
# Copyright (c) 2023 Status Research & Development GmbH
# Licensed under either of
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
# at your option.
# This file may not be copied, modified, or distributed except according to
# those terms.
when (NimMajor, NimMinor) < (1, 4):
{.push raises: [Defect].}
else:
{.push raises: [].}
import chronos
type
AllFuturesFailedError* = object of CatchableError
proc anyCompleted*[T](futs: seq[Future[T]]): Future[Future[T]] {.async.} =
## Returns a future that will complete with the first future that completes.
## If all futures fail or futs is empty, the returned future will fail with AllFuturesFailedError.
var requests = futs
while true:
if requests.len == 0:
raise newException(AllFuturesFailedError, "None of the futures completed successfully")
var raceFut = await one(requests)
if raceFut.completed:
return raceFut
let index = requests.find(raceFut)
requests.del(index)

71
tests/testfuture.nim Normal file
View File

@ -0,0 +1,71 @@
# Nim-Libp2p
# Copyright (c) 2023 Status Research & Development GmbH
# Licensed under either of
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE))
# * MIT license ([LICENSE-MIT](LICENSE-MIT))
# at your option.
# This file may not be copied, modified, or distributed except according to
# those terms.
import chronos
import ../libp2p/utils/future
import ./helpers
suite "Future":
asyncTest "anyCompleted must complete with first completed future":
proc fut1() {.async.} =
await sleepAsync(100.milliseconds)
proc fut2() {.async.} =
await sleepAsync(200.milliseconds)
proc fut3() {.async.} =
raise newException(CatchableError, "fut3")
var f1 = fut1()
var f2 = fut2()
var f3 = fut3()
var f = await anyCompleted(@[f1, f2, f3])
check f == f1
asyncTest "anyCompleted must fail with empty list":
expect AllFuturesFailedError:
discard await anyCompleted(newSeq[Future[void]]())
asyncTest "anyCompleted must fail if all futures fail":
proc fut1() {.async.} =
raise newException(CatchableError, "fut1")
proc fut2() {.async.} =
raise newException(CatchableError, "fut2")
proc fut3() {.async.} =
raise newException(CatchableError, "fut3")
var f1 = fut1()
var f2 = fut2()
var f3 = fut3()
expect AllFuturesFailedError:
discard await anyCompleted(@[f1, f2, f3])
asyncTest "anyCompleted with timeout":
proc fut1() {.async.} =
await sleepAsync(100.milliseconds)
proc fut2() {.async.} =
await sleepAsync(200.milliseconds)
proc fut3() {.async.} =
raise newException(ValueError, "fut3")
var f1 = fut1()
var f2 = fut2()
var f3 = fut3()
var f = anyCompleted(@[f1, f2, f3])
if not await f.withTimeout(50.milliseconds):
f.cancel()
check f.cancelled()

View File

@ -5,7 +5,8 @@ import testvarint,
testminprotobuf, testminprotobuf,
teststreamseq, teststreamseq,
testsemaphore, testsemaphore,
testheartbeat testheartbeat,
testfuture
import testminasn1, import testminasn1,
testrsa, testrsa,