mirror of
https://github.com/withastro/astro.git
synced 2025-01-06 22:10:10 -05:00
ac03218247
* feat: add { extends } to markdown config * test: remark plugins with extends * deps: pnpm lock * chore: changeset * fix: remarkPlugins -> rehypePlugins * docs: update markdown config reference * Revert "feat: add { extends } to markdown config" This reverts commit 5d050bbcf9a2c0d470cae79c4d0a954d489f4e8c. * feat: new "extendDefaultPlugins" flag * docs: update config * nit: We -> Astro applies * fix: backticks on `false` * nit: Note -> REAL note Co-authored-by: Sarah Rainsberger <sarah@rainsberger.ca> * docs: note -> caution Co-authored-by: Sarah Rainsberger <sarah@rainsberger.ca>
148 lines
4.4 KiB
TypeScript
148 lines
4.4 KiB
TypeScript
import type { MarkdownRenderingOptions, MarkdownRenderingResult } from './types';
|
|
|
|
import { loadPlugins } from './load-plugins.js';
|
|
import createCollectHeadings from './rehype-collect-headings.js';
|
|
import rehypeEscape from './rehype-escape.js';
|
|
import rehypeExpressions from './rehype-expressions.js';
|
|
import rehypeIslands from './rehype-islands.js';
|
|
import rehypeJsx from './rehype-jsx.js';
|
|
import remarkEscape from './remark-escape.js';
|
|
import { remarkInitializeAstroData } from './remark-initialize-astro-data.js';
|
|
import remarkMarkAndUnravel from './remark-mark-and-unravel.js';
|
|
import remarkMdxish from './remark-mdxish.js';
|
|
import remarkPrism from './remark-prism.js';
|
|
import scopedStyles from './remark-scoped-styles.js';
|
|
import remarkShiki from './remark-shiki.js';
|
|
import remarkUnwrap from './remark-unwrap.js';
|
|
|
|
import rehypeRaw from 'rehype-raw';
|
|
import rehypeStringify from 'rehype-stringify';
|
|
import markdown from 'remark-parse';
|
|
import markdownToHtml from 'remark-rehype';
|
|
import { unified } from 'unified';
|
|
import { VFile } from 'vfile';
|
|
|
|
export * from './types.js';
|
|
|
|
export const DEFAULT_REMARK_PLUGINS = ['remark-gfm', 'remark-smartypants'];
|
|
export const DEFAULT_REHYPE_PLUGINS = [];
|
|
|
|
/** Shared utility for rendering markdown */
|
|
export async function renderMarkdown(
|
|
content: string,
|
|
opts: MarkdownRenderingOptions
|
|
): Promise<MarkdownRenderingResult> {
|
|
let {
|
|
fileURL,
|
|
syntaxHighlight = 'shiki',
|
|
shikiConfig = {},
|
|
remarkPlugins = [],
|
|
rehypePlugins = [],
|
|
remarkRehype = {},
|
|
extendDefaultPlugins = false,
|
|
isAstroFlavoredMd = false,
|
|
} = opts;
|
|
const input = new VFile({ value: content, path: fileURL });
|
|
const scopedClassName = opts.$?.scopedClassName;
|
|
const { headings, rehypeCollectHeadings } = createCollectHeadings();
|
|
|
|
let parser = unified()
|
|
.use(markdown)
|
|
.use(remarkInitializeAstroData)
|
|
.use(isAstroFlavoredMd ? [remarkMdxish, remarkMarkAndUnravel, remarkUnwrap, remarkEscape] : []);
|
|
|
|
if (extendDefaultPlugins || (remarkPlugins.length === 0 && rehypePlugins.length === 0)) {
|
|
remarkPlugins = [...DEFAULT_REMARK_PLUGINS, ...remarkPlugins];
|
|
rehypePlugins = [...DEFAULT_REHYPE_PLUGINS, ...rehypePlugins];
|
|
}
|
|
|
|
const loadedRemarkPlugins = await Promise.all(loadPlugins(remarkPlugins));
|
|
const loadedRehypePlugins = await Promise.all(loadPlugins(rehypePlugins));
|
|
|
|
loadedRemarkPlugins.forEach(([plugin, pluginOpts]) => {
|
|
parser.use([[plugin, pluginOpts]]);
|
|
});
|
|
|
|
if (scopedClassName) {
|
|
parser.use([scopedStyles(scopedClassName)]);
|
|
}
|
|
|
|
if (syntaxHighlight === 'shiki') {
|
|
parser.use([await remarkShiki(shikiConfig, scopedClassName)]);
|
|
} else if (syntaxHighlight === 'prism') {
|
|
parser.use([remarkPrism(scopedClassName)]);
|
|
}
|
|
|
|
parser.use([
|
|
[
|
|
markdownToHtml as any,
|
|
{
|
|
allowDangerousHtml: true,
|
|
passThrough: isAstroFlavoredMd
|
|
? [
|
|
'raw',
|
|
'mdxFlowExpression',
|
|
'mdxJsxFlowElement',
|
|
'mdxJsxTextElement',
|
|
'mdxTextExpression',
|
|
]
|
|
: [],
|
|
...remarkRehype,
|
|
},
|
|
],
|
|
]);
|
|
|
|
loadedRehypePlugins.forEach(([plugin, pluginOpts]) => {
|
|
parser.use([[plugin, pluginOpts]]);
|
|
});
|
|
|
|
parser
|
|
.use(
|
|
isAstroFlavoredMd
|
|
? [rehypeJsx, rehypeExpressions, rehypeEscape, rehypeIslands, rehypeCollectHeadings]
|
|
: [rehypeCollectHeadings, rehypeRaw]
|
|
)
|
|
.use(rehypeStringify, { allowDangerousHtml: true });
|
|
|
|
let vfile: VFile;
|
|
try {
|
|
vfile = await parser.process(input);
|
|
} catch (err) {
|
|
// Ensure that the error message contains the input filename
|
|
// to make it easier for the user to fix the issue
|
|
err = prefixError(err, `Failed to parse Markdown file "${input.path}"`);
|
|
// eslint-disable-next-line no-console
|
|
console.error(err);
|
|
throw err;
|
|
}
|
|
|
|
return {
|
|
metadata: { headings, source: content, html: String(vfile.value) },
|
|
code: String(vfile.value),
|
|
vfile,
|
|
};
|
|
}
|
|
|
|
function prefixError(err: any, prefix: string) {
|
|
// If the error is an object with a `message` property, attempt to prefix the message
|
|
if (err && err.message) {
|
|
try {
|
|
err.message = `${prefix}:\n${err.message}`;
|
|
return err;
|
|
} catch (error) {
|
|
// Any errors here are ok, there's fallback code below
|
|
}
|
|
}
|
|
|
|
// If that failed, create a new error with the desired message and attempt to keep the stack
|
|
const wrappedError = new Error(`${prefix}${err ? `: ${err}` : ''}`);
|
|
try {
|
|
wrappedError.stack = err.stack;
|
|
// @ts-ignore
|
|
wrappedError.cause = err;
|
|
} catch (error) {
|
|
// It's ok if we could not set the stack or cause - the message is the most important part
|
|
}
|
|
|
|
return wrappedError;
|
|
}
|