Skip to content

Commit 09fd729

Browse files
authored
Code actions for exhaustive switches (#812)
* code actions for inserting exhaustive switches for selected expressions, single identifiers, and switch cases being written * expand options automatically * fix
1 parent e46ff27 commit 09fd729

11 files changed

+382
-19
lines changed

analysis/src/Cli.ml

+4-2
Original file line numberDiff line numberDiff line change
@@ -120,9 +120,11 @@ let main () =
120120
~pos:(int_of_string line_start, int_of_string line_end)
121121
~maxLength ~debug:false
122122
| [_; "codeLens"; path] -> Commands.codeLens ~path ~debug:false
123-
| [_; "codeAction"; path; line; col; currentFile] ->
123+
| [_; "codeAction"; path; startLine; startCol; endLine; endCol; currentFile]
124+
->
124125
Commands.codeAction ~path
125-
~pos:(int_of_string line, int_of_string col)
126+
~startPos:(int_of_string startLine, int_of_string startCol)
127+
~endPos:(int_of_string endLine, int_of_string endCol)
126128
~currentFile ~debug:false
127129
| [_; "codemod"; path; line; col; typ; hint] ->
128130
let typ =

analysis/src/Codemod.ml

+1-6
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,6 @@ let rec collectPatterns p =
55
| Ppat_or (p1, p2) -> collectPatterns p1 @ [p2]
66
| _ -> [p]
77

8-
let mkFailWithExp () =
9-
Ast_helper.Exp.apply
10-
(Ast_helper.Exp.ident {txt = Lident "failwith"; loc = Location.none})
11-
[(Nolabel, Ast_helper.Exp.constant (Pconst_string ("TODO", None)))]
12-
138
let transform ~path ~pos ~debug ~typ ~hint =
149
let structure, printExpr, _ = Xform.parseImplementation ~filename:path in
1510
match typ with
@@ -24,7 +19,7 @@ let transform ~path ~pos ~debug ~typ ~hint =
2419
let cases =
2520
collectPatterns pattern
2621
|> List.map (fun (p : Parsetree.pattern) ->
27-
Ast_helper.Exp.case p (mkFailWithExp ()))
22+
Ast_helper.Exp.case p (TypeUtils.Codegen.mkFailWithExp ()))
2823
in
2924
let result = ref None in
3025
let mkIterator ~pos ~result =

analysis/src/Commands.ml

+20-7
Original file line numberDiff line numberDiff line change
@@ -80,8 +80,8 @@ let signatureHelp ~path ~pos ~currentFile ~debug =
8080
in
8181
print_endline (Protocol.stringifySignatureHelp result)
8282

83-
let codeAction ~path ~pos ~currentFile ~debug =
84-
Xform.extractCodeActions ~path ~pos ~currentFile ~debug
83+
let codeAction ~path ~startPos ~endPos ~currentFile ~debug =
84+
Xform.extractCodeActions ~path ~startPos ~endPos ~currentFile ~debug
8585
|> CodeActions.stringifyCodeActions |> print_endline
8686

8787
let definition ~path ~pos ~debug =
@@ -268,7 +268,9 @@ let test ~path =
268268
let lines = text |> String.split_on_char '\n' in
269269
let processLine i line =
270270
let createCurrentFile () =
271-
let currentFile, cout = Filename.open_temp_file "def" "txt" in
271+
let currentFile, cout =
272+
Filename.open_temp_file "def" ("txt." ^ Filename.extension path)
273+
in
272274
let removeLineComment l =
273275
let len = String.length l in
274276
let rec loop i =
@@ -372,13 +374,24 @@ let test ~path =
372374
^ string_of_int col);
373375
typeDefinition ~path ~pos:(line, col) ~debug:true
374376
| "xfm" ->
375-
print_endline
376-
("Xform " ^ path ^ " " ^ string_of_int line ^ ":"
377-
^ string_of_int col);
377+
let currentFile = createCurrentFile () in
378+
(* +2 is to ensure that the character ^ points to is what's considered the end of the selection. *)
379+
let endCol = col + try String.index rest '^' + 2 with _ -> 0 in
380+
let endPos = (line, endCol) in
381+
let startPos = (line, col) in
382+
if startPos = endPos then
383+
print_endline
384+
("Xform " ^ path ^ " " ^ string_of_int line ^ ":"
385+
^ string_of_int col)
386+
else
387+
print_endline
388+
("Xform " ^ path ^ " start: " ^ Pos.toString startPos
389+
^ ", end: " ^ Pos.toString endPos);
378390
let codeActions =
379-
Xform.extractCodeActions ~path ~pos:(line, col) ~currentFile:path
391+
Xform.extractCodeActions ~path ~startPos ~endPos ~currentFile
380392
~debug:true
381393
in
394+
Sys.remove currentFile;
382395
codeActions
383396
|> List.iter (fun {Protocol.title; edit = {documentChanges}} ->
384397
Printf.printf "Hit: %s\n" title;

analysis/src/CompletionFrontEnd.ml

+20-2
Original file line numberDiff line numberDiff line change
@@ -222,7 +222,8 @@ let completePipeChain (exp : Parsetree.expression) =
222222
exprToContextPath exp |> Option.map (fun ctxPath -> (ctxPath, pexp_loc))
223223
| _ -> None
224224

225-
let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor ~text =
225+
let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor
226+
?findThisExprLoc text =
226227
let offsetNoWhite = Utils.skipWhite text (offset - 1) in
227228
let posNoWhite =
228229
let line, col = posCursor in
@@ -777,6 +778,12 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor ~text =
777778
(Pos.toString posCursor) (Pos.toString posNoWhite)
778779
(Loc.toString expr.pexp_loc)
779780
in
781+
(match findThisExprLoc with
782+
| Some loc when expr.pexp_loc = loc -> (
783+
match exprToContextPath expr with
784+
| None -> ()
785+
| Some ctxPath -> setResult (Cpath ctxPath))
786+
| _ -> ());
780787
let setPipeResult ~(lhs : Parsetree.expression) ~id =
781788
match completePipeChain lhs with
782789
| None -> (
@@ -1228,5 +1235,16 @@ let completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor ~text =
12281235
let completionWithParser ~debug ~path ~posCursor ~currentFile ~text =
12291236
match Pos.positionToOffset text posCursor with
12301237
| Some offset ->
1231-
completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor ~text
1238+
completionWithParser1 ~currentFile ~debug ~offset ~path ~posCursor text
12321239
| None -> None
1240+
1241+
let findTypeOfExpressionAtLoc ~debug ~path ~posCursor ~currentFile loc =
1242+
let textOpt = Files.readFile currentFile in
1243+
match textOpt with
1244+
| None | Some "" -> None
1245+
| Some text -> (
1246+
match Pos.positionToOffset text posCursor with
1247+
| Some offset ->
1248+
completionWithParser1 ~findThisExprLoc:loc ~currentFile ~debug ~offset
1249+
~path ~posCursor text
1250+
| None -> None)

analysis/src/Loc.ml

+4
Original file line numberDiff line numberDiff line change
@@ -8,3 +8,7 @@ let toString (loc : t) =
88
(if loc.loc_ghost then "__ghost__" else "") ^ (loc |> range |> Range.toString)
99

1010
let hasPos ~pos loc = start loc <= pos && pos < end_ loc
11+
12+
(** Allows the character after the end to be included. Ie when the cursor is at the
13+
end of the word, like `someIdentifier<cursor>`. Useful in some scenarios. *)
14+
let hasPosInclusiveEnd ~pos loc = start loc <= pos && pos <= end_ loc

analysis/src/TypeUtils.ml

+74
Original file line numberDiff line numberDiff line change
@@ -606,3 +606,77 @@ let unwrapCompletionTypeIfOption (t : SharedTypes.completionType) =
606606
match t with
607607
| Toption (_, ExtractedType unwrapped) -> unwrapped
608608
| _ -> t
609+
610+
module Codegen = struct
611+
let mkFailWithExp () =
612+
Ast_helper.Exp.apply
613+
(Ast_helper.Exp.ident {txt = Lident "failwith"; loc = Location.none})
614+
[(Nolabel, Ast_helper.Exp.constant (Pconst_string ("TODO", None)))]
615+
616+
let mkConstructPat ?payload name =
617+
Ast_helper.Pat.construct
618+
{Asttypes.txt = Longident.Lident name; loc = Location.none}
619+
payload
620+
621+
let mkTagPat ?payload name = Ast_helper.Pat.variant name payload
622+
623+
let any () = Ast_helper.Pat.any ()
624+
625+
let rec extractedTypeToExhaustivePatterns ~env ~full extractedType =
626+
match extractedType with
627+
| Tvariant v ->
628+
Some
629+
(v.constructors
630+
|> List.map (fun (c : SharedTypes.Constructor.t) ->
631+
mkConstructPat
632+
?payload:
633+
(match c.args with
634+
| Args [] -> None
635+
| _ -> Some (any ()))
636+
c.cname.txt))
637+
| Tpolyvariant v ->
638+
Some
639+
(v.constructors
640+
|> List.map (fun (c : SharedTypes.polyVariantConstructor) ->
641+
mkTagPat
642+
?payload:
643+
(match c.args with
644+
| [] -> None
645+
| _ -> Some (any ()))
646+
c.name))
647+
| Toption (_, innerType) ->
648+
let extractedType =
649+
match innerType with
650+
| ExtractedType t -> Some t
651+
| TypeExpr t -> extractType t ~env ~package:full.package
652+
in
653+
let expandedBranches =
654+
match extractedType with
655+
| None -> []
656+
| Some extractedType -> (
657+
match extractedTypeToExhaustivePatterns ~env ~full extractedType with
658+
| None -> []
659+
| Some patterns -> patterns)
660+
in
661+
Some
662+
([
663+
mkConstructPat "None";
664+
mkConstructPat ~payload:(Ast_helper.Pat.any ()) "Some";
665+
]
666+
@ (expandedBranches
667+
|> List.map (fun (pat : Parsetree.pattern) ->
668+
mkConstructPat ~payload:pat "Some")))
669+
| Tbool _ -> Some [mkConstructPat "true"; mkConstructPat "false"]
670+
| _ -> None
671+
672+
let extractedTypeToExhaustiveCases ~env ~full extractedType =
673+
let patterns = extractedTypeToExhaustivePatterns ~env ~full extractedType in
674+
675+
match patterns with
676+
| None -> None
677+
| Some patterns ->
678+
Some
679+
(patterns
680+
|> List.map (fun (pat : Parsetree.pattern) ->
681+
Ast_helper.Exp.case pat (mkFailWithExp ())))
682+
end

0 commit comments

Comments
 (0)