mirror of https://github.com/vuejs/vitepress
userquin/feat-include-ids-and-anchor-check
parent
c343d938fc
commit
1feab5c05f
@ -0,0 +1,113 @@
|
|||||||
|
import type { SiteConfig } from '../config'
|
||||||
|
import fg from 'fast-glob'
|
||||||
|
import { task } from '../utils/task'
|
||||||
|
import fs from 'fs-extra'
|
||||||
|
import { parse, walkSync, ELEMENT_NODE } from 'ultrahtml'
|
||||||
|
import { dirname, join, resolve } from 'path'
|
||||||
|
|
||||||
|
export async function checkIdsAndAnchorHrefs(siteConfig: SiteConfig) {
|
||||||
|
await task('checking for duplicate ids and bad anchor hrefs', async () => {
|
||||||
|
for await (const error of collectErrors(siteConfig)) {
|
||||||
|
// TODO: use picocolors here
|
||||||
|
console.error(error)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/* exporting this function for testing purposes */
|
||||||
|
export async function* collectErrors(siteConfig: SiteConfig) {
|
||||||
|
const outDir = siteConfig.outDir
|
||||||
|
const files = new Set(
|
||||||
|
siteConfig.pages.map((page) =>
|
||||||
|
`${siteConfig.rewrites.map[page] || page}`
|
||||||
|
.replace(/\\/g, '/')
|
||||||
|
.replace(/\.md$/, '.html')
|
||||||
|
)
|
||||||
|
)
|
||||||
|
// add public html files to the list: i.e. VP docs has public/pure.html
|
||||||
|
for await (const entry of fg.stream('*.html', {
|
||||||
|
cwd: outDir,
|
||||||
|
deep: 1
|
||||||
|
})) {
|
||||||
|
files.add(entry.toString().replace(/\\/g, '/'))
|
||||||
|
}
|
||||||
|
const checkHtmlExt = siteConfig.site.cleanUrls === false
|
||||||
|
const stream = fg.stream('**/*.html', {
|
||||||
|
cwd: siteConfig.outDir
|
||||||
|
})
|
||||||
|
for await (const entry of stream) {
|
||||||
|
const localLinks = new Set<string>()
|
||||||
|
const localIds = new Set<string>()
|
||||||
|
const localErrors: string[] = []
|
||||||
|
const content = parse(
|
||||||
|
await fs.promises.readFile(resolve(outDir, entry.toString()), 'utf8')
|
||||||
|
)
|
||||||
|
// collect id headings and href anchors
|
||||||
|
walkSync(content, (node) => {
|
||||||
|
if (node.type === ELEMENT_NODE) {
|
||||||
|
const id = node.attributes.id
|
||||||
|
if (id) {
|
||||||
|
if (localIds.has(id)) localErrors.push(`duplicate id="${id}"`)
|
||||||
|
else localIds.add(id)
|
||||||
|
}
|
||||||
|
if (node.name.toLowerCase() === 'a') {
|
||||||
|
const href = node.attributes.href
|
||||||
|
if (
|
||||||
|
!href ||
|
||||||
|
href.startsWith('http://') ||
|
||||||
|
href.startsWith('https://')
|
||||||
|
)
|
||||||
|
return
|
||||||
|
localLinks.add(href)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// check for local hrefs and external links
|
||||||
|
for (const href of localLinks) {
|
||||||
|
// 1) check for local heading ids
|
||||||
|
if (href[0] === '#') {
|
||||||
|
const id = href.slice(1)
|
||||||
|
if (!localIds.has(id))
|
||||||
|
localErrors.push(`missing local id for "${href}"`)
|
||||||
|
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// 2) check for external links
|
||||||
|
// Remove parameters and hash
|
||||||
|
let localLink = href.split(/[#?]/).shift()
|
||||||
|
if (!localLink) continue
|
||||||
|
|
||||||
|
// Append .html
|
||||||
|
if (checkHtmlExt) {
|
||||||
|
if (!localLink.endsWith('/')) {
|
||||||
|
localLink += 'index.html'
|
||||||
|
}
|
||||||
|
if (!localLink.endsWith('.html')) {
|
||||||
|
localErrors.push(`bad href link "${href}"`)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (localLink === '/') localLink = '/index.html'
|
||||||
|
if (!localLink.endsWith('.html')) localLink += '.html'
|
||||||
|
}
|
||||||
|
// Get absolute link
|
||||||
|
if (localLink.startsWith('.')) {
|
||||||
|
localLink =
|
||||||
|
'/' + join(dirname(entry.toString()), localLink).replace(/\\/g, '/')
|
||||||
|
}
|
||||||
|
if (!localLink.startsWith('/')) {
|
||||||
|
localErrors.push(`bad href link "${href}"`)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
localLink = localLink.slice(1)
|
||||||
|
if (!localLink) localLink = 'index.html'
|
||||||
|
|
||||||
|
// Check if target html page exists
|
||||||
|
if (!files.has(localLink)) {
|
||||||
|
localErrors.push(`bad href link "${href}" (missing file)`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (localErrors.length)
|
||||||
|
yield `\n${entry}\n${localErrors.map((e) => `\t${e}`).join('\n')}`
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue