package moonpool

  1. Overview
  2. Docs

Module Moonpool.FutSource

Futures.

A future of type 'a t represents the result of a computation that will yield a value of type 'a.

Typically, the computation is running on a thread pool Runner.t and will proceed on some worker. Once set, a future cannot change. It either succeeds (storing a Ok x with x: 'a), or fail (storing a Error (exn, bt) with an exception and the corresponding backtrace).

Using spawn, it's possible to start a bunch of tasks, obtaining futures, and then use await to get their result in the desired order.

Combinators such as map and join_array can be used to produce futures from other futures (in a monadic way). Some combinators take a on argument to specify a runner on which the intermediate computation takes place; for example map ~on:pool ~f fut maps the value in fut using function f, applicatively; the call to f happens on the runner pool (once fut resolves successfully with a value). Be aware that these combinators do not preserve local storage.

Sourcetype 'a or_error = ('a, Exn_bt.t) result

A future with a result of type 'a.

Sourcetype 'a promise = private 'a t

A promise, which can be fulfilled exactly once to set the corresponding future. This is a private alias of 'a t since 0.7, previously it was opaque.

Sourceval make : unit -> 'a t * 'a promise

Make a new future with the associated promise.

Sourceval make_promise : unit -> 'a promise

Same as make but returns a single promise (which can be upcast to a future). This is useful mostly to preserve memory, you probably don't need it.

How to upcast to a future in the worst case:

  let prom = Fut.make_promise ()
  let fut = (prom : _ Fut.promise :> _ Fut.t)
  • since 0.7
Sourceval on_result : 'a t -> ('a or_error -> unit) -> unit

on_result fut f registers f to be called in the future when fut is set; or calls f immediately if fut is already set.

NOTE: it's ill advised to do meaningful work inside the callback f. Instead, try to spawn another task on the runner, or use await.

Sourceval on_result_ignore : _ t -> (Exn_bt.t option -> unit) -> unit

on_result_ignore fut f registers f to be called in the future when fut is set; or calls f immediately if fut is already set. It does not pass the result, only a success/error signal.

  • since 0.7
Sourceexception Already_fulfilled
Sourceval try_cancel : _ promise -> Exn_bt.t -> bool

try_cancel promise ebt tries to cancel the promise using the given exception, returning true. It returns false if the promise is already resolved.

  • since 0.9
Sourceval cancel : _ promise -> Exn_bt.t -> unit

Silent version of try_cancel, ignoring the result.

  • since 0.9
Sourceval fulfill : 'a promise -> 'a or_error -> unit

Fullfill the promise, setting the future at the same time.

Sourceval fulfill_idempotent : 'a promise -> 'a or_error -> unit

Fullfill the promise, setting the future at the same time. Does nothing if the promise is already fulfilled.

Sourceval return : 'a -> 'a t

Already settled future, with a result

Sourceval fail : exn -> Printexc.raw_backtrace -> _ t

Already settled future, with a failure

Sourceval fail_exn_bt : Exn_bt.t -> _ t

Fail from a bundle of exception and backtrace

  • since 0.6
Sourceval of_result : 'a or_error -> 'a t

Already resolved future from a result.

Sourceval is_resolved : _ t -> bool

is_resolved fut is true iff fut is resolved.

Sourceval peek : 'a t -> 'a or_error option

peek fut returns Some r if fut is currently resolved with r, and None if fut is not resolved yet.

Sourceexception Not_ready
  • since 0.2
Sourceval get_or_fail : 'a t -> 'a or_error

get_or_fail fut obtains the result from fut if it's fulfilled (i.e. if peek fut returns Some res, get_or_fail fut returns res).

  • since 0.2
Sourceval get_or_fail_exn : 'a t -> 'a

get_or_fail_exn fut obtains the result from fut if it's fulfilled, like get_or_fail. If the result is an Error _, the exception inside is re-raised.

  • since 0.2
Sourceval is_done : _ t -> bool

Is the future resolved? This is the same as peek fut |> Option.is_some.

  • since 0.2
Sourceval is_success : _ t -> bool

Checks if the future is resolved with Ok _ as a result.

  • since 0.6
Sourceval is_failed : _ t -> bool

Checks if the future is resolved with Error _ as a result.

  • since 0.6
Sourceval raise_if_failed : _ t -> unit

raise_if_failed fut raises e if fut failed with e.

  • since 0.6

Combinators

Sourceval spawn : on:Runner.t -> (unit -> 'a) -> 'a t

spaw ~on f runs f() on the given runner on, and return a future that will hold its result.

Sourceval spawn_on_current_runner : (unit -> 'a) -> 'a t

This must be run from inside a runner, and schedules the new task on it as well.

See Runner.get_current_runner to see how the runner is found.

  • since 0.5
  • raises Failure

    if run from outside a runner.

Sourceval reify_error : 'a t -> 'a or_error t

reify_error fut turns a failing future into a non-failing one that contain Error (exn, bt). A non-failing future returning x is turned into Ok x.

  • since 0.4
Sourceval map : ?on:Runner.t -> f:('a -> 'b) -> 'a t -> 'b t

map ?on ~f fut returns a new future fut2 that resolves with f x if fut resolved with x; and fails with e if fut fails with e or f x raises e.

  • parameter on

    if provided, f runs on the given runner

Sourceval bind : ?on:Runner.t -> f:('a -> 'b t) -> 'a t -> 'b t

bind ?on ~f fut returns a new future fut2 that resolves like the future f x if fut resolved with x; and fails with e if fut fails with e or f x raises e.

This does not preserve local storage of fut inside f.

  • parameter on

    if provided, f runs on the given runner

Sourceval bind_reify_error : ?on:Runner.t -> f:('a or_error -> 'b t) -> 'a t -> 'b t

bind_reify_error ?on ~f fut returns a new future fut2 that resolves like the future f (Ok x) if fut resolved with x; and resolves like the future f (Error (exn, bt)) if fut fails with exn and backtrace bt.

This does not preserve local storage of fut inside f.

  • parameter on

    if provided, f runs on the given runner

  • since 0.4
Sourceval join : 'a t t -> 'a t

join fut is fut >>= Fun.id. It joins the inner layer of the future.

  • since 0.2
Sourceval both : 'a t -> 'b t -> ('a * 'b) t

both a b succeeds with x, y if a succeeds with x and b succeeds with y, or fails if any of them fails.

Sourceval choose : 'a t -> 'b t -> ('a, 'b) Either.t t

choose a b succeeds Left x or Right y if a succeeds with x or b succeeds with y, or fails if both of them fails. If they both succeed, it is not specified which result is used.

Sourceval choose_same : 'a t -> 'a t -> 'a t

choose_same a b succeeds with the value of one of a or b if they succeed, or fails if both fail. If they both succeed, it is not specified which result is used.

Sourceval join_array : 'a t array -> 'a array t

Wait for all the futures in the array. Fails if any future fails.

Sourceval join_list : 'a t list -> 'a list t

Wait for all the futures in the list. Fails if any future fails.

Sourcemodule Advanced : sig ... end

Advanced primitives for synchronization

Sourceval map_list : f:('a -> 'b t) -> 'a list -> 'b list t

map_list ~f l is like join_list @@ List.map f l.

  • since 0.5.1
Sourceval wait_array : _ t array -> unit t

wait_array arr waits for all futures in arr to resolve. It discards the individual results of futures in arr. It fails if any future fails.

Sourceval wait_list : _ t list -> unit t

wait_list l waits for all futures in l to resolve. It discards the individual results of futures in l. It fails if any future fails.

Sourceval for_ : on:Runner.t -> int -> (int -> unit) -> unit t

for_ ~on n f runs f 0, f 1, …, f (n-1) on the runner, and returns a future that resolves when all the tasks have resolved, or fails as soon as one task has failed.

Sourceval for_array : on:Runner.t -> 'a array -> (int -> 'a -> unit) -> unit t

for_array ~on arr f runs f 0 arr.(0), …, f (n-1) arr.(n-1) in the runner (where n = Array.length arr), and returns a future that resolves when all the tasks are done, or fails if any of them fails.

  • since 0.2
Sourceval for_list : on:Runner.t -> 'a list -> ('a -> unit) -> unit t

for_list ~on l f is like for_array ~on (Array.of_list l) f.

  • since 0.2

Await

This suspends the current task using an OCaml 5 algebraic effect, and makes preparations for the task to be resumed once the future has been resolved.

Sourceval await : 'a t -> 'a

await fut suspends the current tasks until fut is fulfilled, then resumes the task on this same runner (but possibly on a different thread/domain).

  • since 0.3

This must only be run from inside the runner itself. The runner must support Suspend_.

Sourceval yield : unit -> unit

Like Moonpool.yield.

  • since 0.10

Blocking

Sourceval wait_block : 'a t -> 'a or_error

wait_block fut blocks the current thread until fut is resolved, and returns its value.

NOTE: A word of warning: this will monopolize the calling thread until the future resolves. This can also easily cause deadlocks, if enough threads in a pool call wait_block on futures running on the same pool or a pool depending on it.

A good rule to avoid deadlocks is to run this from outside of any pool, or to have an acyclic order between pools where wait_block is only called from a pool on futures evaluated in a pool that comes lower in the hierarchy. If this rule is broken, it is possible for all threads in a pool to wait for futures that can only make progress on these same threads, hence the deadlock.

Sourceval wait_block_exn : 'a t -> 'a

Same as wait_block but re-raises the exception if the future failed.

NOTE: do check the cautionary note in wait_block concerning deadlocks.

Infix operators

These combinators run on either the current pool (if present), or on the same thread that just fulfilled the previous future if not.

They were previously present as module Infix_local and val infix, but are now simplified.

  • since 0.5
Sourcemodule Infix : sig ... end
include module type of Infix
Sourceval (>|=) : 'a t -> ('a -> 'b) -> 'b t
Sourceval (>>=) : 'a t -> ('a -> 'b t) -> 'b t
Sourceval (let+) : 'a t -> ('a -> 'b) -> 'b t
Sourceval (and+) : 'a t -> 'b t -> ('a * 'b) t
Sourceval (let*) : 'a t -> ('a -> 'b t) -> 'b t
Sourceval (and*) : 'a t -> 'b t -> ('a * 'b) t
Sourcemodule Infix_local = Infix