Allow skipping checks with @("nolint(...)") and @nolint("...") (#936)
Co-authored-by: Axel Ricard <contact@axelricard.fr> Co-authored-by: WebFreak001 <gh@webfreak.org>
This commit is contained in:
parent
69d824f4f7
commit
1e8f1ec9e6
|
@ -1,7 +1,8 @@
|
||||||
module dscanner.analysis.base;
|
module dscanner.analysis.base;
|
||||||
|
|
||||||
import dparse.ast;
|
import dparse.ast;
|
||||||
import dparse.lexer : IdType, str, Token;
|
import dparse.lexer : IdType, str, Token, tok;
|
||||||
|
import dscanner.analysis.nolint;
|
||||||
import dsymbol.scope_ : Scope;
|
import dsymbol.scope_ : Scope;
|
||||||
import std.array;
|
import std.array;
|
||||||
import std.container;
|
import std.container;
|
||||||
|
@ -405,6 +406,35 @@ public:
|
||||||
unittest_.accept(this);
|
unittest_.accept(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Visits a module declaration.
|
||||||
|
*
|
||||||
|
* When overriden, make sure to keep this structure
|
||||||
|
*/
|
||||||
|
override void visit(const(Module) mod)
|
||||||
|
{
|
||||||
|
if (mod.moduleDeclaration !is null)
|
||||||
|
{
|
||||||
|
with (noLint.push(NoLintFactory.fromModuleDeclaration(mod.moduleDeclaration)))
|
||||||
|
mod.accept(this);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mod.accept(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Visits a declaration.
|
||||||
|
*
|
||||||
|
* When overriden, make sure to disable and reenable error messages
|
||||||
|
*/
|
||||||
|
override void visit(const(Declaration) decl)
|
||||||
|
{
|
||||||
|
with (noLint.push(NoLintFactory.fromDeclaration(decl)))
|
||||||
|
decl.accept(this);
|
||||||
|
}
|
||||||
|
|
||||||
AutoFix.CodeReplacement[] resolveAutoFix(
|
AutoFix.CodeReplacement[] resolveAutoFix(
|
||||||
const Module mod,
|
const Module mod,
|
||||||
scope const(Token)[] tokens,
|
scope const(Token)[] tokens,
|
||||||
|
@ -423,6 +453,7 @@ protected:
|
||||||
|
|
||||||
bool inAggregate;
|
bool inAggregate;
|
||||||
bool skipTests;
|
bool skipTests;
|
||||||
|
NoLint noLint;
|
||||||
|
|
||||||
template visitTemplate(T)
|
template visitTemplate(T)
|
||||||
{
|
{
|
||||||
|
@ -437,42 +468,58 @@ protected:
|
||||||
deprecated("Use the overload taking start and end locations or a Node instead")
|
deprecated("Use the overload taking start and end locations or a Node instead")
|
||||||
void addErrorMessage(size_t line, size_t column, string key, string message)
|
void addErrorMessage(size_t line, size_t column, string key, string message)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
_messages.insert(Message(fileName, line, column, key, message, getName()));
|
_messages.insert(Message(fileName, line, column, key, message, getName()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(const BaseNode node, string key, string message, AutoFix[] autofixes = null)
|
void addErrorMessage(const BaseNode node, string key, string message, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
addErrorMessage(Message.Diagnostic.from(fileName, node, message), key, autofixes);
|
addErrorMessage(Message.Diagnostic.from(fileName, node, message), key, autofixes);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(const Token token, string key, string message, AutoFix[] autofixes = null)
|
void addErrorMessage(const Token token, string key, string message, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
addErrorMessage(Message.Diagnostic.from(fileName, token, message), key, autofixes);
|
addErrorMessage(Message.Diagnostic.from(fileName, token, message), key, autofixes);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(const Token[] tokens, string key, string message, AutoFix[] autofixes = null)
|
void addErrorMessage(const Token[] tokens, string key, string message, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
addErrorMessage(Message.Diagnostic.from(fileName, tokens, message), key, autofixes);
|
addErrorMessage(Message.Diagnostic.from(fileName, tokens, message), key, autofixes);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(size_t[2] index, size_t line, size_t[2] columns, string key, string message, AutoFix[] autofixes = null)
|
void addErrorMessage(size_t[2] index, size_t line, size_t[2] columns, string key, string message, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
addErrorMessage(index, [line, line], columns, key, message, autofixes);
|
addErrorMessage(index, [line, line], columns, key, message, autofixes);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(size_t[2] index, size_t[2] lines, size_t[2] columns, string key, string message, AutoFix[] autofixes = null)
|
void addErrorMessage(size_t[2] index, size_t[2] lines, size_t[2] columns, string key, string message, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
auto d = Message.Diagnostic.from(fileName, index, lines, columns, message);
|
auto d = Message.Diagnostic.from(fileName, index, lines, columns, message);
|
||||||
_messages.insert(Message(d, key, getName(), autofixes));
|
_messages.insert(Message(d, key, getName(), autofixes));
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(Message.Diagnostic diagnostic, string key, AutoFix[] autofixes = null)
|
void addErrorMessage(Message.Diagnostic diagnostic, string key, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
_messages.insert(Message(diagnostic, key, getName(), autofixes));
|
_messages.insert(Message(diagnostic, key, getName(), autofixes));
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(Message.Diagnostic diagnostic, Message.Diagnostic[] supplemental, string key, AutoFix[] autofixes = null)
|
void addErrorMessage(Message.Diagnostic diagnostic, Message.Diagnostic[] supplemental, string key, AutoFix[] autofixes = null)
|
||||||
{
|
{
|
||||||
|
if (noLint.containsCheck(key))
|
||||||
|
return;
|
||||||
_messages.insert(Message(diagnostic, supplemental, key, getName(), autofixes));
|
_messages.insert(Message(diagnostic, supplemental, key, getName(), autofixes));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,271 @@
|
||||||
|
module dscanner.analysis.nolint;
|
||||||
|
|
||||||
|
@safe:
|
||||||
|
|
||||||
|
import dparse.ast;
|
||||||
|
import dparse.lexer;
|
||||||
|
|
||||||
|
import std.algorithm : canFind;
|
||||||
|
import std.regex : matchAll, regex;
|
||||||
|
import std.string : lastIndexOf, strip;
|
||||||
|
import std.typecons;
|
||||||
|
|
||||||
|
struct NoLint
|
||||||
|
{
|
||||||
|
bool containsCheck(scope const(char)[] check) const
|
||||||
|
{
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
if (disabledChecks.get((() @trusted => cast(string) check)(), 0) > 0)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
auto dot = check.lastIndexOf('.');
|
||||||
|
if (dot == -1)
|
||||||
|
break;
|
||||||
|
check = check[0 .. dot];
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// automatic pop when returned value goes out of scope
|
||||||
|
Poppable push(in Nullable!NoLint other) scope
|
||||||
|
{
|
||||||
|
if (other.isNull)
|
||||||
|
return Poppable(null);
|
||||||
|
|
||||||
|
foreach (key, value; other.get.getDisabledChecks)
|
||||||
|
this.disabledChecks[key] += value;
|
||||||
|
|
||||||
|
return Poppable(() => this.pop(other));
|
||||||
|
}
|
||||||
|
|
||||||
|
package:
|
||||||
|
const(int[string]) getDisabledChecks() const
|
||||||
|
{
|
||||||
|
return this.disabledChecks;
|
||||||
|
}
|
||||||
|
|
||||||
|
void pushCheck(in string check)
|
||||||
|
{
|
||||||
|
disabledChecks[check]++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void merge(in Nullable!NoLint other)
|
||||||
|
{
|
||||||
|
if (other.isNull)
|
||||||
|
return;
|
||||||
|
|
||||||
|
foreach (key, value; other.get.getDisabledChecks)
|
||||||
|
this.disabledChecks[key] += value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void pop(in Nullable!NoLint other)
|
||||||
|
{
|
||||||
|
if (other.isNull)
|
||||||
|
return;
|
||||||
|
|
||||||
|
foreach (key, value; other.get.getDisabledChecks)
|
||||||
|
{
|
||||||
|
assert(this.disabledChecks.get(key, 0) >= value);
|
||||||
|
|
||||||
|
this.disabledChecks[key] -= value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct Poppable
|
||||||
|
{
|
||||||
|
~this()
|
||||||
|
{
|
||||||
|
if (onPop)
|
||||||
|
onPop();
|
||||||
|
onPop = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void delegate() onPop;
|
||||||
|
}
|
||||||
|
|
||||||
|
int[string] disabledChecks;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct NoLintFactory
|
||||||
|
{
|
||||||
|
static Nullable!NoLint fromModuleDeclaration(in ModuleDeclaration moduleDeclaration)
|
||||||
|
{
|
||||||
|
NoLint noLint;
|
||||||
|
|
||||||
|
foreach (atAttribute; moduleDeclaration.atAttributes)
|
||||||
|
noLint.merge(NoLintFactory.fromAtAttribute(atAttribute));
|
||||||
|
|
||||||
|
if (!noLint.getDisabledChecks.length)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return noLint.nullable;
|
||||||
|
}
|
||||||
|
|
||||||
|
static Nullable!NoLint fromDeclaration(in Declaration declaration)
|
||||||
|
{
|
||||||
|
NoLint noLint;
|
||||||
|
foreach (attribute; declaration.attributes)
|
||||||
|
noLint.merge(NoLintFactory.fromAttribute(attribute));
|
||||||
|
|
||||||
|
if (!noLint.getDisabledChecks.length)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return noLint.nullable;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static Nullable!NoLint fromAttribute(const(Attribute) attribute)
|
||||||
|
{
|
||||||
|
if (attribute is null)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return NoLintFactory.fromAtAttribute(attribute.atAttribute);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static Nullable!NoLint fromAtAttribute(const(AtAttribute) atAttribute)
|
||||||
|
{
|
||||||
|
if (atAttribute is null)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
auto ident = atAttribute.identifier;
|
||||||
|
auto argumentList = atAttribute.argumentList;
|
||||||
|
|
||||||
|
if (argumentList !is null)
|
||||||
|
{
|
||||||
|
if (ident.text.length)
|
||||||
|
return NoLintFactory.fromStructUda(ident, argumentList);
|
||||||
|
else
|
||||||
|
return NoLintFactory.fromStringUda(argumentList);
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return nullNoLint;
|
||||||
|
}
|
||||||
|
|
||||||
|
// @nolint("..")
|
||||||
|
static Nullable!NoLint fromStructUda(in Token ident, in ArgumentList argumentList)
|
||||||
|
in (ident.text.length && argumentList !is null)
|
||||||
|
{
|
||||||
|
if (ident.text != "nolint")
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
NoLint noLint;
|
||||||
|
|
||||||
|
foreach (nodeExpr; argumentList.items)
|
||||||
|
{
|
||||||
|
if (auto unaryExpr = cast(const UnaryExpression) nodeExpr)
|
||||||
|
{
|
||||||
|
auto primaryExpression = unaryExpr.primaryExpression;
|
||||||
|
if (primaryExpression is null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (primaryExpression.primary != tok!"stringLiteral")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
noLint.pushCheck(primaryExpression.primary.text.strip("\""));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!noLint.getDisabledChecks().length)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return noLint.nullable;
|
||||||
|
}
|
||||||
|
|
||||||
|
// @("nolint(..)")
|
||||||
|
static Nullable!NoLint fromStringUda(in ArgumentList argumentList)
|
||||||
|
in (argumentList !is null)
|
||||||
|
{
|
||||||
|
NoLint noLint;
|
||||||
|
|
||||||
|
foreach (nodeExpr; argumentList.items)
|
||||||
|
{
|
||||||
|
if (auto unaryExpr = cast(const UnaryExpression) nodeExpr)
|
||||||
|
{
|
||||||
|
auto primaryExpression = unaryExpr.primaryExpression;
|
||||||
|
if (primaryExpression is null)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (primaryExpression.primary != tok!"stringLiteral")
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto str = primaryExpression.primary.text.strip("\"");
|
||||||
|
Nullable!NoLint currNoLint = NoLintFactory.fromString(str);
|
||||||
|
noLint.merge(currNoLint);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!noLint.getDisabledChecks().length)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return noLint.nullable;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Transform a string with form "nolint(abc, efg)"
|
||||||
|
// into a NoLint struct
|
||||||
|
static Nullable!NoLint fromString(in string str)
|
||||||
|
{
|
||||||
|
static immutable re = regex(`[\w-_.]+`, "g");
|
||||||
|
auto matches = matchAll(str, re);
|
||||||
|
|
||||||
|
if (!matches)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
const udaName = matches.hit;
|
||||||
|
if (udaName != "nolint")
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
matches.popFront;
|
||||||
|
|
||||||
|
NoLint noLint;
|
||||||
|
|
||||||
|
while (matches)
|
||||||
|
{
|
||||||
|
noLint.pushCheck(matches.hit);
|
||||||
|
matches.popFront;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!noLint.getDisabledChecks.length)
|
||||||
|
return nullNoLint;
|
||||||
|
|
||||||
|
return noLint.nullable;
|
||||||
|
}
|
||||||
|
|
||||||
|
static nullNoLint = Nullable!NoLint.init;
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest
|
||||||
|
{
|
||||||
|
const s1 = "nolint(abc)";
|
||||||
|
const s2 = "nolint(abc, efg, hij)";
|
||||||
|
const s3 = " nolint ( abc , efg ) ";
|
||||||
|
const s4 = "nolint(dscanner.style.abc_efg-ijh)";
|
||||||
|
const s5 = "OtherUda(abc)";
|
||||||
|
const s6 = "nolint(dscanner)";
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s1).get.containsCheck("abc"));
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s2).get.containsCheck("abc"));
|
||||||
|
assert(NoLintFactory.fromString(s2).get.containsCheck("efg"));
|
||||||
|
assert(NoLintFactory.fromString(s2).get.containsCheck("hij"));
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s3).get.containsCheck("abc"));
|
||||||
|
assert(NoLintFactory.fromString(s3).get.containsCheck("efg"));
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s4).get.containsCheck("dscanner.style.abc_efg-ijh"));
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s5).isNull);
|
||||||
|
|
||||||
|
assert(NoLintFactory.fromString(s6).get.containsCheck("dscanner"));
|
||||||
|
assert(!NoLintFactory.fromString(s6).get.containsCheck("dscanner2"));
|
||||||
|
assert(NoLintFactory.fromString(s6).get.containsCheck("dscanner.foo"));
|
||||||
|
|
||||||
|
import std.stdio : stderr, writeln;
|
||||||
|
|
||||||
|
(() @trusted => stderr.writeln("Unittest for NoLint passed."))();
|
||||||
|
}
|
|
@ -14,6 +14,7 @@ import std.conv;
|
||||||
import std.format;
|
import std.format;
|
||||||
import dscanner.analysis.helpers;
|
import dscanner.analysis.helpers;
|
||||||
import dscanner.analysis.base;
|
import dscanner.analysis.base;
|
||||||
|
import dscanner.analysis.nolint;
|
||||||
import dsymbol.scope_ : Scope;
|
import dsymbol.scope_ : Scope;
|
||||||
|
|
||||||
final class StyleChecker : BaseAnalyzer
|
final class StyleChecker : BaseAnalyzer
|
||||||
|
@ -33,6 +34,9 @@ final class StyleChecker : BaseAnalyzer
|
||||||
|
|
||||||
override void visit(const ModuleDeclaration dec)
|
override void visit(const ModuleDeclaration dec)
|
||||||
{
|
{
|
||||||
|
with (noLint.push(NoLintFactory.fromModuleDeclaration(dec)))
|
||||||
|
dec.accept(this);
|
||||||
|
|
||||||
foreach (part; dec.moduleName.identifiers)
|
foreach (part; dec.moduleName.identifiers)
|
||||||
{
|
{
|
||||||
if (part.text.matchFirst(moduleNameRegex).length == 0)
|
if (part.text.matchFirst(moduleNameRegex).length == 0)
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
module dscanner.analysis.useless_initializer;
|
module dscanner.analysis.useless_initializer;
|
||||||
|
|
||||||
import dscanner.analysis.base;
|
import dscanner.analysis.base;
|
||||||
|
import dscanner.analysis.nolint;
|
||||||
import dscanner.utils : safeAccess;
|
import dscanner.utils : safeAccess;
|
||||||
import containers.dynamicarray;
|
import containers.dynamicarray;
|
||||||
import containers.hashmap;
|
import containers.hashmap;
|
||||||
|
@ -92,7 +93,10 @@ public:
|
||||||
override void visit(const(Declaration) decl)
|
override void visit(const(Declaration) decl)
|
||||||
{
|
{
|
||||||
_inStruct.insert(decl.structDeclaration !is null);
|
_inStruct.insert(decl.structDeclaration !is null);
|
||||||
decl.accept(this);
|
|
||||||
|
with (noLint.push(NoLintFactory.fromDeclaration(decl)))
|
||||||
|
decl.accept(this);
|
||||||
|
|
||||||
if (_inStruct.length > 1 && _inStruct[$-2] && decl.constructor &&
|
if (_inStruct.length > 1 && _inStruct[$-2] && decl.constructor &&
|
||||||
((decl.constructor.parameters && decl.constructor.parameters.parameters.length == 0) ||
|
((decl.constructor.parameters && decl.constructor.parameters.parameters.length == 0) ||
|
||||||
!decl.constructor.parameters))
|
!decl.constructor.parameters))
|
||||||
|
@ -361,6 +365,45 @@ public:
|
||||||
NotKnown nk = NotKnown.init;
|
NotKnown nk = NotKnown.init;
|
||||||
}, sac);
|
}, sac);
|
||||||
|
|
||||||
|
// passes
|
||||||
|
assertAnalyzerWarnings(q{
|
||||||
|
@("nolint(dscanner.useless-initializer)")
|
||||||
|
int a = 0;
|
||||||
|
int a = 0; /+
|
||||||
|
^ [warn]: X +/
|
||||||
|
|
||||||
|
@("nolint(dscanner.useless-initializer)")
|
||||||
|
int f() {
|
||||||
|
int a = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct nolint { string s; }
|
||||||
|
|
||||||
|
@nolint("dscanner.useless-initializer")
|
||||||
|
int a = 0;
|
||||||
|
int a = 0; /+
|
||||||
|
^ [warn]: X +/
|
||||||
|
|
||||||
|
@("nolint(other_check, dscanner.useless-initializer, another_one)")
|
||||||
|
int a = 0;
|
||||||
|
|
||||||
|
@nolint("other_check", "another_one", "dscanner.useless-initializer")
|
||||||
|
int a = 0;
|
||||||
|
|
||||||
|
}, sac);
|
||||||
|
|
||||||
|
// passes (disable check at module level)
|
||||||
|
assertAnalyzerWarnings(q{
|
||||||
|
@("nolint(dscanner.useless-initializer)")
|
||||||
|
module my_module;
|
||||||
|
|
||||||
|
int a = 0;
|
||||||
|
|
||||||
|
int f() {
|
||||||
|
int a = 0;
|
||||||
|
}
|
||||||
|
}, sac);
|
||||||
|
|
||||||
stderr.writeln("Unittest for UselessInitializerChecker passed.");
|
stderr.writeln("Unittest for UselessInitializerChecker passed.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue