diff --git a/.gitmodules b/.gitmodules index c977bf8..9838e3c 100644 --- a/.gitmodules +++ b/.gitmodules @@ -11,3 +11,6 @@ [submodule "dsymbol"] path = dsymbol url = https://github.com/Hackerpilot/dsymbol.git +[submodule "libddoc"] + path = libddoc + url = https://github.com/economicmodeling/libddoc diff --git a/dub.json b/dub.json index d6873dd..3b0e457 100644 --- a/dub.json +++ b/dub.json @@ -10,6 +10,7 @@ "libdparse": "~>0.7.0", "dsymbol": "~>0.2.0", "inifiled": ">=0.0.6", - "emsi_containers": "~>0.5.3" + "emsi_containers": "~>0.5.3", + "libddoc": "~>0.2.0" }, } diff --git a/libddoc b/libddoc new file mode 160000 index 0000000..73f2761 --- /dev/null +++ b/libddoc @@ -0,0 +1 @@ +Subproject commit 73f2761d859b0364b0b5f77e6316b87ef7052d4f diff --git a/makefile b/makefile index fab85b8..bf0a71c 100644 --- a/makefile +++ b/makefile @@ -11,11 +11,13 @@ SRC := \ $(shell find inifiled/source/ -name "*.d")\ $(shell find libdparse/src/std/experimental/ -name "*.d")\ $(shell find libdparse/src/dparse/ -name "*.d")\ + $(shell find libddoc/src -name "*.d")\ $(shell find src/ -name "*.d") INCLUDE_PATHS = \ -Iinifiled/source -Isrc\ -Ilibdparse/src\ - -Idsymbol/src -Icontainers/src + -Idsymbol/src -Icontainers/src\ + -Ilibddoc/src VERSIONS = DEBUG_VERSIONS = -version=dparse_verbose DMD_FLAGS = -w -inline -release -O -J. -od${OBJ_DIR} -version=StdLoggerDisableWarning diff --git a/src/analysis/config.d b/src/analysis/config.d index 32ccc75..fd1c9f2 100644 --- a/src/analysis/config.d +++ b/src/analysis/config.d @@ -154,6 +154,9 @@ struct StaticAnalysisConfig @INI("Check for explicitly annotated unittests") string explicitly_annotated_unittests = Check.disabled; + @INI("Check for properly documented public functions (Returns, Params)") + string properly_documented_public_functions = Check.disabled; + @INI("Check for useless usage of the final attribute") string final_attribute_check = Check.disabled; } diff --git a/src/analysis/properly_documented_public_functions.d b/src/analysis/properly_documented_public_functions.d new file mode 100644 index 0000000..1be7802 --- /dev/null +++ b/src/analysis/properly_documented_public_functions.d @@ -0,0 +1,669 @@ +// 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.properly_documented_public_functions; + +import dparse.lexer; +import dparse.ast; +import analysis.base : BaseAnalyzer; + +import std.format : format; +import std.range.primitives; +import std.stdio; + +/** + * Requires each public function to contain the following ddoc sections + - PARAMS: + - if the function has at least one parameter + - every parameter must have a ddoc params entry (applies for template paramters too) + - Ddoc params entries without a parameter trigger warnings as well + - RETURNS: (except if it's void, only functions) + */ +class ProperlyDocumentedPublicFunctions : BaseAnalyzer +{ + enum string MISSING_PARAMS_KEY = "dscanner.style.doc_missing_params"; + enum string MISSING_PARAMS_MESSAGE = "Parameter %s isn't documented in the `Params` section"; + enum string MISSING_TEMPLATE_PARAMS_MESSAGE + = "Template parameters %s isn't documented in the `Params` section"; + + enum string NON_EXISTENT_PARAMS_KEY = "dscanner.style.doc_non_existing_params"; + enum string NON_EXISTENT_PARAMS_MESSAGE = "Ddoced parameter %s isn't a function parameter"; + + enum string MISSING_RETURNS_KEY = "dscanner.style.doc_missing_returns"; + enum string MISSING_RETURNS_MESSAGE = "A public function needs to contain a `Returns` section"; + + /// + this(string fileName, bool skipTests = false) + { + super(fileName, null, skipTests); + } + + override void visit(const Module mod) + { + islastSeenVisibilityLabelPublic = true; + mod.accept(this); + postCheckSeenDdocParams(); + } + + override void visit(const Declaration decl) + { + import std.algorithm.searching : any; + import std.algorithm.iteration : map; + + // skip private symbols + enum tokPrivate = tok!"private", + tokProtected = tok!"protected", + tokPackage = tok!"package", + tokPublic = tok!"public"; + + if (decl.attributes.length > 0) + { + const bool isPublic = !decl.attributes.map!`a.attribute`.any!(x => x == tokPrivate || + x == tokProtected || + x == tokPackage); + // recognize label blocks + if (!hasDeclaration(decl)) + islastSeenVisibilityLabelPublic = isPublic; + + if (!isPublic) + return; + } + + if (islastSeenVisibilityLabelPublic || decl.attributes.map!`a.attribute`.any!(x => x == tokPublic)) + { + if (decl.functionDeclaration !is null || + decl.templateDeclaration !is null || + decl.mixinTemplateDeclaration !is null || + decl.classDeclaration !is null || + decl.structDeclaration !is null) + decl.accept(this); + } + } + + override void visit(const TemplateDeclaration decl) + { + setLastDdocParams(decl.name.line, decl.name.column, decl.comment); + checkDdocParams(decl.name.line, decl.name.column, decl.templateParameters); + decl.accept(this); + } + + override void visit(const MixinTemplateDeclaration decl) + { + decl.accept(this); + } + + override void visit(const StructDeclaration decl) + { + setLastDdocParams(decl.name.line, decl.name.column, decl.comment); + checkDdocParams(decl.name.line, decl.name.column, decl.templateParameters); + decl.accept(this); + } + + override void visit(const ClassDeclaration decl) + { + setLastDdocParams(decl.name.line, decl.name.column, decl.comment); + checkDdocParams(decl.name.line, decl.name.column, decl.templateParameters); + decl.accept(this); + } + + override void visit(const FunctionDeclaration decl) + { + import std.algorithm.searching : any; + + // ignore header declaration for now + if (decl.functionBody is null) + return; + + auto comment = setLastDdocParams(decl.name.line, decl.name.column, decl.comment); + checkDdocParams(decl.name.line, decl.name.column, decl.parameters); + checkDdocParams(decl.name.line, decl.name.column, decl.templateParameters); + + enum voidType = tok!"void"; + if (decl.returnType is null || decl.returnType.type2.builtinType != voidType) + if (!(comment.isDitto || comment.sections.any!(s => s.name == "Returns"))) + addErrorMessage(decl.name.line, decl.name.column, MISSING_RETURNS_KEY, MISSING_RETURNS_MESSAGE); + } + + alias visit = BaseAnalyzer.visit; + +private: + bool islastSeenVisibilityLabelPublic; + + static struct Function + { + bool active; + size_t line, column; + const(string)[] ddocParams; + bool[string] params; + } + Function lastSeenFun; + + // find invalid ddoc parameters (i.e. they don't occur in a function declaration) + void postCheckSeenDdocParams() + { + import std.format : format; + + if (lastSeenFun.active) + foreach (p; lastSeenFun.ddocParams) + if (p !in lastSeenFun.params) + addErrorMessage(lastSeenFun.line, lastSeenFun.column, NON_EXISTENT_PARAMS_KEY, + NON_EXISTENT_PARAMS_MESSAGE.format(p)); + + lastSeenFun.active = false; + } + + auto setLastDdocParams(size_t line, size_t column, string commentText) + { + import ddoc.comments : parseComment; + import std.algorithm.searching : find; + import std.algorithm.iteration : map; + import std.array : array; + + const comment = parseComment(commentText, null); + if (!comment.isDitto) + { + // check old function for invalid ddoc params + if (lastSeenFun.active) + postCheckSeenDdocParams(); + + const paramSection = comment.sections.find!(s => s.name == "Params"); + if (paramSection.empty) + { + lastSeenFun = Function(true, line, column, null); + } + else + { + auto ddocParams = paramSection[0].mapping.map!(a => a[0]).array; + lastSeenFun = Function(true, line, column, ddocParams); + } + } + + return comment; + } + + void checkDdocParams(size_t line, size_t column, const Parameters params) + { + import std.algorithm.searching : canFind; + + if (lastSeenFun.active && params !is null) + foreach (p; params.parameters) + { + if (!lastSeenFun.ddocParams.canFind(p.name.text)) + addErrorMessage(line, column, MISSING_PARAMS_KEY, + format(MISSING_PARAMS_MESSAGE, p.name.text)); + else + lastSeenFun.params[p.name.text] = true; + } + } + + void checkDdocParams(size_t line, size_t column, const TemplateParameters templateParams) + { + import std.algorithm.searching : canFind; + + if (lastSeenFun.active && templateParams !is null && templateParams.templateParameterList !is null) + foreach (p; templateParams.templateParameterList.items) + { + auto name = templateParamName(p); + assert(name, "Invalid template parameter name."); // this shouldn't happen + if (!lastSeenFun.ddocParams.canFind(name)) + addErrorMessage(line, column, MISSING_PARAMS_KEY, + format(MISSING_TEMPLATE_PARAMS_MESSAGE, name)); + else + lastSeenFun.params[name] = true; + } + } + + static string templateParamName(const TemplateParameter p) + { + if (p.templateTypeParameter) + return p.templateTypeParameter.identifier.text; + if (p.templateValueParameter) + return p.templateValueParameter.identifier.text; + if (p.templateAliasParameter) + return p.templateAliasParameter.identifier.text; + if (p.templateTupleParameter) + return p.templateTupleParameter.identifier.text; + if (p.templateThisParameter) + return p.templateThisParameter.templateTypeParameter.identifier.text; + + return null; + } + + bool hasDeclaration(const Declaration decl) + { + import std.meta : AliasSeq; + alias properties = AliasSeq!( + "aliasDeclaration", + "aliasThisDeclaration", + "anonymousEnumDeclaration", + "attributeDeclaration", + "classDeclaration", + "conditionalDeclaration", + "constructor", + "debugSpecification", + "destructor", + "enumDeclaration", + "eponymousTemplateDeclaration", + "functionDeclaration", + "importDeclaration", + "interfaceDeclaration", + "invariant_", + "mixinDeclaration", + "mixinTemplateDeclaration", + "postblit", + "pragmaDeclaration", + "sharedStaticConstructor", + "sharedStaticDestructor", + "staticAssertDeclaration", + "staticConstructor", + "staticDestructor", + "structDeclaration", + "templateDeclaration", + "unionDeclaration", + "unittest_", + "variableDeclaration", + "versionSpecification", + ); + if (decl.declarations !is null) + return false; + + auto isNull = true; + foreach (property; properties) + if (mixin("decl." ~ property ~ " !is null")) + isNull = false; + + return !isNull; + } +} + +version(unittest) +{ + import std.stdio : stderr; + import std.format : format; + import analysis.config : StaticAnalysisConfig, Check; + import analysis.helpers : assertAnalyzerWarnings; + + StaticAnalysisConfig sac = {properly_documented_public_functions: Check.enabled}; +} + +// missing params +unittest +{ + assertAnalyzerWarnings(q{ + /** + Some text + */ + void foo(int k){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_PARAMS_MESSAGE.format("k") + ), sac); + + assertAnalyzerWarnings(q{ + /** + Some text + */ + void foo(int K)(){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("K") + ), sac); + + assertAnalyzerWarnings(q{ + /** + Some text + */ + struct Foo(Bar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("Bar") + ), sac); + + assertAnalyzerWarnings(q{ + /** + Some text + */ + class Foo(Bar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("Bar") + ), sac); + + assertAnalyzerWarnings(q{ + /** + Some text + */ + template Foo(Bar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("Bar") + ), sac); + + + // test no parameters + assertAnalyzerWarnings(q{ + /** Some text */ + void foo(){} + }c, sac); + + assertAnalyzerWarnings(q{ + /** Some text */ + struct Foo(){} + }c, sac); + + assertAnalyzerWarnings(q{ + /** Some text */ + class Foo(){} + }c, sac); + + assertAnalyzerWarnings(q{ + /** Some text */ + template Foo(){} + }c, sac); + +} + +// missing returns (only functions) +unittest +{ + assertAnalyzerWarnings(q{ + /** + Some text + */ + int foo(){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_RETURNS_MESSAGE, + ), sac); +} + +// ignore private +unittest +{ + assertAnalyzerWarnings(q{ + /** + Some text + */ + private void foo(int k){} + }c, sac); + + // with block + assertAnalyzerWarnings(q{ + private: + /** + Some text + */ + private void foo(int k){} + public int bar(){} // [warn]: %s + public: + int foobar(){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_RETURNS_MESSAGE, + ProperlyDocumentedPublicFunctions.MISSING_RETURNS_MESSAGE, + ), sac); + + // with block (template) + assertAnalyzerWarnings(q{ + private: + /** + Some text + */ + private template foo(int k){} + public template bar(T){} // [warn]: %s + public: + template foobar(T){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("T"), + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("T"), + ), sac); + + // with block (struct) + assertAnalyzerWarnings(q{ + private: + /** + Some text + */ + private struct foo(int k){} + public struct bar(T){} // [warn]: %s + public: + struct foobar(T){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("T"), + ProperlyDocumentedPublicFunctions.MISSING_TEMPLATE_PARAMS_MESSAGE.format("T"), + ), sac); +} + + +// test parameter names +unittest +{ + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * + * Returns: + * A long description. + */ +int foo(int k){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_PARAMS_MESSAGE.format("k") + ), sac); + + assertAnalyzerWarnings(q{ +/** +Description. + +Params: +val = A stupid parameter +k = A stupid parameter + +Returns: +A long description. +*/ +int foo(int k){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.NON_EXISTENT_PARAMS_MESSAGE.format("val") + ), sac); + + assertAnalyzerWarnings(q{ +/** +Description. + +Params: + +Returns: +A long description. +*/ +int foo(int k){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_PARAMS_MESSAGE.format("k") + ), sac); + + assertAnalyzerWarnings(q{ +/** +Description. + +Params: +foo = A stupid parameter +bad = A stupid parameter (does not exist) +foobar = A stupid parameter + +Returns: +A long description. +*/ +int foo(int foo, int foobar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.NON_EXISTENT_PARAMS_MESSAGE.format("bad") + ), sac); + + assertAnalyzerWarnings(q{ +/** +Description. + +Params: +foo = A stupid parameter +bad = A stupid parameter (does not exist) +foobar = A stupid parameter + +Returns: +A long description. +*/ +struct foo(int foo, int foobar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.NON_EXISTENT_PARAMS_MESSAGE.format("bad") + ), sac); + + // properly documented + assertAnalyzerWarnings(q{ +/** +Description. + +Params: +foo = A stupid parameter +bar = A stupid parameter + +Returns: +A long description. +*/ +int foo(int foo, int bar){} + }c, sac); + + assertAnalyzerWarnings(q{ +/** +Description. + +Params: +foo = A stupid parameter +bar = A stupid parameter + +Returns: +A long description. +*/ +struct foo(int foo, int bar){} + }c, sac); +} + +// support ditto +unittest +{ + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * k = A stupid parameter + * + * Returns: + * A long description. + */ +int foo(int k){} + +/// ditto +int bar(int k){} + }c, sac); + + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * k = A stupid parameter + * K = A stupid parameter + * + * Returns: + * A long description. + */ +int foo(int k){} + +/// ditto +struct Bar(K){} + }c, sac); + + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * k = A stupid parameter + * f = A stupid parameter + * + * Returns: + * A long description. + */ +int foo(int k){} + +/// ditto +int bar(int f){} + }c, sac); + + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * k = A stupid parameter + * + * Returns: + * A long description. + */ +int foo(int k){} + +/// ditto +int bar(int bar){} // [warn]: %s + }c.format( + ProperlyDocumentedPublicFunctions.MISSING_PARAMS_MESSAGE.format("bar") + ), sac); + + assertAnalyzerWarnings(q{ +/** + * Description. + * + * Params: + * k = A stupid parameter + * bar = A stupid parameter + * f = A stupid parameter + * + * Returns: + * A long description. + * See_Also: + * $(REF takeExactly, std,range) + */ +int foo(int k){} // [warn]: %s + +/// ditto +int bar(int bar){} + }c.format( + ProperlyDocumentedPublicFunctions.NON_EXISTENT_PARAMS_MESSAGE.format("f") + ), sac); +} + + // check correct ddoc headers +unittest +{ + assertAnalyzerWarnings(q{ +/++ + Counts elements in the given + $(REF_ALTTEXT forward range, isForwardRange, std,range,primitives) + until the given predicate is true for one of the given $(D needles). + + Params: + val = A stupid parameter + + Returns: Awesome values. + +/ +string bar(string val){} + }c, sac); + + assertAnalyzerWarnings(q{ +/++ + Counts elements in the given + $(REF_ALTTEXT forward range, isForwardRange, std,range,primitives) + until the given predicate is true for one of the given $(D needles). + + Params: + val = A stupid parameter + + Returns: Awesome values. + +/ +template bar(string val){} + }c, sac); + + stderr.writeln("Unittest for ProperlyDocumentedPublicFunctions passed."); +} diff --git a/src/analysis/run.d b/src/analysis/run.d index 0bc5ec4..4c4ced6 100644 --- a/src/analysis/run.d +++ b/src/analysis/run.d @@ -62,6 +62,7 @@ import analysis.lambda_return_check; import analysis.auto_function; import analysis.imports_sortedness; import analysis.explicitly_annotated_unittests; +import analysis.properly_documented_public_functions; import analysis.final_attribute; import dsymbol.string_interning : internString; @@ -370,6 +371,10 @@ MessageSet analyze(string fileName, const Module m, const StaticAnalysisConfig a checks ~= new ExplicitlyAnnotatedUnittestCheck(fileName, analysisConfig.explicitly_annotated_unittests == Check.skipTests && !ut); + if (analysisConfig.properly_documented_public_functions != Check.disabled) + checks ~= new ProperlyDocumentedPublicFunctions(fileName, + analysisConfig.properly_documented_public_functions == Check.skipTests && !ut); + if (analysisConfig.final_attribute_check != Check.disabled) checks ~= new FinalAttributeChecker(fileName, analysisConfig.final_attribute_check == Check.skipTests && !ut);