## Nim-Codex ## Copyright (c) 2021 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 std/options import pkg/results import pkg/chronos import pkg/questionable/results export results type CodexError* = object of CatchableError # base codex error CodexResult*[T] = Result[T, ref CodexError] FinishedFailed*[T] = tuple[success: seq[Future[T]], failure: seq[Future[T]]] template mapFailure*[T, V, E]( exp: Result[T, V], exc: typedesc[E] ): Result[T, ref CatchableError] = ## Convert `Result[T, E]` to `Result[E, ref CatchableError]` ## exp.mapErr( proc(e: V): ref CatchableError = (ref exc)(msg: $e) ) template mapFailure*[T, V](exp: Result[T, V]): Result[T, ref CatchableError] = mapFailure(exp, CodexError) # TODO: using a template here, causes bad codegen func toFailure*[T](exp: Option[T]): Result[T, ref CatchableError] {.inline.} = if exp.isSome: success exp.get else: T.failure("Option is None") proc allFinishedFailed*[T](futs: seq[Future[T]]): Future[FinishedFailed[T]] {.async.} = ## Check if all futures have finished or failed ## ## TODO: wip, not sure if we want this - at the minimum, ## we should probably avoid the async transform var res: FinishedFailed[T] = (@[], @[]) await allFutures(futs) for f in futs: if f.failed: res.failure.add f else: res.success.add f return res