markdownlint/lib/rules.js

771 lines
22 KiB
JavaScript
Raw Normal View History

"use strict";
var shared = require("./shared");
// Returns the indent for a token
2015-03-04 18:23:19 -08:00
function indentFor(token) {
return token.line.length - token.line.trimLeft().length;
}
// Returns the heading style for a heading token
2015-03-04 18:23:19 -08:00
function headingStyleFor(token) {
if ((token.map[1] - token.map[0]) === 1) {
if (/#\s*$/.test(token.line)) {
return "atx_closed";
}
return "atx";
}
return "setext";
}
// Returns the unordered list style for a list item token
2015-03-04 18:23:19 -08:00
function unorderedListStyleFor(token) {
switch (token.line.trimLeft().substr(0, 1)) {
case "-":
return "dash";
case "+":
return "plus";
case "*":
default:
return "asterisk";
}
}
// Filters a list of tokens by type
function filterTokens(tokens, typeA, typeB) {
2015-03-04 18:23:19 -08:00
return tokens.filter(function filterToken(token) {
return ((token.type === typeA) || (token.type === typeB));
2015-03-04 18:23:19 -08:00
});
}
// Calls the provided function for each line (with context)
function forEachLine(params, callback) {
// Identify lines in code blocks
var codeLines = [];
filterTokens(params.tokens, "code_block")
.forEach(function forToken(token) {
for (var i = token.map[0]; i < token.map[1]; i++) {
codeLines.push(i);
}
});
// Identify lines in code fences (with info about transitions)
var inFence = false;
params.lines.forEach(function forLine(line, lineIndex) {
var onFence = /^(```|~~~)/.test(line);
if (onFence) {
inFence = !inFence;
}
var inCodeBlock = (codeLines.indexOf(lineIndex) !== -1);
callback(line, lineIndex, inFence || inCodeBlock, onFence);
});
}
2015-04-15 18:24:42 -07:00
// Calls the provided function for each specified inline child token
function forEachInlineChild(params, type, callback) {
filterTokens(params.tokens, "inline")
.forEach(function forToken(token) {
2015-04-15 18:24:42 -07:00
filterTokens(token.children, type)
.forEach(function forChild(child) {
callback(child, token);
});
});
}
// Calls the provided function for each heading's content
2015-03-11 18:40:46 -07:00
function forEachHeading(params, callback) {
var heading = null;
params.tokens.forEach(function forToken(token) {
if (token.type === "heading_open") {
heading = token;
} else if (token.type === "heading_close") {
heading = null;
} else if ((token.type === "inline") && heading) {
callback(heading, token.content);
}
});
}
// Returns (nested) lists as a flat array (in order)
function flattenLists(tokens, filterBy) {
var lists = [];
var stack = [];
var current = null;
var lastWithMap = null;
tokens.forEach(function forToken(token) {
if ((token.type === "bullet_list_open") ||
(token.type === "ordered_list_open")) {
// Save current context and start a new one
stack.push(current);
current = {
"ordered": (token.type === "ordered_list_open"),
"open": token,
"items": [],
"nesting": stack.length - 1,
"lastLineIndex": -1,
"insert": lists.length
};
} else if ((token.type === "bullet_list_close") ||
(token.type === "ordered_list_close")) {
// Finalize current context and restore previous
current.lastLineIndex = lastWithMap.map[1];
if ((filterBy === undefined) || (filterBy === current.ordered)) {
lists.splice(current.insert, 0, current);
delete current.insert;
}
current = stack.pop();
} else if (token.type === "list_item_open") {
// Add list item
current.items.push(token);
} else if (token.map) {
// Track last token with map
lastWithMap = token;
}
});
return lists;
}
module.exports = [
{
"name": "MD001",
"desc": "Header levels should only increment by one level at a time",
"tags": [ "headers" ],
"func": function MD001(params, errors) {
var prevLevel = 0;
2015-03-04 18:23:19 -08:00
filterTokens(params.tokens, "heading_open")
.forEach(function forToken(token) {
var level = parseInt(token.tag.slice(1), 10);
if (prevLevel && (level > prevLevel + 1)) {
2015-03-04 18:23:19 -08:00
errors.push(token.lineNumber);
}
prevLevel = level;
2015-03-04 18:23:19 -08:00
});
}
},
{
"name": "MD002",
"desc": "First header should be a h1 header",
"tags": [ "headers" ],
"func": function MD002(params, errors) {
params.tokens.every(function forToken(token) {
if (token.type === "heading_open") {
if (token.tag !== "h1") {
errors.push(token.lineNumber);
}
return false;
}
return true;
});
}
},
{
"name": "MD003",
"desc": "Header style",
"tags": [ "headers" ],
"func": function MD003(params, errors) {
var style = params.options.style || "consistent";
2015-03-04 18:23:19 -08:00
var headings = filterTokens(params.tokens, "heading_open");
if ((style === "consistent") && headings.length) {
2015-03-04 18:23:19 -08:00
style = headingStyleFor(headings[0]);
}
headings.forEach(function forToken(token) {
2015-03-04 18:23:19 -08:00
if (headingStyleFor(token) !== style) {
errors.push(token.lineNumber);
}
});
}
},
{
"name": "MD004",
"desc": "Unordered list style",
"tags": [ "bullet", "ul" ],
"func": function MD004(params, errors) {
var style = params.options.style || "consistent";
flattenLists(params.tokens, false).forEach(function forList(list) {
if (style === "consistent") {
style = unorderedListStyleFor(list.items[0]);
}
list.items.forEach(function forItem(item) {
if (unorderedListStyleFor(item) !== style) {
errors.push(item.lineNumber);
}
});
});
}
},
2015-03-02 23:30:40 -08:00
{
"name": "MD005",
"desc": "Inconsistent indentation for list items at the same level",
"tags": [ "bullet", "ul", "indentation" ],
2015-03-02 23:30:40 -08:00
"func": function MD005(params, errors) {
flattenLists(params.tokens).forEach(function forList(list) {
var indent = indentFor(list.items[0]);
list.items.forEach(function forItem(item) {
if (indentFor(item) !== indent) {
errors.push(item.lineNumber);
2015-03-04 18:23:19 -08:00
}
});
});
2015-03-02 23:30:40 -08:00
}
},
2015-03-03 09:29:13 -08:00
{
"name": "MD006",
"desc": "Consider starting bulleted lists at the beginning of the line",
"tags": [ "bullet", "ul", "indentation" ],
2015-03-03 09:29:13 -08:00
"func": function MD006(params, errors) {
flattenLists(params.tokens, false).forEach(function forList(list) {
if (!list.nesting && indentFor(list.open)) {
errors.push(list.open.lineNumber);
2015-03-03 09:29:13 -08:00
}
});
}
},
2015-03-02 23:30:40 -08:00
{
"name": "MD007",
"desc": "Unordered list indentation",
"tags": [ "bullet", "ul", "indentation" ],
2015-03-02 23:30:40 -08:00
"func": function MD007(params, errors) {
var optionsIndent = params.options.indent || 2;
var prevIndent = 0;
flattenLists(params.tokens, false).forEach(function forList(list) {
var indent = indentFor(list.open);
if ((indent > prevIndent) &&
((indent - prevIndent) !== optionsIndent)) {
errors.push(list.open.lineNumber);
}
prevIndent = indent;
});
2015-03-02 23:30:40 -08:00
}
},
2015-03-03 09:29:13 -08:00
{
"name": "MD009",
"desc": "Trailing spaces",
"tags": [ "whitespace" ],
2015-03-03 09:29:13 -08:00
"func": function MD009(params, errors) {
2015-04-14 09:07:25 -07:00
var brSpaces = params.options.br_spaces || 0;
2015-03-03 09:29:13 -08:00
params.lines.forEach(function forLine(line, lineIndex) {
2015-04-14 09:07:25 -07:00
if (/\s$/.test(line) &&
((brSpaces < 2) ||
(line.length - line.trimRight().length !== brSpaces))) {
2015-03-03 09:29:13 -08:00
errors.push(lineIndex + 1);
}
});
}
},
{
"name": "MD010",
"desc": "Hard tabs",
"tags": [ "whitespace", "hard_tab" ],
2015-03-03 09:29:13 -08:00
"func": function MD010(params, errors) {
params.lines.forEach(function forLine(line, lineIndex) {
if (/\t/.test(line)) {
2015-03-03 09:29:13 -08:00
errors.push(lineIndex + 1);
}
});
}
},
{
"name": "MD011",
"desc": "Reversed link syntax",
"tags": [ "links" ],
"func": function MD011(params, errors) {
2015-04-15 18:24:42 -07:00
forEachInlineChild(params, "text", function forToken(token) {
if (/\([^)]+\)\[[^\]]+\]/.test(token.content)) {
errors.push(token.lineNumber);
}
});
}
},
{
"name": "MD012",
"desc": "Multiple consecutive blank lines",
"tags": [ "whitespace", "blank_lines" ],
"func": function MD012(params, errors) {
var prevLine = "-";
forEachLine(params, function forLine(line, lineIndex, inCode) {
line = line.trim();
if (!inCode && !line.length && !prevLine.length) {
errors.push(lineIndex + 1);
}
prevLine = line;
});
}
},
2015-03-04 18:36:30 -08:00
{
"name": "MD013",
"desc": "Line length",
"tags": [ "line_length" ],
2015-03-04 18:36:30 -08:00
"func": function MD013(params, errors) {
var lineLength = params.options.line_length || 80;
var re = new RegExp("^.{" + lineLength + "}.*\\s");
2015-03-04 18:36:30 -08:00
params.lines.forEach(function forLine(line, lineIndex) {
if (re.test(line)) {
2015-03-04 18:36:30 -08:00
errors.push(lineIndex + 1);
}
});
}
},
2015-03-05 23:30:01 -08:00
{
"name": "MD014",
"desc": "Dollar signs used before commands without showing output",
"tags": [ "code" ],
2015-03-05 23:30:01 -08:00
"func": function MD014(params, errors) {
filterTokens(params.tokens, "code_block", "fence")
.forEach(function forToken(token) {
2015-03-11 18:40:46 -07:00
if (token.content && token.content.split(shared.newLineRe)
2015-03-05 23:30:01 -08:00
.filter(function filterLine(line) {
return line;
}).every(function forLine(line) {
return /^\$\s/.test(line);
})) {
errors.push(token.lineNumber);
}
});
}
},
{
"name": "MD018",
"desc": "No space after hash on atx style header",
"tags": [ "headers", "atx", "spaces" ],
"func": function MD018(params, errors) {
forEachLine(params, function forLine(line, lineIndex, inCode) {
if (!inCode && /^#+[^#\s]/.test(line) && !/#$/.test(line)) {
errors.push(lineIndex + 1);
}
});
}
},
{
"name": "MD019",
"desc": "Multiple spaces after hash on atx style header",
"tags": [ "headers", "atx", "spaces" ],
"func": function MD019(params, errors) {
filterTokens(params.tokens, "heading_open")
.forEach(function forToken(token) {
if ((headingStyleFor(token) === "atx") &&
/^#+\s\s/.test(token.line)) {
errors.push(token.lineNumber);
2015-03-06 18:16:16 -08:00
}
});
}
},
{
"name": "MD020",
"desc": "No space inside hashes on closed atx style header",
"tags": [ "headers", "atx_closed", "spaces" ],
2015-03-06 18:16:16 -08:00
"func": function MD020(params, errors) {
forEachLine(params, function forLine(line, lineIndex, inCode) {
if (!inCode && /^#+[^#]*[^\\]#+$/.test(line) &&
2015-03-06 18:16:16 -08:00
(/^#+[^#\s]/.test(line) || /[^#\s]#+$/.test(line))) {
errors.push(lineIndex + 1);
}
});
}
},
{
"name": "MD021",
"desc": "Multiple spaces inside hashes on closed atx style header",
"tags": [ "headers", "atx_closed", "spaces" ],
2015-03-06 18:16:16 -08:00
"func": function MD021(params, errors) {
filterTokens(params.tokens, "heading_open")
.forEach(function forToken(token) {
if ((headingStyleFor(token) === "atx_closed") &&
(/^#+\s\s/.test(token.line) || /\s\s#+$/.test(token.line))) {
errors.push(token.lineNumber);
2015-03-05 23:30:01 -08:00
}
});
}
},
{
"name": "MD022",
"desc": "Headers should be surrounded by blank lines",
"tags": [ "headers", "blank_lines" ],
"func": function MD022(params, errors) {
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) {
errors.push(token.lineNumber);
}
prevHeadingLineNumber = token.lineNumber;
} else if (token.type === "heading_close") {
needBlankLine = true;
} else if (token.type === "inline") {
token.content.split(shared.newLineRe)
.forEach(function forLine(line, offset) {
if (/^(-+|=+)\s*$/.test(line)) {
errors.push(token.map[0] + offset);
}
});
}
if (token.map) {
if (needBlankLine) {
if ((token.map[0] - prevMaxLineIndex) === 0) {
errors.push(prevHeadingLineNumber);
}
needBlankLine = false;
}
prevMaxLineIndex = Math.max(prevMaxLineIndex, token.map[1]);
}
});
}
},
2015-03-11 09:30:34 -07:00
{
"name": "MD023",
"desc": "Headers must start at the beginning of the line",
"tags": [ "headers", "spaces" ],
2015-03-11 09:30:34 -07:00
"func": function MD023(params, errors) {
filterTokens(params.tokens, "heading_open")
.forEach(function forToken(token) {
if (/^\s/.test(token.line)) {
errors.push(token.lineNumber);
}
});
}
},
{
"name": "MD024",
"desc": "Multiple headers with the same content",
"tags": [ "headers" ],
2015-03-11 09:30:34 -07:00
"func": function MD024(params, errors) {
2015-03-11 18:40:46 -07:00
var knownContent = [];
forEachHeading(params, function forHeading(heading, content) {
if (knownContent.indexOf(content) === -1) {
knownContent.push(content);
} else {
errors.push(heading.lineNumber);
2015-03-11 09:30:34 -07:00
}
});
}
},
{
"name": "MD025",
"desc": "Multiple top level headers in the same document",
"tags": [ "headers" ],
2015-03-11 09:30:34 -07:00
"func": function MD025(params, errors) {
var hasTopLevelHeading = false;
filterTokens(params.tokens, "heading_open")
.forEach(function forToken(token) {
if (token.tag === "h1") {
2015-03-11 09:30:34 -07:00
if (hasTopLevelHeading) {
errors.push(token.lineNumber);
} else if (token.lineNumber === 1) {
hasTopLevelHeading = true;
}
}
});
}
},
2015-03-11 18:40:46 -07:00
{
"name": "MD026",
"desc": "Trailing punctuation in header",
"tags": [ "headers" ],
2015-03-11 18:40:46 -07:00
"func": function MD026(params, errors) {
var punctuation = params.options.punctuation || ".,;:!?";
var re = new RegExp("[" + punctuation + "]$");
forEachHeading(params, function forHeading(heading, content) {
if (re.test(content)) {
errors.push(heading.lineNumber);
}
});
}
},
{
"name": "MD027",
"desc": "Multiple spaces after blockquote symbol",
"tags": [ "blockquote", "whitespace", "indentation" ],
2015-03-11 18:40:46 -07:00
"func": function MD027(params, errors) {
var inBlockquote = false;
params.tokens.forEach(function forToken(token) {
if (token.type === "blockquote_open") {
inBlockquote = true;
} else if (token.type === "blockquote_close") {
inBlockquote = false;
} else if ((token.type === "inline") && inBlockquote) {
token.content.split(shared.newLineRe)
.forEach(function forLine(line, offset) {
if (/^\s/.test(line) ||
(!offset && /^\s*>\s\s/.test(token.line))) {
errors.push(token.lineNumber + offset);
}
});
}
});
}
},
{
"name": "MD028",
"desc": "Blank line inside blockquote",
"tags": [ "blockquote", "whitespace" ],
"func": function MD028(params, errors) {
var prevToken = {};
params.tokens.forEach(function forToken(token) {
if ((token.type === "blockquote_open") &&
(prevToken.type === "blockquote_close")) {
errors.push(token.lineNumber - 1);
}
prevToken = token;
});
}
},
2015-03-07 22:46:45 -08:00
{
"name": "MD029",
"desc": "Ordered list item prefix",
"tags": [ "ol" ],
2015-03-07 22:46:45 -08:00
"func": function MD029(params, errors) {
var style = params.options.style || "one";
flattenLists(params.tokens, true).forEach(function forList(list) {
var number = 1;
list.items.forEach(function forItem(item) {
var re = new RegExp("^\\s*" + String(number) + "\\. ");
if (!re.test(item.line)) {
errors.push(item.lineNumber);
2015-03-07 22:46:45 -08:00
}
if (style === "ordered") {
number++;
}
});
});
}
},
{
"name": "MD030",
"desc": "Spaces after list markers",
"tags": [ "ol", "ul", "whitespace" ],
"func": function MD030(params, errors) {
var ulSingle = params.options.ul_single || 1;
var olSingle = params.options.ol_single || 1;
var ulMulti = params.options.ul_multi || 1;
var olMulti = params.options.ol_multi || 1;
flattenLists(params.tokens).forEach(function forList(list) {
var lineCount = list.lastLineIndex - list.open.map[0];
var allSingle = lineCount === list.items.length;
var expectedSpaces = list.ordered ?
(allSingle ? olSingle : olMulti) :
(allSingle ? ulSingle : ulMulti);
list.items.forEach(function forItem(item) {
var match = /^\s*\S+(\s+)/.exec(item.line);
if (match[1].length !== expectedSpaces) {
errors.push(item.lineNumber);
}
});
2015-03-07 22:46:45 -08:00
});
}
},
{
"name": "MD031",
"desc": "Fenced code blocks should be surrounded by blank lines",
"tags": [ "code", "blank_lines" ],
"func": function MD031(params, errors) {
forEachLine(params, function forLine(line, lineIndex, inCode, onFence) {
if (onFence &&
((inCode && (lineIndex - 1 >= 0) &&
params.lines[lineIndex - 1].length) ||
(!inCode && (lineIndex + 1 < params.lines.length) &&
params.lines[lineIndex + 1].length))) {
errors.push(lineIndex + 1);
}
});
}
},
{
"name": "MD032",
"desc": "Lists should be surrounded by blank lines",
"tags": [ "bullet", "ul", "ol", "blank_lines" ],
"func": function MD032(params, errors) {
var inList = false;
var prevLine = "";
forEachLine(params, function forLine(line, lineIndex, inCode, onFence) {
if (!inCode || onFence) {
var listMarker = /^([\*\+\-]|(\d+\.))\s/.test(line.trim());
if (listMarker && !inList && !/^($|\s)/.test(prevLine)) {
2015-03-03 09:29:13 -08:00
errors.push(lineIndex + 1);
} else if (!listMarker && inList && !/^($|\s)/.test(line)) {
2015-03-03 09:29:13 -08:00
errors.push(lineIndex);
}
inList = listMarker;
}
inList = inList && !onFence;
prevLine = line;
});
}
},
{
"name": "MD033",
"desc": "Inline HTML",
"tags": [ "html" ],
"func": function MD033(params, errors) {
filterTokens(params.tokens, "html_inline", "html_block")
.forEach(function forToken(token) {
errors.push(token.lineNumber);
});
}
},
{
"name": "MD034",
"desc": "Bare URL used",
"tags": [ "links", "url" ],
"func": function MD034(params, errors) {
filterTokens(params.tokens, "inline")
.forEach(function forToken(token) {
var inLink = false;
token.children.forEach(function forChild(child) {
if (child.type === "link_open") {
inLink = true;
} else if (child.type === "link_close") {
inLink = false;
} else if ((child.type === "text") &&
!inLink &&
/https?:\/\//.test(child.content)) {
errors.push(child.lineNumber);
}
});
});
}
2015-04-14 09:40:16 -07:00
},
{
"name": "MD035",
"desc": "Horizontal rule style",
"tags": [ "hr" ],
"func": function MD035(params, errors) {
var style = params.options.style || "consistent";
var horizontalRules = filterTokens(params.tokens, "hr");
if ((style === "consistent") && horizontalRules.length) {
style = horizontalRules[0].line;
}
horizontalRules.forEach(function forToken(token) {
if (token.line !== style) {
errors.push(token.lineNumber);
}
});
}
2015-04-14 22:37:56 -07:00
},
{
"name": "MD036",
"desc": "Emphasis used instead of a header",
"tags": [ "headers", "emphasis" ],
"func": function MD036(params, errors) {
function base(token) {
if (token.type === "paragraph_open") {
return function inParagraph(t) {
if ((t.type === "inline") &&
(t.children.length === 3) &&
((t.children[0].type === "strong_open") ||
(t.children[0].type === "em_open")) &&
(t.children[1].type === "text")) {
errors.push(t.lineNumber);
2015-04-14 22:37:56 -07:00
}
};
} else if (token.type === "blockquote_open") {
return function inBlockquote(t) {
if (t.type !== "blockquote_close") {
return inBlockquote;
}
};
}
}
var state = base;
params.tokens.forEach(function forToken(token) {
state = state(token) || base;
});
}
},
{
"name": "MD037",
"desc": "Spaces inside emphasis markers",
"tags": [ "whitespace", "emphasis" ],
"func": function MD037(params, errors) {
2015-04-15 18:24:42 -07:00
forEachInlineChild(params, "text", function forToken(token) {
if (/\s(\*\*?|__?)\s.+\1/.test(token.content) ||
/(\*\*?|__?).+\s\1\s/.test(token.content)) {
errors.push(token.lineNumber);
}
});
}
2015-04-15 18:24:42 -07:00
},
{
"name": "MD038",
"desc": "Spaces inside code span elements",
"tags": [ "whitespace", "code" ],
"func": function MD038(params, errors) {
forEachInlineChild(params, "code_inline",
function forToken(token, inline) {
if (inline.content.indexOf("`" + token.content + "`") === -1) {
errors.push(token.lineNumber);
}
});
}
},
{
"name": "MD039",
"desc": "Spaces inside link text",
"tags": [ "whitespace", "links" ],
"func": function MD039(params, errors) {
filterTokens(params.tokens, "inline")
.forEach(function forToken(token) {
var inLink = false;
token.children.forEach(function forChild(child) {
if (child.type === "link_open") {
inLink = true;
} else if (child.type === "link_close") {
inLink = false;
} else if ((child.type === "text") &&
inLink &&
(child.content.trim().length !== child.content.length)) {
errors.push(child.lineNumber);
}
});
});
}
2015-04-16 09:13:56 -07:00
},
{
"name": "MD040",
"desc": "Fenced code blocks should have a language specified",
"tags": [ "code", "language" ],
"func": function MD040(params, errors) {
filterTokens(params.tokens, "fence")
.forEach(function forToken(token) {
if (!token.info.trim()) {
errors.push(token.lineNumber);
}
});
}
}
];