mirror of
https://github.com/DavidAnson/markdownlint.git
synced 2025-12-16 22:10:13 +01:00
Reimplement MD025/single-title/single-h1 using micromark tokens.
This commit is contained in:
parent
6daaa43410
commit
ea9659841e
5 changed files with 82 additions and 38 deletions
|
|
@ -4671,8 +4671,9 @@ module.exports = {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
const { addErrorContext, filterTokens, frontMatterHasTitle } =
|
const { addErrorContext, frontMatterHasTitle } = __webpack_require__(/*! ../helpers */ "../helpers/helpers.js");
|
||||||
__webpack_require__(/*! ../helpers */ "../helpers/helpers.js");
|
const { filterByTypes, getHeadingLevel, inHtmlFlow } =
|
||||||
|
__webpack_require__(/*! ../helpers/micromark.cjs */ "../helpers/micromark.cjs");
|
||||||
|
|
||||||
// eslint-disable-next-line jsdoc/valid-types
|
// eslint-disable-next-line jsdoc/valid-types
|
||||||
/** @type import("./markdownlint").Rule */
|
/** @type import("./markdownlint").Rule */
|
||||||
|
|
@ -4680,26 +4681,41 @@ module.exports = {
|
||||||
"names": [ "MD025", "single-title", "single-h1" ],
|
"names": [ "MD025", "single-title", "single-h1" ],
|
||||||
"description": "Multiple top-level headings in the same document",
|
"description": "Multiple top-level headings in the same document",
|
||||||
"tags": [ "headings" ],
|
"tags": [ "headings" ],
|
||||||
"parser": "markdownit",
|
"parser": "micromark",
|
||||||
"function": function MD025(params, onError) {
|
"function": function MD025(params, onError) {
|
||||||
const level = Number(params.config.level || 1);
|
const level = Number(params.config.level || 1);
|
||||||
const tag = "h" + level;
|
|
||||||
const foundFrontMatterTitle =
|
const foundFrontMatterTitle =
|
||||||
frontMatterHasTitle(
|
frontMatterHasTitle(
|
||||||
params.frontMatterLines,
|
params.frontMatterLines,
|
||||||
params.config.front_matter_title
|
params.config.front_matter_title
|
||||||
);
|
);
|
||||||
let hasTopLevelHeading = false;
|
let hasTopLevelHeading = false;
|
||||||
filterTokens(params, "heading_open", function forToken(token) {
|
const headings = filterByTypes(
|
||||||
if (token.tag === tag) {
|
params.parsers.micromark.tokens,
|
||||||
|
[ "atxHeading", "setextHeading" ]
|
||||||
|
);
|
||||||
|
for (const heading of headings) {
|
||||||
|
const headingLevel = getHeadingLevel(heading);
|
||||||
|
if ((headingLevel === level) && !inHtmlFlow(heading)) {
|
||||||
if (hasTopLevelHeading || foundFrontMatterTitle) {
|
if (hasTopLevelHeading || foundFrontMatterTitle) {
|
||||||
addErrorContext(onError, token.lineNumber,
|
const headingTexts = filterByTypes(
|
||||||
token.line.trim());
|
heading.children,
|
||||||
} else if (token.lineNumber === 1) {
|
[ "atxHeadingText", "setextHeadingText" ]
|
||||||
|
);
|
||||||
|
const headingText = headingTexts.
|
||||||
|
map((token) => token.text).
|
||||||
|
join(" ").
|
||||||
|
replace(/[\r\n]+/g, " ");
|
||||||
|
addErrorContext(
|
||||||
|
onError,
|
||||||
|
heading.startLine,
|
||||||
|
headingText
|
||||||
|
);
|
||||||
|
} else if (heading.startLine === 1) {
|
||||||
hasTopLevelHeading = true;
|
hasTopLevelHeading = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
36
lib/md025.js
36
lib/md025.js
|
|
@ -2,8 +2,9 @@
|
||||||
|
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
const { addErrorContext, filterTokens, frontMatterHasTitle } =
|
const { addErrorContext, frontMatterHasTitle } = require("../helpers");
|
||||||
require("../helpers");
|
const { filterByTypes, getHeadingLevel, inHtmlFlow } =
|
||||||
|
require("../helpers/micromark.cjs");
|
||||||
|
|
||||||
// eslint-disable-next-line jsdoc/valid-types
|
// eslint-disable-next-line jsdoc/valid-types
|
||||||
/** @type import("./markdownlint").Rule */
|
/** @type import("./markdownlint").Rule */
|
||||||
|
|
@ -11,25 +12,40 @@ module.exports = {
|
||||||
"names": [ "MD025", "single-title", "single-h1" ],
|
"names": [ "MD025", "single-title", "single-h1" ],
|
||||||
"description": "Multiple top-level headings in the same document",
|
"description": "Multiple top-level headings in the same document",
|
||||||
"tags": [ "headings" ],
|
"tags": [ "headings" ],
|
||||||
"parser": "markdownit",
|
"parser": "micromark",
|
||||||
"function": function MD025(params, onError) {
|
"function": function MD025(params, onError) {
|
||||||
const level = Number(params.config.level || 1);
|
const level = Number(params.config.level || 1);
|
||||||
const tag = "h" + level;
|
|
||||||
const foundFrontMatterTitle =
|
const foundFrontMatterTitle =
|
||||||
frontMatterHasTitle(
|
frontMatterHasTitle(
|
||||||
params.frontMatterLines,
|
params.frontMatterLines,
|
||||||
params.config.front_matter_title
|
params.config.front_matter_title
|
||||||
);
|
);
|
||||||
let hasTopLevelHeading = false;
|
let hasTopLevelHeading = false;
|
||||||
filterTokens(params, "heading_open", function forToken(token) {
|
const headings = filterByTypes(
|
||||||
if (token.tag === tag) {
|
params.parsers.micromark.tokens,
|
||||||
|
[ "atxHeading", "setextHeading" ]
|
||||||
|
);
|
||||||
|
for (const heading of headings) {
|
||||||
|
const headingLevel = getHeadingLevel(heading);
|
||||||
|
if ((headingLevel === level) && !inHtmlFlow(heading)) {
|
||||||
if (hasTopLevelHeading || foundFrontMatterTitle) {
|
if (hasTopLevelHeading || foundFrontMatterTitle) {
|
||||||
addErrorContext(onError, token.lineNumber,
|
const headingTexts = filterByTypes(
|
||||||
token.line.trim());
|
heading.children,
|
||||||
} else if (token.lineNumber === 1) {
|
[ "atxHeadingText", "setextHeadingText" ]
|
||||||
|
);
|
||||||
|
const headingText = headingTexts.
|
||||||
|
map((token) => token.text).
|
||||||
|
join(" ").
|
||||||
|
replace(/[\r\n]+/g, " ");
|
||||||
|
addErrorContext(
|
||||||
|
onError,
|
||||||
|
heading.startLine,
|
||||||
|
headingText
|
||||||
|
);
|
||||||
|
} else if (heading.startLine === 1) {
|
||||||
hasTopLevelHeading = true;
|
hasTopLevelHeading = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,9 @@
|
||||||
# Heading 1
|
# Heading 1
|
||||||
|
|
||||||
# Heading 2 {MD025}
|
# Heading 2 {MD025}
|
||||||
|
|
||||||
|
<p>
|
||||||
|
# Not heading
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- markdownlint-disable-file no-inline-html -->
|
||||||
|
|
|
||||||
|
|
@ -653,7 +653,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# Heading',
|
errorContext: 'Heading',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -2564,7 +2564,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '## Another one {MD025}',
|
errorContext: 'Another one {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -5589,7 +5589,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '> # Quoted heading {MD025}',
|
errorContext: 'Quoted heading {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -5603,7 +5603,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '> > # Double-quoted heading {M...',
|
errorContext: 'Double-quoted heading {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -5617,7 +5617,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '> # Quoted heading in list {MD...',
|
errorContext: 'Quoted heading in list {MD022}...',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -5631,7 +5631,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '> > # Double-quoted heading in...',
|
errorContext: 'Double-quoted heading in list ...',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -13304,7 +13304,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '# Top level heading {MD025}',
|
errorContext: 'Top level heading {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -13556,7 +13556,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '# Another {MD025}',
|
errorContext: 'Another {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -13653,7 +13653,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '# Top level heading {MD025}',
|
errorContext: 'Top level heading {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -14855,7 +14855,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# A',
|
errorContext: 'A',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -14869,7 +14869,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# A',
|
errorContext: 'A',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -14883,7 +14883,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# Heading duplicate content si...',
|
errorContext: 'Heading duplicate content sibl...',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -15147,7 +15147,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '# Heading 2 {MD025}',
|
errorContext: 'Heading 2 {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -15164,6 +15164,12 @@ Generated by [AVA](https://avajs.dev).
|
||||||
fixed: `# Heading 1␊
|
fixed: `# Heading 1␊
|
||||||
␊
|
␊
|
||||||
# Heading 2 {MD025}␊
|
# Heading 2 {MD025}␊
|
||||||
|
␊
|
||||||
|
<p>␊
|
||||||
|
# Not heading␊
|
||||||
|
</p>␊
|
||||||
|
␊
|
||||||
|
<!-- markdownlint-disable-file no-inline-html -->␊
|
||||||
`,
|
`,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -16421,7 +16427,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: 'Some text {MD022} {MD025}',
|
errorContext: 'Some text {MD022} {MD025} Head...',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -16435,7 +16441,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: 'Some text {MD022} {MD025}',
|
errorContext: 'Some text {MD022} {MD025} Head...',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -19179,7 +19185,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
{
|
{
|
||||||
errors: [
|
errors: [
|
||||||
{
|
{
|
||||||
errorContext: '# Heading {MD025}',
|
errorContext: 'Heading {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -23886,7 +23892,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# heading1 {MD025}',
|
errorContext: 'heading1 {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
@ -23939,7 +23945,7 @@ Generated by [AVA](https://avajs.dev).
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
errorContext: '# header1 {MD025}',
|
errorContext: 'header1 {MD025}',
|
||||||
errorDetail: null,
|
errorDetail: null,
|
||||||
errorRange: null,
|
errorRange: null,
|
||||||
fixInfo: null,
|
fixInfo: null,
|
||||||
|
|
|
||||||
Binary file not shown.
Loading…
Add table
Add a link
Reference in a new issue