From 2a13145bca6ef107cb7d80f61c8e34b297d4c723 Mon Sep 17 00:00:00 2001 From: Matthew McQuaid <47466291+mmcqd@users.noreply.github.com> Date: Wed, 20 Sep 2023 02:32:09 -0700 Subject: [PATCH] feat: custom printers for unhandled effects (#19) Co-authored-by: favonia --- src/Reader.ml | 5 +++++ src/Reader.mli | 3 +++ src/Sequencer.ml | 5 +++++ src/Sequencer.mli | 3 +++ src/State.ml | 6 ++++++ src/State.mli | 3 +++ src/UniqueID.ml | 7 +++++++ src/UniqueID.mli | 3 +++ test/TestReader.ml | 1 + test/TestSequencer.ml | 1 + test/TestState.ml | 1 + 11 files changed, 38 insertions(+) diff --git a/src/Reader.ml b/src/Reader.ml index c9c71fc..7d713fc 100644 --- a/src/Reader.ml +++ b/src/Reader.ml @@ -10,6 +10,7 @@ sig val read : unit -> env val scope : (env -> env) -> (unit -> 'a) -> 'a val run : env:env -> (unit -> 'a) -> 'a + val register_printer : ([`Read] -> string option) -> unit end module Make (P : Param) = @@ -30,4 +31,8 @@ struct | _ -> None } let scope f c = run ~env:(f @@ read ()) c + + let register_printer f = Printexc.register_printer @@ function + | Effect.Unhandled Read -> f `Read + | _ -> None end diff --git a/src/Reader.mli b/src/Reader.mli index 6556956..69ad47f 100644 --- a/src/Reader.mli +++ b/src/Reader.mli @@ -41,6 +41,9 @@ sig val run : env:env -> (unit -> 'a) -> 'a (** [run t] runs the thunk [t] which may perform reading effects. *) + + (**/**) + val register_printer : ([`Read] -> string option) -> unit end module Make (P : Param) : S with type env = P.env diff --git a/src/Sequencer.ml b/src/Sequencer.ml index 4e69a46..3140f04 100644 --- a/src/Sequencer.ml +++ b/src/Sequencer.ml @@ -9,6 +9,7 @@ sig val yield : elt -> unit val run : (unit -> unit) -> elt Seq.t + val register_printer : ([`Yield of elt] -> string option) -> unit end module Make (P : Param) = @@ -27,4 +28,8 @@ struct | Yield x -> Option.some @@ fun (k : (a, _) continuation) -> Seq.Cons (x, continue k) | _ -> None } + + let register_printer f = Printexc.register_printer @@ function + | Effect.Unhandled (Yield elt) -> f (`Yield elt) + | _ -> None end diff --git a/src/Sequencer.mli b/src/Sequencer.mli index 36fd390..8575856 100644 --- a/src/Sequencer.mli +++ b/src/Sequencer.mli @@ -35,6 +35,9 @@ sig val run : (unit -> unit) -> elt Seq.t (** [run t] runs the thunk [t] which may perform sequencing effects. *) + + (**/**) + val register_printer : ([`Yield of elt] -> string option) -> unit end module Make (P : Param) : S with type elt = P.elt diff --git a/src/State.ml b/src/State.ml index 84ca186..0b9eb50 100644 --- a/src/State.ml +++ b/src/State.ml @@ -11,6 +11,7 @@ sig val set : state -> unit val modify : (state -> state) -> unit val run : init:state -> (unit -> 'a) -> 'a + val register_printer : ([`Get | `Set of state] -> string option) -> unit end module Make (P : Param) = @@ -37,4 +38,9 @@ struct | _ -> None } let modify f = set @@ f @@ get () + + let register_printer f = Printexc.register_printer @@ function + | Effect.Unhandled Get -> f `Get + | Effect.Unhandled (Set state) -> f (`Set state) + | _ -> None end diff --git a/src/State.mli b/src/State.mli index 723f89f..b829b55 100644 --- a/src/State.mli +++ b/src/State.mli @@ -40,6 +40,9 @@ sig val run : init:state -> (unit -> 'a) -> 'a (** [run t] runs the thunk [t] which may perform state effects. *) + + (**/**) + val register_printer : ([`Get | `Set of state] -> string option) -> unit end module Make (P : Param) : S with type state = P.state diff --git a/src/UniqueID.ml b/src/UniqueID.ml index c7f87e7..3c415be 100644 --- a/src/UniqueID.ml +++ b/src/UniqueID.ml @@ -21,6 +21,7 @@ sig val retrieve : id -> elt val export : unit -> elt Seq.t val run : ?init:elt Seq.t -> (unit -> 'a) -> 'a + val register_printer : ([`Register of elt | `Retrieve of id | `Export] -> string option) -> unit end module Make (P : Param) = @@ -66,4 +67,10 @@ struct | Export -> Option.some @@ fun (k : (a, _) continuation) -> continue k @@ Seq.map snd @@ M.to_seq @@ Eff.get () | _ -> None } + + let register_printer f = Printexc.register_printer @@ function + | Effect.Unhandled (Insert elt) -> f (`Register elt) + | Effect.Unhandled (Select id) -> f (`Retrieve id) + | Effect.Unhandled Export -> f `Export + | _ -> None end diff --git a/src/UniqueID.mli b/src/UniqueID.mli index efda34b..2b460a9 100644 --- a/src/UniqueID.mli +++ b/src/UniqueID.mli @@ -54,6 +54,9 @@ sig @param init The initial storage, which should be the output of some previous {!val:export}. *) + + (**/**) + val register_printer : ([`Register of elt | `Retrieve of id | `Export] -> string option) -> unit end module Make (P : Param) : S with type elt = P.elt diff --git a/test/TestReader.ml b/test/TestReader.ml index 77a089d..a12dcc8 100644 --- a/test/TestReader.ml +++ b/test/TestReader.ml @@ -18,6 +18,7 @@ struct let read () = U.perform ReaderMonad.read let scope f m = U.perform @@ ReaderMonad.scope f @@ U.run m let run ~env f = U.run f env + let register_printer _ = () end type cmd = ReadAndPrint | Scope of (int -> int) * prog diff --git a/test/TestSequencer.ml b/test/TestSequencer.ml index 83030b0..68c95f7 100644 --- a/test/TestSequencer.ml +++ b/test/TestSequencer.ml @@ -26,6 +26,7 @@ struct type elt = int let yield x = U.perform (SequencerMonad.yield x) let run f = output_to_seq @@ snd @@ U.run f + let register_printer _ = () end type cmd = Yield of int diff --git a/test/TestState.ml b/test/TestState.ml index cf5d638..91d120b 100644 --- a/test/TestState.ml +++ b/test/TestState.ml @@ -20,6 +20,7 @@ struct let set s = U.perform @@ StateMonad.set s let modify f = U.perform @@ StateMonad.modify f let run ~init f = fst @@ U.run f init + let register_printer _ = () end type cmd = Set of int | GetAndPrint | Mod of (int -> int)