[doclint] simplify error reporting
This patch simplifies error reporting, making it straight-forward. It also adds a few tests to make doclint more robust.
This commit is contained in:
parent
c468c451c5
commit
56619baa64
@ -12,23 +12,18 @@ class Documentation {
|
|||||||
/**
|
/**
|
||||||
* @param {!Documentation} actual
|
* @param {!Documentation} actual
|
||||||
* @param {!Documentation} expected
|
* @param {!Documentation} expected
|
||||||
|
* @return {!Array<string>}
|
||||||
*/
|
*/
|
||||||
static diff(actual, expected) {
|
static diff(actual, expected) {
|
||||||
// Diff classes.
|
const errors = [];
|
||||||
const result = {
|
|
||||||
extraClasses: [],
|
|
||||||
missingClasses: [],
|
|
||||||
extraMethods: [],
|
|
||||||
missingMethods: [],
|
|
||||||
extraProperties: [],
|
|
||||||
missingProperties: [],
|
|
||||||
badArguments: [],
|
|
||||||
};
|
|
||||||
const actualClasses = Array.from(actual.classes.keys()).sort();
|
const actualClasses = Array.from(actual.classes.keys()).sort();
|
||||||
const expectedClasses = Array.from(expected.classes.keys()).sort();
|
const expectedClasses = Array.from(expected.classes.keys()).sort();
|
||||||
let classesDiff = diff(actualClasses, expectedClasses);
|
let classesDiff = diff(actualClasses, expectedClasses);
|
||||||
result.extraClasses.push(...classesDiff.extra);
|
for (let className of classesDiff.extra)
|
||||||
result.missingClasses.push(...classesDiff.missing);
|
errors.push(`Non-existing class found: ${className}`);
|
||||||
|
for (let className of classesDiff.missing)
|
||||||
|
errors.push(`Class not found: ${className}`);
|
||||||
|
|
||||||
for (let className of classesDiff.equal) {
|
for (let className of classesDiff.equal) {
|
||||||
const actualClass = actual.classes.get(className);
|
const actualClass = actual.classes.get(className);
|
||||||
@ -36,29 +31,63 @@ class Documentation {
|
|||||||
const actualMethods = Array.from(actualClass.methods.keys()).sort();
|
const actualMethods = Array.from(actualClass.methods.keys()).sort();
|
||||||
const expectedMethods = Array.from(expectedClass.methods.keys()).sort();
|
const expectedMethods = Array.from(expectedClass.methods.keys()).sort();
|
||||||
const methodDiff = diff(actualMethods, expectedMethods);
|
const methodDiff = diff(actualMethods, expectedMethods);
|
||||||
result.extraMethods.push(...(methodDiff.extra.map(methodName => className + '.' + methodName)));
|
for (let methodName of methodDiff.extra)
|
||||||
result.missingMethods.push(...(methodDiff.missing.map(methodName => className + '.' + methodName)));
|
errors.push(`Non-existing method found: ${className}.${methodName}()`);
|
||||||
|
for (let methodName of methodDiff.missing)
|
||||||
|
errors.push(`Method not found: ${className}.${methodName}()`);
|
||||||
|
|
||||||
for (let methodName of methodDiff.equal) {
|
for (let methodName of methodDiff.equal) {
|
||||||
const actualMethod = actualClass.methods.get(methodName);
|
const actualMethod = actualClass.methods.get(methodName);
|
||||||
const expectedMethod = expectedClass.methods.get(methodName);
|
const expectedMethod = expectedClass.methods.get(methodName);
|
||||||
const actualArgs = actualMethod.args.map(arg => arg.name);
|
const actualArgs = Array.from(actualMethod.args.keys());
|
||||||
const expectedArgs = expectedMethod.args.map(arg => arg.name);
|
const expectedArgs = Array.from(expectedMethod.args.keys());
|
||||||
const argDiff = diff(actualArgs, expectedArgs);
|
const argDiff = diff(actualArgs, expectedArgs);
|
||||||
if (argDiff.extra.length || argDiff.missing.length) {
|
if (argDiff.extra.length || argDiff.missing.length) {
|
||||||
result.badArguments.push({
|
let text = [`Method ${className}.${methodName}() fails to describe its parameters:`];
|
||||||
method: `${className}.${methodName}`,
|
for (let arg of argDiff.missing)
|
||||||
missingArgs: argDiff.missing,
|
text.push(`- Argument not found: ${arg}`);
|
||||||
extraArgs: argDiff.extra
|
for (let arg of argDiff.extra)
|
||||||
});
|
text.push(`- Non-existing argument found: ${arg}`);
|
||||||
|
errors.push(text.join('\n'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const actualProperties = actualClass.propertiesArray.slice().sort();
|
const actualProperties = Array.from(actualClass.properties.keys()).sort();
|
||||||
const expectedProperties = expectedClass.propertiesArray.slice().sort();
|
const expectedProperties = Array.from(expectedClass.properties.keys()).sort();
|
||||||
const propertyDiff = diff(actualProperties, expectedProperties);
|
const propertyDiff = diff(actualProperties, expectedProperties);
|
||||||
result.extraProperties.push(...(propertyDiff.extra.map(propertyName => className + '.' + propertyName)));
|
for (let propertyName of propertyDiff.extra)
|
||||||
result.missingProperties.push(...(propertyDiff.missing.map(propertyName => className + '.' + propertyName)));
|
errors.push(`Non-existing property found: ${className}.${propertyName}`);
|
||||||
|
for (let propertyName of propertyDiff.missing)
|
||||||
|
errors.push(`Property not found: ${className}.${propertyName}`);
|
||||||
}
|
}
|
||||||
return result;
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {!Documentation} doc
|
||||||
|
* @return {!Array<string>}
|
||||||
|
*/
|
||||||
|
static validate(doc) {
|
||||||
|
const errors = [];
|
||||||
|
let classes = new Set();
|
||||||
|
// Report duplicates.
|
||||||
|
for (let cls of doc.classesArray) {
|
||||||
|
if (classes.has(cls.name))
|
||||||
|
errors.push(`Duplicate declaration of class ${cls.name}`);
|
||||||
|
classes.add(cls.name);
|
||||||
|
let methods = new Set();
|
||||||
|
for (let method of cls.methodsArray) {
|
||||||
|
if (methods.has(method.name))
|
||||||
|
errors.push(`Duplicate declaration of method ${cls.name}.${method.name}()`);
|
||||||
|
methods.add(method.name);
|
||||||
|
let args = new Set();
|
||||||
|
for (let arg of method.argsArray) {
|
||||||
|
if (args.has(arg.name))
|
||||||
|
errors.push(`Duplicate declaration of argument ${cls.name}.${method.name} "${arg.name}"`);
|
||||||
|
args.add(arg.name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errors;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -82,11 +111,14 @@ Documentation.Class = class {
|
|||||||
Documentation.Method = class {
|
Documentation.Method = class {
|
||||||
/**
|
/**
|
||||||
* @param {string} name
|
* @param {string} name
|
||||||
* @param {!Array<!Documentation.Argument>} args
|
* @param {!Array<!Documentation.Argument>} argsArray
|
||||||
*/
|
*/
|
||||||
constructor(name, args) {
|
constructor(name, argsArray) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.args = args;
|
this.argsArray = argsArray;
|
||||||
|
this.args = new Map();
|
||||||
|
for (let arg of argsArray)
|
||||||
|
this.args.set(arg.name, arg);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -39,79 +39,46 @@ async function lint(page, docsFolderPath, jsFolderPath) {
|
|||||||
let jsResult = await jsBuilder(jsFolderPath);
|
let jsResult = await jsBuilder(jsFolderPath);
|
||||||
let jsDocumentation = filterJSDocumentation(jsResult);
|
let jsDocumentation = filterJSDocumentation(jsResult);
|
||||||
let mdDocumentation = mdResult.documentation;
|
let mdDocumentation = mdResult.documentation;
|
||||||
let diff = Documentation.diff(mdDocumentation, jsDocumentation);
|
|
||||||
|
|
||||||
let jsErrors = [];
|
let jsErrors = [];
|
||||||
let mdErrors = [];
|
let mdErrors = Documentation.diff(mdDocumentation, jsDocumentation);
|
||||||
|
|
||||||
// Report all markdown parse errors.
|
// Report all markdown parse errors.
|
||||||
mdErrors.push(...mdResult.errors);
|
mdErrors.push(...mdResult.errors);
|
||||||
{
|
|
||||||
// Report duplicate JavaScript classes.
|
jsErrors.push(...Documentation.validate(jsDocumentation));
|
||||||
let jsClasses = new Map();
|
mdErrors.push(...Documentation.validate(mdDocumentation));
|
||||||
for (let jsClass of jsDocumentation.classesArray) {
|
mdErrors.push(...lintMarkdown(mdDocumentation));
|
||||||
if (jsClasses.has(jsClass.name))
|
|
||||||
jsErrors.push(`Duplicate declaration of class ${jsClass.name}`);
|
|
||||||
jsClasses.set(jsClass.name, jsClass);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
// Report duplicate MarkDown classes.
|
|
||||||
let mdClasses = new Map();
|
|
||||||
for (let mdClass of mdDocumentation.classesArray) {
|
|
||||||
if (mdClasses.has(mdClass.name))
|
|
||||||
mdErrors.push(`Duplicate declaration of class ${mdClass.name}`);
|
|
||||||
mdClasses.set(mdClass.name, mdClass);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
// Make sure class constructors are defined before other methods.
|
|
||||||
for (let mdClass of mdDocumentation.classesArray) {
|
|
||||||
let constructorMethod = mdClass.methods.get('constructor');
|
|
||||||
if (!constructorMethod)
|
|
||||||
continue;
|
|
||||||
if (mdClass.methodsArray[0] !== constructorMethod)
|
|
||||||
mdErrors.push(`Constructor of ${mdClass.name} should go before other methods`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
{
|
|
||||||
// Methods should be sorted alphabetically.
|
|
||||||
for (let mdClass of mdDocumentation.classesArray) {
|
|
||||||
for (let i = 0; i < mdClass.methodsArray.length - 1; ++i) {
|
|
||||||
// Constructor should always go first.
|
|
||||||
if (mdClass.methodsArray[i].name === 'constructor')
|
|
||||||
continue;
|
|
||||||
let method1 = mdClass.methodsArray[i];
|
|
||||||
let method2 = mdClass.methodsArray[i + 1];
|
|
||||||
if (method1.name > method2.name)
|
|
||||||
mdErrors.push(`${mdClass.name}.${method1.name} breaks alphabetic sorting inside class ${mdClass.name}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Report non-existing and missing classes.
|
|
||||||
mdErrors.push(...diff.extraClasses.map(className => `Non-existing class found: ${className}`));
|
|
||||||
mdErrors.push(...diff.missingClasses.map(className => `Class not found: ${className}`));
|
|
||||||
mdErrors.push(...diff.extraMethods.map(methodName => `Non-existing method found: ${methodName}`));
|
|
||||||
mdErrors.push(...diff.missingMethods.map(methodName => `Method not found: ${methodName}`));
|
|
||||||
mdErrors.push(...diff.extraProperties.map(propertyName => `Non-existing property found: ${propertyName}`));
|
|
||||||
mdErrors.push(...diff.missingProperties.map(propertyName => `Property not found: ${propertyName}`));
|
|
||||||
{
|
|
||||||
// Report badly described arguments.
|
|
||||||
for (let badArgument of diff.badArguments) {
|
|
||||||
let text = [`Method ${badArgument.method} fails to describe its parameters:`];
|
|
||||||
for (let missing of badArgument.missingArgs)
|
|
||||||
text.push(`- Missing description for "${missing}"`);
|
|
||||||
for (let extra of badArgument.extraArgs)
|
|
||||||
text.push(`- Described non-existing parameter "${extra}"`);
|
|
||||||
mdErrors.push(text.join('\n'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Push all errors with proper prefixes
|
// Push all errors with proper prefixes
|
||||||
let errors = jsErrors.map(error => '[JavaScript] ' + error);
|
let errors = jsErrors.map(error => '[JavaScript] ' + error);
|
||||||
errors.push(...mdErrors.map(error => '[MarkDown] ' + error));
|
errors.push(...mdErrors.map(error => '[MarkDown] ' + error));
|
||||||
return errors;
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {!Documentation} doc
|
||||||
|
* @return {!Array<string>}
|
||||||
|
*/
|
||||||
|
function lintMarkdown(doc) {
|
||||||
|
const errors = [];
|
||||||
|
// Methods should be sorted alphabetically.
|
||||||
|
for (let cls of doc.classesArray) {
|
||||||
|
for (let i = 0; i < cls.methodsArray.length - 1; ++i) {
|
||||||
|
// Constructor always goes first.
|
||||||
|
if (cls.methodsArray[i].name === 'constructor') {
|
||||||
|
if (i > 0)
|
||||||
|
errors.push(`Constructor of ${cls.name} should go before other methods`);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let method1 = cls.methodsArray[i];
|
||||||
|
let method2 = cls.methodsArray[i + 1];
|
||||||
|
if (method1.name > method2.name)
|
||||||
|
errors.push(`${cls.name}.${method1.name} breaks alphabetic methods sorting inside class ${cls.name}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {!Documentation} jsDocumentation
|
* @param {!Documentation} jsDocumentation
|
||||||
* @return {!Documentation}
|
* @return {!Documentation}
|
||||||
|
14
utils/doclint/test/06-duplicates/doc.md
Normal file
14
utils/doclint/test/06-duplicates/doc.md
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
### class: Bar
|
||||||
|
|
||||||
|
### class: Foo
|
||||||
|
|
||||||
|
#### foo.test()
|
||||||
|
|
||||||
|
#### foo.test()
|
||||||
|
|
||||||
|
#### foo.title(arg, arg)
|
||||||
|
- `arg` <[number]>
|
||||||
|
- `arg` <[number]>
|
||||||
|
|
||||||
|
### class: Bar
|
||||||
|
|
10
utils/doclint/test/06-duplicates/foo.js
Normal file
10
utils/doclint/test/06-duplicates/foo.js
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
class Foo {
|
||||||
|
test() {
|
||||||
|
}
|
||||||
|
|
||||||
|
title(arg) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class Bar {
|
||||||
|
}
|
9
utils/doclint/test/07-sorting/doc.md
Normal file
9
utils/doclint/test/07-sorting/doc.md
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
### class: Foo
|
||||||
|
|
||||||
|
#### foo.aaa()
|
||||||
|
|
||||||
|
#### new Foo()
|
||||||
|
|
||||||
|
#### foo.ccc()
|
||||||
|
|
||||||
|
#### foo.bbb()
|
10
utils/doclint/test/07-sorting/foo.js
Normal file
10
utils/doclint/test/07-sorting/foo.js
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
class Foo {
|
||||||
|
constructor() {
|
||||||
|
}
|
||||||
|
|
||||||
|
aaa() {}
|
||||||
|
|
||||||
|
bbb() {}
|
||||||
|
|
||||||
|
ccc() {}
|
||||||
|
}
|
@ -1,2 +1,2 @@
|
|||||||
[MarkDown] Non-existing method found: Foo.proceed
|
[MarkDown] Non-existing method found: Foo.proceed()
|
||||||
[MarkDown] Method not found: Foo.stop
|
[MarkDown] Method not found: Foo.stop()
|
@ -1,3 +1,3 @@
|
|||||||
[MarkDown] Method not found: Foo.constructor
|
[MarkDown] Method not found: Foo.constructor()
|
||||||
[MarkDown] Non-existing property found: Foo.c
|
[MarkDown] Non-existing property found: Foo.c
|
||||||
[MarkDown] Property not found: Foo.b
|
[MarkDown] Property not found: Foo.b
|
@ -1,4 +1,4 @@
|
|||||||
|
[MarkDown] Method Foo.constructor() fails to describe its parameters:
|
||||||
|
- Argument not found: arg3
|
||||||
|
- Non-existing argument found: arg2
|
||||||
[MarkDown] Heading arguments for "foo.test(fileNames)" do not match described ones, i.e. "fileNames" != "filePaths"
|
[MarkDown] Heading arguments for "foo.test(fileNames)" do not match described ones, i.e. "fileNames" != "filePaths"
|
||||||
[MarkDown] Method Foo.constructor fails to describe its parameters:
|
|
||||||
- Missing description for "arg3"
|
|
||||||
- Described non-existing parameter "arg2"
|
|
3
utils/doclint/test/golden/06-duplicates.txt
Normal file
3
utils/doclint/test/golden/06-duplicates.txt
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[MarkDown] Duplicate declaration of method Foo.test()
|
||||||
|
[MarkDown] Duplicate declaration of argument Foo.title "arg"
|
||||||
|
[MarkDown] Duplicate declaration of class Bar
|
2
utils/doclint/test/golden/07-sorting.txt
Normal file
2
utils/doclint/test/golden/07-sorting.txt
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
[MarkDown] Constructor of Foo should go before other methods
|
||||||
|
[MarkDown] Foo.ccc breaks alphabetic methods sorting inside class Foo
|
@ -36,6 +36,8 @@ describe('doclint', function() {
|
|||||||
it('03-property-errors', SX(test));
|
it('03-property-errors', SX(test));
|
||||||
it('04-bad-arguments', SX(test));
|
it('04-bad-arguments', SX(test));
|
||||||
it('05-outdated-toc', SX(test));
|
it('05-outdated-toc', SX(test));
|
||||||
|
it('06-duplicates', SX(test));
|
||||||
|
it('07-sorting', SX(test));
|
||||||
});
|
});
|
||||||
|
|
||||||
async function test() {
|
async function test() {
|
||||||
|
Loading…
Reference in New Issue
Block a user