mirror of
https://github.com/DavidAnson/markdownlint.git
synced 2025-09-22 05:40:48 +02:00
Move each rule implementation into its own file (fixes #83).
This commit is contained in:
parent
49e36f817c
commit
9ba143555d
42 changed files with 1289 additions and 1028 deletions
22
lib/md001.js
Normal file
22
lib/md001.js
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD001", "header-increment" ],
|
||||||
|
"description": "Header levels should only increment by one level at a time",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD001(params, onError) {
|
||||||
|
var prevLevel = 0;
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
var level = parseInt(token.tag.slice(1), 10);
|
||||||
|
if (prevLevel && (level > prevLevel)) {
|
||||||
|
shared.addErrorDetailIf(onError, token.lineNumber,
|
||||||
|
"h" + (prevLevel + 1), "h" + level);
|
||||||
|
}
|
||||||
|
prevLevel = level;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
22
lib/md002.js
Normal file
22
lib/md002.js
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD002", "first-header-h1" ],
|
||||||
|
"description": "First header should be a top level header",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD002(params, onError) {
|
||||||
|
var level = params.config.level || 1;
|
||||||
|
var tag = "h" + level;
|
||||||
|
params.tokens.every(function forToken(token) {
|
||||||
|
if (token.type === "heading_open") {
|
||||||
|
shared.addErrorDetailIf(onError, token.lineNumber, tag, token.tag);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
41
lib/md003.js
Normal file
41
lib/md003.js
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD003", "header-style" ],
|
||||||
|
"description": "Header style",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD003(params, onError) {
|
||||||
|
var style = params.config.style || "consistent";
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
var styleForToken = shared.headingStyleFor(token);
|
||||||
|
if (style === "consistent") {
|
||||||
|
style = styleForToken;
|
||||||
|
}
|
||||||
|
if (styleForToken !== style) {
|
||||||
|
var h12 = /h[12]/.test(token.tag);
|
||||||
|
var setextWithAtx =
|
||||||
|
(style === "setext_with_atx") &&
|
||||||
|
((h12 && (styleForToken === "setext")) ||
|
||||||
|
(!h12 && (styleForToken === "atx")));
|
||||||
|
var setextWithAtxClosed =
|
||||||
|
(style === "setext_with_atx_closed") &&
|
||||||
|
((h12 && (styleForToken === "setext")) ||
|
||||||
|
(!h12 && (styleForToken === "atx_closed")));
|
||||||
|
if (!setextWithAtx && !setextWithAtxClosed) {
|
||||||
|
var expected = style;
|
||||||
|
if (style === "setext_with_atx") {
|
||||||
|
expected = h12 ? "setext" : "atx";
|
||||||
|
} else if (style === "setext_with_atx_closed") {
|
||||||
|
expected = h12 ? "setext" : "atx_closed";
|
||||||
|
}
|
||||||
|
shared.addErrorDetailIf(onError, token.lineNumber,
|
||||||
|
expected, styleForToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
41
lib/md004.js
Normal file
41
lib/md004.js
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD004", "ul-style" ],
|
||||||
|
"description": "Unordered list style",
|
||||||
|
"tags": [ "bullet", "ul" ],
|
||||||
|
"function": function MD004(params, onError) {
|
||||||
|
var style = params.config.style || "consistent";
|
||||||
|
var expectedStyle = style;
|
||||||
|
var nestingStyles = [];
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
if (list.unordered) {
|
||||||
|
if (expectedStyle === "consistent") {
|
||||||
|
expectedStyle = shared.unorderedListStyleFor(list.items[0]);
|
||||||
|
}
|
||||||
|
list.items.forEach(function forItem(item) {
|
||||||
|
var itemStyle = shared.unorderedListStyleFor(item);
|
||||||
|
if (style === "sublist") {
|
||||||
|
var nesting = list.nesting;
|
||||||
|
if (!nestingStyles[nesting] &&
|
||||||
|
(itemStyle !== nestingStyles[nesting - 1])) {
|
||||||
|
nestingStyles[nesting] = itemStyle;
|
||||||
|
} else {
|
||||||
|
shared.addErrorDetailIf(onError, item.lineNumber,
|
||||||
|
nestingStyles[nesting], itemStyle, null,
|
||||||
|
shared.rangeFromRegExp(item.line, shared.listItemMarkerRe));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
shared.addErrorDetailIf(onError, item.lineNumber,
|
||||||
|
expectedStyle, itemStyle, null,
|
||||||
|
shared.rangeFromRegExp(item.line, shared.listItemMarkerRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md005.js
Normal file
21
lib/md005.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD005", "list-indent" ],
|
||||||
|
"description": "Inconsistent indentation for list items at the same level",
|
||||||
|
"tags": [ "bullet", "ul", "indentation" ],
|
||||||
|
"function": function MD005(params, onError) {
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
var indent = shared.indentFor(list.items[0]);
|
||||||
|
list.items.forEach(function forItem(item) {
|
||||||
|
shared.addErrorDetailIf(onError, item.lineNumber, indent,
|
||||||
|
shared.indentFor(item), null,
|
||||||
|
shared.rangeFromRegExp(item.line, shared.listItemMarkerRe));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md006.js
Normal file
21
lib/md006.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD006", "ul-start-left" ],
|
||||||
|
"description":
|
||||||
|
"Consider starting bulleted lists at the beginning of the line",
|
||||||
|
"tags": [ "bullet", "ul", "indentation" ],
|
||||||
|
"function": function MD006(params, onError) {
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
if (list.unordered && !list.nesting) {
|
||||||
|
shared.addErrorDetailIf(onError, list.open.lineNumber,
|
||||||
|
0, shared.indentFor(list.open), null,
|
||||||
|
shared.rangeFromRegExp(list.open.line, shared.listItemMarkerRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
26
lib/md007.js
Normal file
26
lib/md007.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD007", "ul-indent" ],
|
||||||
|
"description": "Unordered list indentation",
|
||||||
|
"tags": [ "bullet", "ul", "indentation" ],
|
||||||
|
"function": function MD007(params, onError) {
|
||||||
|
var optionsIndent = params.config.indent || 2;
|
||||||
|
var prevIndent = 0;
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
if (list.unordered && list.parentsUnordered) {
|
||||||
|
var indent = shared.indentFor(list.open);
|
||||||
|
if (indent > prevIndent) {
|
||||||
|
shared.addErrorDetailIf(onError, list.open.lineNumber,
|
||||||
|
prevIndent + optionsIndent, indent, null,
|
||||||
|
shared.rangeFromRegExp(list.open.line, shared.listItemMarkerRe));
|
||||||
|
}
|
||||||
|
prevIndent = indent;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
37
lib/md009.js
Normal file
37
lib/md009.js
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var trailingSpaceRe = /\s+$/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD009", "no-trailing-spaces" ],
|
||||||
|
"description": "Trailing spaces",
|
||||||
|
"tags": [ "whitespace" ],
|
||||||
|
"function": function MD009(params, onError) {
|
||||||
|
var brSpaces = params.config.br_spaces || 0;
|
||||||
|
var listItemEmptyLines = params.config.list_item_empty_lines;
|
||||||
|
var allowListItemEmptyLines =
|
||||||
|
(listItemEmptyLines === undefined) ? false : !!listItemEmptyLines;
|
||||||
|
var listItemLineNumbers = [];
|
||||||
|
if (allowListItemEmptyLines) {
|
||||||
|
shared.filterTokens(params, "list_item_open", function forToken(token) {
|
||||||
|
for (var i = token.map[0]; i < token.map[1]; i++) {
|
||||||
|
listItemLineNumbers.push(i + 1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
shared.forEachLine(params, function forLine(line, lineIndex) {
|
||||||
|
var lineNumber = lineIndex + 1;
|
||||||
|
if (trailingSpaceRe.test(line) &&
|
||||||
|
(listItemLineNumbers.indexOf(lineNumber) === -1)) {
|
||||||
|
var expected = (brSpaces < 2) ? 0 : brSpaces;
|
||||||
|
shared.addErrorDetailIf(onError, lineNumber,
|
||||||
|
expected, line.length - shared.trimRight(line).length, null,
|
||||||
|
shared.rangeFromRegExp(line, trailingSpaceRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
24
lib/md010.js
Normal file
24
lib/md010.js
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var tabRe = /\t+/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD010", "no-hard-tabs" ],
|
||||||
|
"description": "Hard tabs",
|
||||||
|
"tags": [ "whitespace", "hard_tab" ],
|
||||||
|
"function": function MD010(params, onError) {
|
||||||
|
var codeBlocks = params.config.code_blocks;
|
||||||
|
var includeCodeBlocks = (codeBlocks === undefined) ? true : !!codeBlocks;
|
||||||
|
shared.forEachLine(params, function forLine(line, lineIndex, inCode) {
|
||||||
|
if (tabRe.test(line) && (!inCode || includeCodeBlocks)) {
|
||||||
|
shared.addError(onError, lineIndex + 1,
|
||||||
|
"Column: " + (line.indexOf("\t") + 1), null,
|
||||||
|
shared.rangeFromRegExp(line, tabRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
22
lib/md011.js
Normal file
22
lib/md011.js
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var reversedLinkRe = /\([^)]+\)\[[^\]^][^\]]*]/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD011", "no-reversed-links" ],
|
||||||
|
"description": "Reversed link syntax",
|
||||||
|
"tags": [ "links" ],
|
||||||
|
"function": function MD011(params, onError) {
|
||||||
|
shared.forEachInlineChild(params, "text", function forToken(token) {
|
||||||
|
var match = reversedLinkRe.exec(token.content);
|
||||||
|
if (match) {
|
||||||
|
shared.addError(onError, token.lineNumber, match[0], null,
|
||||||
|
shared.rangeFromRegExp(token.line, reversedLinkRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md012.js
Normal file
21
lib/md012.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD012", "no-multiple-blanks" ],
|
||||||
|
"description": "Multiple consecutive blank lines",
|
||||||
|
"tags": [ "whitespace", "blank_lines" ],
|
||||||
|
"function": function MD012(params, onError) {
|
||||||
|
var maximum = params.config.maximum || 1;
|
||||||
|
var count = 0;
|
||||||
|
shared.forEachLine(params, function forLine(line, lineIndex, inCode) {
|
||||||
|
count = (inCode || line.trim().length) ? 0 : count + 1;
|
||||||
|
if (maximum < count) {
|
||||||
|
shared.addErrorDetailIf(onError, lineIndex + 1, maximum, count);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
63
lib/md013.js
Normal file
63
lib/md013.js
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var labelRe = /^\s*\[.*[^\\]]:/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD013", "line-length" ],
|
||||||
|
"description": "Line length",
|
||||||
|
"tags": [ "line_length" ],
|
||||||
|
"function": function MD013(params, onError) {
|
||||||
|
var lineLength = params.config.line_length || 80;
|
||||||
|
var codeBlocks = params.config.code_blocks;
|
||||||
|
var includeCodeBlocks = (codeBlocks === undefined) ? true : !!codeBlocks;
|
||||||
|
var tables = params.config.tables;
|
||||||
|
var includeTables = (tables === undefined) ? true : !!tables;
|
||||||
|
var headers = params.config.headers;
|
||||||
|
var includeHeaders = (headers === undefined) ? true : !!headers;
|
||||||
|
var headerLineNumbers = [];
|
||||||
|
if (!includeHeaders) {
|
||||||
|
shared.forEachHeading(params, function forHeading(heading) {
|
||||||
|
headerLineNumbers.push(heading.lineNumber);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
var tokenTypeMap = {
|
||||||
|
"em_open": "e",
|
||||||
|
"em_close": "E",
|
||||||
|
"link_open": "l",
|
||||||
|
"link_close": "L",
|
||||||
|
"strong_open": "s",
|
||||||
|
"strong_close": "S",
|
||||||
|
"text": "T"
|
||||||
|
};
|
||||||
|
var linkOnlyLineNumbers = [];
|
||||||
|
shared.filterTokens(params, "inline", function forToken(token) {
|
||||||
|
var childTokenTypes = "";
|
||||||
|
token.children.forEach(function forChild(child) {
|
||||||
|
if (child.type !== "text" || child.content !== "") {
|
||||||
|
childTokenTypes += tokenTypeMap[child.type] || "x";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (/^[es]*lT?L[ES]*$/.test(childTokenTypes)) {
|
||||||
|
linkOnlyLineNumbers.push(token.lineNumber);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
var longLineRe = new RegExp("^(.{" + lineLength + "})(.*\\s.*)$");
|
||||||
|
shared.forEachLine(params,
|
||||||
|
function forLine(line, lineIndex, inCode, onFence, inTable) {
|
||||||
|
var lineNumber = lineIndex + 1;
|
||||||
|
if ((includeCodeBlocks || !inCode) &&
|
||||||
|
(includeTables || !inTable) &&
|
||||||
|
(includeHeaders || (headerLineNumbers.indexOf(lineNumber)) < 0) &&
|
||||||
|
(linkOnlyLineNumbers.indexOf(lineNumber) < 0) &&
|
||||||
|
longLineRe.test(line) &&
|
||||||
|
!labelRe.test(line)) {
|
||||||
|
shared.addErrorDetailIf(onError, lineNumber, lineLength,
|
||||||
|
line.length, null, shared.rangeFromRegExp(line, longLineRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
28
lib/md014.js
Normal file
28
lib/md014.js
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var dollarCommandRe = /^(\s*)(\$\s)/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD014", "commands-show-output" ],
|
||||||
|
"description": "Dollar signs used before commands without showing output",
|
||||||
|
"tags": [ "code" ],
|
||||||
|
"function": function MD014(params, onError) {
|
||||||
|
[ "code_block", "fence" ].forEach(function forType(type) {
|
||||||
|
shared.filterTokens(params, type, function forToken(token) {
|
||||||
|
var allBlank = true;
|
||||||
|
if (token.content && token.content.split(shared.newLineRe)
|
||||||
|
.every(function forLine(line) {
|
||||||
|
return !line || (allBlank = false) || dollarCommandRe.test(line);
|
||||||
|
}) && !allBlank) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
token.content.split(shared.newLineRe)[0].trim(), null, null,
|
||||||
|
shared.rangeFromRegExp(token.line, dollarCommandRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
19
lib/md018.js
Normal file
19
lib/md018.js
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD018", "no-missing-space-atx" ],
|
||||||
|
"description": "No space after hash on atx style header",
|
||||||
|
"tags": [ "headers", "atx", "spaces" ],
|
||||||
|
"function": function MD018(params, onError) {
|
||||||
|
shared.forEachLine(params, function forLine(line, lineIndex, inCode) {
|
||||||
|
if (!inCode && /^#+[^#\s]/.test(line) && !/#$/.test(line)) {
|
||||||
|
shared.addErrorContext(onError, lineIndex + 1, line.trim(), null,
|
||||||
|
null, shared.rangeFromRegExp(line, shared.atxHeaderSpaceRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md019.js
Normal file
21
lib/md019.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD019", "no-multiple-space-atx" ],
|
||||||
|
"description": "Multiple spaces after hash on atx style header",
|
||||||
|
"tags": [ "headers", "atx", "spaces" ],
|
||||||
|
"function": function MD019(params, onError) {
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
if ((shared.headingStyleFor(token) === "atx") &&
|
||||||
|
/^#+\s\s/.test(token.line)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line.trim(),
|
||||||
|
null, null,
|
||||||
|
shared.rangeFromRegExp(token.line, shared.atxHeaderSpaceRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
25
lib/md020.js
Normal file
25
lib/md020.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var atxClosedHeaderNoSpaceRe = /(?:^#+[^#\s])|(?:[^#\s]#+\s*$)/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD020", "no-missing-space-closed-atx" ],
|
||||||
|
"description": "No space inside hashes on closed atx style header",
|
||||||
|
"tags": [ "headers", "atx_closed", "spaces" ],
|
||||||
|
"function": function MD020(params, onError) {
|
||||||
|
shared.forEachLine(params, function forLine(line, lineIndex, inCode) {
|
||||||
|
if (!inCode && /^#+[^#]*[^\\]#+$/.test(line)) {
|
||||||
|
var left = /^#+[^#\s]/.test(line);
|
||||||
|
var right = /[^#\s]#+$/.test(line);
|
||||||
|
if (left || right) {
|
||||||
|
shared.addErrorContext(onError, lineIndex + 1, line.trim(), left,
|
||||||
|
right, shared.rangeFromRegExp(line, atxClosedHeaderNoSpaceRe));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
26
lib/md021.js
Normal file
26
lib/md021.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var atxClosedHeaderSpaceRe = /(?:^#+\s\s+?\S)|(?:\S\s\s+?#+\s*$)/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD021", "no-multiple-space-closed-atx" ],
|
||||||
|
"description": "Multiple spaces inside hashes on closed atx style header",
|
||||||
|
"tags": [ "headers", "atx_closed", "spaces" ],
|
||||||
|
"function": function MD021(params, onError) {
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
if (shared.headingStyleFor(token) === "atx_closed") {
|
||||||
|
var left = /^#+\s\s/.test(token.line);
|
||||||
|
var right = /\s\s#+$/.test(token.line);
|
||||||
|
if (left || right) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line.trim(),
|
||||||
|
left, right,
|
||||||
|
shared.rangeFromRegExp(token.line, atxClosedHeaderSpaceRe));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
39
lib/md022.js
Normal file
39
lib/md022.js
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD022", "blanks-around-headers" ],
|
||||||
|
"description": "Headers should be surrounded by blank lines",
|
||||||
|
"tags": [ "headers", "blank_lines" ],
|
||||||
|
"function": function MD022(params, onError) {
|
||||||
|
var prevHeadingLineNumber = 0;
|
||||||
|
var prevMaxLineIndex = -1;
|
||||||
|
var needBlankLine = false;
|
||||||
|
params.tokens.forEach(function forToken(token) {
|
||||||
|
if (token.type === "heading_open") {
|
||||||
|
if ((token.map[0] - prevMaxLineIndex) === 0) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
token.line.trim());
|
||||||
|
}
|
||||||
|
} else if (token.type === "heading_close") {
|
||||||
|
needBlankLine = true;
|
||||||
|
}
|
||||||
|
if (token.map) {
|
||||||
|
if (needBlankLine) {
|
||||||
|
if ((token.map[0] - prevMaxLineIndex) === 0) {
|
||||||
|
shared.addErrorContext(onError, prevHeadingLineNumber,
|
||||||
|
params.lines[prevHeadingLineNumber - 1].trim());
|
||||||
|
}
|
||||||
|
needBlankLine = false;
|
||||||
|
}
|
||||||
|
prevMaxLineIndex = Math.max(prevMaxLineIndex, token.map[1]);
|
||||||
|
}
|
||||||
|
if (token.type === "heading_open") {
|
||||||
|
prevHeadingLineNumber = token.lineNumber;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md023.js
Normal file
21
lib/md023.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var spaceBeforeHeaderRe = /^\s+\S/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD023", "header-start-left" ],
|
||||||
|
"description": "Headers must start at the beginning of the line",
|
||||||
|
"tags": [ "headers", "spaces" ],
|
||||||
|
"function": function MD023(params, onError) {
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
if (spaceBeforeHeaderRe.test(token.line)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line, null,
|
||||||
|
null, shared.rangeFromRegExp(token.line, spaceBeforeHeaderRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
22
lib/md024.js
Normal file
22
lib/md024.js
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD024", "no-duplicate-header" ],
|
||||||
|
"description": "Multiple headers with the same content",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD024(params, onError) {
|
||||||
|
var knownContent = [];
|
||||||
|
shared.forEachHeading(params, function forHeading(heading, content) {
|
||||||
|
if (knownContent.indexOf(content) === -1) {
|
||||||
|
knownContent.push(content);
|
||||||
|
} else {
|
||||||
|
shared.addErrorContext(onError, heading.lineNumber,
|
||||||
|
heading.line.trim());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
26
lib/md025.js
Normal file
26
lib/md025.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD025", "single-h1" ],
|
||||||
|
"description": "Multiple top level headers in the same document",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD025(params, onError) {
|
||||||
|
var level = params.config.level || 1;
|
||||||
|
var tag = "h" + level;
|
||||||
|
var hasTopLevelHeading = false;
|
||||||
|
shared.filterTokens(params, "heading_open", function forToken(token) {
|
||||||
|
if (token.tag === tag) {
|
||||||
|
if (hasTopLevelHeading) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
token.line.trim());
|
||||||
|
} else if (token.lineNumber === 1) {
|
||||||
|
hasTopLevelHeading = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
23
lib/md026.js
Normal file
23
lib/md026.js
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD026", "no-trailing-punctuation" ],
|
||||||
|
"description": "Trailing punctuation in header",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD026(params, onError) {
|
||||||
|
var punctuation = params.config.punctuation || ".,;:!?";
|
||||||
|
var trailingPunctuationRe = new RegExp("[" + punctuation + "]$");
|
||||||
|
shared.forEachHeading(params, function forHeading(heading, content) {
|
||||||
|
var match = trailingPunctuationRe.exec(content);
|
||||||
|
if (match) {
|
||||||
|
shared.addError(onError, heading.lineNumber,
|
||||||
|
"Punctuation: '" + match[0] + "'", null,
|
||||||
|
shared.rangeFromRegExp(heading.line, trailingPunctuationRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
44
lib/md027.js
Normal file
44
lib/md027.js
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var spaceAfterBlockQuote = /^\s*(?:>\s+)+\S/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD027", "no-multiple-space-blockquote" ],
|
||||||
|
"description": "Multiple spaces after blockquote symbol",
|
||||||
|
"tags": [ "blockquote", "whitespace", "indentation" ],
|
||||||
|
"function": function MD027(params, onError) {
|
||||||
|
var blockquoteNesting = 0;
|
||||||
|
var listItemNesting = 0;
|
||||||
|
params.tokens.forEach(function forToken(token) {
|
||||||
|
if (token.type === "blockquote_open") {
|
||||||
|
blockquoteNesting++;
|
||||||
|
} else if (token.type === "blockquote_close") {
|
||||||
|
blockquoteNesting--;
|
||||||
|
} else if (token.type === "list_item_open") {
|
||||||
|
listItemNesting++;
|
||||||
|
} else if (token.type === "list_item_close") {
|
||||||
|
listItemNesting--;
|
||||||
|
} else if ((token.type === "inline") && (blockquoteNesting > 0)) {
|
||||||
|
var multipleSpaces = listItemNesting ?
|
||||||
|
/^(\s*>)+\s\s+>/.test(token.line) :
|
||||||
|
/^(\s*>)+\s\s/.test(token.line);
|
||||||
|
if (multipleSpaces) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line, null,
|
||||||
|
null, shared.rangeFromRegExp(token.line, spaceAfterBlockQuote));
|
||||||
|
}
|
||||||
|
token.content.split(shared.newLineRe)
|
||||||
|
.forEach(function forLine(line, offset) {
|
||||||
|
if (/^\s/.test(line)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber + offset,
|
||||||
|
"> " + line, null, null,
|
||||||
|
shared.rangeFromRegExp(line, spaceAfterBlockQuote));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md028.js
Normal file
21
lib/md028.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD028", "no-blanks-blockquote" ],
|
||||||
|
"description": "Blank line inside blockquote",
|
||||||
|
"tags": [ "blockquote", "whitespace" ],
|
||||||
|
"function": function MD028(params, onError) {
|
||||||
|
var prevToken = {};
|
||||||
|
params.tokens.forEach(function forToken(token) {
|
||||||
|
if ((token.type === "blockquote_open") &&
|
||||||
|
(prevToken.type === "blockquote_close")) {
|
||||||
|
shared.addError(onError, token.lineNumber - 1);
|
||||||
|
}
|
||||||
|
prevToken = token;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
37
lib/md029.js
Normal file
37
lib/md029.js
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var numberRe = /^[\s>]*([^.)]*)[.)]/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD029", "ol-prefix" ],
|
||||||
|
"description": "Ordered list item prefix",
|
||||||
|
"tags": [ "ol" ],
|
||||||
|
"function": function MD029(params, onError) {
|
||||||
|
var style = params.config.style || "one_or_ordered";
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
if (!list.unordered) {
|
||||||
|
var listStyle = style;
|
||||||
|
if (listStyle === "one_or_ordered") {
|
||||||
|
var second = (list.items.length > 1) &&
|
||||||
|
numberRe.exec(list.items[1].line);
|
||||||
|
listStyle = (second && (second[1] !== "1")) ? "ordered" : "one";
|
||||||
|
}
|
||||||
|
var number = 1;
|
||||||
|
list.items.forEach(function forItem(item) {
|
||||||
|
var match = numberRe.exec(item.line);
|
||||||
|
shared.addErrorDetailIf(onError, item.lineNumber,
|
||||||
|
String(number), !match || match[1],
|
||||||
|
"Style: " + (listStyle === "one" ? "1/1/1" : "1/2/3"),
|
||||||
|
shared.rangeFromRegExp(item.line, shared.listItemMarkerRe));
|
||||||
|
if (listStyle === "ordered") {
|
||||||
|
number++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
30
lib/md030.js
Normal file
30
lib/md030.js
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD030", "list-marker-space" ],
|
||||||
|
"description": "Spaces after list markers",
|
||||||
|
"tags": [ "ol", "ul", "whitespace" ],
|
||||||
|
"function": function MD030(params, onError) {
|
||||||
|
var ulSingle = params.config.ul_single || 1;
|
||||||
|
var olSingle = params.config.ol_single || 1;
|
||||||
|
var ulMulti = params.config.ul_multi || 1;
|
||||||
|
var olMulti = params.config.ol_multi || 1;
|
||||||
|
shared.flattenLists(params).forEach(function forList(list) {
|
||||||
|
var lineCount = list.lastLineIndex - list.open.map[0];
|
||||||
|
var allSingle = lineCount === list.items.length;
|
||||||
|
var expectedSpaces = list.unordered ?
|
||||||
|
(allSingle ? ulSingle : ulMulti) :
|
||||||
|
(allSingle ? olSingle : olMulti);
|
||||||
|
list.items.forEach(function forItem(item) {
|
||||||
|
var match = /^[\s>]*\S+(\s+)/.exec(item.line);
|
||||||
|
shared.addErrorDetailIf(onError, item.lineNumber,
|
||||||
|
expectedSpaces, (match ? match[1].length : 0), null,
|
||||||
|
shared.rangeFromRegExp(item.line, shared.listItemMarkerRe));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
20
lib/md031.js
Normal file
20
lib/md031.js
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD031", "blanks-around-fences" ],
|
||||||
|
"description": "Fenced code blocks should be surrounded by blank lines",
|
||||||
|
"tags": [ "code", "blank_lines" ],
|
||||||
|
"function": function MD031(params, onError) {
|
||||||
|
var lines = params.lines;
|
||||||
|
shared.forEachLine(params, function forLine(line, i, inCode, onFence) {
|
||||||
|
if (((onFence > 0) && (i - 1 >= 0) && lines[i - 1].length) ||
|
||||||
|
((onFence < 0) && (i + 1 < lines.length) && lines[i + 1].length)) {
|
||||||
|
shared.addErrorContext(onError, i + 1, lines[i].trim());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
38
lib/md032.js
Normal file
38
lib/md032.js
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var listItemMarkerInterruptsRe = /^[\s>]*(?:[*+-]|1\.)\s+/;
|
||||||
|
var blankOrListRe = /^[\s>]*($|\s)/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD032", "blanks-around-lists" ],
|
||||||
|
"description": "Lists should be surrounded by blank lines",
|
||||||
|
"tags": [ "bullet", "ul", "ol", "blank_lines" ],
|
||||||
|
"function": function MD032(params, onError) {
|
||||||
|
var inList = false;
|
||||||
|
var prevLine = "";
|
||||||
|
shared.forEachLine(params,
|
||||||
|
function forLine(line, lineIndex, inCode, onFence) {
|
||||||
|
if (!inCode || onFence) {
|
||||||
|
var lineTrim = line.trim();
|
||||||
|
var listMarker = shared.listItemMarkerRe.test(lineTrim);
|
||||||
|
if (listMarker && !inList && !blankOrListRe.test(prevLine)) {
|
||||||
|
// Check whether this list prefix can interrupt a paragraph
|
||||||
|
if (listItemMarkerInterruptsRe.test(lineTrim)) {
|
||||||
|
shared.addErrorContext(onError, lineIndex + 1, lineTrim);
|
||||||
|
} else {
|
||||||
|
listMarker = false;
|
||||||
|
}
|
||||||
|
} else if (!listMarker && inList && !blankOrListRe.test(line)) {
|
||||||
|
shared.addErrorContext(onError, lineIndex, lineTrim);
|
||||||
|
}
|
||||||
|
inList = listMarker;
|
||||||
|
}
|
||||||
|
prevLine = line;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
41
lib/md033.js
Normal file
41
lib/md033.js
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var htmlRe = /<[^>]*>/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD033", "no-inline-html" ],
|
||||||
|
"description": "Inline HTML",
|
||||||
|
"tags": [ "html" ],
|
||||||
|
"function": function MD033(params, onError) {
|
||||||
|
var allowedElements = (params.config.allowed_elements || [])
|
||||||
|
.map(function forElement(element) {
|
||||||
|
return element.toLowerCase();
|
||||||
|
});
|
||||||
|
function forToken(token) {
|
||||||
|
token.content.split(shared.newLineRe)
|
||||||
|
.forEach(function forLine(line, offset) {
|
||||||
|
var allowed = (line.match(/<[^/\s>!]*/g) || [])
|
||||||
|
.filter(function forElement(element) {
|
||||||
|
return element.length > 1;
|
||||||
|
})
|
||||||
|
.map(function forElement(element) {
|
||||||
|
return element.slice(1).toLowerCase();
|
||||||
|
})
|
||||||
|
.filter(function forElement(element) {
|
||||||
|
return allowedElements.indexOf(element) === -1;
|
||||||
|
});
|
||||||
|
if (allowed.length) {
|
||||||
|
shared.addError(onError, token.lineNumber + offset,
|
||||||
|
"Element: " + allowed[0], null,
|
||||||
|
shared.rangeFromRegExp(token.line, htmlRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
shared.filterTokens(params, "html_block", forToken);
|
||||||
|
shared.forEachInlineChild(params, "html_inline", forToken);
|
||||||
|
}
|
||||||
|
};
|
29
lib/md034.js
Normal file
29
lib/md034.js
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD034", "no-bare-urls" ],
|
||||||
|
"description": "Bare URL used",
|
||||||
|
"tags": [ "links", "url" ],
|
||||||
|
"function": function MD034(params, onError) {
|
||||||
|
shared.filterTokens(params, "inline", function forToken(token) {
|
||||||
|
var inLink = false;
|
||||||
|
token.children.forEach(function forChild(child) {
|
||||||
|
var match = null;
|
||||||
|
if (child.type === "link_open") {
|
||||||
|
inLink = true;
|
||||||
|
} else if (child.type === "link_close") {
|
||||||
|
inLink = false;
|
||||||
|
} else if ((child.type === "text") &&
|
||||||
|
!inLink &&
|
||||||
|
(match = shared.bareUrlRe.exec(child.content))) {
|
||||||
|
shared.addErrorContext(onError, child.lineNumber, match[0], null,
|
||||||
|
null, shared.rangeFromRegExp(child.line, shared.bareUrlRe));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
21
lib/md035.js
Normal file
21
lib/md035.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD035", "hr-style" ],
|
||||||
|
"description": "Horizontal rule style",
|
||||||
|
"tags": [ "hr" ],
|
||||||
|
"function": function MD035(params, onError) {
|
||||||
|
var style = params.config.style || "consistent";
|
||||||
|
shared.filterTokens(params, "hr", function forToken(token) {
|
||||||
|
var lineTrim = token.line.trim();
|
||||||
|
if (style === "consistent") {
|
||||||
|
style = lineTrim;
|
||||||
|
}
|
||||||
|
shared.addErrorDetailIf(onError, token.lineNumber, style, lineTrim);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
54
lib/md036.js
Normal file
54
lib/md036.js
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD036", "no-emphasis-as-header" ],
|
||||||
|
"description": "Emphasis used instead of a header",
|
||||||
|
"tags": [ "headers", "emphasis" ],
|
||||||
|
"function": function MD036(params, onError) {
|
||||||
|
var punctuation = params.config.punctuation || ".,;:!?";
|
||||||
|
var re = new RegExp("[" + punctuation + "]$");
|
||||||
|
function base(token) {
|
||||||
|
if (token.type === "paragraph_open") {
|
||||||
|
return function inParagraph(t) {
|
||||||
|
// Always paragraph_open/inline/paragraph_close,
|
||||||
|
// omit (t.type === "inline")
|
||||||
|
var children = t.children.filter(function notEmptyText(child) {
|
||||||
|
return (child.type !== "text") || (child.content !== "");
|
||||||
|
});
|
||||||
|
if ((children.length === 3) &&
|
||||||
|
((children[0].type === "strong_open") ||
|
||||||
|
(children[0].type === "em_open")) &&
|
||||||
|
(children[1].type === "text") &&
|
||||||
|
!re.test(children[1].content)) {
|
||||||
|
shared.addErrorContext(onError, t.lineNumber,
|
||||||
|
children[1].content);
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
};
|
||||||
|
} else if (token.type === "blockquote_open") {
|
||||||
|
return function inBlockquote(t) {
|
||||||
|
if (t.type !== "blockquote_close") {
|
||||||
|
return inBlockquote;
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
};
|
||||||
|
} else if (token.type === "list_item_open") {
|
||||||
|
return function inListItem(t) {
|
||||||
|
if (t.type !== "list_item_close") {
|
||||||
|
return inListItem;
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
}
|
||||||
|
var state = base;
|
||||||
|
params.tokens.forEach(function forToken(token) {
|
||||||
|
state = state(token);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
29
lib/md037.js
Normal file
29
lib/md037.js
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD037", "no-space-in-emphasis" ],
|
||||||
|
"description": "Spaces inside emphasis markers",
|
||||||
|
"tags": [ "whitespace", "emphasis" ],
|
||||||
|
"function": function MD037(params, onError) {
|
||||||
|
shared.forEachInlineChild(params, "text", function forToken(token) {
|
||||||
|
var left = true;
|
||||||
|
var match = /\s(\*\*?|__?)\s.+\1/.exec(token.content);
|
||||||
|
if (!match) {
|
||||||
|
left = false;
|
||||||
|
match = /(\*\*?|__?).+\s\1\s/.exec(token.content);
|
||||||
|
}
|
||||||
|
if (match) {
|
||||||
|
var text = match[0].trim();
|
||||||
|
var line = params.lines[token.lineNumber - 1];
|
||||||
|
var column = line.indexOf(text) + 1;
|
||||||
|
var length = text.length;
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
text, left, !left, [ column, length ]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
34
lib/md038.js
Normal file
34
lib/md038.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var inlineCodeSpansRe = /(?:^|[^\\])((`+)((?:.*?[^`])|)\2(?!`))/g;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD038", "no-space-in-code" ],
|
||||||
|
"description": "Spaces inside code span elements",
|
||||||
|
"tags": [ "whitespace", "code" ],
|
||||||
|
"function": function MD038(params, onError) {
|
||||||
|
shared.forEachInlineChild(params, "code_inline",
|
||||||
|
function forToken(token) {
|
||||||
|
var line = params.lines[token.lineNumber - 1];
|
||||||
|
var match = null;
|
||||||
|
while ((match = inlineCodeSpansRe.exec(line)) !== null) {
|
||||||
|
var inlineCodeSpan = match[1];
|
||||||
|
var content = match[3];
|
||||||
|
var length = inlineCodeSpan.length;
|
||||||
|
var column = match.index + 1 + (match[0].length - length);
|
||||||
|
var range = [ column, length ];
|
||||||
|
if (/^\s([^`]|$)/.test(content)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
inlineCodeSpan, true, false, range);
|
||||||
|
} else if (/[^`]\s$/.test(content)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
inlineCodeSpan, false, true, range);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
36
lib/md039.js
Normal file
36
lib/md039.js
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var spaceInLinkRe = /\[(?:\s+(?:[^\]]*?)\s*|(?:[^\]]*?)\s+)](?=\(\S*\))/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD039", "no-space-in-links" ],
|
||||||
|
"description": "Spaces inside link text",
|
||||||
|
"tags": [ "whitespace", "links" ],
|
||||||
|
"function": function MD039(params, onError) {
|
||||||
|
shared.filterTokens(params, "inline", function forToken(token) {
|
||||||
|
var inLink = false;
|
||||||
|
var linkText = "";
|
||||||
|
token.children.forEach(function forChild(child) {
|
||||||
|
if (child.type === "link_open") {
|
||||||
|
inLink = true;
|
||||||
|
linkText = "";
|
||||||
|
} else if (child.type === "link_close") {
|
||||||
|
inLink = false;
|
||||||
|
var left = shared.trimLeft(linkText).length !== linkText.length;
|
||||||
|
var right = shared.trimRight(linkText).length !== linkText.length;
|
||||||
|
if (left || right) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber,
|
||||||
|
"[" + linkText + "]", left, right,
|
||||||
|
shared.rangeFromRegExp(token.line, spaceInLinkRe));
|
||||||
|
}
|
||||||
|
} else if (inLink) {
|
||||||
|
linkText += child.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
18
lib/md040.js
Normal file
18
lib/md040.js
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD040", "fenced-code-language" ],
|
||||||
|
"description": "Fenced code blocks should have a language specified",
|
||||||
|
"tags": [ "code", "language" ],
|
||||||
|
"function": function MD040(params, onError) {
|
||||||
|
shared.filterTokens(params, "fence", function forToken(token) {
|
||||||
|
if (!token.info.trim()) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
36
lib/md041.js
Normal file
36
lib/md041.js
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD041", "first-line-h1" ],
|
||||||
|
"description": "First line in file should be a top level header",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD041(params, onError) {
|
||||||
|
var level = params.config.level || 1;
|
||||||
|
var frontMatterTitle = params.config.front_matter_title;
|
||||||
|
var tag = "h" + level;
|
||||||
|
var frontMatterTitleRe =
|
||||||
|
new RegExp(frontMatterTitle || "^\\s*title\\s*[:=]", "i");
|
||||||
|
params.tokens.every(function forToken(token, index) {
|
||||||
|
if (token.type === "heading_open") {
|
||||||
|
if (!((token.lineNumber === 1) || (index > 0)) ||
|
||||||
|
(token.tag !== tag)) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
} else if (token.type === "html_block") {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (((frontMatterTitle !== undefined) && !frontMatterTitle) ||
|
||||||
|
!params.frontMatterLines.some(function forLine(line) {
|
||||||
|
return frontMatterTitleRe.test(line);
|
||||||
|
})) {
|
||||||
|
shared.addErrorContext(onError, token.lineNumber, token.line);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
40
lib/md042.js
Normal file
40
lib/md042.js
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
var emptyLinkRe = /\[[^\]]*](?:\((?:#?|(?:<>))\))/;
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD042", "no-empty-links" ],
|
||||||
|
"description": "No empty links",
|
||||||
|
"tags": [ "links" ],
|
||||||
|
"function": function MD042(params, onError) {
|
||||||
|
shared.filterTokens(params, "inline", function forToken(token) {
|
||||||
|
var inLink = false;
|
||||||
|
var linkText = "";
|
||||||
|
var emptyLink = false;
|
||||||
|
token.children.forEach(function forChild(child) {
|
||||||
|
if (child.type === "link_open") {
|
||||||
|
inLink = true;
|
||||||
|
linkText = "";
|
||||||
|
child.attrs.forEach(function forAttr(attr) {
|
||||||
|
if (attr[0] === "href" && (!attr[1] || (attr[1] === "#"))) {
|
||||||
|
emptyLink = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (child.type === "link_close") {
|
||||||
|
inLink = false;
|
||||||
|
if (emptyLink) {
|
||||||
|
shared.addErrorContext(onError, child.lineNumber,
|
||||||
|
"[" + linkText + "]()", null, null,
|
||||||
|
shared.rangeFromRegExp(child.line, emptyLinkRe));
|
||||||
|
}
|
||||||
|
} else if (inLink) {
|
||||||
|
linkText += child.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
44
lib/md043.js
Normal file
44
lib/md043.js
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD043", "required-headers" ],
|
||||||
|
"description": "Required header structure",
|
||||||
|
"tags": [ "headers" ],
|
||||||
|
"function": function MD043(params, onError) {
|
||||||
|
var requiredHeaders = params.config.headers;
|
||||||
|
if (requiredHeaders) {
|
||||||
|
var levels = {};
|
||||||
|
[ 1, 2, 3, 4, 5, 6 ].forEach(function forLevel(level) {
|
||||||
|
levels["h" + level] = "######".substr(-level);
|
||||||
|
});
|
||||||
|
var i = 0;
|
||||||
|
var optional = false;
|
||||||
|
var errorCount = 0;
|
||||||
|
shared.forEachHeading(params, function forHeading(heading, content) {
|
||||||
|
if (!errorCount) {
|
||||||
|
var actual = levels[heading.tag] + " " + content;
|
||||||
|
var expected = requiredHeaders[i++] || "[None]";
|
||||||
|
if (expected === "*") {
|
||||||
|
optional = true;
|
||||||
|
} else if (expected.toLowerCase() === actual.toLowerCase()) {
|
||||||
|
optional = false;
|
||||||
|
} else if (optional) {
|
||||||
|
i--;
|
||||||
|
} else {
|
||||||
|
shared.addErrorDetailIf(onError, heading.lineNumber,
|
||||||
|
expected, actual);
|
||||||
|
errorCount++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if ((i < requiredHeaders.length) && !errorCount) {
|
||||||
|
shared.addErrorContext(onError, params.lines.length,
|
||||||
|
requiredHeaders[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
47
lib/md044.js
Normal file
47
lib/md044.js
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD044", "proper-names" ],
|
||||||
|
"description": "Proper names should have the correct capitalization",
|
||||||
|
"tags": [ "spelling" ],
|
||||||
|
"function": function MD044(params, onError) {
|
||||||
|
var names = params.config.names || [];
|
||||||
|
var codeBlocks = params.config.code_blocks;
|
||||||
|
var includeCodeBlocks = (codeBlocks === undefined) ? true : !!codeBlocks;
|
||||||
|
names.forEach(function forName(name) {
|
||||||
|
var escapedName = shared.escapeForRegExp(name);
|
||||||
|
var namePattern = "\\S*\\b(" + escapedName + ")\\b\\S*";
|
||||||
|
var anyNameRe = new RegExp(namePattern, "gi");
|
||||||
|
function forToken(token) {
|
||||||
|
var fenceOffset = (token.type === "fence") ? 1 : 0;
|
||||||
|
token.content.split(shared.newLineRe)
|
||||||
|
.forEach(function forLine(line, index) {
|
||||||
|
var match = null;
|
||||||
|
while ((match = anyNameRe.exec(line)) !== null) {
|
||||||
|
var fullMatch = match[0];
|
||||||
|
if (!shared.bareUrlRe.test(fullMatch)) {
|
||||||
|
var wordMatch = fullMatch
|
||||||
|
.replace(/^\W*/, "").replace(/\W*$/, "");
|
||||||
|
if (names.indexOf(wordMatch) === -1) {
|
||||||
|
var lineNumber = token.lineNumber + index + fenceOffset;
|
||||||
|
var range = [ match.index + 1, wordMatch.length ];
|
||||||
|
shared.addErrorDetailIf(onError, lineNumber,
|
||||||
|
name, match[1], null, range);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
shared.forEachInlineChild(params, "text", forToken);
|
||||||
|
if (includeCodeBlocks) {
|
||||||
|
shared.forEachInlineChild(params, "code_inline", forToken);
|
||||||
|
shared.filterTokens(params, "code_block", forToken);
|
||||||
|
shared.filterTokens(params, "fence", forToken);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
18
lib/md045.js
Normal file
18
lib/md045.js
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var shared = require("./shared");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD045", "no-alt-text" ],
|
||||||
|
"description": "Images should have alternate text (alt text)",
|
||||||
|
"tags": [ "accessibility", "images" ],
|
||||||
|
"function": function MD045(params, onError) {
|
||||||
|
shared.forEachInlineChild(params, "image", function forToken(token) {
|
||||||
|
if (token.content === "") {
|
||||||
|
shared.addError(onError, token.lineNumber);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
1069
lib/rules.js
1069
lib/rules.js
File diff suppressed because it is too large
Load diff
Loading…
Add table
Add a link
Reference in a new issue