mirror of
https://github.com/vacp2p/nim-libp2p-experimental.git
synced 2025-01-13 20:04:43 +00:00
Add anyCompleted proc (#853)
This commit is contained in:
parent
0e28d3b828
commit
266c7b117a
35
libp2p/utils/future.nim
Normal file
35
libp2p/utils/future.nim
Normal 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
71
tests/testfuture.nim
Normal 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()
|
@ -5,7 +5,8 @@ import testvarint,
|
|||||||
testminprotobuf,
|
testminprotobuf,
|
||||||
teststreamseq,
|
teststreamseq,
|
||||||
testsemaphore,
|
testsemaphore,
|
||||||
testheartbeat
|
testheartbeat,
|
||||||
|
testfuture
|
||||||
|
|
||||||
import testminasn1,
|
import testminasn1,
|
||||||
testrsa,
|
testrsa,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user