Skip to content

Commit

Permalink
Rely on upstream Merlin analyses (ocaml#1383)
Browse files Browse the repository at this point in the history
* Use `Typedtree_util` to extract toplevel identifier

* Use Inlay_hint from merlin-lib

* refactoring signature help

* Bump minimum `merlin-lib` version

---------

Co-authored-by: xvw <[email protected]>
Co-authored-by: PizieDust <[email protected]>
  • Loading branch information
3 people authored Oct 1, 2024
1 parent d5519dd commit beea7e1
Show file tree
Hide file tree
Showing 4 changed files with 36 additions and 309 deletions.
2 changes: 1 addition & 1 deletion ocaml-lsp-server.opam
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ depends: [
"csexp" {>= "1.5"}
"ocamlformat-rpc-lib" {>= "0.21.0"}
"odoc" {with-doc}
"merlin-lib" {>= "5.0" & < "6.0"}
"merlin-lib" {>= "5.2" & < "6.0"}
]
dev-repo: "git+https://github.com/ocaml/ocaml-lsp.git"
build: [
Expand Down
21 changes: 4 additions & 17 deletions ocaml-lsp-server/src/inference.ml
Original file line number Diff line number Diff line change
Expand Up @@ -126,23 +126,10 @@ let infer_intf (state : State.t) intf_doc =
(** Extracts an [Ident.t] from all variants that have one at the top level. For
many of the other variants, it would be possible to extract a list of IDs,
but that's not needed for the update-signatures code action. *)
let top_level_id (item : Typedtree.signature_item) =
match item.sig_desc with
| Typedtree.Tsig_value { val_id; _ } -> Some val_id
| Typedtree.Tsig_module { md_id; _ } -> md_id
| Typedtree.Tsig_modsubst { ms_id; _ } -> Some ms_id
| Typedtree.Tsig_modtype { mtd_id; _ } -> Some mtd_id
| Typedtree.Tsig_modtypesubst { mtd_id; _ } -> Some mtd_id
| Typedtree.Tsig_type _
| Typedtree.Tsig_typesubst _
| Typedtree.Tsig_typext _
| Typedtree.Tsig_exception _
| Typedtree.Tsig_recmodule _
| Typedtree.Tsig_open _
| Typedtree.Tsig_include _
| Typedtree.Tsig_class _
| Typedtree.Tsig_class_type _
| Typedtree.Tsig_attribute _ -> None
let top_level_id item =
match Merlin_analysis.Typedtree_utils.extract_toplevel_identifier item with
| [ ident ] -> Some ident
| _ -> None
;;

(** Represents an item that's present in the existing interface and has a
Expand Down
146 changes: 25 additions & 121 deletions ocaml-lsp-server/src/inlay_hints.ml
Original file line number Diff line number Diff line change
@@ -1,109 +1,15 @@
open Import
open Fiber.O

let range_overlaps_loc range loc =
match Range.of_loc_opt loc with
| Some range' -> Range.overlaps range range'
| None -> false
;;

let outline_type ~env typ =
Ocaml_typing.Printtyp.wrap_printing_env env (fun () ->
Format.asprintf "@[<h>: %a@]" Ocaml_typing.Printtyp.type_scheme typ)
let outline_type typ =
typ
|> Format.asprintf "@[<h>: %s@]"
|> String.extract_words ~is_word_char:(function
| ' ' | '\t' | '\n' -> false
| _ -> true)
|> String.concat ~sep:" "
;;

let hint_binding_iter
?(hint_let_bindings = false)
?(hint_pattern_variables = false)
typedtree
range
k
=
let module I = Ocaml_typing.Tast_iterator in
(* to be used for pattern variables in match cases, but not for function
arguments *)
let case hint_lhs (iter : I.iterator) (case : _ Typedtree.case) =
if hint_lhs then iter.pat iter case.c_lhs;
Option.iter case.c_guard ~f:(iter.expr iter);
iter.expr iter case.c_rhs
in
let value_binding hint_lhs (iter : I.iterator) (vb : Typedtree.value_binding) =
if range_overlaps_loc range vb.vb_loc
then
if not hint_lhs
then iter.expr iter vb.vb_expr
else (
match vb.vb_expr.exp_desc with
| Texp_function _ -> iter.expr iter vb.vb_expr
| _ -> I.default_iterator.value_binding iter vb)
in
let expr (iter : I.iterator) (e : Typedtree.expression) =
if range_overlaps_loc range e.exp_loc
then (
match e.exp_desc with
| Texp_function
( _
, Tfunction_cases
{ cases =
[ { c_rhs = { exp_desc = Texp_let (_, [ { vb_pat; _ } ], body); _ }; _ }
]
; _
} ) ->
iter.pat iter vb_pat;
iter.expr iter body
| Texp_let (_, vbs, body) ->
List.iter vbs ~f:(value_binding hint_let_bindings iter);
iter.expr iter body
| Texp_letop { body; _ } -> case hint_let_bindings iter body
| Texp_match (expr, cases, _) ->
iter.expr iter expr;
List.iter cases ~f:(case hint_pattern_variables iter)
(* Stop iterating when we see a ghost location to avoid annotating generated code *)
| _ when e.exp_loc.loc_ghost && not inside_test -> ()
| _ -> I.default_iterator.expr iter e)
in
let structure_item (iter : I.iterator) (item : Typedtree.structure_item) =
if range_overlaps_loc range item.str_loc
then (
match item.str_desc with
| Typedtree.Tstr_value (_, vbs) ->
List.iter vbs ~f:(fun (vb : Typedtree.value_binding) -> expr iter vb.vb_expr)
(* Stop iterating when we see a ghost location to avoid annotating generated code *)
| _ when item.str_loc.loc_ghost && not inside_test -> ()
| _ -> I.default_iterator.structure_item iter item)
in
let pat (type k) iter (pat : k Typedtree.general_pattern) =
if range_overlaps_loc range pat.pat_loc
then (
let has_constraint =
List.exists pat.pat_extra ~f:(fun (extra, _, _) ->
match extra with
| Typedtree.Tpat_constraint _ -> true
| _ -> false)
in
if not has_constraint
then (
I.default_iterator.pat iter pat;
match pat.pat_desc with
| Tpat_var _ when not pat.pat_loc.loc_ghost ->
k pat.pat_env pat.pat_type pat.pat_loc
| _ -> ()))
in
let iterator =
{ I.default_iterator with
expr
; structure_item
; pat
; value_binding = value_binding true
}
in
iterator.structure iterator typedtree
;;

let compute (state : State.t) { InlayHintParams.range; textDocument = { uri }; _ } =
let doc =
let store = state.store in
Expand All @@ -116,36 +22,34 @@ let compute (state : State.t) { InlayHintParams.range; textDocument = { uri }; _
let+ hints =
let hint_let_bindings =
Option.map state.configuration.data.inlay_hints ~f:(fun c -> c.hint_let_bindings)
|> Option.value ~default:false
in
let hint_pattern_variables =
Option.map state.configuration.data.inlay_hints ~f:(fun c ->
c.hint_pattern_variables)
|> Option.value ~default:false
in
Document.Merlin.with_pipeline_exn ~name:"inlay-hints" doc (fun pipeline ->
let hints = ref [] in
(match Mtyper.get_typedtree (Mpipeline.typer_result pipeline) with
| `Interface _ -> ()
| `Implementation typedtree ->
hint_binding_iter
?hint_let_bindings
?hint_pattern_variables
typedtree
range
(fun env type_ loc ->
let hint =
let label = outline_type ~env type_ in
let open Option.O in
let+ position = Position.of_lexical_position loc.loc_end in
InlayHint.create
~kind:Type
~position
~label:(`String label)
~paddingLeft:false
~paddingRight:false
()
in
Option.iter hint ~f:(fun hint -> hints := hint :: !hints)));
!hints)
let start = range.start |> Position.logical
and stop = range.end_ |> Position.logical in
let command =
Query_protocol.Inlay_hints
(start, stop, hint_let_bindings, hint_pattern_variables, not inside_test)
in
let hints = Query_commands.dispatch pipeline command in
List.filter_map
~f:(fun (pos, label) ->
let open Option.O in
let+ position = Position.of_lexical_position pos in
let label = `String (outline_type label) in
InlayHint.create
~kind:Type
~position
~label
~paddingLeft:false
~paddingRight:false
())
hints)
in
Some hints
;;
Loading

0 comments on commit beea7e1

Please sign in to comment.