2015-02-23 23:39:20 -08:00
|
|
|
"use strict";
|
|
|
|
|
2015-03-08 23:08:43 -07:00
|
|
|
var shared = require("./shared");
|
|
|
|
|
2016-10-31 22:53:46 -07:00
|
|
|
// Range regular expressions
|
|
|
|
var atxClosedHeaderNoSpaceRe = /(?:^#+\S)|(?:\S#+\s*$)/;
|
|
|
|
var atxClosedHeaderSpaceRe = /(?:^#+\s\s+?\S)|(?:\S\s\s+?#+\s*$)/;
|
|
|
|
var atxHeaderSpaceRe = /^#+\s*\S/;
|
|
|
|
var bareUrlRe = /(?:http|ftp)s?:\/\/[^\s]*/;
|
|
|
|
var dollarCommandRe = /^(\s*)(\$\s)/;
|
2017-01-15 15:14:41 -08:00
|
|
|
var emptyLinkRe = /\[[^\]]*](?:\((?:#?|(?:<>))\))|(?:\[[^\]]*])/;
|
2016-10-31 22:53:46 -07:00
|
|
|
var htmlRe = /<[^>]*>/;
|
2016-11-02 22:09:16 -07:00
|
|
|
var listItemMarkerRe = /^[\s>]*(?:[*+-]|\d+\.)\s+/;
|
2017-01-15 14:41:58 -08:00
|
|
|
var listItemMarkerInterruptsRe = /^[\s>]*(?:[*+-]|1\.)\s+/;
|
2016-11-02 22:09:16 -07:00
|
|
|
var reversedLinkRe = /\([^)]+\)\[[^\]^][^\]]*]/;
|
2016-10-31 22:53:46 -07:00
|
|
|
var spaceAfterBlockQuote = />\s+\S/;
|
|
|
|
var spaceBeforeHeaderRe = /^\s+\S/;
|
|
|
|
var spaceInsideCodeRe = /`(?:(?:\s[^`]*)|(?:[^`]*\s))`/;
|
|
|
|
var spaceInsideEmphasisRe = /(\*\*?|__?)(?:(?:\s.+)|(?:.+\s))\1/;
|
2016-11-02 22:09:16 -07:00
|
|
|
var spaceInsideLinkRe = /\[(?:(?:\s[^\]]*)|(?:[^\]]*\s))](?=\(\S*\))/;
|
2016-10-31 22:53:46 -07:00
|
|
|
var tabRe = /\t+/;
|
|
|
|
var trailingPunctuationRe = /.$/;
|
|
|
|
var trailingSpaceRe = /\s+$/;
|
|
|
|
var defaultLineLength = 80;
|
|
|
|
function longLineReFunc(options) {
|
|
|
|
var lineLength = options.line_length || defaultLineLength;
|
|
|
|
return new RegExp("^(.{" + lineLength + "})(.*\\s.*)$");
|
|
|
|
}
|
|
|
|
|
2016-10-16 21:46:02 -07:00
|
|
|
// Escapes a string for use in a RegExp
|
|
|
|
function escapeForRegExp(str) {
|
2016-11-02 22:09:16 -07:00
|
|
|
return str.replace(/[-/\\^$*+?.()|[\]{}]/g, "\\$&");
|
2016-10-16 21:46:02 -07:00
|
|
|
}
|
|
|
|
|
2015-03-16 22:31:18 -07:00
|
|
|
// Returns the indent for a token
|
2015-03-04 18:23:19 -08:00
|
|
|
function indentFor(token) {
|
2016-09-10 19:23:50 -07:00
|
|
|
var line = token.line.replace(/^[\s>]*(> |>)/, "");
|
|
|
|
return line.length - line.trimLeft().length;
|
2015-02-25 18:19:36 -08:00
|
|
|
}
|
|
|
|
|
2015-03-16 22:31:18 -07:00
|
|
|
// Returns the heading style for a heading token
|
2015-03-04 18:23:19 -08:00
|
|
|
function headingStyleFor(token) {
|
2015-03-18 23:14:44 -07:00
|
|
|
if ((token.map[1] - token.map[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-16 22:31:18 -07:00
|
|
|
// Returns the unordered list style for a list item token
|
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-05-29 18:18:20 -07:00
|
|
|
// Calls the provided function for each matching token
|
2015-06-12 09:37:11 -07:00
|
|
|
function filterTokens(params, type, callback) {
|
|
|
|
(params.tokenLists[type] || []).forEach(callback);
|
2015-03-04 18:23:19 -08:00
|
|
|
}
|
|
|
|
|
2015-03-16 22:31:18 -07:00
|
|
|
// Calls the provided function for each line (with context)
|
2015-03-09 00:31:07 -07:00
|
|
|
function forEachLine(params, callback) {
|
2015-06-16 17:50:55 -07:00
|
|
|
if (!params.forEachLine) {
|
|
|
|
var lineMetadata = new Array(params.lines.length);
|
2015-08-27 22:49:59 -07:00
|
|
|
var fenceStart = null;
|
2015-06-16 17:50:55 -07:00
|
|
|
var inFence = false;
|
|
|
|
// Find fenced code by pattern (parser ignores "``` close fence")
|
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
|
|
|
var metadata = 0;
|
2015-08-27 22:49:59 -07:00
|
|
|
var match = /^(`{3,}|~{3,})/.exec(line);
|
|
|
|
var fence = match && match[1];
|
|
|
|
if (fence &&
|
|
|
|
(!inFence || (fence.substr(0, fenceStart.length) === fenceStart))) {
|
2016-01-09 22:20:36 -08:00
|
|
|
metadata = inFence ? 2 : 6;
|
2015-08-27 22:49:59 -07:00
|
|
|
fenceStart = inFence ? null : fence;
|
2015-06-16 17:50:55 -07:00
|
|
|
inFence = !inFence;
|
|
|
|
} else if (inFence) {
|
|
|
|
metadata = 1;
|
|
|
|
}
|
|
|
|
lineMetadata[lineIndex] = metadata;
|
|
|
|
});
|
|
|
|
// Find code blocks normally
|
|
|
|
filterTokens(params, "code_block", function forToken(token) {
|
|
|
|
for (var i = token.map[0]; i < token.map[1]; i++) {
|
|
|
|
lineMetadata[i] = 1;
|
|
|
|
}
|
|
|
|
});
|
2016-01-09 22:20:36 -08:00
|
|
|
// Find tables normally
|
|
|
|
filterTokens(params, "table_open", function forToken(token) {
|
|
|
|
for (var i = token.map[0]; i < token.map[1]; i++) {
|
|
|
|
lineMetadata[i] += 8;
|
|
|
|
}
|
|
|
|
});
|
2015-06-16 17:50:55 -07:00
|
|
|
params.forEachLine = lineMetadata;
|
|
|
|
}
|
|
|
|
// Invoke callback
|
2015-03-09 00:31:07 -07:00
|
|
|
params.lines.forEach(function forLine(line, lineIndex) {
|
2015-06-16 17:50:55 -07:00
|
|
|
var metadata = params.forEachLine[lineIndex];
|
2016-01-09 22:20:36 -08:00
|
|
|
callback(
|
|
|
|
line,
|
|
|
|
lineIndex,
|
|
|
|
!!(metadata & 7),
|
|
|
|
(((metadata & 6) >> 1) || 2) - 2,
|
|
|
|
!!(metadata & 8));
|
2015-03-09 00:31:07 -07:00
|
|
|
});
|
2015-02-23 23:39:20 -08:00
|
|
|
}
|
|
|
|
|
2015-04-15 18:24:42 -07:00
|
|
|
// Calls the provided function for each specified inline child token
|
|
|
|
function forEachInlineChild(params, type, callback) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "inline", function forToken(token) {
|
|
|
|
token.children.forEach(function forChild(child) {
|
|
|
|
if (child.type === type) {
|
|
|
|
callback(child, token);
|
|
|
|
}
|
2015-04-15 17:50:01 -07:00
|
|
|
});
|
2015-05-29 18:18:20 -07:00
|
|
|
});
|
2015-04-15 17:50:01 -07:00
|
|
|
}
|
|
|
|
|
2015-03-16 22:31:18 -07:00
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-03-16 22:31:18 -07:00
|
|
|
// Returns (nested) lists as a flat array (in order)
|
2015-06-11 18:33:40 -07:00
|
|
|
function flattenLists(params) {
|
|
|
|
if (!params.flattenLists) {
|
|
|
|
var lists = [];
|
|
|
|
var stack = [];
|
|
|
|
var current = null;
|
|
|
|
var lastWithMap = { "map": [ 0, 1 ] };
|
|
|
|
params.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 = {
|
2016-06-26 11:56:48 -07:00
|
|
|
"unordered": (token.type === "bullet_list_open"),
|
|
|
|
"parentsUnordered": !current ||
|
|
|
|
(current.unordered && current.parentsUnordered),
|
2015-06-11 18:33:40 -07:00
|
|
|
"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];
|
2015-03-10 23:10:06 -07:00
|
|
|
lists.splice(current.insert, 0, current);
|
|
|
|
delete current.insert;
|
2015-06-11 18:33:40 -07:00
|
|
|
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;
|
2015-03-10 23:10:06 -07:00
|
|
|
}
|
2015-06-11 18:33:40 -07:00
|
|
|
});
|
|
|
|
params.flattenLists = lists;
|
|
|
|
}
|
|
|
|
return params.flattenLists;
|
2015-03-10 23:10:06 -07:00
|
|
|
}
|
|
|
|
|
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-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "header-increment" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
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-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2015-05-29 18:18:20 -07:00
|
|
|
var level = parseInt(token.tag.slice(1), 10);
|
2016-10-16 21:46:02 -07:00
|
|
|
if (prevLevel && (level > prevLevel)) {
|
|
|
|
errors.addDetailIf(token.lineNumber,
|
|
|
|
"h" + (prevLevel + 1), "h" + level);
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
prevLevel = level;
|
|
|
|
});
|
2015-02-25 18:19:36 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-02-25 18:00:08 -08:00
|
|
|
{
|
|
|
|
"name": "MD002",
|
2016-07-04 23:23:29 -07:00
|
|
|
"desc": "First header should be a top level header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "first-header-h1" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD002(params, errors) {
|
2016-07-04 23:23:29 -07:00
|
|
|
var level = params.options.level || 1;
|
|
|
|
var tag = "h" + level;
|
2015-02-27 22:06:54 -08:00
|
|
|
params.tokens.every(function forToken(token) {
|
2015-02-25 18:19:36 -08:00
|
|
|
if (token.type === "heading_open") {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(token.lineNumber, tag, token.tag);
|
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",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "header-style" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD003(params, errors) {
|
|
|
|
var style = params.options.style || "consistent";
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2015-07-29 22:17:33 -07:00
|
|
|
var styleForToken = headingStyleFor(token);
|
2015-05-29 18:18:20 -07:00
|
|
|
if (style === "consistent") {
|
2015-07-29 22:17:33 -07:00
|
|
|
style = styleForToken;
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
2016-02-24 21:11:58 -08:00
|
|
|
if (styleForToken !== style) {
|
|
|
|
var h12 = /h[12]/.test(token.tag);
|
|
|
|
var setextWithAtx =
|
|
|
|
(style === "setext_with_atx") &&
|
|
|
|
((h12 && (styleForToken === "setext")) ||
|
2016-10-16 21:46:02 -07:00
|
|
|
(!h12 && (styleForToken === "atx")));
|
2016-02-24 21:11:58 -08:00
|
|
|
var setextWithAtxClosed =
|
|
|
|
(style === "setext_with_atx_closed") &&
|
|
|
|
((h12 && (styleForToken === "setext")) ||
|
2016-10-16 21:46:02 -07:00
|
|
|
(!h12 && (styleForToken === "atx_closed")));
|
2016-02-24 21:11:58 -08:00
|
|
|
if (!setextWithAtx && !setextWithAtxClosed) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var expected = style;
|
|
|
|
if (style === "setext_with_atx") {
|
|
|
|
expected = h12 ? "setext" : "atx";
|
|
|
|
} else if (style === "setext_with_atx_closed") {
|
|
|
|
expected = h12 ? "setext" : "atx_closed";
|
|
|
|
}
|
|
|
|
errors.addDetailIf(token.lineNumber, expected, styleForToken);
|
2016-02-24 21:11:58 -08:00
|
|
|
}
|
2015-02-27 22:06:54 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-01 22:56:52 -08:00
|
|
|
{
|
|
|
|
"name": "MD004",
|
|
|
|
"desc": "Unordered list style",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "bullet", "ul" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "ul-style" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-01 22:56:52 -08:00
|
|
|
"func": function MD004(params, errors) {
|
|
|
|
var style = params.options.style || "consistent";
|
2016-07-04 14:32:40 -07:00
|
|
|
var expectedStyle = style;
|
|
|
|
var nestingStyles = [];
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2016-06-26 11:56:48 -07:00
|
|
|
if (list.unordered) {
|
2016-07-04 14:32:40 -07:00
|
|
|
if (expectedStyle === "consistent") {
|
|
|
|
expectedStyle = unorderedListStyleFor(list.items[0]);
|
2015-03-10 23:10:06 -07:00
|
|
|
}
|
2015-06-11 18:33:40 -07:00
|
|
|
list.items.forEach(function forItem(item) {
|
2016-07-04 14:32:40 -07:00
|
|
|
var itemStyle = unorderedListStyleFor(item);
|
|
|
|
if (style === "sublist") {
|
|
|
|
var nesting = list.nesting;
|
|
|
|
if (!nestingStyles[nesting] &&
|
|
|
|
(itemStyle !== nestingStyles[nesting - 1])) {
|
|
|
|
nestingStyles[nesting] = itemStyle;
|
2016-10-16 21:46:02 -07:00
|
|
|
} else {
|
|
|
|
errors.addDetailIf(item.lineNumber,
|
|
|
|
nestingStyles[nesting], itemStyle);
|
2016-07-04 14:32:40 -07:00
|
|
|
}
|
2016-10-16 21:46:02 -07:00
|
|
|
} else {
|
|
|
|
errors.addDetailIf(item.lineNumber, expectedStyle, itemStyle);
|
2015-06-11 18:33:40 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
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",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "bullet", "ul", "indentation" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "list-indent" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-02 23:30:40 -08:00
|
|
|
"func": function MD005(params, errors) {
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2015-03-10 23:10:06 -07:00
|
|
|
var indent = indentFor(list.items[0]);
|
|
|
|
list.items.forEach(function forItem(item) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(item.lineNumber, indent, indentFor(item));
|
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",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "bullet", "ul", "indentation" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "ul-start-left" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-03 09:29:13 -08:00
|
|
|
"func": function MD006(params, errors) {
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2016-10-16 21:46:02 -07:00
|
|
|
if (list.unordered && !list.nesting) {
|
|
|
|
errors.addDetailIf(list.open.lineNumber, 0, indentFor(list.open));
|
2015-03-03 09:29:13 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-02 23:30:40 -08:00
|
|
|
{
|
|
|
|
"name": "MD007",
|
|
|
|
"desc": "Unordered list indentation",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "bullet", "ul", "indentation" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "ul-indent" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-02 23:30:40 -08:00
|
|
|
"func": function MD007(params, errors) {
|
|
|
|
var optionsIndent = params.options.indent || 2;
|
|
|
|
var prevIndent = 0;
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2016-06-26 11:56:48 -07:00
|
|
|
if (list.unordered && list.parentsUnordered) {
|
2015-06-11 18:33:40 -07:00
|
|
|
var indent = indentFor(list.open);
|
2016-10-16 21:46:02 -07:00
|
|
|
if (indent > prevIndent) {
|
|
|
|
errors.addDetailIf(list.open.lineNumber,
|
|
|
|
prevIndent + optionsIndent, indent);
|
2015-06-11 18:33:40 -07:00
|
|
|
}
|
|
|
|
prevIndent = indent;
|
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": "MD009",
|
|
|
|
"desc": "Trailing spaces",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "whitespace" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-trailing-spaces" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": trailingSpaceRe,
|
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) {
|
2016-10-31 22:53:46 -07:00
|
|
|
if (trailingSpaceRe.test(line)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var expected = (brSpaces < 2) ? 0 : brSpaces;
|
|
|
|
errors.addDetailIf(lineIndex + 1,
|
|
|
|
expected, line.length - line.trimRight().length);
|
2015-03-03 09:29:13 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD010",
|
|
|
|
"desc": "Hard tabs",
|
2015-03-17 22:34:47 -07:00
|
|
|
"tags": [ "whitespace", "hard_tab" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-hard-tabs" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": tabRe,
|
2015-03-03 09:29:13 -08:00
|
|
|
"func": function MD010(params, errors) {
|
2016-09-29 21:25:10 -07:00
|
|
|
var codeBlocks = params.options.code_blocks;
|
|
|
|
var includeCodeBlocks = (codeBlocks === undefined) ? true : !!codeBlocks;
|
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode) {
|
2016-10-31 22:53:46 -07:00
|
|
|
if (tabRe.test(line) && (!inCode || includeCodeBlocks)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetail(lineIndex + 1,
|
|
|
|
"Column: " + (line.indexOf("\t") + 1));
|
2015-03-03 09:29:13 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-03 18:28:59 -08:00
|
|
|
{
|
|
|
|
"name": "MD011",
|
|
|
|
"desc": "Reversed link syntax",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "links" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-reversed-links" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": reversedLinkRe,
|
2015-03-03 18:28:59 -08:00
|
|
|
"func": function MD011(params, errors) {
|
2015-04-15 18:24:42 -07:00
|
|
|
forEachInlineChild(params, "text", function forToken(token) {
|
2016-10-31 22:53:46 -07:00
|
|
|
var match = reversedLinkRe.exec(token.content);
|
2016-10-16 21:46:02 -07:00
|
|
|
if (match) {
|
|
|
|
errors.addDetail(token.lineNumber, match[0]);
|
2015-04-15 17:50:01 -07:00
|
|
|
}
|
|
|
|
});
|
2015-03-03 18:28:59 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:09:46 -08:00
|
|
|
{
|
|
|
|
"name": "MD012",
|
|
|
|
"desc": "Multiple consecutive blank lines",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "whitespace", "blank_lines" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-multiple-blanks" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-03-04 18:09:46 -08:00
|
|
|
"func": function MD012(params, errors) {
|
2016-10-03 21:42:44 -07:00
|
|
|
var maximum = params.options.maximum || 1;
|
|
|
|
var count = 0;
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode) {
|
2016-10-03 21:42:44 -07:00
|
|
|
count = (inCode || line.trim().length) ? 0 : count + 1;
|
|
|
|
if (maximum < count) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(lineIndex + 1, maximum, count);
|
2015-03-04 18:09:46 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:36:30 -08:00
|
|
|
{
|
|
|
|
"name": "MD013",
|
|
|
|
"desc": "Line length",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "line_length" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "line-length" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": longLineReFunc,
|
2015-03-04 18:36:30 -08:00
|
|
|
"func": function MD013(params, errors) {
|
2016-10-31 22:53:46 -07:00
|
|
|
var lineLength = params.options.line_length || defaultLineLength;
|
2016-01-09 22:20:36 -08:00
|
|
|
var codeBlocks = params.options.code_blocks;
|
|
|
|
var includeCodeBlocks = (codeBlocks === undefined) ? true : !!codeBlocks;
|
|
|
|
var tables = params.options.tables;
|
|
|
|
var includeTables = (tables === undefined) ? true : !!tables;
|
2017-02-11 16:20:24 -08:00
|
|
|
var headers = params.options.headers;
|
|
|
|
var includeHeaders = (headers === undefined) ? true : !!headers;
|
|
|
|
var headerLineNumbers = [];
|
|
|
|
if (!includeHeaders) {
|
|
|
|
forEachHeading(params, function forHeading(heading) {
|
|
|
|
headerLineNumbers.push(heading.lineNumber);
|
|
|
|
});
|
|
|
|
}
|
2016-10-31 22:53:46 -07:00
|
|
|
var re = longLineReFunc(params.options);
|
2016-01-09 22:20:36 -08:00
|
|
|
forEachLine(params,
|
|
|
|
function forLine(line, lineIndex, inCode, onFence, inTable) {
|
2017-02-11 16:20:24 -08:00
|
|
|
var lineNumber = lineIndex + 1;
|
2016-01-09 22:20:36 -08:00
|
|
|
if ((includeCodeBlocks || !inCode) &&
|
|
|
|
(includeTables || !inTable) &&
|
2017-02-11 16:20:24 -08:00
|
|
|
(includeHeaders || (headerLineNumbers.indexOf(lineNumber)) < 0) &&
|
2016-01-09 22:20:36 -08:00
|
|
|
re.test(line)) {
|
2017-02-11 16:20:24 -08:00
|
|
|
errors.addDetailIf(lineNumber, lineLength, line.length);
|
2016-01-09 22:20:36 -08:00
|
|
|
}
|
|
|
|
});
|
2015-03-04 18:36:30 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-05 23:30:01 -08:00
|
|
|
{
|
|
|
|
"name": "MD014",
|
|
|
|
"desc": "Dollar signs used before commands without showing output",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "code" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "commands-show-output" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": dollarCommandRe,
|
2015-03-05 23:30:01 -08:00
|
|
|
"func": function MD014(params, errors) {
|
2015-05-29 18:18:20 -07:00
|
|
|
[ "code_block", "fence" ].forEach(function forType(type) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, type, function forToken(token) {
|
2016-01-04 22:22:35 -08:00
|
|
|
var allBlank = true;
|
2015-03-11 18:40:46 -07:00
|
|
|
if (token.content && token.content.split(shared.newLineRe)
|
2015-05-31 22:29:01 -07:00
|
|
|
.every(function forLine(line) {
|
2016-10-31 22:53:46 -07:00
|
|
|
return !line || (allBlank = false) || dollarCommandRe.test(line);
|
2016-01-04 22:22:35 -08:00
|
|
|
}) && !allBlank) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber,
|
|
|
|
token.content.split(shared.newLineRe)[0].trim());
|
2015-03-06 09:21:55 -08:00
|
|
|
}
|
|
|
|
});
|
2015-05-29 18:18:20 -07:00
|
|
|
});
|
2015-03-06 09:21:55 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD018",
|
|
|
|
"desc": "No space after hash on atx style header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "atx", "spaces" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-missing-space-atx" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": atxHeaderSpaceRe,
|
2015-03-06 09:21:55 -08:00
|
|
|
"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)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(lineIndex + 1, line.trim());
|
2015-03-06 09:21:55 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD019",
|
|
|
|
"desc": "Multiple spaces after hash on atx style header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "atx", "spaces" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-multiple-space-atx" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": atxHeaderSpaceRe,
|
2015-03-06 09:21:55 -08:00
|
|
|
"func": function MD019(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2015-05-29 18:18:20 -07:00
|
|
|
if ((headingStyleFor(token) === "atx") &&
|
|
|
|
/^#+\s\s/.test(token.line)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line.trim());
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
});
|
2015-03-06 18:16:16 -08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD020",
|
|
|
|
"desc": "No space inside hashes on closed atx style header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "atx_closed", "spaces" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-missing-space-closed-atx" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": atxClosedHeaderNoSpaceRe,
|
2015-03-06 18:16:16 -08:00
|
|
|
"func": function MD020(params, errors) {
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode) {
|
2016-10-16 21:46:02 -07:00
|
|
|
if (!inCode && /^#+[^#]*[^\\]#+$/.test(line)) {
|
|
|
|
var left = /^#+[^#\s]/.test(line);
|
|
|
|
var right = /[^#\s]#+$/.test(line);
|
|
|
|
if (left || right) {
|
|
|
|
errors.addContext(lineIndex + 1, line.trim(), left, right);
|
|
|
|
}
|
2015-03-06 18:16:16 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD021",
|
|
|
|
"desc": "Multiple spaces inside hashes on closed atx style header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "atx_closed", "spaces" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-multiple-space-closed-atx" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": atxClosedHeaderSpaceRe,
|
2015-03-06 18:16:16 -08:00
|
|
|
"func": function MD021(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2016-10-16 21:46:02 -07:00
|
|
|
if (headingStyleFor(token) === "atx_closed") {
|
|
|
|
var left = /^#+\s\s/.test(token.line);
|
|
|
|
var right = /\s\s#+$/.test(token.line);
|
|
|
|
if (left || right) {
|
|
|
|
errors.addContext(token.lineNumber, token.line.trim(), left, right);
|
|
|
|
}
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
});
|
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",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "blank_lines" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "blanks-around-headers" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-03-08 23:08:43 -07:00
|
|
|
"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") {
|
2015-03-18 23:14:44 -07:00
|
|
|
if ((token.map[0] - prevMaxLineIndex) === 0) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line.trim());
|
2015-03-08 23:08:43 -07:00
|
|
|
}
|
|
|
|
} else if (token.type === "heading_close") {
|
|
|
|
needBlankLine = true;
|
|
|
|
}
|
2015-03-18 23:14:44 -07:00
|
|
|
if (token.map) {
|
2015-03-08 23:08:43 -07:00
|
|
|
if (needBlankLine) {
|
2015-03-18 23:14:44 -07:00
|
|
|
if ((token.map[0] - prevMaxLineIndex) === 0) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(prevHeadingLineNumber,
|
|
|
|
params.lines[prevHeadingLineNumber - 1].trim());
|
2015-03-08 23:08:43 -07:00
|
|
|
}
|
|
|
|
needBlankLine = false;
|
|
|
|
}
|
2015-03-18 23:14:44 -07:00
|
|
|
prevMaxLineIndex = Math.max(prevMaxLineIndex, token.map[1]);
|
2015-03-08 23:08:43 -07:00
|
|
|
}
|
2016-06-23 22:41:47 -07:00
|
|
|
if (token.type === "heading_open") {
|
|
|
|
prevHeadingLineNumber = token.lineNumber;
|
|
|
|
}
|
2015-03-08 23:08:43 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-11 09:30:34 -07:00
|
|
|
{
|
|
|
|
"name": "MD023",
|
|
|
|
"desc": "Headers must start at the beginning of the line",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers", "spaces" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "header-start-left" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": spaceBeforeHeaderRe,
|
2015-03-11 09:30:34 -07:00
|
|
|
"func": function MD023(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2016-10-31 22:53:46 -07:00
|
|
|
if (spaceBeforeHeaderRe.test(token.line)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line);
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
});
|
2015-03-11 09:30:34 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD024",
|
|
|
|
"desc": "Multiple headers with the same content",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-duplicate-header" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
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 {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(heading.lineNumber, heading.line.trim());
|
2015-03-11 09:30:34 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD025",
|
|
|
|
"desc": "Multiple top level headers in the same document",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "single-h1" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-03-11 09:30:34 -07:00
|
|
|
"func": function MD025(params, errors) {
|
2016-07-04 23:23:29 -07:00
|
|
|
var level = params.options.level || 1;
|
|
|
|
var tag = "h" + level;
|
2015-03-11 09:30:34 -07:00
|
|
|
var hasTopLevelHeading = false;
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "heading_open", function forToken(token) {
|
2016-07-04 23:23:29 -07:00
|
|
|
if (token.tag === tag) {
|
2015-05-29 18:18:20 -07:00
|
|
|
if (hasTopLevelHeading) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line.trim());
|
2015-05-29 18:18:20 -07:00
|
|
|
} else if (token.lineNumber === 1) {
|
|
|
|
hasTopLevelHeading = true;
|
2015-03-11 09:30:34 -07:00
|
|
|
}
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
});
|
2015-03-11 09:30:34 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-11 18:40:46 -07:00
|
|
|
{
|
|
|
|
"name": "MD026",
|
|
|
|
"desc": "Trailing punctuation in header",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-trailing-punctuation" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": trailingPunctuationRe,
|
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) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var match = re.exec(content);
|
|
|
|
if (match) {
|
|
|
|
errors.addDetail(heading.lineNumber,
|
|
|
|
"Punctuation: '" + match[0] + "'");
|
2015-03-11 18:40:46 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD027",
|
|
|
|
"desc": "Multiple spaces after blockquote symbol",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "blockquote", "whitespace", "indentation" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-multiple-space-blockquote" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": spaceAfterBlockQuote,
|
2015-03-11 18:40:46 -07:00
|
|
|
"func": function MD027(params, errors) {
|
2016-09-09 10:30:22 +09:00
|
|
|
var blockquoteNesting = 0;
|
2016-10-27 22:09:05 -07:00
|
|
|
var listItemNesting = 0;
|
2015-03-11 18:40:46 -07:00
|
|
|
params.tokens.forEach(function forToken(token) {
|
|
|
|
if (token.type === "blockquote_open") {
|
2016-09-09 10:30:22 +09:00
|
|
|
blockquoteNesting++;
|
2015-03-11 18:40:46 -07:00
|
|
|
} else if (token.type === "blockquote_close") {
|
2016-09-09 10:30:22 +09:00
|
|
|
blockquoteNesting--;
|
2016-10-27 22:09:05 -07:00
|
|
|
} else if (token.type === "list_item_open") {
|
|
|
|
listItemNesting++;
|
|
|
|
} else if (token.type === "list_item_close") {
|
|
|
|
listItemNesting--;
|
2016-09-10 19:23:50 -07:00
|
|
|
} else if ((token.type === "inline") && (blockquoteNesting > 0)) {
|
2016-10-27 22:09:05 -07:00
|
|
|
var multipleSpaces = listItemNesting ?
|
|
|
|
/^(\s*>)+\s\s+>/.test(token.line) :
|
|
|
|
/^(\s*>)+\s\s/.test(token.line);
|
|
|
|
if (multipleSpaces) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line);
|
|
|
|
}
|
2015-03-11 18:40:46 -07:00
|
|
|
token.content.split(shared.newLineRe)
|
|
|
|
.forEach(function forLine(line, offset) {
|
2016-10-16 21:46:02 -07:00
|
|
|
if (/^\s/.test(line)) {
|
|
|
|
errors.addContext(token.lineNumber + offset, "> " + line);
|
2015-03-11 18:40:46 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-04 18:09:46 -08:00
|
|
|
{
|
|
|
|
"name": "MD028",
|
|
|
|
"desc": "Blank line inside blockquote",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "blockquote", "whitespace" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-blanks-blockquote" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-03-04 18:09:46 -08:00
|
|
|
"func": function MD028(params, errors) {
|
|
|
|
var prevToken = {};
|
|
|
|
params.tokens.forEach(function forToken(token) {
|
|
|
|
if ((token.type === "blockquote_open") &&
|
|
|
|
(prevToken.type === "blockquote_close")) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.add(token.lineNumber - 1);
|
2015-03-04 18:09:46 -08:00
|
|
|
}
|
|
|
|
prevToken = token;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-07 22:46:45 -08:00
|
|
|
{
|
|
|
|
"name": "MD029",
|
|
|
|
"desc": "Ordered list item prefix",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "ol" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "ol-prefix" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-07 22:46:45 -08:00
|
|
|
"func": function MD029(params, errors) {
|
|
|
|
var style = params.options.style || "one";
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2016-06-26 11:56:48 -07:00
|
|
|
if (!list.unordered) {
|
2015-06-11 18:33:40 -07:00
|
|
|
var number = 1;
|
|
|
|
list.items.forEach(function forItem(item) {
|
2016-11-02 22:09:16 -07:00
|
|
|
var match = /^[\s>]*([^.)]*)[.)]/.exec(item.line);
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(item.lineNumber,
|
|
|
|
String(number), !match || match[1]);
|
2015-06-11 18:33:40 -07:00
|
|
|
if (style === "ordered") {
|
|
|
|
number++;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2015-03-10 23:10:06 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD030",
|
|
|
|
"desc": "Spaces after list markers",
|
2015-03-16 22:31:18 -07:00
|
|
|
"tags": [ "ol", "ul", "whitespace" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "list-marker-space" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": listItemMarkerRe,
|
2015-03-10 23:10:06 -07:00
|
|
|
"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;
|
2015-06-11 18:33:40 -07:00
|
|
|
flattenLists(params).forEach(function forList(list) {
|
2015-03-18 23:14:44 -07:00
|
|
|
var lineCount = list.lastLineIndex - list.open.map[0];
|
2015-03-10 23:10:06 -07:00
|
|
|
var allSingle = lineCount === list.items.length;
|
2016-06-26 11:56:48 -07:00
|
|
|
var expectedSpaces = list.unordered ?
|
|
|
|
(allSingle ? ulSingle : ulMulti) :
|
|
|
|
(allSingle ? olSingle : olMulti);
|
2015-03-10 23:10:06 -07:00
|
|
|
list.items.forEach(function forItem(item) {
|
2016-09-10 19:23:50 -07:00
|
|
|
var match = /^[\s>]*\S+(\s+)/.exec(item.line);
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(item.lineNumber,
|
|
|
|
expectedSpaces, !match || match[1].length);
|
2015-03-10 23:10:06 -07:00
|
|
|
});
|
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-03-16 22:31:18 -07:00
|
|
|
"tags": [ "code", "blank_lines" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "blanks-around-fences" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD031(params, errors) {
|
2015-06-16 17:50:55 -07:00
|
|
|
var lines = params.lines;
|
|
|
|
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)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(i + 1, lines[i].trim());
|
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-03-16 22:31:18 -07:00
|
|
|
"tags": [ "bullet", "ul", "ol", "blank_lines" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "blanks-around-lists" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-02-27 22:06:54 -08:00
|
|
|
"func": function MD032(params, errors) {
|
2016-11-03 22:16:03 -07:00
|
|
|
var blankOrListRe = /^[\s>]*($|\s)/;
|
2015-02-24 23:50:37 -08:00
|
|
|
var inList = false;
|
|
|
|
var prevLine = "";
|
2015-03-09 00:31:07 -07:00
|
|
|
forEachLine(params, function forLine(line, lineIndex, inCode, onFence) {
|
|
|
|
if (!inCode || onFence) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var lineTrim = line.trim();
|
2016-10-31 22:53:46 -07:00
|
|
|
var listMarker = listItemMarkerRe.test(lineTrim);
|
2016-11-03 22:16:03 -07:00
|
|
|
if (listMarker && !inList && !blankOrListRe.test(prevLine)) {
|
2017-01-15 14:41:58 -08:00
|
|
|
// Check whether this list prefix can interrupt a paragraph
|
|
|
|
if (listItemMarkerInterruptsRe.test(lineTrim)) {
|
|
|
|
errors.addContext(lineIndex + 1, lineTrim);
|
|
|
|
} else {
|
|
|
|
listMarker = false;
|
|
|
|
}
|
2016-11-03 22:16:03 -07:00
|
|
|
} else if (!listMarker && inList && !blankOrListRe.test(line)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(lineIndex, lineTrim);
|
2015-02-24 23:50:37 -08:00
|
|
|
}
|
|
|
|
inList = listMarker;
|
2015-02-24 23:35:34 -08:00
|
|
|
}
|
2015-02-24 23:50:37 -08:00
|
|
|
prevLine = line;
|
|
|
|
});
|
|
|
|
}
|
2015-04-13 08:47:15 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD033",
|
|
|
|
"desc": "Inline HTML",
|
|
|
|
"tags": [ "html" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-inline-html" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": htmlRe,
|
2015-04-13 08:47:15 -07:00
|
|
|
"func": function MD033(params, errors) {
|
2015-12-15 21:30:37 -08:00
|
|
|
var allowedElements = (params.options.allowed_elements || [])
|
|
|
|
.map(function forElement(element) {
|
|
|
|
return element.toLowerCase();
|
|
|
|
});
|
2015-09-26 16:55:33 -07:00
|
|
|
function forToken(token) {
|
2015-12-15 21:30:37 -08:00
|
|
|
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();
|
|
|
|
})
|
2016-10-16 21:46:02 -07:00
|
|
|
.filter(function forElement(element) {
|
|
|
|
return allowedElements.indexOf(element) === -1;
|
2015-12-15 21:30:37 -08:00
|
|
|
});
|
2016-10-16 21:46:02 -07:00
|
|
|
if (allowed.length) {
|
|
|
|
errors.addDetail(token.lineNumber + offset,
|
|
|
|
"Element: " + allowed[0]);
|
2015-12-15 21:30:37 -08:00
|
|
|
}
|
|
|
|
});
|
2015-09-26 16:55:33 -07:00
|
|
|
}
|
|
|
|
filterTokens(params, "html_block", forToken);
|
|
|
|
forEachInlineChild(params, "html_inline", forToken);
|
2015-04-13 08:47:15 -07:00
|
|
|
}
|
2015-04-14 00:01:57 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD034",
|
|
|
|
"desc": "Bare URL used",
|
|
|
|
"tags": [ "links", "url" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-bare-urls" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": bareUrlRe,
|
2015-04-14 00:01:57 -07:00
|
|
|
"func": function MD034(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "inline", function forToken(token) {
|
2015-05-29 18:18:20 -07:00
|
|
|
var inLink = false;
|
|
|
|
token.children.forEach(function forChild(child) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var match = null;
|
2015-05-29 18:18:20 -07:00
|
|
|
if (child.type === "link_open") {
|
|
|
|
inLink = true;
|
|
|
|
} else if (child.type === "link_close") {
|
|
|
|
inLink = false;
|
|
|
|
} else if ((child.type === "text") &&
|
|
|
|
!inLink &&
|
2016-10-31 22:53:46 -07:00
|
|
|
(match = bareUrlRe.exec(child.content))) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(child.lineNumber, match[0]);
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
2015-04-14 00:01:57 -07:00
|
|
|
});
|
2015-05-29 18:18:20 -07:00
|
|
|
});
|
2015-04-14 00:01:57 -07:00
|
|
|
}
|
2015-04-14 09:40:16 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD035",
|
|
|
|
"desc": "Horizontal rule style",
|
|
|
|
"tags": [ "hr" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "hr-style" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-04-14 09:40:16 -07:00
|
|
|
"func": function MD035(params, errors) {
|
|
|
|
var style = params.options.style || "consistent";
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "hr", function forToken(token) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var lineTrim = token.line.trim();
|
2015-05-29 18:18:20 -07:00
|
|
|
if (style === "consistent") {
|
2016-10-16 21:46:02 -07:00
|
|
|
style = lineTrim;
|
2015-04-14 09:40:16 -07:00
|
|
|
}
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(token.lineNumber, style, lineTrim);
|
2015-04-14 09:40:16 -07:00
|
|
|
});
|
|
|
|
}
|
2015-04-14 22:37:56 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD036",
|
|
|
|
"desc": "Emphasis used instead of a header",
|
|
|
|
"tags": [ "headers", "emphasis" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-emphasis-as-header" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-04-14 22:37:56 -07:00
|
|
|
"func": function MD036(params, errors) {
|
2016-09-24 16:15:54 -07:00
|
|
|
var punctuation = params.options.punctuation || ".,;:!?";
|
|
|
|
var re = new RegExp("[" + punctuation + "]$");
|
2015-04-14 22:37:56 -07:00
|
|
|
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")) &&
|
2016-09-24 16:15:54 -07:00
|
|
|
(t.children[1].type === "text") &&
|
|
|
|
!re.test(t.children[1].content)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(t.lineNumber, t.children[1].content);
|
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;
|
|
|
|
}
|
|
|
|
};
|
2016-09-30 22:54:16 -07:00
|
|
|
} else if (token.type === "list_item_open") {
|
|
|
|
return function inListItem(t) {
|
|
|
|
if (t.type !== "list_item_close") {
|
|
|
|
return inListItem;
|
|
|
|
}
|
|
|
|
};
|
2015-04-14 22:37:56 -07:00
|
|
|
}
|
|
|
|
}
|
2016-07-05 14:49:33 -07:00
|
|
|
var state = base;
|
2015-04-14 22:37:56 -07:00
|
|
|
params.tokens.forEach(function forToken(token) {
|
|
|
|
state = state(token) || base;
|
|
|
|
});
|
|
|
|
}
|
2015-04-15 17:50:01 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD037",
|
|
|
|
"desc": "Spaces inside emphasis markers",
|
|
|
|
"tags": [ "whitespace", "emphasis" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-space-in-emphasis" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": spaceInsideEmphasisRe,
|
2015-04-15 17:50:01 -07:00
|
|
|
"func": function MD037(params, errors) {
|
2015-04-15 18:24:42 -07:00
|
|
|
forEachInlineChild(params, "text", function forToken(token) {
|
2016-10-16 21:46:02 -07:00
|
|
|
var left = /\s(\*\*?|__?)\s.+\1/.exec(token.content);
|
|
|
|
var right = /(\*\*?|__?).+\s\1\s/.exec(token.content);
|
|
|
|
if (left) {
|
|
|
|
errors.addContext(token.lineNumber, left[0].trim());
|
|
|
|
} else if (right) {
|
|
|
|
errors.addContext(token.lineNumber, right[0].trim(), false, true);
|
2015-04-15 17:50:01 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2015-04-15 18:24:42 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD038",
|
|
|
|
"desc": "Spaces inside code span elements",
|
|
|
|
"tags": [ "whitespace", "code" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-space-in-code" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": spaceInsideCodeRe,
|
2015-04-15 18:24:42 -07:00
|
|
|
"func": function MD038(params, errors) {
|
2017-02-18 22:56:06 -08:00
|
|
|
var inlineCodeSpansRe = /(`+)((?:.*?[^`])|)\1(?!`)/g;
|
2015-04-15 18:24:42 -07:00
|
|
|
forEachInlineChild(params, "code_inline",
|
2017-02-18 22:56:06 -08:00
|
|
|
function forToken(token) {
|
|
|
|
var line = params.lines[token.lineNumber - 1];
|
|
|
|
var match = null;
|
|
|
|
while ((match = inlineCodeSpansRe.exec(line)) !== null) {
|
|
|
|
var inlineCodeSpan = match[0];
|
|
|
|
var content = match[2];
|
|
|
|
if (/^\s([^`]|$)/.test(content)) {
|
|
|
|
errors.addContext(token.lineNumber, inlineCodeSpan);
|
|
|
|
} else if (/[^`]\s$/.test(content)) {
|
|
|
|
errors.addContext(token.lineNumber, inlineCodeSpan, false, true);
|
|
|
|
}
|
2015-04-15 18:24:42 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD039",
|
|
|
|
"desc": "Spaces inside link text",
|
|
|
|
"tags": [ "whitespace", "links" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "no-space-in-links" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": spaceInsideLinkRe,
|
2015-04-15 18:24:42 -07:00
|
|
|
"func": function MD039(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "inline", function forToken(token) {
|
2015-05-29 18:18:20 -07:00
|
|
|
var inLink = false;
|
2016-10-16 21:46:02 -07:00
|
|
|
var linkText = "";
|
2015-05-29 18:18:20 -07:00
|
|
|
token.children.forEach(function forChild(child) {
|
|
|
|
if (child.type === "link_open") {
|
|
|
|
inLink = true;
|
2016-10-16 21:46:02 -07:00
|
|
|
linkText = "";
|
2015-05-29 18:18:20 -07:00
|
|
|
} else if (child.type === "link_close") {
|
|
|
|
inLink = false;
|
2016-10-16 21:46:02 -07:00
|
|
|
var left = linkText.trimLeft().length !== linkText.length;
|
|
|
|
var right = linkText.trimRight().length !== linkText.length;
|
|
|
|
if (left || right) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(
|
|
|
|
token.lineNumber, "[" + linkText + "]", left, right);
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
} else if (inLink) {
|
2016-10-16 21:46:02 -07:00
|
|
|
linkText += child.content;
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
2015-04-15 18:24:42 -07:00
|
|
|
});
|
2015-05-29 18:18:20 -07:00
|
|
|
});
|
2015-04-15 18:24:42 -07:00
|
|
|
}
|
2015-04-16 09:13:56 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD040",
|
|
|
|
"desc": "Fenced code blocks should have a language specified",
|
|
|
|
"tags": [ "code", "language" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "fenced-code-language" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-04-16 09:13:56 -07:00
|
|
|
"func": function MD040(params, errors) {
|
2015-06-12 09:37:11 -07:00
|
|
|
filterTokens(params, "fence", function forToken(token) {
|
2015-05-29 18:18:20 -07:00
|
|
|
if (!token.info.trim()) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(token.lineNumber, token.line);
|
2015-05-29 18:18:20 -07:00
|
|
|
}
|
|
|
|
});
|
2015-04-16 09:13:56 -07:00
|
|
|
}
|
2015-07-20 22:06:48 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD041",
|
|
|
|
"desc": "First line in file should be a top level header",
|
|
|
|
"tags": [ "headers" ],
|
2016-01-12 21:29:17 -08:00
|
|
|
"aliases": [ "first-line-h1" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2015-07-20 22:06:48 -07:00
|
|
|
"func": function MD041(params, errors) {
|
2016-07-04 23:23:29 -07:00
|
|
|
var level = params.options.level || 1;
|
|
|
|
var tag = "h" + level;
|
2015-07-20 22:06:48 -07:00
|
|
|
var firstHeader = null;
|
|
|
|
params.tokens.every(function forToken(token) {
|
|
|
|
if (token.type === "heading_open") {
|
|
|
|
firstHeader = token;
|
|
|
|
return false;
|
|
|
|
} else if (token.lineNumber > 1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
if (!firstHeader ||
|
|
|
|
(firstHeader.lineNumber !== 1) ||
|
2016-07-04 23:23:29 -07:00
|
|
|
(firstHeader.tag !== tag)) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(1, params.lines[0]);
|
2015-07-20 22:06:48 -07:00
|
|
|
}
|
|
|
|
}
|
2016-06-27 22:19:02 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD042",
|
|
|
|
"desc": "No empty links",
|
|
|
|
"tags": [ "links" ],
|
|
|
|
"aliases": [ "no-empty-links" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": emptyLinkRe,
|
2016-07-02 22:37:52 -07:00
|
|
|
"func": function MD042(params, errors) {
|
2016-10-16 21:46:02 -07:00
|
|
|
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) {
|
|
|
|
errors.addContext(child.lineNumber, "[" + linkText + "]");
|
|
|
|
}
|
|
|
|
} else if (inLink) {
|
|
|
|
linkText += child.content;
|
2016-06-27 22:19:02 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2016-07-02 22:37:52 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD043",
|
|
|
|
"desc": "Required header structure",
|
|
|
|
"tags": [ "headers" ],
|
|
|
|
"aliases": [ "required-headers" ],
|
2016-10-31 22:53:46 -07:00
|
|
|
"regexp": null,
|
2016-07-02 22:37:52 -07:00
|
|
|
"func": function MD043(params, errors) {
|
|
|
|
var requiredHeaders = params.options.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;
|
|
|
|
forEachHeading(params, function forHeading(heading, content) {
|
|
|
|
if (!errors.length) {
|
|
|
|
var actual = levels[heading.tag] + " " + content;
|
|
|
|
var expected = requiredHeaders[i++] || "";
|
|
|
|
if (expected === "*") {
|
|
|
|
optional = true;
|
|
|
|
} else if (expected.toLowerCase() === actual.toLowerCase()) {
|
|
|
|
optional = false;
|
|
|
|
} else if (optional) {
|
|
|
|
i--;
|
|
|
|
} else {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addDetailIf(heading.lineNumber, expected, actual);
|
2016-07-02 22:37:52 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if ((i < requiredHeaders.length) && !errors.length) {
|
2016-10-16 21:46:02 -07:00
|
|
|
errors.addContext(params.lines.length, requiredHeaders[i]);
|
2016-07-02 22:37:52 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-12-22 13:34:18 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
"name": "MD044",
|
|
|
|
"desc": "Proper names should have the correct capitalization",
|
|
|
|
"tags": [ "spelling" ],
|
|
|
|
"aliases": [ "proper-names" ],
|
|
|
|
"regexp": null,
|
|
|
|
"func": function MD044(params, errors) {
|
|
|
|
var names = params.options.names || [];
|
|
|
|
names.forEach(function forName(name) {
|
|
|
|
var escapedName = escapeForRegExp(name);
|
|
|
|
var namePattern = "\\b" + escapedName + "\\b";
|
|
|
|
var anyNameRe = new RegExp(namePattern, "gi");
|
|
|
|
forEachLine(params, function forLine(line, lineIndex) {
|
|
|
|
var matches = line.match(anyNameRe) || [];
|
|
|
|
matches.forEach(function forMatch(match) {
|
|
|
|
errors.addDetailIf(lineIndex + 1, name, match);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2015-02-23 23:39:20 -08:00
|
|
|
}
|
2015-02-24 23:50:37 -08:00
|
|
|
];
|