Skip to content

Commit

Permalink
Merge pull request #845 from ejgallego/payload
Browse files Browse the repository at this point in the history
[coq] Abstract the feedback / error payloads better.
  • Loading branch information
ejgallego authored Oct 3, 2024
2 parents 4012c75 + ee95987 commit d75491c
Show file tree
Hide file tree
Showing 13 changed files with 87 additions and 50 deletions.
2 changes: 1 addition & 1 deletion controller/request.ml
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ module R = struct

let print_err ~name e =
match e with
| Coq.Protect.Error.Anomaly (_loc, msg) | User (_loc, msg) ->
| Coq.Protect.Error.Anomaly { msg; _ } | User { msg; _ } ->
Format.asprintf "Error in %s request: %a" name Pp.pp_with msg

let of_execution ~name ~f x : t =
Expand Down
3 changes: 2 additions & 1 deletion controller/rq_goals.ml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,8 @@
(************************************************************************)

(* Replace by ppx when we can print goals properly in the client *)
let mk_message (range, level, text) = Lsp.JFleche.Message.{ range; level; text }
let mk_message (level, { Coq.Message.Payload.range; msg }) =
Lsp.JFleche.Message.{ range; level; text = msg }

let mk_messages node =
Option.map Fleche.Doc.Node.messages node
Expand Down
5 changes: 3 additions & 2 deletions coq/init.ml
Original file line number Diff line number Diff line change
Expand Up @@ -36,9 +36,10 @@ let coq_lvl_to_severity (lvl : Feedback.level) =
| Feedback.Warning -> warning
| Feedback.Error -> error

let add_message lvl loc msg q =
let add_message lvl range msg q =
let lvl = coq_lvl_to_severity lvl in
q := (loc, lvl, msg) :: !q
let payload = Message.Payload.make ?range msg in
q := (lvl, payload) :: !q

let mk_fb_handler q Feedback.{ contents; _ } =
match contents with
Expand Down
14 changes: 13 additions & 1 deletion coq/message.ml
Original file line number Diff line number Diff line change
@@ -1,2 +1,14 @@
(** Messages from Coq *)
type 'l t = 'l option * Lang.Diagnostic.Severity.t * Pp.t
module Payload = struct
type 'l t =
{ range : 'l option
; msg : Pp.t
}

let make ?range msg = { range; msg }
let map ~f { range; msg } = { range = Option.map f range; msg }
end

type 'l t = Lang.Diagnostic.Severity.t * 'l Payload.t

let map ~f (lvl, pl) = (lvl, Payload.map ~f pl)
22 changes: 21 additions & 1 deletion coq/message.mli
Original file line number Diff line number Diff line change
@@ -1,2 +1,22 @@
(** Messages from Coq *)
type 'l t = 'l option * Lang.Diagnostic.Severity.t * Pp.t

(** Coq provides payload to our layer via two different mechanisms:
- feedback messages
- error exceptions
In both cases, the payload is the same, and it comes via different ways due
to historical reasons. We abstract the payload as to better handle the
common paths. *)
module Payload : sig
type 'l t =
{ range : 'l option
; msg : Pp.t
}

val make : ?range:'l -> Pp.t -> 'l t
val map : f:('l -> 'm) -> 'l t -> 'm t
end

type 'l t = Lang.Diagnostic.Severity.t * 'l Payload.t

val map : f:('l -> 'm) -> 'l t -> 'm t
22 changes: 11 additions & 11 deletions coq/protect.ml
Original file line number Diff line number Diff line change
@@ -1,9 +1,7 @@
module Error = struct
type 'l payload = 'l option * Pp.t

type 'l t =
| User of 'l payload
| Anomaly of 'l payload
| User of 'l Message.Payload.t
| Anomaly of 'l Message.Payload.t

let map ~f = function
| User e -> User (f e)
Expand All @@ -15,7 +13,9 @@ module R = struct
| Completed of ('a, 'l Error.t) result
| Interrupted (* signal sent, eval didn't complete *)

let error e = Completed (Error (Error.User (None, e)))
let error msg =
let payload = Message.Payload.make msg in
Completed (Error (Error.User payload))

let map ~f = function
| Completed (Result.Ok r) -> Completed (Result.Ok (f r))
Expand All @@ -28,7 +28,7 @@ module R = struct
| Interrupted -> Interrupted

let map_loc ~f =
let f (loc, msg) = (Option.map f loc, msg) in
let f = Message.Payload.map ~f in
map_error ~f
end

Expand All @@ -41,11 +41,12 @@ let eval_exn ~token ~f x =
R.Interrupted
| exception exn ->
let e, info = Exninfo.capture exn in
let loc = Loc.(get_loc info) in
let range = Loc.(get_loc info) in
let msg = CErrors.iprint (e, info) in
let payload = Message.Payload.make ?range msg in
Vernacstate.Interp.invalidate_cache ();
if CErrors.is_anomaly e then R.Completed (Error (Anomaly (loc, msg)))
else R.Completed (Error (User (loc, msg)))
if CErrors.is_anomaly e then R.Completed (Error (Anomaly payload))
else R.Completed (Error (User payload))

let _bind_exn ~f x =
match x with
Expand All @@ -68,10 +69,9 @@ module E = struct
{ r; feedback }

let map ~f { r; feedback } = { r = R.map ~f r; feedback }
let map_message ~f (loc, lvl, msg) = (Option.map f loc, lvl, msg)

let map_loc ~f { r; feedback } =
{ r = R.map_loc ~f r; feedback = List.map (map_message ~f) feedback }
{ r = R.map_loc ~f r; feedback = List.map (Message.map ~f) feedback }

let bind ~f { r; feedback } =
match r with
Expand Down
8 changes: 3 additions & 5 deletions coq/protect.mli
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,9 @@
As of today this includes feedback and exceptions. *)
module Error : sig
type 'l payload = 'l option * Pp.t

type 'l t = private
| User of 'l payload
| Anomaly of 'l payload
| User of 'l Message.Payload.t
| Anomaly of 'l Message.Payload.t
end

(** This "monad" could be related to "Runners in action" (Ahman, Bauer), thanks
Expand All @@ -22,7 +20,7 @@ module R : sig
val map : f:('a -> 'b) -> ('a, 'l) t -> ('b, 'l) t

val map_error :
f:('l Error.payload -> 'm Error.payload) -> ('a, 'l) t -> ('a, 'm) t
f:('l Message.Payload.t -> 'm Message.Payload.t) -> ('a, 'l) t -> ('a, 'm) t

(** Update the loc stored in the result, this is used by our cache-aware
location *)
Expand Down
4 changes: 2 additions & 2 deletions coq/workspace.ml
Original file line number Diff line number Diff line change
Expand Up @@ -387,8 +387,8 @@ let guess ~token ~debug ~cmdline ~dir =
ignore feedback;
match r with
| Protect.R.Interrupted -> Error "Workspace Scanning Interrupted"
| Protect.R.Completed (Error (User (_, msg)))
| Protect.R.Completed (Error (Anomaly (_, msg))) ->
| Protect.R.Completed (Error (User { msg; _ }))
| Protect.R.Completed (Error (Anomaly { msg; _ })) ->
Error (Format.asprintf "Workspace Scanning Errored: %a" Pp.pp_with msg)
| Protect.R.Completed (Ok workspace) -> Ok workspace

Expand Down
38 changes: 21 additions & 17 deletions fleche/doc.ml
Original file line number Diff line number Diff line change
Expand Up @@ -116,8 +116,9 @@ module Node = struct
module Message = struct
type t = Lang.Range.t Coq.Message.t

let feedback_to_message ~lines (loc, lvl, msg) =
(Coq.Utils.to_orange ~lines loc, lvl, msg)
let feedback_to_message ~lines message =
let f = Coq.Utils.to_range ~lines in
Coq.Message.map ~f message
end

type t =
Expand Down Expand Up @@ -197,9 +198,10 @@ end = struct
let data = extra_diagnostics_of_ast stm_range ast in
make ?data err_range Lang.Diagnostic.Severity.error msg

let of_feed ~drange (range, severity, message) =
let of_feed ~drange (severity, payload) =
let { Coq.Message.Payload.range; msg } = payload in
let range = Option.default drange range in
make range severity message
make range severity msg

type partition_kind =
| Left
Expand All @@ -224,7 +226,7 @@ end = struct
else if !Config.v.show_notices_as_diagnostics then 4
else 3
in
let f (_, lvl, _) =
let f (lvl, _) =
(* warning = 2 *)
if lvl = Lang.Diagnostic.Severity.warning then Both
else if lvl < cutoff then Left
Expand Down Expand Up @@ -354,8 +356,9 @@ let empty_doc ~uri ~contents ~version ~env ~root ~nodes ~completed =
let completed = completed init_range in
{ uri; contents; toc; version; env; root; nodes; diags_dirty; completed }

let error_doc ~loc ~message ~uri ~contents ~version ~env =
let feedback = [ (loc, Diags.err, Pp.str message) ] in
let error_doc ~range ~message ~uri ~contents ~version ~env =
let payload = Coq.Message.Payload.make ?range (Pp.str message) in
let feedback = [ (Diags.err, payload) ] in
let root = env.Env.init in
let nodes = [] in
let completed range = Completion.Failed range in
Expand All @@ -365,7 +368,8 @@ let conv_error_doc ~raw ~uri ~version ~env ~root err =
let contents = Contents.make_raw ~raw in
let lines = contents.lines in
let err =
(None, Diags.err, Pp.(str "Error in document conversion: " ++ str err))
let msg = Pp.(str "Error in document conversion: " ++ str err) in
(Diags.err, Coq.Message.Payload.make msg)
in
(* No execution to add *)
let stats = None in
Expand Down Expand Up @@ -394,14 +398,14 @@ let handle_doc_creation_exec ~token ~env ~uri ~version ~contents =
match r with
| Interrupted ->
let message = "Document Creation Interrupted!" in
let loc = None in
error_doc ~loc ~message ~uri ~version ~contents ~env
| Completed (Error (User (loc, err_msg)))
| Completed (Error (Anomaly (loc, err_msg))) ->
let range = None in
error_doc ~range ~message ~uri ~version ~contents ~env
| Completed (Error (User { range; msg = err_msg }))
| Completed (Error (Anomaly { range; msg = err_msg })) ->
let message =
Format.asprintf "Doc.create, internal error: @[%a@]" Pp.pp_with err_msg
in
error_doc ~loc ~message ~uri ~version ~contents ~env
error_doc ~range ~message ~uri ~version ~contents ~env
| Completed (Ok doc) -> (doc, [])
in
let state = doc.root in
Expand Down Expand Up @@ -749,15 +753,15 @@ let parse_action ~token ~lines ~st last_tok doc_handle =
| Ok (Some ast) ->
let () = if Debug.parsing then DDebug.parsed_sentence ~ast in
(Process ast, [], feedback, time)
| Error (Anomaly (_, msg)) | Error (User (None, msg)) ->
| Error (Anomaly { msg; _ }) | Error (User { range = None; msg }) ->
(* We don't have a better alternative :(, usually missing error loc here
means an anomaly, so we stop *)
let err_range = last_tok in
let parse_diags =
[ Diags.error ~err_range ~msg ~stm_range:err_range () ]
in
(EOF (Failed last_tok), parse_diags, feedback, time)
| Error (User (Some err_range, msg)) ->
| Error (User { range = Some err_range; msg }) ->
Coq.Parsing.discard_to_dot doc_handle;
let last_tok = Coq.Parsing.Parsable.loc doc_handle in
let last_tok_range = Coq.Utils.to_range ~lines last_tok in
Expand Down Expand Up @@ -816,8 +820,8 @@ let node_of_coq_result ~token ~doc ~range ~prev ~ast ~st ~parsing_diags
~diags:[] ~feedback ~info
in
Continue { state; last_tok; node }
| Error (Coq.Protect.Error.Anomaly (err_range, msg) as coq_err)
| Error (User (err_range, msg) as coq_err) ->
| Error (Coq.Protect.Error.Anomaly { range = err_range; msg } as coq_err)
| Error (User { range = err_range; msg } as coq_err) ->
let err_range = Option.default range err_range in
let err_diags = [ Diags.error ~err_range ~msg ~stm_range:range ~ast () ] in
let contents, nodes = (doc.contents, doc.nodes) in
Expand Down
2 changes: 1 addition & 1 deletion fleche/doc.mli
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ module Node : sig
end

module Message : sig
type t = Lang.Range.t option * Lang.Diagnostic.Severity.t * Pp.t
type t = Lang.Range.t Coq.Message.t
end

type t = private
Expand Down
4 changes: 2 additions & 2 deletions petanque/agent.ml
Original file line number Diff line number Diff line change
Expand Up @@ -137,9 +137,9 @@ let execute_precommands ~token ~memo ~pre_commands ~(node : Fleche.Doc.Node.t) =
let protect_to_result (r : _ Coq.Protect.E.t) : (_, _) Result.t =
match r with
| { r = Interrupted; feedback = _ } -> Error Error.Interrupted
| { r = Completed (Error (User (_loc, msg))); feedback = _ } ->
| { r = Completed (Error (User { msg; _ })); feedback = _ } ->
Error (Error.Coq (Pp.string_of_ppcmds msg))
| { r = Completed (Error (Anomaly (_loc, msg))); feedback = _ } ->
| { r = Completed (Error (Anomaly { msg; _ })); feedback = _ } ->
Error (Error.Anomaly (Pp.string_of_ppcmds msg))
| { r = Completed (Ok r); feedback = _ } -> Ok r

Expand Down
6 changes: 4 additions & 2 deletions plugins/explain_errors/main.ml
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,10 @@ let pp_goals ~token ~st =
| Some proof -> (
match Coq.Print.pr_goals ~token ~proof with
| { Coq.Protect.E.r = Completed (Ok goals); _ } -> goals
| { Coq.Protect.E.r = Completed (Error (User msg | Anomaly msg)); _ } ->
Pp.(str "error when printing goals: " ++ snd msg)
| { Coq.Protect.E.r =
Completed (Error (User { msg; _ } | Anomaly { msg; _ }))
; _
} -> Pp.(str "error when printing goals: " ++ msg)
| { Coq.Protect.E.r = Interrupted; _ } ->
Pp.str "goal printing was interrupted")

Expand Down
7 changes: 3 additions & 4 deletions plugins/goaldump/main.ml
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,10 @@ let of_execution ~io ~what (v : (_, _) Coq.Protect.E.t) =
| { r; feedback = _ } -> (
match r with
| Coq.Protect.R.Completed (Ok goals) -> goals
| Coq.Protect.R.Completed (Error (Anomaly err))
| Coq.Protect.R.Completed (Error (User err)) ->
| Coq.Protect.R.Completed (Error (Anomaly { msg; _ }))
| Coq.Protect.R.Completed (Error (User { msg; _ })) ->
let lvl = Io.Level.Error in
Io.Report.msg ~io ~lvl "error when retrieving %s: %a" what Pp.pp_with
(snd err);
Io.Report.msg ~io ~lvl "error when retrieving %s: %a" what Pp.pp_with msg;
None
| Coq.Protect.R.Interrupted -> None)

Expand Down

0 comments on commit d75491c

Please sign in to comment.