mirror of
https://github.com/DavidAnson/markdownlint.git
synced 2025-09-22 05:40:48 +02:00
This commit is contained in:
parent
62f5c85238
commit
33ee1cd85e
13 changed files with 250 additions and 24 deletions
|
@ -107,6 +107,7 @@ playground for learning and exploring.
|
||||||
* **[MD048](doc/Rules.md#md048)** *code-fence-style* - Code fence style
|
* **[MD048](doc/Rules.md#md048)** *code-fence-style* - Code fence style
|
||||||
* **[MD049](doc/Rules.md#md049)** *emphasis-style* - Emphasis style should be consistent
|
* **[MD049](doc/Rules.md#md049)** *emphasis-style* - Emphasis style should be consistent
|
||||||
* **[MD050](doc/Rules.md#md050)** *strong-style* - Strong style should be consistent
|
* **[MD050](doc/Rules.md#md050)** *strong-style* - Strong style should be consistent
|
||||||
|
* **[MD051](doc/Rules.md#md051)** *valid-link-fragments* - Link fragments should be valid
|
||||||
|
|
||||||
<!-- markdownlint-restore -->
|
<!-- markdownlint-restore -->
|
||||||
|
|
||||||
|
@ -142,7 +143,7 @@ rules at once.
|
||||||
* **indentation** - MD005, MD006, MD007, MD027
|
* **indentation** - MD005, MD006, MD007, MD027
|
||||||
* **language** - MD040
|
* **language** - MD040
|
||||||
* **line_length** - MD013
|
* **line_length** - MD013
|
||||||
* **links** - MD011, MD034, MD039, MD042
|
* **links** - MD011, MD034, MD039, MD042, MD051
|
||||||
* **ol** - MD029, MD030, MD032
|
* **ol** - MD029, MD030, MD032
|
||||||
* **spaces** - MD018, MD019, MD020, MD021, MD023
|
* **spaces** - MD018, MD019, MD020, MD021, MD023
|
||||||
* **spelling** - MD044
|
* **spelling** - MD044
|
||||||
|
|
|
@ -4383,6 +4383,58 @@ module.exports = {
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/***/ }),
|
||||||
|
|
||||||
|
/***/ "../lib/md051.js":
|
||||||
|
/*!***********************!*\
|
||||||
|
!*** ../lib/md051.js ***!
|
||||||
|
\***********************/
|
||||||
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
var _a = __webpack_require__(/*! ../helpers */ "../helpers/helpers.js"), addError = _a.addError, forEachHeading = _a.forEachHeading, filterTokens = _a.filterTokens;
|
||||||
|
/**
|
||||||
|
* Converts a Markdown heading into an HTML fragment
|
||||||
|
* according to the rules used by GitHub.
|
||||||
|
*
|
||||||
|
* @param {string} string The string to convert.
|
||||||
|
* @returns {string} The converted string.
|
||||||
|
*/
|
||||||
|
function convertHeadingToHTMLFragment(string) {
|
||||||
|
return "#" + string
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/ /g, "-")
|
||||||
|
.replace(/[^-_a-z0-9]/g, "");
|
||||||
|
}
|
||||||
|
module.exports = {
|
||||||
|
"names": ["MD051", "valid-link-fragments"],
|
||||||
|
"description": "Link fragments should be valid",
|
||||||
|
"tags": ["links"],
|
||||||
|
"function": function MD051(params, onError) {
|
||||||
|
var validLinkFragments = [];
|
||||||
|
forEachHeading(params, function (_heading, content) {
|
||||||
|
validLinkFragments.push(convertHeadingToHTMLFragment(content));
|
||||||
|
});
|
||||||
|
filterTokens(params, "inline", function (token) {
|
||||||
|
token.children.forEach(function (child) {
|
||||||
|
var lineNumber = child.lineNumber, type = child.type, attrs = child.attrs;
|
||||||
|
if (type === "link_open") {
|
||||||
|
var href = attrs.find(function (attr) { return attr[0] === "href"; });
|
||||||
|
if (href !== undefined &&
|
||||||
|
href[1].startsWith("#") &&
|
||||||
|
!validLinkFragments.includes(href[1])) {
|
||||||
|
var detail = "Link Fragment is invalid";
|
||||||
|
addError(onError, lineNumber, detail, href[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/***/ }),
|
/***/ }),
|
||||||
|
|
||||||
/***/ "../lib/rules.js":
|
/***/ "../lib/rules.js":
|
||||||
|
@ -4441,7 +4493,8 @@ var rules = [
|
||||||
__webpack_require__(/*! ./md047 */ "../lib/md047.js"),
|
__webpack_require__(/*! ./md047 */ "../lib/md047.js"),
|
||||||
__webpack_require__(/*! ./md048 */ "../lib/md048.js"),
|
__webpack_require__(/*! ./md048 */ "../lib/md048.js"),
|
||||||
__webpack_require__(/*! ./md049 */ "../lib/md049.js"),
|
__webpack_require__(/*! ./md049 */ "../lib/md049.js"),
|
||||||
__webpack_require__(/*! ./md050 */ "../lib/md050.js")
|
__webpack_require__(/*! ./md050 */ "../lib/md050.js"),
|
||||||
|
__webpack_require__(/*! ./md051 */ "../lib/md051.js")
|
||||||
];
|
];
|
||||||
rules.forEach(function (rule) {
|
rules.forEach(function (rule) {
|
||||||
var name = rule.names[0].toLowerCase();
|
var name = rule.names[0].toLowerCase();
|
||||||
|
|
29
doc/Rules.md
29
doc/Rules.md
|
@ -56,7 +56,8 @@ Aliases: first-heading-h1, first-header-h1
|
||||||
Parameters: level (number; default 1)
|
Parameters: level (number; default 1)
|
||||||
|
|
||||||
> Note: *MD002 has been deprecated and is disabled by default.*
|
> Note: *MD002 has been deprecated and is disabled by default.*
|
||||||
> [MD041/first-line-heading](#md041) offers an improved implementation.
|
> [MD041/first-line-heading](#md041---first-line-in-a-file-should-be-a-top-level-heading)
|
||||||
|
offers an improved implementation.
|
||||||
|
|
||||||
This rule is intended to ensure document headings start at the top level and
|
This rule is intended to ensure document headings start at the top level and
|
||||||
is triggered when the first heading in the document isn't an h1 heading:
|
is triggered when the first heading in the document isn't an h1 heading:
|
||||||
|
@ -783,7 +784,8 @@ The `lines_above` and `lines_below` parameters can be used to specify a differen
|
||||||
number of blank lines (including 0) above or below each heading.
|
number of blank lines (including 0) above or below each heading.
|
||||||
|
|
||||||
Note: If `lines_above` or `lines_below` are configured to require more than one
|
Note: If `lines_above` or `lines_below` are configured to require more than one
|
||||||
blank line, [MD012/no-multiple-blanks](#md012) should also be customized.
|
blank line, [MD012/no-multiple-blanks](#md012---multiple-consecutive-blank-lines)
|
||||||
|
should also be customized.
|
||||||
|
|
||||||
Rationale: Aside from aesthetic reasons, some parsers, including `kramdown`, will
|
Rationale: Aside from aesthetic reasons, some parsers, including `kramdown`, will
|
||||||
not parse headings that don't have a blank line before, and will parse them as
|
not parse headings that don't have a blank line before, and will parse them as
|
||||||
|
@ -1984,3 +1986,26 @@ The configured strong style can be a specific symbol to use ("asterisk",
|
||||||
"underscore"), or can require that usage be consistent within the document.
|
"underscore"), or can require that usage be consistent within the document.
|
||||||
|
|
||||||
Rationale: Consistent formatting makes it easier to understand a document.
|
Rationale: Consistent formatting makes it easier to understand a document.
|
||||||
|
|
||||||
|
<a name="md051"></a>
|
||||||
|
|
||||||
|
## MD051 - Link fragments should be valid
|
||||||
|
|
||||||
|
Tags: links
|
||||||
|
|
||||||
|
Aliases: valid-link-fragments
|
||||||
|
|
||||||
|
This rule is triggered if a link fragment does not correspond to a
|
||||||
|
heading within the document:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
# Title
|
||||||
|
|
||||||
|
[Link](#invalid-fragment)
|
||||||
|
```
|
||||||
|
|
||||||
|
To fix this issue, ensure that the heading exists,
|
||||||
|
here you could replace `#invalid-fragment` by `#title`.
|
||||||
|
|
||||||
|
It's not part of the CommonMark specification,
|
||||||
|
for example [GitHub turn headings into links](https://docs.github.com/en/get-started/writing-on-github/getting-started-with-writing-and-formatting-on-github/basic-writing-and-formatting-syntax#section-links).
|
||||||
|
|
46
lib/md051.js
Normal file
46
lib/md051.js
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const { addError, forEachHeading, filterTokens } = require("../helpers");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a Markdown heading into an HTML fragment
|
||||||
|
* according to the rules used by GitHub.
|
||||||
|
*
|
||||||
|
* @param {string} string The string to convert.
|
||||||
|
* @returns {string} The converted string.
|
||||||
|
*/
|
||||||
|
function convertHeadingToHTMLFragment(string) {
|
||||||
|
return "#" + string
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/ /g, "-")
|
||||||
|
.replace(/[^-_a-z0-9]/g, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"names": [ "MD051", "valid-link-fragments" ],
|
||||||
|
"description": "Link fragments should be valid",
|
||||||
|
"tags": [ "links" ],
|
||||||
|
"function": function MD051(params, onError) {
|
||||||
|
const validLinkFragments = [];
|
||||||
|
forEachHeading(params, (_heading, content) => {
|
||||||
|
validLinkFragments.push(convertHeadingToHTMLFragment(content));
|
||||||
|
});
|
||||||
|
filterTokens(params, "inline", (token) => {
|
||||||
|
token.children.forEach((child) => {
|
||||||
|
const { lineNumber, type, attrs } = child;
|
||||||
|
if (type === "link_open") {
|
||||||
|
const href = attrs.find((attr) => attr[0] === "href");
|
||||||
|
if (href !== undefined &&
|
||||||
|
href[1].startsWith("#") &&
|
||||||
|
!validLinkFragments.includes(href[1])
|
||||||
|
) {
|
||||||
|
const detail = "Link Fragment is invalid";
|
||||||
|
addError(onError, lineNumber, detail, href[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
|
@ -50,7 +50,8 @@ const rules = [
|
||||||
require("./md047"),
|
require("./md047"),
|
||||||
require("./md048"),
|
require("./md048"),
|
||||||
require("./md049"),
|
require("./md049"),
|
||||||
require("./md050")
|
require("./md050"),
|
||||||
|
require("./md051")
|
||||||
];
|
];
|
||||||
rules.forEach((rule) => {
|
rules.forEach((rule) => {
|
||||||
const name = rule.names[0].toLowerCase();
|
const name = rule.names[0].toLowerCase();
|
||||||
|
|
|
@ -262,5 +262,8 @@
|
||||||
"MD050": {
|
"MD050": {
|
||||||
// Strong style should be consistent
|
// Strong style should be consistent
|
||||||
"style": "consistent"
|
"style": "consistent"
|
||||||
}
|
},
|
||||||
|
|
||||||
|
// MD051/valid-link-fragments - Link fragments should be valid
|
||||||
|
"MD051": true
|
||||||
}
|
}
|
|
@ -236,4 +236,7 @@ MD049:
|
||||||
# MD050/strong-style - Strong style should be consistent
|
# MD050/strong-style - Strong style should be consistent
|
||||||
MD050:
|
MD050:
|
||||||
# Strong style should be consistent
|
# Strong style should be consistent
|
||||||
style: "consistent"
|
style: "consistent"
|
||||||
|
|
||||||
|
# MD051/valid-link-fragments - Link fragments should be valid
|
||||||
|
MD051: true
|
|
@ -878,6 +878,16 @@
|
||||||
"strong-style": {
|
"strong-style": {
|
||||||
"$ref": "#/properties/MD050"
|
"$ref": "#/properties/MD050"
|
||||||
},
|
},
|
||||||
|
"MD051": {
|
||||||
|
"description": "MD051/valid-link-fragments - Link fragments should be valid",
|
||||||
|
"type": "boolean",
|
||||||
|
"default": true
|
||||||
|
},
|
||||||
|
"valid-link-fragments": {
|
||||||
|
"description": "MD051/valid-link-fragments - Link fragments should be valid",
|
||||||
|
"type": "boolean",
|
||||||
|
"default": true
|
||||||
|
},
|
||||||
"headings": {
|
"headings": {
|
||||||
"description": "headings - MD001, MD002, MD003, MD018, MD019, MD020, MD021, MD022, MD023, MD024, MD025, MD026, MD036, MD041, MD043",
|
"description": "headings - MD001, MD002, MD003, MD018, MD019, MD020, MD021, MD022, MD023, MD024, MD025, MD026, MD036, MD041, MD043",
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
|
@ -914,7 +924,7 @@
|
||||||
"default": true
|
"default": true
|
||||||
},
|
},
|
||||||
"links": {
|
"links": {
|
||||||
"description": "links - MD011, MD034, MD039, MD042",
|
"description": "links - MD011, MD034, MD039, MD042, MD051",
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"default": true
|
"default": true
|
||||||
},
|
},
|
||||||
|
|
|
@ -314,5 +314,31 @@
|
||||||
"MD050",
|
"MD050",
|
||||||
"strong-style"
|
"strong-style"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"errorContext": "#",
|
||||||
|
"errorDetail": "Link Fragment is invalid",
|
||||||
|
"errorRange": null,
|
||||||
|
"fixInfo": null,
|
||||||
|
"lineNumber": 5,
|
||||||
|
"ruleDescription": "Link fragments should be valid",
|
||||||
|
"ruleInformation": "https://github.com/DavidAnson/markdownlint/blob/v0.0.0/doc/Rules.md#md051",
|
||||||
|
"ruleNames": [
|
||||||
|
"MD051",
|
||||||
|
"valid-link-fragments"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"errorContext": "#one",
|
||||||
|
"errorDetail": "Link Fragment is invalid",
|
||||||
|
"errorRange": null,
|
||||||
|
"fixInfo": null,
|
||||||
|
"lineNumber": 17,
|
||||||
|
"ruleDescription": "Link fragments should be valid",
|
||||||
|
"ruleInformation": "https://github.com/DavidAnson/markdownlint/blob/v0.0.0/doc/Rules.md#md051",
|
||||||
|
"ruleNames": [
|
||||||
|
"MD051",
|
||||||
|
"valid-link-fragments"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
|
@ -12,22 +12,24 @@
|
||||||
|
|
||||||
[text]( <> "title" ) {MD042}
|
[text]( <> "title" ) {MD042}
|
||||||
|
|
||||||
[text](#) {MD042}
|
[text](#) {MD042} {MD051}
|
||||||
|
|
||||||
[text]( # ) {MD042}
|
[text]( # ) {MD042} {MD051}
|
||||||
|
|
||||||
[text](# "title") {MD042}
|
[text](# "title") {MD042} {MD051}
|
||||||
|
|
||||||
[text]( # "title" ) {MD042}
|
[text]( # "title" ) {MD042} {MD051}
|
||||||
|
|
||||||
[text][frag] {MD042}
|
[text][frag] {MD042} {MD051}
|
||||||
|
|
||||||
[text][ frag ] {MD042}
|
[text][ frag ] {MD042} {MD051}
|
||||||
|
|
||||||
[frag]: #
|
[frag]: #
|
||||||
|
|
||||||
## Non-empty links
|
## Non-empty links
|
||||||
|
|
||||||
|
### frag
|
||||||
|
|
||||||
[text](link)
|
[text](link)
|
||||||
|
|
||||||
[text]( link )
|
[text]( link )
|
||||||
|
|
|
@ -841,7 +841,7 @@ test.cb("customFileSystemAsync", (t) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
test.cb("readme", (t) => {
|
test.cb("readme", (t) => {
|
||||||
t.plan(119);
|
t.plan(121);
|
||||||
const tagToRules = {};
|
const tagToRules = {};
|
||||||
rules.forEach(function forRule(rule) {
|
rules.forEach(function forRule(rule) {
|
||||||
rule.tags.forEach(function forTag(tag) {
|
rule.tags.forEach(function forTag(tag) {
|
||||||
|
@ -917,7 +917,7 @@ test.cb("readme", (t) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
test.cb("rules", (t) => {
|
test.cb("rules", (t) => {
|
||||||
t.plan(352);
|
t.plan(359);
|
||||||
fs.readFile("doc/Rules.md", "utf8",
|
fs.readFile("doc/Rules.md", "utf8",
|
||||||
(err, contents) => {
|
(err, contents) => {
|
||||||
t.falsy(err);
|
t.falsy(err);
|
||||||
|
@ -1094,7 +1094,7 @@ test("validateConfigExampleJson", async(t) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
test("allBuiltInRulesHaveValidUrl", (t) => {
|
test("allBuiltInRulesHaveValidUrl", (t) => {
|
||||||
t.plan(138);
|
t.plan(141);
|
||||||
rules.forEach(function forRule(rule) {
|
rules.forEach(function forRule(rule) {
|
||||||
t.truthy(rule.information);
|
t.truthy(rule.information);
|
||||||
t.true(Object.getPrototypeOf(rule.information) === URL.prototype);
|
t.true(Object.getPrototypeOf(rule.information) === URL.prototype);
|
||||||
|
|
|
@ -78,21 +78,21 @@ Text [link](https://example.com/link`link`link`link) text `code`.
|
||||||
|
|
||||||
Text [link](https://example.com/link "title`title") text `code`.
|
Text [link](https://example.com/link "title`title") text `code`.
|
||||||
|
|
||||||
Text [link](#link`link) text `code`.
|
Text [link](#link`link) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link] (#link`link) text `code`. {MD038}
|
Text [link] (#link`link) text `code`. {MD038}
|
||||||
|
|
||||||
Text [link[link](#link`link) text `code`.
|
Text [link[link](#link`link) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link(link](#link`link) text `code`.
|
Text [link(link](#link`link) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link)link](#link`link) text `code`.
|
Text [link)link](#link`link) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link](#link[link`link) text `code`.
|
Text [link](#link[linklink) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link](#link]link`link) text `code`.
|
Text [link](#link[linklink) text `code`. {MD051}
|
||||||
|
|
||||||
Text [link](#link(link`link) text `code`. {MD038}
|
Text [link](#link[linklink) text `code`. {MD051}
|
||||||
|
|
||||||
Text [`link`](xref:custom.link`1) text `code`.
|
Text [`link`](xref:custom.link`1) text `code`.
|
||||||
|
|
||||||
|
|
56
test/valid-link-fragments.md
Normal file
56
test/valid-link-fragments.md
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
# Valid/Invalid Link Fragments
|
||||||
|
|
||||||
|
## Valid Fragments
|
||||||
|
|
||||||
|
[Valid](#validinvalid-link-fragments)
|
||||||
|
|
||||||
|
[Valid](#valid-fragments)
|
||||||
|
|
||||||
|
[Valid](#valid-h3-heading)
|
||||||
|
|
||||||
|
[Valid](#valid-heading-with-underscores-_)
|
||||||
|
|
||||||
|
[Valid](#valid-heading-with-quotes--and-double-quotes-)
|
||||||
|
|
||||||
|
[Valid](#-valid-heading-with-emoji)
|
||||||
|
|
||||||
|
[Valid](#valid-heading--with-emoji-2)
|
||||||
|
|
||||||
|
[Valid](#valid-closed-atx-heading)
|
||||||
|
|
||||||
|
[Valid](#valid-setex-heading)
|
||||||
|
|
||||||
|
### Valid H3 Heading
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
### Valid Heading With Underscores _
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
### Valid Heading With Quotes ' And Double Quotes "
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
### 🚀 Valid Heading With Emoji
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
### Valid Heading 👀 With Emoji 2
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
<!-- markdownlint-disable-next-line MD003 -->
|
||||||
|
### Valid Closed ATX Heading ###
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
<!-- markdownlint-disable-next-line MD003 -->
|
||||||
|
Valid Setex Heading
|
||||||
|
----
|
||||||
|
|
||||||
|
Text
|
||||||
|
|
||||||
|
## Invalid Fragments
|
||||||
|
|
||||||
|
[Invalid](#invalid-fragments-not-exist) {MD051}
|
Loading…
Add table
Add a link
Reference in a new issue