add CLI API to get autofixes at location

This commit is contained in:
WebFreak001 2023-07-08 22:26:59 +02:00 committed by Jan Jurzitza
parent 53c9536332
commit 43caad72a8
3 changed files with 139 additions and 6 deletions

View File

@ -101,6 +101,48 @@ dscanner -S -f github source/
dscanner -S -f '{filepath}({line}:{column})[{type}]: {message}' source/
```
To collect automatic issue fixes for a given location use
```sh
# collecting automatic issue fixes
# --resolveMessage <line>:<column> <filename>
dscanner --resolveMessage 11:3 file.d
# --resolveMessage b<byteIndex> <filename>
dscanner --resolveMessage b512 file.d
# <filename> may be omitted to read from stdin
```
outputs JSON:
```json
// list of available auto-fixes at the given location
[
{
"name": "Make function const",
// byte range `[start, end)` what code to replace
// this is sorted by range[0]
"replacements": [
// replace: range[0] < range[1], newText != ""
{"range": [10, 14], "newText": "const "},
// insert: range[0] == range[1], newText != ""
{"range": [20, 20], "newText": "auto"},
// remove: range[0] < range[1], newText == ""
{"range": [30, 40], "newText": ""},
]
}
]
```
Algorithm to apply replacements:
```d
foreach_reverse (r; replacements)
codeBytes = codeBytes[0 .. r.range[0]] ~ r.newText ~ codeBytes[r.range[1] .. $];
```
Replacements are non-overlapping, sorted by `range[0]` in ascending order. When
combining multiple different replacements, you first need to sort them by
`range[0]` to apply using the algorithm above.
## Other features
### Token Count

View File

@ -480,6 +480,90 @@ bool autofix(string[] fileNames, const StaticAnalysisConfig config, string error
return hasErrors;
}
void listAutofixes(
StaticAnalysisConfig config,
string resolveMessage,
bool usingStdin,
string fileName,
StringCache* cache,
ref ModuleCache moduleCache
)
{
import dparse.parser : parseModule;
import dscanner.analysis.base : Message;
import std.format : format;
import std.json : JSONValue;
union RequestedLocation
{
struct
{
uint line, column;
}
ulong bytes;
}
RequestedLocation req;
bool isBytes = resolveMessage[0] == 'b';
if (isBytes)
req.bytes = resolveMessage[1 .. $].to!ulong;
else
{
auto parts = resolveMessage.findSplit(":");
req.line = parts[0].to!uint;
req.column = parts[2].to!uint;
}
bool matchesCursor(Message m)
{
return isBytes
? req.bytes >= m.startIndex && req.bytes <= m.endIndex
: req.line >= m.startLine && req.line <= m.endLine
&& (req.line > m.startLine || req.column >= m.startColumn)
&& (req.line < m.endLine || req.column <= m.endColumn);
}
RollbackAllocator rba;
LexerConfig lexerConfig;
lexerConfig.fileName = fileName;
lexerConfig.stringBehavior = StringBehavior.source;
auto tokens = getTokensForParser(usingStdin ? readStdin()
: readFile(fileName), lexerConfig, cache);
auto mod = parseModule(tokens, fileName, &rba, toDelegate(&doNothing));
auto messages = analyze(fileName, mod, config, moduleCache, tokens);
with (stdout.lockingTextWriter)
{
put("[");
foreach (message; messages[].filter!matchesCursor)
{
resolveAutoFixes(message, fileName, moduleCache, tokens, mod, config);
foreach (i, autofix; message.autofixes)
{
put(i == 0 ? "\n" : ",\n");
put("\t{\n");
put(format!"\t\t\"name\": %s,\n"(JSONValue(autofix.name)));
put("\t\t\"replacements\": [");
foreach (j, replacement; autofix.expectReplacements)
{
put(j == 0 ? "\n" : ",\n");
put(format!"\t\t\t{\"range\": [%d, %d], \"newText\": %s}"(
replacement.range[0],
replacement.range[1],
JSONValue(replacement.newText)));
}
put("\n");
put("\t\t]\n");
put("\t}");
}
}
put("\n]");
}
stdout.flush();
}
private struct UserSelect
{
import std.string : strip;
@ -942,9 +1026,8 @@ private void resolveMessageFromCheck(
}
}
AutoFix.CodeReplacement[] resolveAutoFix(const Message message,
const AutoFix.ResolveContext resolve, string fileName,
ref ModuleCache moduleCache, scope const(char)[] rawCode,
void resolveAutoFixes(ref Message message, string fileName,
ref ModuleCache moduleCache,
scope const(Token)[] tokens, const Module m,
const StaticAnalysisConfig analysisConfig,
const AutoFixFormatting overrideFormattingConfig = AutoFixFormatting.invalid)
@ -973,7 +1056,8 @@ AutoFix.CodeReplacement[] resolveAutoFix(const Message message,
{
if (check.getName() == message.checkName)
{
return check.resolveAutoFix(m, tokens, message, resolve, formattingConfig);
resolveMessageFromCheck(message, check, m, tokens, formattingConfig);
return;
}
}

View File

@ -64,6 +64,7 @@ else
bool report;
bool skipTests;
bool applySingleFixes;
string resolveMessage;
string reportFormat;
string reportFile;
string symbolName;
@ -98,6 +99,7 @@ else
"report", &report,
"reportFormat", &reportFormat,
"reportFile", &reportFile,
"resolveMessage", &resolveMessage,
"applySingle", &applySingleFixes,
"I", &importPaths,
"version", &printVersion,
@ -213,7 +215,7 @@ else
immutable optionCount = count!"a"([sloc, highlight, ctags, tokenCount,
syntaxCheck, ast, imports, outline, tokenDump, styleCheck,
defaultConfig, report, autofix,
defaultConfig, report, autofix, resolveMessage.length,
symbolName !is null, etags, etagsAll, recursiveImports,
]);
if (optionCount > 1)
@ -286,7 +288,7 @@ else
{
stdout.printEtags(etagsAll, expandArgs(args));
}
else if (styleCheck || autofix)
else if (styleCheck || autofix || resolveMessage.length)
{
StaticAnalysisConfig config = defaultStaticAnalysisConfig();
string s = configLocation is null ? getConfigurationLocation() : configLocation;
@ -299,6 +301,11 @@ else
{
return .autofix(expandArgs(args), config, errorFormat, cache, moduleCache, applySingleFixes) ? 1 : 0;
}
else if (resolveMessage.length)
{
listAutofixes(config, resolveMessage, usingStdin, usingStdin ? "stdin" : args[1], &cache, moduleCache);
return 0;
}
else if (report)
{
switch (reportFormat)