From 20dca2a0c74d77c1b3929151c28f07fb15cf27c5 Mon Sep 17 00:00:00 2001 From: Hackerpilot Date: Wed, 20 Aug 2014 18:49:44 -0700 Subject: [PATCH] Add JSON report output for the Sonar plugin --- .gitignore | 5 +- analysis/base.d | 10 +++- analysis/builtin_property_names.d | 8 +-- analysis/constructors.d | 7 +-- analysis/del.d | 3 +- analysis/duplicate_attribute.d | 2 +- analysis/enumarrayliteral.d | 7 +-- analysis/fish.d | 5 +- analysis/helpers.d | 11 ++-- analysis/ifelsesame.d | 4 ++ analysis/length_subtraction.d | 1 + analysis/numbers.d | 2 +- analysis/objectconst.d | 4 +- analysis/opequals_without_tohash.d | 8 +-- analysis/pokemon.d | 7 +-- analysis/range.d | 5 +- analysis/run.d | 82 ++++++++++++++++++++++++------ analysis/style.d | 14 ++--- analysis/unused.d | 4 +- libdparse | 2 +- main.d | 15 ++++-- sonar-project.properties | 5 ++ 22 files changed, 151 insertions(+), 60 deletions(-) create mode 100644 sonar-project.properties diff --git a/.gitignore b/.gitignore index 367f9e8..e0bd4b9 100755 --- a/.gitignore +++ b/.gitignore @@ -16,4 +16,7 @@ # D Scanner binaries dscanner -dscanner.o \ No newline at end of file +dscanner.o + +# Static analysis reports +dscanner-report.json diff --git a/analysis/base.d b/analysis/base.d index 5b62689..883433b 100644 --- a/analysis/base.d +++ b/analysis/base.d @@ -7,9 +7,15 @@ import std.array; struct Message { + /// Name of the file where the warning was triggered string fileName; + /// Line number where the warning was triggered size_t line; + /// Column number where the warning was triggered (in bytes) size_t column; + /// Name of the warning + string key; + /// Warning message string message; } @@ -47,9 +53,9 @@ protected: import core.vararg; - void addErrorMessage(size_t line, size_t column, string message) + void addErrorMessage(size_t line, size_t column, string key, string message) { - _messages.insert(Message(fileName, line, column, message)); + _messages.insert(Message(fileName, line, column, key, message)); } /** diff --git a/analysis/builtin_property_names.d b/analysis/builtin_property_names.d index cb10918..5ba02e3 100644 --- a/analysis/builtin_property_names.d +++ b/analysis/builtin_property_names.d @@ -29,6 +29,8 @@ class BuiltinPropertyNameCheck : BaseAnalyzer { alias visit = BaseAnalyzer.visit; + enum string KEY = "dscanner.confusing.builtin_property_names"; + this(string fileName) { super(fileName); @@ -38,7 +40,7 @@ class BuiltinPropertyNameCheck : BaseAnalyzer { if (depth > 0 && isBuiltinProperty(fd.name.text)) { - addErrorMessage(fd.name.line, fd.name.column, generateErrorMessage(fd.name.text)); + addErrorMessage(fd.name.line, fd.name.column, KEY, generateErrorMessage(fd.name.text)); } fd.accept(this); } @@ -48,14 +50,14 @@ class BuiltinPropertyNameCheck : BaseAnalyzer if (depth > 0) foreach (i; ad.identifiers) { if (isBuiltinProperty(i.text)) - addErrorMessage(i.line, i.column, generateErrorMessage(i.text)); + addErrorMessage(i.line, i.column, KEY, generateErrorMessage(i.text)); } } override void visit(const Declarator d) { if (depth > 0 && isBuiltinProperty(d.name.text)) - addErrorMessage(d.name.line, d.name.column, generateErrorMessage(d.name.text)); + addErrorMessage(d.name.line, d.name.column, KEY, generateErrorMessage(d.name.text)); } override void visit(const StructBody sb) diff --git a/analysis/constructors.d b/analysis/constructors.d index 66ad116..051a068 100644 --- a/analysis/constructors.d +++ b/analysis/constructors.d @@ -28,9 +28,9 @@ class ConstructorCheck : BaseAnalyzer if (hasNoArgConstructor && hasDefaultArgConstructor) { addErrorMessage(classDeclaration.name.line, - classDeclaration.name.column, "This class has a zero-argument" - ~ " constructor as well as a constructor with one default" - ~ " argument. This can be confusing."); + classDeclaration.name.column, "dscanner.confusing.constructor_args", + "This class has a zero-argument constructor as well as a" + ~ " constructor with one default argument. This can be confusing."); } hasDefaultArgConstructor = oldHasDefault; hasNoArgConstructor = oldHasNoArg; @@ -54,6 +54,7 @@ class ConstructorCheck : BaseAnalyzer && constructor.parameters.parameters[0].default_ !is null) { addErrorMessage(constructor.line, constructor.column, + "dscanner.confusing.struct_constructor_default_args", "This struct constructor can never be called with its " ~ "default argument."); } diff --git a/analysis/del.d b/analysis/del.d index 223c121..6613220 100644 --- a/analysis/del.d +++ b/analysis/del.d @@ -25,7 +25,8 @@ class DeleteCheck : BaseAnalyzer override void visit(const DeleteExpression d) { - addErrorMessage(d.line, d.column, "Avoid using the 'delete' keyword."); + addErrorMessage(d.line, d.column, "dscanner.deprecated.delete_keyword", + "Avoid using the 'delete' keyword."); d.accept(this); } } diff --git a/analysis/duplicate_attribute.d b/analysis/duplicate_attribute.d index 0f1ee32..d91fa81 100644 --- a/analysis/duplicate_attribute.d +++ b/analysis/duplicate_attribute.d @@ -91,7 +91,7 @@ class DuplicateAttributeCheck : BaseAnalyzer if (hasAttribute) { string message = "Attribute '%s' is duplicated.".format(attributeName); - addErrorMessage(line, column, message); + addErrorMessage(line, column, "dscanner.unnecessary.duplicate_attribute", message); } // Mark it as having that attribute diff --git a/analysis/enumarrayliteral.d b/analysis/enumarrayliteral.d index 2b3c980..545851a 100644 --- a/analysis/enumarrayliteral.d +++ b/analysis/enumarrayliteral.d @@ -51,9 +51,10 @@ class EnumArrayLiteralCheck : BaseAnalyzer if (initializer.nonVoidInitializer is null) continue; if (initializer.nonVoidInitializer.arrayInitializer is null) continue; addErrorMessage(autoDec.identifiers[i].line, - autoDec.identifiers[i].column, "This enum may lead to " - ~ "unnecessary allocation at run-time. Use 'static immutable " - ~ autoDec.identifiers[i].text ~ " = [ ...' instead."); + autoDec.identifiers[i].column, "dscanner.performance.enum_array_literal", + "This enum may lead to unnecessary allocation at run-time." + ~ " Use 'static immutable " ~ autoDec.identifiers[i].text + ~ " = [ ...' instead."); } } autoDec.accept(this); diff --git a/analysis/fish.d b/analysis/fish.d index 3d998bc..fe39afc 100644 --- a/analysis/fish.d +++ b/analysis/fish.d @@ -18,6 +18,8 @@ class FloatOperatorCheck : BaseAnalyzer { alias visit = BaseAnalyzer.visit; + enum string KEY = "dscanner.deprecated.floating_point_operators"; + this(string fileName) { super(fileName); @@ -34,7 +36,8 @@ class FloatOperatorCheck : BaseAnalyzer || r.operator == tok!"!>=" || r.operator == tok!"!<=") { - addErrorMessage(r.line, r.column, "Avoid using the deprecated floating-point operators."); + addErrorMessage(r.line, r.column, KEY, + "Avoid using the deprecated floating-point operators."); } r.accept(this); } diff --git a/analysis/helpers.d b/analysis/helpers.d index 87587c4..b5d0033 100644 --- a/analysis/helpers.d +++ b/analysis/helpers.d @@ -12,6 +12,7 @@ import std.stdio; import std.d.ast; import analysis.config; import analysis.run; +import analysis.base; S between(S)(S value, S before, S after) @@ -54,23 +55,23 @@ void assertAnalyzerWarnings(string code, const StaticAnalysisConfig config, stri import analysis.run; // Run the code and get any warnings - string[] rawWarnings = analyze("test", cast(ubyte[]) code, config); + MessageSet rawWarnings = analyze("test", cast(ubyte[]) code, config); string[] codeLines = code.split("\n"); // Get the warnings ordered by line string[size_t] warnings; - for (size_t i=0; i 0) - output.writeln(results.join("\n")); + MessageSet results = analyze(fileName, code, config, staticAnalyze, report); + if (report) + { + foreach (result; results[]) + { + writeJSON(result.key, result.fileName, result.line, result.column, result.message); + } + } + else + { + foreach (result; results[]) + writefln("%s(%d:%d)[warn]: %s", result.fileName, result.line, + result.column, result.message); + } + } + + if (report) + { + writeln(); + writeln(" ]"); + writeln("}"); } } // For a string -string[] analyze(string fileName, ubyte[] code, const StaticAnalysisConfig analysisConfig, bool staticAnalyze = true) +MessageSet analyze(string fileName, ubyte[] code, const StaticAnalysisConfig analysisConfig, + bool staticAnalyze = true, bool report = false) { import std.parallelism; @@ -72,12 +123,16 @@ string[] analyze(string fileName, ubyte[] code, const StaticAnalysisConfig analy foreach (message; lexer.messages) { - messageFunction(fileName, message.line, message.column, message.message, - message.isError); + if (report) + messageFunctionJSON(fileName, message.line, message.column, message.message, + message.isError); + else + messageFunction(fileName, message.line, message.column, message.message, + message.isError); } ParseAllocator p = new ParseAllocator; - Module m = parseModule(tokens, fileName, p, &messageFunction); + Module m = parseModule(tokens, fileName, p, report ? &messageFunctionJSON : &messageFunction); if (!staticAnalyze) return null; @@ -110,12 +165,7 @@ string[] analyze(string fileName, ubyte[] code, const StaticAnalysisConfig analy foreach (check; checks) foreach (message; check.messages) set.insert(message); - - string[] results; - foreach (message; set[]) - results ~= "%s(%d:%d)[warn]: %s".format(message.fileName, message.line, - message.column, message.message); p.deallocateAll(); - return results; + return set; } diff --git a/analysis/style.d b/analysis/style.d index ca62883..21ca59e 100644 --- a/analysis/style.d +++ b/analysis/style.d @@ -20,10 +20,10 @@ class StyleChecker : BaseAnalyzer { alias visit = ASTVisitor.visit; - // FIXME: All variable and function names seem to never match this - enum varFunNameRegex = `^([\p{Ll}_][_\w\d]*|[\p{Lu}\d_]+)$`; - enum aggregateNameRegex = `^\p{Lu}[\w\d]*$`; - enum moduleNameRegex = `^[\p{Ll}_\d]+$`; + enum string varFunNameRegex = `^([\p{Ll}_][_\w\d]*|[\p{Lu}\d_]+)$`; + enum string aggregateNameRegex = `^\p{Lu}[\w\d]*$`; + enum string moduleNameRegex = `^[\p{Ll}_\d]+$`; + enum string KEY = "dscanner.style.phobos_naming_convention"; this(string fileName) { @@ -35,7 +35,7 @@ class StyleChecker : BaseAnalyzer foreach (part; dec.moduleName.identifiers) { if (part.text.matchFirst(moduleNameRegex).length == 0) - addErrorMessage(part.line, part.column, "Module/package name '" + addErrorMessage(part.line, part.column, KEY, "Module/package name '" ~ part.text ~ "' does not match style guidelines."); } } @@ -53,7 +53,7 @@ class StyleChecker : BaseAnalyzer void checkLowercaseName(string type, ref const Token name) { if (name.text.matchFirst(varFunNameRegex).length == 0) - addErrorMessage(name.line, name.column, type ~ " name '" + addErrorMessage(name.line, name.column, KEY, type ~ " name '" ~ name.text ~ "' does not match style guidelines."); } @@ -86,7 +86,7 @@ class StyleChecker : BaseAnalyzer void checkAggregateName(string aggregateType, ref const Token name) { if (name.text.matchFirst(aggregateNameRegex).length == 0) - addErrorMessage(name.line, name.column, aggregateType + addErrorMessage(name.line, name.column, KEY, aggregateType ~ " name '" ~ name.text ~ "' does not match style guidelines."); } } diff --git a/analysis/unused.d b/analysis/unused.d index 36c80fa..d642cfb 100644 --- a/analysis/unused.d +++ b/analysis/unused.d @@ -323,8 +323,10 @@ class UnusedVariableCheck : BaseAnalyzer { if (!uu.isRef && tree.length > 1) addErrorMessage(uu.line, uu.column, + uu.isParameter ? "dscanner.suspicious.unused_parameter" + : "dscanner.suspicious.unused_variable", (uu.isParameter ? "Parameter " : "Variable ") - ~ uu.name ~ " is never used."); + ~ uu.name ~ " is never used."); } tree = tree[0 .. $ - 1]; } diff --git a/libdparse b/libdparse index f693be2..4feb92c 160000 --- a/libdparse +++ b/libdparse @@ -1 +1 @@ -Subproject commit f693be2b2c0fb5b015493c7441e14b888647ef0d +Subproject commit 4feb92c544a9e09fcc56051d962b669327cee386 diff --git a/main.d b/main.d index 9358021..839bc79 100644 --- a/main.d +++ b/main.d @@ -56,6 +56,7 @@ int run(string[] args) bool tokenDump; bool styleCheck; bool defaultConfig; + bool report; string symbolName; string configLocation; @@ -67,7 +68,7 @@ int run(string[] args) "ast|xml", &ast, "imports|i", &imports, "outline|o", &outline, "tokenDump", &tokenDump, "styleCheck|S", &styleCheck, "defaultConfig", &defaultConfig, "declaration|d", &symbolName, - "config", &configLocation, "muffinButton", &muffin); + "config", &configLocation, "report", &report,"muffinButton", &muffin); } catch (ConvException e) { @@ -98,7 +99,7 @@ int run(string[] args) auto optionCount = count!"a"([sloc, highlight, ctags, tokenCount, syntaxCheck, ast, imports, outline, tokenDump, styleCheck, defaultConfig, - symbolName !is null]); + report, symbolName !is null]); if (optionCount > 1) { stderr.writeln("Too many options specified"); @@ -110,6 +111,9 @@ int run(string[] args) return 1; } + // --report implies --styleCheck + if (report) styleCheck = true; + StringCache cache = StringCache(StringCache.defaultBucketCount); if (defaultConfig) { @@ -156,11 +160,11 @@ int run(string[] args) string s = configLocation is null ? getConfigurationLocation() : configLocation; if (s.exists()) readINIFile(config, s); - stdout.analyze(expandArgs(args), config); + analyze(expandArgs(args), config, true, report); } else if (syntaxCheck) { - stdout.syntaxCheck(expandArgs(args)); + .syntaxCheck(expandArgs(args)); } else { @@ -342,6 +346,9 @@ options: accurate than "grep". Searches the given files and directories, or the current working directory if none are specified. + --report [sourceFiles sourceDirectories] + Generate a static analysis report in JSON format. Implies --styleCheck. + --config configFile Use the given configuration file instead of the default located in $HOME/.config/dscanner/dscanner.ini diff --git a/sonar-project.properties b/sonar-project.properties new file mode 100644 index 0000000..05a21f5 --- /dev/null +++ b/sonar-project.properties @@ -0,0 +1,5 @@ +sonar.projectKey=dscanner +sonar.projectName=D Scanner +sonar.projectVersion=1.0 +sonar.sourceEncoding=UTF-8 +sonar.sources=.