mirror of
https://github.com/DavidAnson/markdownlint.git
synced 2025-09-22 05:40:48 +02:00
301 lines
8.2 KiB
JavaScript
301 lines
8.2 KiB
JavaScript
"use strict";
|
|
|
|
function indentFrom(token) {
|
|
return token.line.length - token.line.trimLeft().length;
|
|
}
|
|
|
|
function headingStyleFrom(token) {
|
|
if ((token.lines[1] - token.lines[0]) === 1) {
|
|
if (token.line.match(/#\s*$/)) {
|
|
return "atx_closed";
|
|
}
|
|
return "atx";
|
|
}
|
|
return "setext";
|
|
}
|
|
|
|
function unorderedListStyleFrom(token) {
|
|
switch (token.line.trimLeft().substr(0, 1)) {
|
|
case "*":
|
|
return "asterisk";
|
|
case "-":
|
|
return "dash";
|
|
case "+":
|
|
return "plus";
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function padAndTrim(lines) {
|
|
return [].concat(
|
|
"",
|
|
lines.map(function mapLine(line) {
|
|
return line.trim();
|
|
}),
|
|
"");
|
|
}
|
|
|
|
module.exports = [
|
|
{
|
|
"name": "MD001",
|
|
"desc": "Header levels should only increment by one level at a time",
|
|
"func": function MD001(params, errors) {
|
|
var prevLevel = 0;
|
|
params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "heading_open");
|
|
}).forEach(function forToken(token) {
|
|
if (prevLevel && (token.hLevel > prevLevel + 1)) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
prevLevel = token.hLevel;
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD002",
|
|
"desc": "First header should be a h1 header",
|
|
"func": function MD002(params, errors) {
|
|
params.tokens.every(function forToken(token) {
|
|
if (token.type === "heading_open") {
|
|
if (token.hLevel !== 1) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD003",
|
|
"desc": "Header style",
|
|
"func": function MD003(params, errors) {
|
|
var style = params.options.style || "consistent";
|
|
var headings = params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "heading_open");
|
|
});
|
|
if ((style === "consistent") && headings.length) {
|
|
style = headingStyleFrom(headings[0]);
|
|
}
|
|
headings.forEach(function forToken(token) {
|
|
if (headingStyleFrom(token) !== style) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD004",
|
|
"desc": "Unordered list style",
|
|
"func": function MD004(params, errors) {
|
|
var style = params.options.style || "consistent";
|
|
var listItems = params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "list_item_open");
|
|
});
|
|
if ((style === "consistent") && listItems.length) {
|
|
style = unorderedListStyleFrom(listItems[0]);
|
|
}
|
|
listItems.forEach(function forToken(token) {
|
|
if (unorderedListStyleFrom(token) !== style) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD005",
|
|
"desc": "Inconsistent indentation for list items at the same level",
|
|
"func": function MD005(params, errors) {
|
|
var indentLevels = [];
|
|
params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "list_item_open");
|
|
}).forEach(function forToken(token) {
|
|
var indentLevel = indentFrom(token);
|
|
if (!indentLevels[token.level]) {
|
|
indentLevels[token.level] = indentLevel;
|
|
} else if (indentLevel !== indentLevels[token.level]) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD006",
|
|
"desc": "Consider starting bulleted lists at the beginning of the line",
|
|
"func": function MD006(params, errors) {
|
|
var inList = 0;
|
|
params.tokens.filter(function filterToken(token) {
|
|
switch (token.type) {
|
|
case "bullet_list_open":
|
|
inList++;
|
|
return inList === 1;
|
|
case "bullet_list_close":
|
|
inList--;
|
|
return false;
|
|
default:
|
|
return false;
|
|
}
|
|
}).forEach(function forToken(token) {
|
|
if (indentFrom(token) !== 0) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD007",
|
|
"desc": "Unordered list indentation",
|
|
"func": function MD007(params, errors) {
|
|
var optionsIndent = params.options.indent || 2;
|
|
var prevIndent = 0;
|
|
params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "bullet_list_open");
|
|
}).forEach(function forToken(token) {
|
|
var indent = indentFrom(token);
|
|
if ((indent > prevIndent) &&
|
|
((indent - prevIndent) !== optionsIndent)) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
prevIndent = indent;
|
|
});
|
|
}
|
|
},
|
|
|
|
// MD008 does not exist
|
|
|
|
{
|
|
"name": "MD009",
|
|
"desc": "Trailing spaces",
|
|
"func": function MD009(params, errors) {
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
if (line.match(/\s$/)) {
|
|
errors.push(lineIndex + 1);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD010",
|
|
"desc": "Hard tabs",
|
|
"func": function MD010(params, errors) {
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
if (line.match(/\t/)) {
|
|
errors.push(lineIndex + 1);
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD011",
|
|
"desc": "Reversed link syntax",
|
|
"func": function MD011(params, errors) {
|
|
params.tokens.filter(function filterToken(token) {
|
|
return (token.type === "inline");
|
|
}).forEach(function forToken(token) {
|
|
token.children.filter(function filterChild(child) {
|
|
return (child.type === "text");
|
|
}).forEach(function forChild(child) {
|
|
if (child.content.match(/\([^)]+\)\[[^\]]+\]/)) {
|
|
errors.push(token.lineNumber);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD012",
|
|
"desc": "Multiple consecutive blank lines",
|
|
"func": function MD012(params, errors) {
|
|
var exclusions = [];
|
|
params.tokens.filter(function filterToken(token) {
|
|
return ((token.type === "code_block") || (token.type === "fence"));
|
|
}).forEach(function forToken(token) {
|
|
for (var i = token.lines[0] ; i < token.lines[1] ; i++) {
|
|
exclusions.push(i);
|
|
}
|
|
});
|
|
var prevLine = "-";
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
line = line.trim();
|
|
if (!line.length && !prevLine.length &&
|
|
(exclusions.indexOf(lineIndex) === -1)) {
|
|
errors.push(lineIndex + 1);
|
|
}
|
|
prevLine = line;
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD028",
|
|
"desc": "Blank line inside blockquote",
|
|
"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;
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD031",
|
|
"desc": "Fenced code blocks should be surrounded by blank lines",
|
|
"func": function MD031(params, errors) {
|
|
// Some parsers have trouble detecting fenced code blocks without
|
|
// surrounding whitespace, so examine the lines directly.
|
|
var lines = padAndTrim(params.lines);
|
|
var inCode = false;
|
|
lines.forEach(function forLine(line, lineNumber) {
|
|
if (line.match(/^(```|~~~)/)) {
|
|
inCode = !inCode;
|
|
if ((inCode && lines[lineNumber - 1].length) ||
|
|
(!inCode && lines[lineNumber + 1].length)) {
|
|
errors.push(lineNumber);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
{
|
|
"name": "MD032",
|
|
"desc": "Lists should be surrounded by blank lines",
|
|
"func": function MD032(params, errors) {
|
|
// Some parsers have trouble detecting lists without surrounding
|
|
// whitespace, so examine the lines directly.
|
|
var inList = false;
|
|
var inCode = false;
|
|
var prevLine = "";
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
if (!inCode) {
|
|
var listMarker = line.trim().match(/^([\*\+\-]|(\d+\.))\s/);
|
|
if (listMarker && !inList && !prevLine.match(/^($|\s)/)) {
|
|
errors.push(lineIndex + 1);
|
|
} else if (!listMarker && inList && !line.match(/^($|\s)/)) {
|
|
errors.push(lineIndex);
|
|
}
|
|
inList = listMarker;
|
|
}
|
|
if (line.trim().match(/^(```|~~~)/)) {
|
|
inCode = !inCode;
|
|
inList = false;
|
|
}
|
|
prevLine = line;
|
|
});
|
|
}
|
|
}
|
|
];
|