[doclint] be more permissive with syntax

This patch adds a helpful message if the 'returns' statement is somehow
mistyped.

References #14.
This commit is contained in:
Andrey Lushnikov 2017-07-15 10:20:10 -07:00
parent 6eac22dd87
commit 3b5dbe2308
4 changed files with 25 additions and 7 deletions

View File

@ -19,10 +19,11 @@ class MDOutline {
// Extract headings. // Extract headings.
await page.setContent(html); await page.setContent(html);
const classes = await page.evaluate(() => { const {classes, errors} = await page.evaluate(() => {
let classes = []; let classes = [];
let currentClass = {}; let currentClass = {};
let member = {}; let member = {};
let errors = [];
for (let element of document.body.querySelectorAll('h3, h4, h4 + ul > li')) { for (let element of document.body.querySelectorAll('h3, h4, h4 + ul > li')) {
if (element.matches('h3')) { if (element.matches('h3')) {
currentClass = { currentClass = {
@ -39,18 +40,27 @@ class MDOutline {
currentClass.members.push(member); currentClass.members.push(member);
} else if (element.matches('li') && element.firstChild.matches && element.firstChild.matches('code')) { } else if (element.matches('li') && element.firstChild.matches && element.firstChild.matches('code')) {
member.args.push(element.firstChild.textContent); member.args.push(element.firstChild.textContent);
} else if (element.matches('li') && element.firstChild.nodeType === Element.TEXT_NODE && element.firstChild.textContent.startsWith('returns: ')) { } else if (element.matches('li') && element.firstChild.nodeType === Element.TEXT_NODE && element.firstChild.textContent.toLowerCase().startsWith('retur')) {
member.hasReturn = true; member.hasReturn = true;
const expectedText = 'returns: ';
let actualText = element.firstChild.textContent;
let angleIndex = actualText.indexOf('<');
let spaceIndex = actualText.indexOf(' ');
angleIndex = angleIndex === -1 ? angleText.length : angleIndex;
spaceIndex = spaceIndex === -1 ? spaceIndex.length : spaceIndex + 1;
actualText = actualText.substring(0, Math.min(angleIndex, spaceIndex));
if (actualText !== expectedText)
errors.push(`${member.name} has mistyped 'return' type declaration: expected exactly '${expectedText}', found '${actualText}'.`);
} }
} }
return classes; return {classes, errors};
}); });
return new MDOutline(classes); return new MDOutline(classes, errors);
} }
constructor(classes) { constructor(classes, errors) {
this.classes = []; this.classes = [];
this.errors = []; this.errors = errors;
const classHeading = /^class: (\w+)$/; const classHeading = /^class: (\w+)$/;
const constructorRegex = /^new (\w+)\((.*)\)$/; const constructorRegex = /^new (\w+)\((.*)\)$/;
const methodRegex = /^(\w+)\.(\w+)\((.*)\)$/; const methodRegex = /^(\w+)\.(\w+)\((.*)\)$/;

View File

@ -7,3 +7,5 @@
#### foo.returnNothing() #### foo.returnNothing()
- returns: <[number]> - returns: <[number]>
#### foo.www()
- returns <[string]>

View File

@ -10,6 +10,11 @@ class Foo {
e(); e();
} }
www() {
if (1 === 1)
return 'df';
}
async asyncFunction() { async asyncFunction() {
} }
} }

View File

@ -1,3 +1,4 @@
[MarkDown] Async method Foo.asyncFunction should describe return type Promise [MarkDown] Async method Foo.asyncFunction should describe return type Promise
[MarkDown] Method Foo.return42 is missing return type description [MarkDown] Method Foo.return42 is missing return type description
[MarkDown] Method Foo.returnNothing has unneeded description of return type [MarkDown] Method Foo.returnNothing has unneeded description of return type
[MarkDown] foo.www() has mistyped 'return' type declaration: expected exactly 'returns: ', found 'returns '.