diff --git a/analysis/constructors.d b/analysis/constructors.d index ed4dd76..39c5125 100644 --- a/analysis/constructors.d +++ b/analysis/constructors.d @@ -2,7 +2,10 @@ module analysis.constructors; import std.d.ast; import std.d.lexer; +import std.stdio; import analysis.base; +import analysis.helpers; + class ConstructorCheck : BaseAnalyzer { @@ -84,3 +87,23 @@ private: bool hasNoArgConstructor; bool hasDefaultArgConstructor; } + +unittest +{ + shouldWarn(q{ + class Cat // [warn]: This class has a zero-argument constructor as well as a constructor with one default argument. This can be confusing + { + this() {} + this(string name = "kittie") {} + } + + struct Dog + { + this() {} + this(string name = "doggie") {} // [warn]: This struct constructor can never be called with its default argument. + } + }c, analysis.run.AnalyzerCheck.constructor_check); + + stderr.writeln("Unittest for ConstructorCheck passed."); +} + diff --git a/analysis/del.d b/analysis/del.d index 5ef4472..5d48152 100644 --- a/analysis/del.d +++ b/analysis/del.d @@ -5,9 +5,11 @@ module analysis.del; +import std.stdio; import std.d.ast; import std.d.lexer; import analysis.base; +import analysis.helpers; /** * Checks for use of the deprecated "delete" keyword @@ -27,3 +29,20 @@ class DeleteCheck : BaseAnalyzer d.accept(this); } } + +unittest +{ + shouldWarn(q{ + void testDelete() + { + int[int] data = [1 : 2]; + delete data[1]; // [warn]: Avoid using the delete keyword + + auto a = new Class(); + delete a; // [warn]: Avoid using the delete keyword + } + }c, analysis.run.AnalyzerCheck.delete_check); + + stderr.writeln("Unittest for DeleteCheck passed."); +} + diff --git a/analysis/fish.d b/analysis/fish.d index 71309d7..d46d54f 100644 --- a/analysis/fish.d +++ b/analysis/fish.d @@ -5,9 +5,11 @@ module analysis.fish; +import std.stdio; import std.d.ast; import std.d.lexer; import analysis.base; +import analysis.helpers; /** * Checks for use of the deprecated floating point comparison operators. @@ -37,3 +39,25 @@ class FloatOperatorCheck : BaseAnalyzer r.accept(this); } } + +unittest +{ + shouldWarn(q{ + void testFish() + { + float z = 1.5f; + bool a; + a = z !<>= z; // [warn]: Avoid using the deprecated floating-point operators + a = z !<> z; // [warn]: Avoid using the deprecated floating-point operators + a = z <> z; // [warn]: Avoid using the deprecated floating-point operators + a = z <>= z; // [warn]: Avoid using the deprecated floating-point operators + a = z !> z; // [warn]: Avoid using the deprecated floating-point operators + a = z !>= z; // [warn]: Avoid using the deprecated floating-point operators + a = z !< z; // [warn]: Avoid using the deprecated floating-point operators + a = z !<= z; // [warn]: Avoid using the deprecated floating-point operators + } + }c, analysis.run.AnalyzerCheck.float_operator_check); + + stderr.writeln("Unittest for FloatOperatorCheck passed."); +} + diff --git a/analysis/helpers.d b/analysis/helpers.d new file mode 100644 index 0000000..58bdb44 --- /dev/null +++ b/analysis/helpers.d @@ -0,0 +1,141 @@ +// Copyright (c) 2014, Matthew Brennan Jones +// Distributed under the Boost Software License, Version 1.0. +// (See accompanying file LICENSE_1_0.txt or copy at +// http://www.boost.org/LICENSE_1_0.txt) + +module analysis.helpers; + +import std.string; +import std.traits; + +import std.d.ast; + + +S between(S)(S value, S before, S after) +if (isSomeString!S) { + return value.after(before).before(after); +} + +S before(S)(S value, S separator) +if (isSomeString!S) { + auto i = indexOf(value, separator); + + if (i == -1) + return value; + + return value[0 .. i]; +} + +S after(S)(S value, S separator) +if(isSomeString!S) { + auto i = indexOf(value, separator); + + if( i == -1) + return ""; + + size_t start = i + separator.length; + + return value[start .. $]; +} + +S afterLast(S)(S value, S separator) +if (isSomeString!S) { + size_t i = rindex(value, separator); + + if (i == value.length) + return ""; + + size_t start = i + separator.length; + + return value[start .. $]; +} + +void shouldWarn(string code, analysis.run.AnalyzerCheck analyzers, string file=__FILE__, size_t line=__LINE__) +{ + import analysis.run; + + // Run the code and get any warnings + string[] rawWarnings = analyze("test", cast(ubyte[]) code, analyzers); + string[] codeLines = code.split("\n"); + + // Get the warnings ordered by line + string[size_t] warnings; + for (size_t i=0; i