mirror of
https://github.com/versia-pub/docs.git
synced 2025-12-06 14:28:20 +01:00
142 lines
4.8 KiB
JavaScript
142 lines
4.8 KiB
JavaScript
|
|
import * as fs from "node:fs";
|
||
|
|
import * as path from "node:path";
|
||
|
|
import * as url from "node:url";
|
||
|
|
import { slugifyWithCounter } from "@sindresorhus/slugify";
|
||
|
|
import glob from "fast-glob";
|
||
|
|
import { toString as mdastToString } from "mdast-util-to-string";
|
||
|
|
import { remark } from "remark";
|
||
|
|
import remarkMdx from "remark-mdx";
|
||
|
|
import { createLoader } from "simple-functional-loader";
|
||
|
|
import { filter } from "unist-util-filter";
|
||
|
|
import { SKIP, visit } from "unist-util-visit";
|
||
|
|
|
||
|
|
const __filename = url.fileURLToPath(import.meta.url);
|
||
|
|
const processor = remark().use(remarkMdx).use(extractSections);
|
||
|
|
const slugify = slugifyWithCounter();
|
||
|
|
|
||
|
|
function isObjectExpression(node) {
|
||
|
|
return (
|
||
|
|
node.type === "mdxTextExpression" &&
|
||
|
|
node.data?.estree?.body?.[0]?.expression?.type === "ObjectExpression"
|
||
|
|
);
|
||
|
|
}
|
||
|
|
|
||
|
|
function excludeObjectExpressions(tree) {
|
||
|
|
return filter(tree, (node) => !isObjectExpression(node));
|
||
|
|
}
|
||
|
|
|
||
|
|
function extractSections() {
|
||
|
|
return (tree, { sections }) => {
|
||
|
|
slugify.reset();
|
||
|
|
|
||
|
|
visit(tree, (node) => {
|
||
|
|
if (node.type === "heading" || node.type === "paragraph") {
|
||
|
|
const content = mdastToString(excludeObjectExpressions(node));
|
||
|
|
if (node.type === "heading" && node.depth <= 2) {
|
||
|
|
const hash = node.depth === 1 ? null : slugify(content);
|
||
|
|
sections.push([content, hash, []]);
|
||
|
|
} else {
|
||
|
|
sections.at(-1)?.[2].push(content);
|
||
|
|
}
|
||
|
|
return SKIP;
|
||
|
|
}
|
||
|
|
});
|
||
|
|
};
|
||
|
|
}
|
||
|
|
|
||
|
|
export default function Search(nextConfig = {}) {
|
||
|
|
const cache = new Map();
|
||
|
|
|
||
|
|
return Object.assign({}, nextConfig, {
|
||
|
|
webpack(config, options) {
|
||
|
|
config.module.rules.push({
|
||
|
|
test: __filename,
|
||
|
|
use: [
|
||
|
|
createLoader(function () {
|
||
|
|
const appDir = path.resolve("./app");
|
||
|
|
this.addContextDependency(appDir);
|
||
|
|
|
||
|
|
const files = glob.sync("**/*.mdx", { cwd: appDir });
|
||
|
|
const data = files.map((file) => {
|
||
|
|
const url = `/${file.replace(/(^|\/)page\.mdx$/, "")}`;
|
||
|
|
const mdx = fs.readFileSync(
|
||
|
|
path.join(appDir, file),
|
||
|
|
"utf8",
|
||
|
|
);
|
||
|
|
|
||
|
|
let sections = [];
|
||
|
|
|
||
|
|
if (cache.get(file)?.[0] === mdx) {
|
||
|
|
sections = cache.get(file)[1];
|
||
|
|
} else {
|
||
|
|
const vfile = { value: mdx, sections };
|
||
|
|
processor.runSync(
|
||
|
|
processor.parse(vfile),
|
||
|
|
vfile,
|
||
|
|
);
|
||
|
|
cache.set(file, [mdx, sections]);
|
||
|
|
}
|
||
|
|
|
||
|
|
return { url, sections };
|
||
|
|
});
|
||
|
|
|
||
|
|
// When this file is imported within the application
|
||
|
|
// the following module is loaded:
|
||
|
|
return `
|
||
|
|
import FlexSearch from 'flexsearch'
|
||
|
|
|
||
|
|
let sectionIndex = new FlexSearch.Document({
|
||
|
|
tokenize: 'full',
|
||
|
|
document: {
|
||
|
|
id: 'url',
|
||
|
|
index: 'content',
|
||
|
|
store: ['title', 'pageTitle'],
|
||
|
|
},
|
||
|
|
context: {
|
||
|
|
resolution: 9,
|
||
|
|
depth: 2,
|
||
|
|
bidirectional: true
|
||
|
|
}
|
||
|
|
})
|
||
|
|
|
||
|
|
let data = ${JSON.stringify(data)}
|
||
|
|
|
||
|
|
for (let { url, sections } of data) {
|
||
|
|
for (let [title, hash, content] of sections) {
|
||
|
|
sectionIndex.add({
|
||
|
|
url: url + (hash ? ('#' + hash) : ''),
|
||
|
|
title,
|
||
|
|
content: [title, ...content].join('\\n'),
|
||
|
|
pageTitle: hash ? sections[0][0] : undefined,
|
||
|
|
})
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
export function search(query, options = {}) {
|
||
|
|
let result = sectionIndex.search(query, {
|
||
|
|
...options,
|
||
|
|
enrich: true,
|
||
|
|
})
|
||
|
|
if (result.length === 0) {
|
||
|
|
return []
|
||
|
|
}
|
||
|
|
return result[0].result.map((item) => ({
|
||
|
|
url: item.id,
|
||
|
|
title: item.doc.title,
|
||
|
|
pageTitle: item.doc.pageTitle,
|
||
|
|
}))
|
||
|
|
}
|
||
|
|
`;
|
||
|
|
}),
|
||
|
|
],
|
||
|
|
});
|
||
|
|
|
||
|
|
if (typeof nextConfig.webpack === "function") {
|
||
|
|
return nextConfig.webpack(config, options);
|
||
|
|
}
|
||
|
|
|
||
|
|
return config;
|
||
|
|
},
|
||
|
|
});
|
||
|
|
}
|