From ba2df533cfe8d7d23ef2728cd28425fd95805574 Mon Sep 17 00:00:00 2001 From: Toine Hartman Date: Tue, 19 Nov 2024 13:41:48 +0100 Subject: [PATCH] Fix duplicate locations in rename edits. --- .../lang/rascal/lsp/refactor/Rename.rsc | 48 +++++++++++-------- .../lang/rascal/tests/rename/TestUtils.rsc | 2 + 2 files changed, 29 insertions(+), 21 deletions(-) diff --git a/rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Rename.rsc b/rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Rename.rsc index 5aafb438..3707ccb5 100644 --- a/rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Rename.rsc +++ b/rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Rename.rsc @@ -62,6 +62,8 @@ import util::Reflective; alias Edits = tuple[list[DocumentEdit], map[ChangeAnnotationId, ChangeAnnotation]]; +private str MANDATORY_CHANGE_DESCRIPTION = "These changes are required for a correct renaming. They can be previewed here, but it is not advised to disable them."; + // Rascal compiler-specific extension void throwAnyErrors(list[ModuleMessages] mmsgs) { for (mmsg <- mmsgs) { @@ -192,13 +194,13 @@ private set[IllegalRenameReason] rascalCollectIllegalRenames(WorkspaceInfo ws, s private str rascalEscapeName(str name) = name in getRascalReservedIdentifiers() ? "\\" : name; // Find the smallest trees of defined non-terminal type with a source location in `useDefs` -private map[loc, loc] rascalFindNamesInUseDefs(start[Module] m, set[loc] useDefs) { - map[loc, loc] useDefNameAt = (); +private rel[loc, loc] rascalFindNamesInUseDefs(start[Module] m, set[loc] useDefs) { + rel[loc, loc] nameOfUseDef = {}; useDefsToDo = useDefs; visit(m.top) { case t: appl(prod(_, _, _), _): { if (t.src in useDefsToDo && just(nameLoc) := rascalLocationOfName(t)) { - useDefNameAt[t.src] = nameLoc; + nameOfUseDef += ; useDefsToDo -= t.src; } } @@ -208,7 +210,7 @@ private map[loc, loc] rascalFindNamesInUseDefs(start[Module] m, set[loc] useDefs throw unsupportedRename("Rename unsupported", issues={."> | l <- useDefsToDo}); } - return useDefNameAt; + return nameOfUseDef; } Maybe[loc] rascalLocationOfName(Name n) = just(n.src); @@ -229,25 +231,32 @@ Maybe[loc] rascalLocationOfName(Nonterminal nt) = just(nt.src); Maybe[loc] rascalLocationOfName(NonterminalLabel l) = just(l.src); default Maybe[loc] rascalLocationOfName(Tree t) = nothing(); -private tuple[set[IllegalRenameReason] reasons, list[TextEdit] edits] computeTextEdits(WorkspaceInfo ws, start[Module] m, set[RenameLocation] defs, set[RenameLocation] uses, str name) { +private tuple[set[IllegalRenameReason] reasons, list[TextEdit] edits] computeTextEdits(WorkspaceInfo ws, start[Module] m, set[RenameLocation] defs, set[RenameLocation] uses, str name, ChangeAnnotationRegister registerChangeAnnotation) { if (reasons := rascalCollectIllegalRenames(ws, m, defs.l, uses.l, name), reasons != {}) { return ; } replaceName = rascalEscapeName(name); - set[RenameLocation] renames = defs + uses; - set[loc] renameLocs = renames.l; - map[loc, loc] namesAt = rascalFindNamesInUseDefs(m, renameLocs); - - return <{}, [{just(annotation), *_} := renames[l] - ? replace(namesAt[l], replaceName, annotation = annotation) - : replace(namesAt[l], replaceName) - | l <- renameLocs]>; + rel[loc l, Maybe[ChangeAnnotationId] ann, bool isDef] renames = + { | <- defs} + + { | <- uses}; + rel[loc name, loc useDef] nameOfUseDef = rascalFindNamesInUseDefs(m, renames.l); + + ChangeAnnotationId defAnno = registerChangeAnnotation("Definitions", MANDATORY_CHANGE_DESCRIPTION, false); + ChangeAnnotationId useAnno = registerChangeAnnotation("References", MANDATORY_CHANGE_DESCRIPTION, false); + + // Note: if the implementer of the rename logic has attached annotations to multiple rename suggestions that have the same + // name location, one will be arbitrarily chosen here. This could mean that a `needsConfirmation` annotation is thrown away. + return <{}, [{just(annotation), *_} := renameOpts.ann + ? replace(l, replaceName, annotation = annotation) + : replace(l, replaceName, annotation = any(b <- renameOpts.isDef) ? defAnno : useAnno) + | l <- nameOfUseDef.name + , rel[Maybe[ChangeAnnotationId] ann, bool isDef] renameOpts := renames[nameOfUseDef[l]]]>; } -private tuple[set[IllegalRenameReason] reasons, list[TextEdit] edits] computeTextEdits(WorkspaceInfo ws, loc moduleLoc, set[RenameLocation] defs, set[RenameLocation] uses, str name) = - computeTextEdits(ws, parseModuleWithSpacesCached(moduleLoc), defs, uses, name); +private tuple[set[IllegalRenameReason] reasons, list[TextEdit] edits] computeTextEdits(WorkspaceInfo ws, loc moduleLoc, set[RenameLocation] defs, set[RenameLocation] uses, str name, ChangeAnnotationRegister registerChangeAnnotation) = + computeTextEdits(ws, parseModuleWithSpacesCached(moduleLoc), defs, uses, name, registerChangeAnnotation); private bool rascalIsFunctionLocalDefs(WorkspaceInfo ws, set[loc] defs) { for (d <- defs) { @@ -597,20 +606,17 @@ Edits rascalRenameSymbol(Tree cursorT, set[loc] workspaceFolders, str newName, P return id; }; - set[RenameLocation] registerChangeAnnotations(set[RenameLocation] locs, str label, str description = "These changes are required for a correct renaming. They can be previewed here, but it is not advised to disable them.", bool needsConfirmation = false) = - { | <- locs}; - = rascalGetDefsUses(ws, cur, rascalMayOverloadSameName, registerChangeAnnotation); - rel[loc file, RenameLocation defines] defsPerFile = { | d <- registerChangeAnnotations(defs, "Definitions")}; - rel[loc file, RenameLocation uses] usesPerFile = { | u <- registerChangeAnnotations(uses, "References")}; + rel[loc file, RenameLocation defines] defsPerFile = { | d <- defs}; + rel[loc file, RenameLocation uses] usesPerFile = { | u <- uses}; set[loc] \files = defsPerFile.file + usesPerFile.file; step("checking rename validity", 1); map[loc, tuple[set[IllegalRenameReason] reasons, list[TextEdit] edits]] moduleResults = - (file: | file <- \files, := computeTextEdits(ws, file, defsPerFile[file], usesPerFile[file], newName)); + (file: | file <- \files, := computeTextEdits(ws, file, defsPerFile[file], usesPerFile[file], newName, registerChangeAnnotation)); if (reasons := union({moduleResults[file].reasons | file <- moduleResults}), reasons != {}) { list[str] reasonDescs = toList({describe(r) | r <- reasons}); diff --git a/rascal-lsp/src/main/rascal/lang/rascal/tests/rename/TestUtils.rsc b/rascal-lsp/src/main/rascal/lang/rascal/tests/rename/TestUtils.rsc index 4d049cab..8014979a 100644 --- a/rascal-lsp/src/main/rascal/lang/rascal/tests/rename/TestUtils.rsc +++ b/rascal-lsp/src/main/rascal/lang/rascal/tests/rename/TestUtils.rsc @@ -65,6 +65,8 @@ private DocumentEdit sortChanges(changed(loc l, list[TextEdit] edits)) = changed private default DocumentEdit sortChanges(DocumentEdit e) = e; private void verifyTypeCorrectRenaming(loc root, Edits edits, PathConfig pcfg) { + list[loc] editLocs = [l | /replace(l, _) := edits<0>]; + assert size(editLocs) == size(toSet(editLocs)) : "Duplicate locations in suggested edits - VS Code cannot handle this"; executeDocumentEdits(sortEdits(edits<0>)); remove(pcfg.resources); RascalCompilerConfig ccfg = rascalCompilerConfig(pcfg)[forceCompilationTopModule = true][verbose = false][logPathConfig = false];