Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

API and implementation of codefixes and code actions for Rascal itself #478

Merged
merged 39 commits into from
Oct 31, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
fac1420
scaffolded an initial implementation of codefixes and code actioons f…
jurgenvinju Oct 16, 2024
05c9f9a
added first example code action for Rascal
jurgenvinju Oct 16, 2024
f60cff5
added default
jurgenvinju Oct 16, 2024
62f627f
final fixes
jurgenvinju Oct 16, 2024
ea0991d
added sort imports action
jurgenvinju Oct 16, 2024
46f76ea
fixed broken indentation
jurgenvinju Oct 16, 2024
55f6845
improved asyncronous command handling
jurgenvinju Oct 16, 2024
f08530e
removed superfluous newlines in sorted imports and extends
jurgenvinju Oct 17, 2024
0d0dcac
better grouping for imports and extends
jurgenvinju Oct 17, 2024
6be8c16
syntax better spaced
jurgenvinju Oct 17, 2024
0ab364f
added \'add missing license\' action for Rascal
jurgenvinju Oct 23, 2024
4a83365
grammar rules have 1 line distance
jurgenvinju Oct 23, 2024
0eea6fc
added UI test for Rascal code actions; factored out common assertLine…
jurgenvinju Oct 23, 2024
d799dc2
changed order of imports for testing purposes
jurgenvinju Oct 23, 2024
1e074bc
use arrow keys to skip the other actions
jurgenvinju Oct 23, 2024
17953b9
fixed typo
jurgenvinju Oct 23, 2024
4eecdaa
Merge branch 'main' into code-actions-for-rascal
jurgenvinju Oct 23, 2024
214dd88
reintroduced method lost in merge
jurgenvinju Oct 23, 2024
ed75af4
changed test because the action is not currently focused, it is the t…
jurgenvinju Oct 23, 2024
31f88d8
bringing mo to the mountain, just test the first action instead of th…
jurgenvinju Oct 24, 2024
e853737
added LICENSE to test project for testing purposes of the code action…
jurgenvinju Oct 24, 2024
eec1297
added LICENSE to original files
jurgenvinju Oct 24, 2024
e021fa5
make sure to revert changes made by code actions before we continue t…
jurgenvinju Oct 24, 2024
78250cd
factoring common functionality between Rascal LSP and Parametric DSL …
jurgenvinju Oct 24, 2024
f8b0702
acted on suggestions in review by @davylandman
jurgenvinju Oct 24, 2024
c07ba2e
fixed bug with command going to the wrong LSP server
jurgenvinju Oct 24, 2024
58997ec
adding license action only appears on module name now
jurgenvinju Oct 24, 2024
0644f00
factoring constants
jurgenvinju Oct 24, 2024
bbba2cb
fixed commands for Rascal itself
jurgenvinju Oct 24, 2024
d5362ac
fixed indentation
jurgenvinju Oct 24, 2024
f38c062
factored offset moving and character-width fixing code into reusable …
jurgenvinju Oct 30, 2024
782a23e
factoring reusable code and wrapping more async code
jurgenvinju Oct 30, 2024
af3284d
factored reusable quick-fix action triggering code from two tests as …
jurgenvinju Oct 30, 2024
8ffab57
found another use for toRascalPosition
jurgenvinju Oct 30, 2024
a6019d9
found another use for toRascalPosition
jurgenvinju Oct 30, 2024
4cfe8df
fixed unused imports
jurgenvinju Oct 30, 2024
69e5563
forgot await
jurgenvinju Oct 30, 2024
88867b7
Update utils.ts
jurgenvinju Oct 30, 2024
4d98fc0
Merge branch 'main' into code-actions-for-rascal
jurgenvinju Oct 30, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,9 @@
import org.eclipse.lsp4j.services.WorkspaceService;

public class BaseWorkspaceService implements WorkspaceService, LanguageClientAware {
public static final String RASCAL_LANGUAGE = "Rascal";
public static final String RASCAL_META_COMMAND = "rascal-meta-command";
public static final String RASCAL_COMMAND = "rascal-command";

private final IBaseTextDocumentService documentService;
private final CopyOnWriteArrayList<WorkspaceFolder> workspaceFolders = new CopyOnWriteArrayList<>();
Expand Down Expand Up @@ -108,11 +110,12 @@ public void didChangeWorkspaceFolders(DidChangeWorkspaceFoldersParams params) {

@Override
public CompletableFuture<Object> executeCommand(ExecuteCommandParams params) {
if (params.getCommand().startsWith(RASCAL_META_COMMAND)) {
if (params.getCommand().startsWith(RASCAL_META_COMMAND) || params.getCommand().startsWith(RASCAL_COMMAND)) {
jurgenvinju marked this conversation as resolved.
Show resolved Hide resolved
String languageName = ((JsonPrimitive) params.getArguments().get(0)).getAsString();
String command = ((JsonPrimitive) params.getArguments().get(1)).getAsString();
return documentService.executeCommand(languageName, command).thenApply(v -> v);
}

return CompletableFuture.supplyAsync(() -> params.getCommand() + " was ignored.");
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,4 +44,5 @@ public interface IBaseTextDocumentService extends TextDocumentService {
void unregisterLanguage(LanguageParameter lang);
CompletableFuture<IValue> executeCommand(String languageName, String command);
LineColumnOffsetMap getColumnMap(ISourceLocation file);

}
Original file line number Diff line number Diff line change
Expand Up @@ -29,11 +29,9 @@
import java.io.IOException;
import java.io.Reader;
import java.time.Duration;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.ConcurrentHashMap;
Expand All @@ -50,7 +48,6 @@
import org.checkerframework.checker.nullness.qual.MonotonicNonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.eclipse.lsp4j.CodeAction;
import org.eclipse.lsp4j.CodeActionKind;
import org.eclipse.lsp4j.CodeActionParams;
import org.eclipse.lsp4j.CodeLens;
import org.eclipse.lsp4j.CodeLensOptions;
Expand Down Expand Up @@ -90,7 +87,6 @@
import org.eclipse.lsp4j.TextDocumentItem;
import org.eclipse.lsp4j.TextDocumentSyncKind;
import org.eclipse.lsp4j.VersionedTextDocumentIdentifier;
import org.eclipse.lsp4j.WorkspaceEdit;
import org.eclipse.lsp4j.jsonrpc.ResponseErrorException;
import org.eclipse.lsp4j.jsonrpc.messages.Either;
import org.eclipse.lsp4j.jsonrpc.messages.ResponseError;
Expand All @@ -108,11 +104,10 @@
import org.rascalmpl.vscode.lsp.TextDocumentState;
import org.rascalmpl.vscode.lsp.parametric.model.ParametricFileFacts;
import org.rascalmpl.vscode.lsp.parametric.model.ParametricSummary;
import org.rascalmpl.vscode.lsp.parametric.model.RascalADTs;
import org.rascalmpl.vscode.lsp.parametric.model.ParametricSummary.SummaryLookup;
import org.rascalmpl.vscode.lsp.terminal.ITerminalIDEServer.LanguageParameter;
import org.rascalmpl.vscode.lsp.util.CodeActions;
import org.rascalmpl.vscode.lsp.util.Diagnostics;
import org.rascalmpl.vscode.lsp.util.DocumentChanges;
import org.rascalmpl.vscode.lsp.util.FoldingRanges;
import org.rascalmpl.vscode.lsp.util.DocumentSymbols;
import org.rascalmpl.vscode.lsp.util.SemanticTokenizer;
Expand All @@ -123,16 +118,13 @@
import org.rascalmpl.vscode.lsp.util.locations.Locations;
import org.rascalmpl.vscode.lsp.util.locations.impl.TreeSearch;

import com.google.gson.JsonPrimitive;

import io.usethesource.vallang.IBool;
import io.usethesource.vallang.IConstructor;
import io.usethesource.vallang.IList;
import io.usethesource.vallang.ISourceLocation;
import io.usethesource.vallang.IString;
import io.usethesource.vallang.ITuple;
import io.usethesource.vallang.IValue;
import io.usethesource.vallang.IWithKeywordParameters;
import io.usethesource.vallang.exceptions.FactParseError;

public class ParametricTextDocumentService implements IBaseTextDocumentService, LanguageClientAware {
Expand Down Expand Up @@ -386,80 +378,10 @@ private CodeLens locCommandTupleToCodeLense(String languageName, IValue v) {
ISourceLocation loc = (ISourceLocation) t.get(0);
IConstructor command = (IConstructor) t.get(1);

return new CodeLens(Locations.toRange(loc, columns), constructorToCommand(languageName, command), null);
}

private CodeAction constructorToCodeAction(String languageName, IConstructor codeAction) {
IWithKeywordParameters<?> kw = codeAction.asWithKeywordParameters();
IConstructor command = (IConstructor) kw.getParameter(RascalADTs.CodeActionFields.COMMAND);
IString title = (IString) kw.getParameter(RascalADTs.CodeActionFields.TITLE);
IList edits = (IList) kw.getParameter(RascalADTs.CodeActionFields.EDITS);
IConstructor kind = (IConstructor) kw.getParameter(RascalADTs.CodeActionFields.KIND);

// first deal with the defaults. Must mimick what's in util::LanguageServer with the `data CodeAction` declaration
if (title == null) {
if (command != null) {
title = (IString) command.asWithKeywordParameters().getParameter(RascalADTs.CommandFields.TITLE);
}

if (title == null) {
title = IRascalValueFactory.getInstance().string("");
}
}

CodeAction result = new CodeAction(title.getValue());

if (command != null) {
result.setCommand(constructorToCommand(languageName, command));
}

if (edits != null) {
result.setEdit(new WorkspaceEdit(DocumentChanges.translateDocumentChanges(this, edits)));
}

result.setKind(constructorToCodeActionKind(kind));

return result;
return new CodeLens(Locations.toRange(loc, columns), CodeActions.constructorToCommand(dedicatedLanguageName, languageName, command), null);
}

/**
* Translates `refactor(inline())` to `"refactor.inline"` and `empty()` to `""`, etc.
* `kind == null` signals absence of the optional parameter. This is factorede into
* this private function because otherwise every call has to check it.
*/
private String constructorToCodeActionKind(@Nullable IConstructor kind) {
if (kind == null) {
return CodeActionKind.QuickFix;
}

String name = kind.getName();

if (name.isEmpty()) {
return "";
}
else if (name.length() == 1) {
return name.toUpperCase();
}
else if ("empty".equals(name)) {
return "";
}
else {
var kw = kind.asWithKeywordParameters();
for (String kwn : kw.getParameterNames()) {
String nestedName = constructorToCodeActionKind((IConstructor) kw.getParameter(kwn));
name = name + (nestedName.isEmpty() ? "" : ("." + nestedName));
}
}

return name;
}

private Command constructorToCommand(String languageName, IConstructor command) {
IWithKeywordParameters<?> kw = command.asWithKeywordParameters();
IString possibleTitle = (IString) kw.getParameter(RascalADTs.CommandFields.TITLE);

return new Command(possibleTitle != null ? possibleTitle.getValue() : command.toString(), getRascalMetaCommandName(), Arrays.asList(languageName, command.toString()));
}

private void handleParsingErrors(TextDocumentState file) {
handleParsingErrors(file, file.getCurrentTreeAsync());
Expand Down Expand Up @@ -584,54 +506,30 @@ public CompletableFuture<SemanticTokens> semanticTokensRange(SemanticTokensRange

@Override
public CompletableFuture<List<Either<Command, CodeAction>>> codeAction(CodeActionParams params) {
logger.debug("codeActions: {}", params);
logger.debug("codeAction: {}", params);

final ILanguageContributions contribs = contributions(params.getTextDocument());
final var loc = Locations.toLoc(params.getTextDocument());
final var start = params.getRange().getStart();
// convert to Rascal 1-based line
final var startLine = start.getLine() + 1;
// convert to Rascal UTF-32 column width
final var startColumn = columns.get(loc).translateInverseColumn(start.getLine(), start.getCharacter(), false);
final var emptyListFuture = CompletableFuture.completedFuture(IRascalValueFactory.getInstance().list());

var range = Locations.toRascalRange(params.getTextDocument(), params.getRange(), columns);

// first we make a future stream for filtering out the "fixes" that were optionally sent along with earlier diagnostics
// and which came back with the codeAction's list of relevant (in scope) diagnostics:
// CompletableFuture<Stream<IValue>>
CompletableFuture<Stream<IValue>> quickfixes
= params.getContext().getDiagnostics()
.stream()
.map(Diagnostic::getData)
.filter(Objects::nonNull)
.filter(JsonPrimitive.class::isInstance)
.map(JsonPrimitive.class::cast)
.map(JsonPrimitive::getAsString)
// this is the "magic" resurrection of command terms from the JSON data field
.map(contribs::parseCodeActions)
// this serializes the stream of futures and accumulates their results as a flat list again
.reduce(emptyListFuture, (acc, next) -> acc.thenCombine(next, IList::concat))
.thenApply(IList::stream)
;
var quickfixes = CodeActions.extractActionsFromDiagnostics(params, contribs::parseCodeActions);

// here we dynamically ask the contributions for more actions,
// based on the cursor position in the file and the current parse tree
CompletableFuture<Stream<IValue>> codeActions = recoverExceptions(
getFile(params.getTextDocument())
.getCurrentTreeAsync()
.thenApply(Versioned::get)
.thenCompose(tree -> computeCodeActions(contribs, startLine, startColumn, tree))
.thenCompose(tree -> computeCodeActions(contribs, range.getStart().getLine(), range.getStart().getCharacter(), tree))
.thenApply(IList::stream)
, () -> Stream.<IValue>empty())
;

// final merging the two streams of commmands, and their conversion to LSP Command data-type
return codeActions.thenCombine(quickfixes, (actions, quicks) ->
Stream.concat(quicks, actions)
.map(IConstructor.class::cast)
.map(cons -> constructorToCodeAction(contribs.getName(), cons))
.map(cmd -> Either.<Command,CodeAction>forRight(cmd))
.collect(Collectors.toList())
);
return CodeActions.mergeAndConvertCodeActions(this, dedicatedLanguageName, contribs.getName(), quickfixes, codeActions);
jurgenvinju marked this conversation as resolved.
Show resolved Hide resolved
}

private CompletableFuture<IList> computeCodeActions(final ILanguageContributions contribs, final int startLine, final int startColumn, ITree tree) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -479,14 +479,13 @@ public void invalidate() {
return null;
}

var line = cursor.getLine() + 1;
var translatedOffset = columns.get(file).translateInverseColumn(line, cursor.getCharacter(), false);
var focus = TreeSearch.computeFocusList(tree.get(), line, translatedOffset);
var pos = Locations.toRascalPosition(file, cursor, columns);
var focus = TreeSearch.computeFocusList(tree.get(), pos.getLine(), pos.getCharacter());

InterruptibleFuture<ISet> set = null;

if (focus.isEmpty()) {
logger.trace("{}: could not find substree at line {} and offset {}", logName, line, translatedOffset);
logger.trace("{}: could not find substree at line {} and offset {}", logName, pos.getLine(), pos.getCharacter());
set = InterruptibleFuture.completedFuture(IRascalValueFactory.getInstance().set());
} else {
logger.trace("{}: looked up focus with length: {}, now calling dedicated function", logName, focus.length());
Expand Down
Loading
Loading