Merge pull request #804 from andre2007/reportFile

--reportFile added
This commit is contained in:
Brian Schott 2020-04-14 12:31:48 -07:00 committed by GitHub
commit 6d91031302
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 123 additions and 26 deletions

View File

@ -14,6 +14,8 @@ import std.algorithm;
import std.range; import std.range;
import std.array; import std.array;
import std.functional : toDelegate; import std.functional : toDelegate;
import std.file : mkdirRecurse;
import std.path : dirName;
import dparse.lexer; import dparse.lexer;
import dparse.parser; import dparse.parser;
import dparse.ast; import dparse.ast;
@ -87,7 +89,7 @@ import dsymbol.conversion.second;
import dsymbol.modulecache : ModuleCache; import dsymbol.modulecache : ModuleCache;
import dscanner.utils; import dscanner.utils;
import dscanner.reports : SonarQubeGenericIssueDataReporter; import dscanner.reports : DScannerJsonReporter, SonarQubeGenericIssueDataReporter;
bool first = true; bool first = true;
@ -146,10 +148,14 @@ bool syntaxCheck(string[] fileNames, string errorFormat, ref StringCache stringC
} }
void generateReport(string[] fileNames, const StaticAnalysisConfig config, void generateReport(string[] fileNames, const StaticAnalysisConfig config,
ref StringCache cache, ref ModuleCache moduleCache) ref StringCache cache, ref ModuleCache moduleCache, string reportFile = "")
{ {
writeln("{"); auto reporter = new DScannerJsonReporter();
writeln(` "issues": [`);
auto writeMessages = delegate void(string fileName, size_t line, size_t column, string message, bool isError){
reporter.addMessage(Message(fileName, line, column, "dscanner.syntax", message), isError);
};
first = true; first = true;
StatsCollector stats = new StatsCollector(""); StatsCollector stats = new StatsCollector("");
ulong lineOfCodeCount; ulong lineOfCodeCount;
@ -161,29 +167,26 @@ void generateReport(string[] fileNames, const StaticAnalysisConfig config,
continue; continue;
RollbackAllocator r; RollbackAllocator r;
const(Token)[] tokens; const(Token)[] tokens;
const Module m = parseModule(fileName, code, &r, defaultErrorFormat, cache, true, tokens, &lineOfCodeCount); const Module m = parseModule(fileName, code, &r, cache, tokens, writeMessages, &lineOfCodeCount, null, null);
stats.visit(m); stats.visit(m);
MessageSet results = analyze(fileName, m, config, moduleCache, tokens, true); MessageSet messageSet = analyze(fileName, m, config, moduleCache, tokens, true);
foreach (result; results[]) reporter.addMessageSet(messageSet);
{ }
writeJSON(result);
} string reportFileContent = reporter.getContent(stats, lineOfCodeCount);
if (reportFile == "")
{
writeln(reportFileContent);
}
else
{
mkdirRecurse(reportFile.dirName);
toFile(reportFileContent, reportFile);
} }
writeln();
writeln(" ],");
writefln(` "interfaceCount": %d,`, stats.interfaceCount);
writefln(` "classCount": %d,`, stats.classCount);
writefln(` "functionCount": %d,`, stats.functionCount);
writefln(` "templateCount": %d,`, stats.templateCount);
writefln(` "structCount": %d,`, stats.structCount);
writefln(` "statementCount": %d,`, stats.statementCount);
writefln(` "lineOfCodeCount": %d,`, lineOfCodeCount);
writefln(` "undocumentedPublicSymbols": %d`, stats.undocumentedPublicSymbols);
writeln("}");
} }
void generateSonarQubeGenericIssueDataReport(string[] fileNames, const StaticAnalysisConfig config, void generateSonarQubeGenericIssueDataReport(string[] fileNames, const StaticAnalysisConfig config,
ref StringCache cache, ref ModuleCache moduleCache) ref StringCache cache, ref ModuleCache moduleCache, string reportFile = "")
{ {
auto reporter = new SonarQubeGenericIssueDataReporter(); auto reporter = new SonarQubeGenericIssueDataReporter();
@ -204,7 +207,16 @@ void generateSonarQubeGenericIssueDataReport(string[] fileNames, const StaticAna
reporter.addMessageSet(messageSet); reporter.addMessageSet(messageSet);
} }
writeln(reporter.getContent()); string reportFileContent = reporter.getContent();
if (reportFile == "")
{
writeln(reportFileContent);
}
else
{
mkdirRecurse(reportFile.dirName);
toFile(reportFileContent, reportFile);
}
} }
/** /**

View File

@ -63,6 +63,7 @@ else
bool report; bool report;
bool skipTests; bool skipTests;
string reportFormat; string reportFormat;
string reportFile;
string symbolName; string symbolName;
string configLocation; string configLocation;
string[] importPaths; string[] importPaths;
@ -93,6 +94,7 @@ else
"config", &configLocation, "config", &configLocation,
"report", &report, "report", &report,
"reportFormat", &reportFormat, "reportFormat", &reportFormat,
"reportFile", &reportFile,
"I", &importPaths, "I", &importPaths,
"version", &printVersion, "version", &printVersion,
"muffinButton", &muffin, "muffinButton", &muffin,
@ -157,7 +159,7 @@ else
if (absImportPaths.length) if (absImportPaths.length)
moduleCache.addImportPaths(absImportPaths); moduleCache.addImportPaths(absImportPaths);
if (reportFormat.length) if (reportFormat.length || reportFile.length)
report = true; report = true;
immutable optionCount = count!"a"([sloc, highlight, ctags, tokenCount, syntaxCheck, ast, imports, immutable optionCount = count!"a"([sloc, highlight, ctags, tokenCount, syntaxCheck, ast, imports,
@ -250,10 +252,10 @@ else
goto case; goto case;
case "": case "":
case "dscanner": case "dscanner":
generateReport(expandArgs(args), config, cache, moduleCache); generateReport(expandArgs(args), config, cache, moduleCache, reportFile);
break; break;
case "sonarQubeGenericIssueData": case "sonarQubeGenericIssueData":
generateSonarQubeGenericIssueDataReport(expandArgs(args), config, cache, moduleCache); generateSonarQubeGenericIssueDataReport(expandArgs(args), config, cache, moduleCache, reportFile);
break; break;
} }
} }
@ -407,6 +409,9 @@ Options:
however the exit code will still be zero if errors or warnings are however the exit code will still be zero if errors or warnings are
found. found.
--reportFile <file>
Write report into file instead of STDOUT.
--reportFormat <dscanner | sonarQubeGenericIssueData>... --reportFormat <dscanner | sonarQubeGenericIssueData>...
Specifies the format of the generated report. Specifies the format of the generated report.

View File

@ -10,6 +10,86 @@ import std.algorithm : map;
import std.array : split, array, Appender, appender; import std.array : split, array, Appender, appender;
import dscanner.analysis.base : Message, MessageSet; import dscanner.analysis.base : Message, MessageSet;
import dscanner.analysis.stats_collector;
class DScannerJsonReporter
{
struct Issue
{
Message message;
string type;
}
private Appender!(Issue[]) _issues;
this()
{
_issues = appender!(Issue[]);
}
void addMessageSet(MessageSet messageSet)
{
_issues ~= toIssues(messageSet);
}
void addMessage(Message message, bool isError = false)
{
_issues ~= toIssue(message, isError);
}
string getContent(StatsCollector stats, ulong lineOfCodeCount)
{
JSONValue result = [
"issues" : JSONValue(_issues.data.map!(e => toJson(e)).array),
"interfaceCount": JSONValue(stats.interfaceCount),
"classCount": JSONValue(stats.classCount),
"functionCount": JSONValue(stats.functionCount),
"templateCount": JSONValue(stats.templateCount),
"structCount": JSONValue(stats.structCount),
"statementCount": JSONValue(stats.statementCount),
"lineOfCodeCount": JSONValue(lineOfCodeCount),
"undocumentedPublicSymbols": JSONValue(stats.undocumentedPublicSymbols)
];
return result.toPrettyString();
}
private static JSONValue toJson(Issue issue)
{
// dfmt off
JSONValue js = JSONValue([
"key": JSONValue(issue.message.key),
"fileName": JSONValue(issue.message.fileName),
"line": JSONValue(issue.message.line),
"column": JSONValue(issue.message.column),
"message": JSONValue(issue.message.message),
"type": JSONValue(issue.type)
]);
// dfmt on
if (issue.message.checkName !is null)
{
js["name"] = JSONValue(issue.message.checkName);
}
return js;
}
private static Issue[] toIssues(MessageSet messageSet)
{
return messageSet[].map!(e => toIssue(e)).array;
}
private static Issue toIssue(Message message, bool isError = false)
{
// dfmt off
Issue issue = {
message: message,
type : isError ? "error" : "warn"
};
// dfmt on
return issue;
}
}
class SonarQubeGenericIssueDataReporter class SonarQubeGenericIssueDataReporter
{ {