Skip to content

Commit

Permalink
Merge pull request #533 from usethesource/fix/rename-refactoring/inva…
Browse files Browse the repository at this point in the history
…lid-name-check

Improve rename refactoring name check
  • Loading branch information
toinehartman authored Nov 27, 2024
2 parents 41d6fd6 + 4fa9c79 commit 0a8f478
Show file tree
Hide file tree
Showing 4 changed files with 35 additions and 27 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ data IllegalRenameReason
| definitionsOutsideWorkspace(set[loc] defs)
;

data RuntimeException
data RenameException
= illegalRename(str message, set[IllegalRenameReason] reason)
| unsupportedRename(str message, rel[loc location, str message] issues = {})
| unexpectedFailure(str message)
Expand Down
44 changes: 18 additions & 26 deletions rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Rename.rsc
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ module lang::rascal::lsp::refactor::Rename

import Exception;
import IO;
import Grammar;
import List;
import Location;
import Map;
Expand Down Expand Up @@ -76,35 +77,24 @@ void throwAnyErrors(program(_, msgs)) {
throwAnyErrors(msgs);
}

set[IllegalRenameReason] rascalCheckLegalName(str name, set[IdRole] defRoles) {
set[IllegalRenameReason] tryParseAs(type[&T <: Tree] begin, str idDescription) {
try {
parse(begin, rascalEscapeName(name));
return {};
} catch ParseError(_): {
return {invalidName(name, idDescription)};
}
}
private set[IllegalRenameReason] rascalCheckLegalName(str name, set[IdRole] roles) {
escName = rascalEscapeName(name);
tuple[type[&T <: Tree] as, str desc] asType = <#Name, "identifier">;
if ({moduleId(), *_} := roles) asType = <#QualifiedName, "module name">;
if ({constructorId(), *_} := roles) asType = <#NonterminalLabel, "constructor name">;
if ({fieldId(), *_} := roles) asType = <#NonterminalLabel, "constructor field name">;
if (size(syntaxRoles & roles) > 0) asType = <#Nonterminal, "non-terminal name">;

bool isSyntaxRole = any(role <- defRoles, role in syntaxRoles);
bool isField = any(role <- defRoles, role is fieldId);
bool isConstructor = any(role <- defRoles, role is constructorId);
if (!tryParseAs(asType.as, escName)) return {invalidName(escName, asType.desc)};
return {};
}

set[IllegalRenameReason] reasons = {};
if (isSyntaxRole) {
reasons += tryParseAs(#Nonterminal, "non-terminal name");
}
if (isField) {
reasons += tryParseAs(#NonterminalLabel, "constructor field name");
private void rascalCheckLegalName(str name, Symbol sym) {
escName = rascalEscapeName(name);
g = grammar(#start[Module]);
if (!tryParseAs(type(sym, g.rules), escName)) {
throw illegalRename("\'<escName>\' is not a valid name at this position", {invalidName(escName, "<sym>")});
}
if (isConstructor) {
reasons += tryParseAs(#NonterminalLabel, "constructor name");
}
if (!(isSyntaxRole || isField || isConstructor)) {
reasons += tryParseAs(#Name, "identifier");
}

return reasons;
}

private set[IllegalRenameReason] rascalCheckDefinitionsOutsideWorkspace(WorkspaceInfo ws, set[loc] defs) =
Expand Down Expand Up @@ -531,6 +521,8 @@ Edits rascalRenameSymbol(Tree cursorT, set[loc] workspaceFolders, str newName, P
loc cursorLoc = cursorT.src;
str cursorName = "<cursorT>";
rascalCheckLegalName(newName, typeOf(cursorT));
step("collecting workspace information", 1);
WorkspaceInfo ws = workspaceInfo(
// Get path config
Expand Down
13 changes: 13 additions & 0 deletions rascal-lsp/src/main/rascal/lang/rascal/lsp/refactor/Util.rsc
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ import IO;
import List;
import Location;
import Message;
import ParseTree;
import String;

import util::Maybe;
Expand Down Expand Up @@ -79,6 +80,18 @@ start[Module] parseModuleWithSpacesCached(loc l) {
return parseModuleWithSpacesCached(l, lastModified(l));
}

@synopsis{
Try to parse string `name` as reified type `begin` and return whether this succeeded.
}
bool tryParseAs(type[&T <: Tree] begin, str name, bool allowAmbiguity = false) {
try {
parse(begin, name, allowAmbiguity = allowAmbiguity);
return true;
} catch ParseError(_): {
return false;
}
}

Maybe[&B] flatMap(nothing(), Maybe[&B](&A) _) = nothing();
Maybe[&B] flatMap(just(&A a), Maybe[&B](&A) f) = f(a);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -57,3 +57,6 @@ test bool newNameHasNumericPrefix() = testRename("int foo = 8;", newName = "8abc

@expected{illegalRename}
test bool newNameIsEscapedInvalid() = testRename("int foo = 8;", newName = "\\8int");

@expected{illegalRename}
test bool qualifiedNameWhereNameExpected() = testRename("int foo = 8;", newName = "Foo::foo");

0 comments on commit 0a8f478

Please sign in to comment.