2015-02-23 23:39:20 -08:00
|
|
|
"use strict";
|
|
|
|
|
2015-03-08 23:08:43 -07:00
|
|
|
var shared = require("./shared");
|
|
|
|
|
2015-03-04 18:23:19 -08:00
|
|
|
function indentFor(token) {
|
2015-03-02 23:52:39 -08:00
|
|
|
return token.line.length - token.line.trimLeft().length;
|
2015-02-25 18:19:36 -08:00
|
|
|
}
|
|
|
|
|
2015-03-04 18:23:19 -08:00
|
|
|
function headingStyleFor(token) {
|
2015-02-27 22:06:54 -08:00
|
|
|
if ((token.lines[1] - token.lines[0]) === 1) {
|
2015-03-06 09:21:55 -08:00
|
|
|
if (/#\s*$/.test(token.line)) {
|
2015-02-27 22:06:54 -08:00
|
|
|
return "atx_closed";
|
|
|
|
}
|
|
|
|
return "atx";
|
|
|
|
}
|
|
|
|
return "setext";
|
|
|
|
}
|
|
|
|
|
2015-03-04 18:23:19 -08:00
|
|
|
function unorderedListStyleFor(token) {
|
2015-03-02 23:52:39 -08:00
|
|
|
switch (token.line.trimLeft().substr(0, 1)) {
|
2015-03-01 22:56:52 -08:00
|
|
|
case "-":
|
|
|
|
return "dash";
|
|
|
|
case "+":
|
|
|
|
return "plus";
|
2015-03-10 23:10:06 -07:00
|
|
|
case "*":
|
2015-03-01 22:56:52 -08:00
|
|
|
default:
|
2015-03-10 23:10:06 -07:00
|
|
|
return "asterisk";
|
2015-03-01 22:56:52 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-04 18:23:19 -08:00
|
|
|
function filterTokens(tokens, tokenA, tokenB) {
|
|
|
|
return tokens.filter(function filterToken(token) {
|
|
|
|
return ((token.type === tokenA) || (token.type === tokenB));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-03-09 00:31:07 -07:00
|
|
|
function forEachLine(params, callback) {
|
|
|
|
var codeBlocks = [];
|
|
|
|
filterTokens(params.tokens, "code_block")
|
|
|
|
.forEach(function forToken(token) {
|
|
|
|
for (var i = token.lines[0]; i < token.lines[1]; i++) {
|
|
|
|
codeBlocks.push(i);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
var inFence = false;
|
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
|
|
var onFence = /^(```|~~~)/.test(line);
|
|
|
|
if (onFence) {
|
|
|
|
inFence = !inFence;
|
|
|
|
}
|
|
|
|
var codeBlock = (codeBlocks.indexOf(lineIndex) !== -1);
|
|
|
|
callback(line, lineIndex, inFence || codeBlock, onFence);
|
|
|
|
});
|
2015-02-23 23:39:20 -08:00
|
|
|
}
|
|
|
|
|
2015-03-10 23:10:06 -07:00
|
|
|
function flattenLists(tokens, filterBy) {
|
|
|
|
var lists = [];
|
|
|
|
var stack = [];
|
|
|
|
var current = null;
|
|
|
|
var lastWithLines = null;
|
|
|
|
tokens.forEach(function forToken(token) {
|
|
|
|
if ((token.type === "bullet_list_open") ||
|
|
|
|
(token.type === "ordered_list_open")) {
|
|
|
|
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")) {
|
|
|
|
current.lastLineIndex = lastWithLines.lines[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") {
|
|
|
|
current.items.push(token);
|
|
|
|
} else if (token.lines) {
|
|
|
|
lastWithLines = token;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return lists;
|
|
|
|
}
|
|
|
|
|
2015-02-24 23:50:37 -08:00
|
|
|
module.exports = [
|
2015-02-25 18:19:36 -08:00
|
|
|
{
|
|
|
|
"name": "MD001",
|
|
|
|
"desc": "Header levels should only increment by one level at a time",
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD001(params, errors) {
|
2015-02-25 18:19:36 -08:00
|
|
|
var prevLevel = 0;
|
2015-03-04 18:23:19 -08:00
|
|
|
filterTokens(params.tokens, "heading_open")
|
|
|
|
.forEach(function forToken(token) {
|
|
|
|
if (prevLevel && (token.hLevel > prevLevel + 1)) {
|
|
|
|
errors.push(token.lineNumber);
|
|
|
|
}
|
|
|
|
prevLevel = token.hLevel;
|
|
|
|
});
|
2015-02-25 18:19:36 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-02-25 18:00:08 -08:00
|
|
|
{
|
|
|
|
"name": "MD002",
|
|
|
|
"desc": "First header should be a h1 header",
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD002(params, errors) {
|
|
|
|
params.tokens.every(function forToken(token) {
|
2015-02-25 18:19:36 -08:00
|
|
|
if (token.type === "heading_open") {
|
|
|
|
if (token.hLevel !== 1) {
|
2015-03-02 23:52:39 -08:00
|
|
|
errors.push(token.lineNumber);
|
2015-02-25 18:19:36 -08:00
|
|
|
}
|
2015-02-25 18:00:08 -08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-02-27 22:06:54 -08:00
|
|
|
{
|
|
|
|
"name": "MD003",
|
|
|
|
"desc": "Header style",
|
|
|
|
"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");
|
2015-02-27 22:06:54 -08:00
|
|
|
if ((style === "consistent") && headings.length) {
|
2015-03-04 18:23:19 -08:00
|
|
|
style = headingStyleFor(headings[0]);
|
2015-02-27 22:06:54 -08:00
|
|
|
}
|
|
|
|
headings.forEach(function forToken(token) {
|
2015-03-04 18:23:19 -08:00
|
|
|
if (headingStyleFor(token) !== style) {
|
2015-03-02 23:52:39 -08:00
|
|
|
errors.push(token.lineNumber);
|
2015-02-27 22:06:54 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-01 22:56:52 -08:00
|
|
|
{
|
|
|
|
"name": "MD004",
|
|
|
|
"desc": "Unordered list style",
|
|
|
|
"func": function MD004(params, errors) {
|
|
|
|
var style = params.options.style || "consistent";
|
2015-03-10 23:10:06 -07:00
|
|
|
flattenLists(params.tokens, false).forEach(function forList(list) {
|
|
|
|
if (style === "consistent") {
|
|
|
|
style = unorderedListStyleFor(list.items[0]);
|
2015-03-01 22:56:52 -08:00
|
|
|
}
|
2015-03-10 23:10:06 -07:00
|
|
|
list.items.forEach(function forItem(item) {
|
|
|
|
if (unorderedListStyleFor(item) !== style) {
|
|
|
|
errors.push(item.lineNumber);
|
|
|
|
}
|
|
|
|
});
|
2015-03-01 22:56:52 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-02 23:30:40 -08:00
|
|
|
{
|
|
|
|
"name": "MD005",
|
|
|
|
"desc": "Inconsistent indentation for list items at the same level",
|
|
|
|
"func": function MD005(params, errors) {
|
2015-03-10 23:10:06 -07:00
|
|
|
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-10 23:10:06 -07: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",
|
|
|
|
"func": function MD006(params, errors) {
|
2015-03-10 23:10:06 -07:00
|
|
|
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",
|
|
|
|
"func": function MD007(params, errors) {
|
|
|
|
var optionsIndent = params.options.indent || 2;
|
|
|
|
var prevIndent = 0;
|
2015-03-10 23:10:06 -07:00
|
|
|
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",
|
|
|
|
"func": function MD009(params, errors) {
|
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
2015-03-06 09:21:55 -08:00
|
|
|
if (/\s$/.test(line)) {
|
2015-03-03 09:29:13 -08:00
|
|
|
errors.push(lineIndex + 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD010",
|
|
|
|
"desc": "Hard tabs",
|
|
|
|
"func": function MD010(params, errors) {
|
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
2015-03-06 09:21:55 -08:00
|
|
|
if (/\t/.test(line)) {
|
2015-03-03 09:29:13 -08:00
|
|
|
errors.push(lineIndex + 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-03 18:28:59 -08:00
|
|
|
{
|
|
|
|
"name": "MD011",
|
|
|
|
"desc": "Reversed link syntax",
|
|
|
|
"func": function MD011(params, errors) {
|
2015-03-04 18:23:19 -08:00
|
|
|
filterTokens(params.tokens, "inline")
|
|
|
|
.forEach(function forToken(token) {
|
|
|
|
filterTokens(token.children, "text")
|
|
|
|
.forEach(function forChild(child) {
|
2015-03-06 09:21:55 -08:00
|
|
|
if (/\([^)]+\)\[[^\]]+\]/.test(child.content)) {
|
2015-03-04 18:23:19 -08:00
|
|
|
errors.push(token.lineNumber);
|
|
|
|
}
|
|
|
|
});
|
2015-03-03 18:28:59 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:09:46 -08:00
|
|
|
{
|
|
|
|
"name": "MD012",
|
|
|
|
"desc": "Multiple consecutive blank lines",
|
|
|
|
"func": function MD012(params, errors) {
|
|
|
|
var prevLine = "-";
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode) {
|
2015-03-04 18:09:46 -08:00
|
|
|
line = line.trim();
|
2015-03-09 00:31:07 -07:00
|
|
|
if (!inCode && !line.length && !prevLine.length) {
|
2015-03-04 18:09:46 -08:00
|
|
|
errors.push(lineIndex + 1);
|
|
|
|
}
|
|
|
|
prevLine = line;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:36:30 -08:00
|
|
|
{
|
|
|
|
"name": "MD013",
|
|
|
|
"desc": "Line length",
|
|
|
|
"func": function MD013(params, errors) {
|
|
|
|
var lineLength = params.options.line_length || 80;
|
|
|
|
var regex = new RegExp("^.{" + lineLength + "}.*\\s");
|
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
|
|
if (regex.test(line)) {
|
|
|
|
errors.push(lineIndex + 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-05 23:30:01 -08:00
|
|
|
{
|
|
|
|
"name": "MD014",
|
|
|
|
"desc": "Dollar signs used before commands without showing output",
|
|
|
|
"func": function MD014(params, errors) {
|
|
|
|
filterTokens(params.tokens, "code_block", "fence")
|
|
|
|
.forEach(function forToken(token) {
|
|
|
|
if (token.content && token.content
|
2015-03-08 23:08:43 -07:00
|
|
|
.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);
|
|
|
|
})) {
|
2015-03-06 09:21:55 -08:00
|
|
|
errors.push(token.lineNumber);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD018",
|
|
|
|
"desc": "No space after hash on atx style header",
|
|
|
|
"func": function MD018(params, errors) {
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode) {
|
|
|
|
if (!inCode && /^#+[^#\s]/.test(line) && !/#$/.test(line)) {
|
2015-03-06 09:21:55 -08:00
|
|
|
errors.push(lineIndex + 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD019",
|
|
|
|
"desc": "Multiple spaces after hash on atx style header",
|
|
|
|
"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",
|
|
|
|
"func": function MD020(params, errors) {
|
2015-03-09 00:31:07 -07:00
|
|
|
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",
|
|
|
|
"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
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-08 23:08:43 -07:00
|
|
|
{
|
|
|
|
"name": "MD022",
|
|
|
|
"desc": "Headers should be surrounded by 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.lines[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.lines[0] + offset);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (token.lines) {
|
|
|
|
if (needBlankLine) {
|
|
|
|
if ((token.lines[0] - prevMaxLineIndex) === 0) {
|
|
|
|
errors.push(prevHeadingLineNumber);
|
|
|
|
}
|
|
|
|
needBlankLine = false;
|
|
|
|
}
|
|
|
|
prevMaxLineIndex = Math.max(prevMaxLineIndex, token.lines[1]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-11 09:30:34 -07:00
|
|
|
{
|
|
|
|
"name": "MD023",
|
|
|
|
"desc": "Headers must start at the beginning of the line",
|
|
|
|
"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",
|
|
|
|
"func": function MD024(params, errors) {
|
|
|
|
var content = [];
|
|
|
|
var inHeading = false;
|
|
|
|
params.tokens.forEach(function forToken(token) {
|
|
|
|
if (token.type === "heading_open") {
|
|
|
|
inHeading = true;
|
|
|
|
} else if (token.type === "heading_close") {
|
|
|
|
inHeading = false;
|
|
|
|
} else if ((token.type === "inline") && inHeading) {
|
|
|
|
if (content.indexOf(token.content) === -1) {
|
|
|
|
content.push(token.content);
|
|
|
|
} else {
|
|
|
|
errors.push(token.lineNumber);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD025",
|
|
|
|
"desc": "Multiple top level headers in the same document",
|
|
|
|
"func": function MD025(params, errors) {
|
|
|
|
var hasTopLevelHeading = false;
|
|
|
|
filterTokens(params.tokens, "heading_open")
|
|
|
|
.forEach(function forToken(token) {
|
|
|
|
if (token.hLevel === 1) {
|
|
|
|
if (hasTopLevelHeading) {
|
|
|
|
errors.push(token.lineNumber);
|
|
|
|
} else if (token.lineNumber === 1) {
|
|
|
|
hasTopLevelHeading = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:09:46 -08:00
|
|
|
{
|
|
|
|
"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;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-07 22:46:45 -08:00
|
|
|
{
|
|
|
|
"name": "MD029",
|
|
|
|
"desc": "Ordered list item prefix",
|
|
|
|
"func": function MD029(params, errors) {
|
|
|
|
var style = params.options.style || "one";
|
2015-03-10 23:10:06 -07:00
|
|
|
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++;
|
|
|
|
}
|
2015-03-10 23:10:06 -07:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD030",
|
|
|
|
"desc": "Spaces after list markers",
|
|
|
|
"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.lines[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
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-02-24 23:50:37 -08:00
|
|
|
{
|
|
|
|
"name": "MD031",
|
|
|
|
"desc": "Fenced code blocks should be surrounded by blank lines",
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD031(params, errors) {
|
2015-02-24 23:50:37 -08:00
|
|
|
// Some parsers have trouble detecting fenced code blocks without
|
|
|
|
// surrounding whitespace, so examine the lines directly.
|
2015-03-09 00:31:07 -07:00
|
|
|
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);
|
2015-02-23 23:39:20 -08:00
|
|
|
}
|
2015-02-24 23:50:37 -08:00
|
|
|
});
|
|
|
|
}
|
2015-02-24 23:35:34 -08:00
|
|
|
},
|
|
|
|
|
2015-02-24 23:50:37 -08:00
|
|
|
{
|
|
|
|
"name": "MD032",
|
|
|
|
"desc": "Lists should be surrounded by blank lines",
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD032(params, errors) {
|
2015-02-24 23:50:37 -08:00
|
|
|
// Some parsers have trouble detecting lists without surrounding
|
|
|
|
// whitespace, so examine the lines directly.
|
|
|
|
var inList = false;
|
|
|
|
var prevLine = "";
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode, onFence) {
|
|
|
|
if (!inCode || onFence) {
|
2015-03-06 09:21:55 -08:00
|
|
|
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);
|
2015-03-06 09:21:55 -08:00
|
|
|
} else if (!listMarker && inList && !/^($|\s)/.test(line)) {
|
2015-03-03 09:29:13 -08:00
|
|
|
errors.push(lineIndex);
|
2015-02-24 23:50:37 -08:00
|
|
|
}
|
|
|
|
inList = listMarker;
|
2015-02-24 23:35:34 -08:00
|
|
|
}
|
2015-03-09 00:31:07 -07:00
|
|
|
inList = inList && !onFence;
|
2015-02-24 23:50:37 -08:00
|
|
|
prevLine = line;
|
|
|
|
});
|
|
|
|
}
|
2015-02-23 23:39:20 -08:00
|
|
|
}
|
2015-02-24 23:50:37 -08:00
|
|
|
];
|