853081d1c8
* chore: add rel image error plugin * deps: mdast, mdast types * chore: add rel image throw to mdx * refactor: doc rel image path plugin * fix: respect experimental flag in md remark * chore: changeset * deps: remove mdast package * fix: resolve contentDir from config * fix: apply MDX plugin after user plugins * fix: stub out contentDir
322 lines
10 KiB
TypeScript
322 lines
10 KiB
TypeScript
import { rehypeHeadingIds } from '@astrojs/markdown-remark';
|
|
import type { Image } from 'mdast';
|
|
import { nodeTypes } from '@mdx-js/mdx';
|
|
import type { PluggableList } from '@mdx-js/mdx/lib/core.js';
|
|
import type { Options as MdxRollupPluginOptions } from '@mdx-js/rollup';
|
|
import type { AstroConfig, MarkdownAstroData } from 'astro';
|
|
import type { Literal, MemberExpression } from 'estree';
|
|
import { visit } from 'unist-util-visit';
|
|
import { visit as estreeVisit } from 'estree-util-visit';
|
|
import { bold, yellow } from 'kleur/colors';
|
|
import rehypeRaw from 'rehype-raw';
|
|
import remarkGfm from 'remark-gfm';
|
|
import remarkSmartypants from 'remark-smartypants';
|
|
import type { Data, VFile } from 'vfile';
|
|
import { MdxOptions } from './index.js';
|
|
import { rehypeInjectHeadingsExport } from './rehype-collect-headings.js';
|
|
import rehypeMetaString from './rehype-meta-string.js';
|
|
import remarkPrism from './remark-prism.js';
|
|
import remarkShiki from './remark-shiki.js';
|
|
import { jsToTreeNode, isRelativePath } from './utils.js';
|
|
import { pathToFileURL } from 'node:url';
|
|
|
|
export function recmaInjectImportMetaEnvPlugin({
|
|
importMetaEnv,
|
|
}: {
|
|
importMetaEnv: Record<string, any>;
|
|
}) {
|
|
return (tree: any) => {
|
|
estreeVisit(tree, (node) => {
|
|
if (node.type === 'MemberExpression') {
|
|
// attempt to get "import.meta.env" variable name
|
|
const envVarName = getImportMetaEnvVariableName(node as MemberExpression);
|
|
if (typeof envVarName === 'string') {
|
|
// clear object keys to replace with envVarLiteral
|
|
for (const key in node) {
|
|
delete (node as any)[key];
|
|
}
|
|
const envVarLiteral: Literal = {
|
|
type: 'Literal',
|
|
value: importMetaEnv[envVarName],
|
|
raw: JSON.stringify(importMetaEnv[envVarName]),
|
|
};
|
|
Object.assign(node, envVarLiteral);
|
|
}
|
|
}
|
|
});
|
|
};
|
|
}
|
|
|
|
export function remarkInitializeAstroData() {
|
|
return function (tree: any, vfile: VFile) {
|
|
if (!vfile.data.astro) {
|
|
vfile.data.astro = { frontmatter: {} };
|
|
}
|
|
};
|
|
}
|
|
|
|
export function rehypeApplyFrontmatterExport(pageFrontmatter: Record<string, any>) {
|
|
return function (tree: any, vfile: VFile) {
|
|
const { frontmatter: injectedFrontmatter } = safelyGetAstroData(vfile.data);
|
|
const frontmatter = { ...injectedFrontmatter, ...pageFrontmatter };
|
|
const exportNodes = [
|
|
jsToTreeNode(
|
|
`export const frontmatter = ${JSON.stringify(
|
|
frontmatter
|
|
)};\nexport const _internal = { injectedFrontmatter: ${JSON.stringify(
|
|
injectedFrontmatter
|
|
)} };`
|
|
),
|
|
];
|
|
if (frontmatter.layout) {
|
|
// NOTE(bholmesdev) 08-22-2022
|
|
// Using an async layout import (i.e. `const Layout = (await import...)`)
|
|
// Preserves the dev server import cache when globbing a large set of MDX files
|
|
// Full explanation: 'https://github.com/withastro/astro/pull/4428'
|
|
exportNodes.unshift(
|
|
jsToTreeNode(
|
|
/** @see 'vite-plugin-markdown' for layout props reference */
|
|
`import { jsx as layoutJsx } from 'astro/jsx-runtime';
|
|
|
|
export default async function ({ children }) {
|
|
const Layout = (await import(${JSON.stringify(frontmatter.layout)})).default;
|
|
const { layout, ...content } = frontmatter;
|
|
content.file = file;
|
|
content.url = url;
|
|
content.astro = {};
|
|
Object.defineProperty(content.astro, 'headings', {
|
|
get() {
|
|
throw new Error('The "astro" property is no longer supported! To access "headings" from your layout, try using "Astro.props.headings."')
|
|
}
|
|
});
|
|
Object.defineProperty(content.astro, 'html', {
|
|
get() {
|
|
throw new Error('The "astro" property is no longer supported! To access "html" from your layout, try using "Astro.props.compiledContent()."')
|
|
}
|
|
});
|
|
Object.defineProperty(content.astro, 'source', {
|
|
get() {
|
|
throw new Error('The "astro" property is no longer supported! To access "source" from your layout, try using "Astro.props.rawContent()."')
|
|
}
|
|
});
|
|
return layoutJsx(Layout, {
|
|
file,
|
|
url,
|
|
content,
|
|
frontmatter: content,
|
|
headings: getHeadings(),
|
|
'server:root': true,
|
|
children,
|
|
});
|
|
};`
|
|
)
|
|
);
|
|
}
|
|
tree.children = exportNodes.concat(tree.children);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* `src/content/` does not support relative image paths.
|
|
* This plugin throws an error if any are found
|
|
*/
|
|
function toRemarkContentRelImageError({ srcDir }: { srcDir: URL }) {
|
|
const contentDir = new URL('content/', srcDir);
|
|
return function remarkContentRelImageError() {
|
|
return (tree: any, vfile: VFile) => {
|
|
const isContentFile = pathToFileURL(vfile.path).href.startsWith(contentDir.href);
|
|
if (!isContentFile) return;
|
|
|
|
const relImagePaths = new Set<string>();
|
|
visit(tree, 'image', function raiseError(node: Image) {
|
|
if (isRelativePath(node.url)) {
|
|
relImagePaths.add(node.url);
|
|
}
|
|
});
|
|
if (relImagePaths.size === 0) return;
|
|
|
|
const errorMessage =
|
|
`Relative image paths are not supported in the content/ directory. Place local images in the public/ directory and use absolute paths (see https://docs.astro.build/en/guides/images/#in-markdown-files):\n` +
|
|
[...relImagePaths].map((path) => JSON.stringify(path)).join(',\n');
|
|
|
|
throw new Error(errorMessage);
|
|
};
|
|
};
|
|
}
|
|
|
|
const DEFAULT_REMARK_PLUGINS: PluggableList = [remarkGfm, remarkSmartypants];
|
|
const DEFAULT_REHYPE_PLUGINS: PluggableList = [];
|
|
|
|
export async function getRemarkPlugins(
|
|
mdxOptions: MdxOptions,
|
|
config: AstroConfig
|
|
): Promise<MdxRollupPluginOptions['remarkPlugins']> {
|
|
let remarkPlugins: PluggableList = [
|
|
// Set "vfile.data.astro" for plugins to inject frontmatter
|
|
remarkInitializeAstroData,
|
|
];
|
|
switch (mdxOptions.extendPlugins) {
|
|
case false:
|
|
break;
|
|
case 'astroDefaults':
|
|
remarkPlugins = [...remarkPlugins, ...DEFAULT_REMARK_PLUGINS];
|
|
break;
|
|
default:
|
|
remarkPlugins = [
|
|
...remarkPlugins,
|
|
...(markdownShouldExtendDefaultPlugins(config) ? DEFAULT_REMARK_PLUGINS : []),
|
|
...ignoreStringPlugins(config.markdown.remarkPlugins ?? []),
|
|
];
|
|
break;
|
|
}
|
|
if (config.markdown.syntaxHighlight === 'shiki') {
|
|
remarkPlugins.push([await remarkShiki(config.markdown.shikiConfig)]);
|
|
}
|
|
if (config.markdown.syntaxHighlight === 'prism') {
|
|
remarkPlugins.push(remarkPrism);
|
|
}
|
|
|
|
remarkPlugins = [...remarkPlugins, ...(mdxOptions.remarkPlugins ?? [])];
|
|
|
|
// Apply last in case user plugins resolve relative image paths
|
|
if (config.experimental.contentCollections) {
|
|
remarkPlugins.push(toRemarkContentRelImageError(config));
|
|
}
|
|
return remarkPlugins;
|
|
}
|
|
|
|
export function getRehypePlugins(
|
|
mdxOptions: MdxOptions,
|
|
config: AstroConfig
|
|
): MdxRollupPluginOptions['rehypePlugins'] {
|
|
let rehypePlugins: PluggableList = [
|
|
// ensure `data.meta` is preserved in `properties.metastring` for rehype syntax highlighters
|
|
rehypeMetaString,
|
|
// rehypeRaw allows custom syntax highlighters to work without added config
|
|
[rehypeRaw, { passThrough: nodeTypes }] as any,
|
|
];
|
|
switch (mdxOptions.extendPlugins) {
|
|
case false:
|
|
break;
|
|
case 'astroDefaults':
|
|
rehypePlugins = [...rehypePlugins, ...DEFAULT_REHYPE_PLUGINS];
|
|
break;
|
|
default:
|
|
rehypePlugins = [
|
|
...rehypePlugins,
|
|
...(markdownShouldExtendDefaultPlugins(config) ? DEFAULT_REHYPE_PLUGINS : []),
|
|
...ignoreStringPlugins(config.markdown.rehypePlugins ?? []),
|
|
];
|
|
break;
|
|
}
|
|
|
|
rehypePlugins = [
|
|
...rehypePlugins,
|
|
...(mdxOptions.rehypePlugins ?? []),
|
|
// getHeadings() is guaranteed by TS, so this must be included.
|
|
// We run `rehypeHeadingIds` _last_ to respect any custom IDs set by user plugins.
|
|
rehypeHeadingIds,
|
|
rehypeInjectHeadingsExport,
|
|
];
|
|
return rehypePlugins;
|
|
}
|
|
|
|
function markdownShouldExtendDefaultPlugins(config: AstroConfig): boolean {
|
|
return (
|
|
config.markdown.extendDefaultPlugins ||
|
|
(config.markdown.remarkPlugins.length === 0 && config.markdown.rehypePlugins.length === 0)
|
|
);
|
|
}
|
|
|
|
function ignoreStringPlugins(plugins: any[]) {
|
|
let validPlugins: PluggableList = [];
|
|
let hasInvalidPlugin = false;
|
|
for (const plugin of plugins) {
|
|
if (typeof plugin === 'string') {
|
|
console.warn(yellow(`[MDX] ${bold(plugin)} not applied.`));
|
|
hasInvalidPlugin = true;
|
|
} else if (Array.isArray(plugin) && typeof plugin[0] === 'string') {
|
|
console.warn(yellow(`[MDX] ${bold(plugin[0])} not applied.`));
|
|
hasInvalidPlugin = true;
|
|
} else {
|
|
validPlugins.push(plugin);
|
|
}
|
|
}
|
|
if (hasInvalidPlugin) {
|
|
console.warn(
|
|
`To inherit Markdown plugins in MDX, please use explicit imports in your config instead of "strings." See Markdown docs: https://docs.astro.build/en/guides/markdown-content/#markdown-plugins`
|
|
);
|
|
}
|
|
return validPlugins;
|
|
}
|
|
|
|
/**
|
|
* Copied from markdown utils
|
|
* @see "vite-plugin-utils"
|
|
*/
|
|
function isValidAstroData(obj: unknown): obj is MarkdownAstroData {
|
|
if (typeof obj === 'object' && obj !== null && obj.hasOwnProperty('frontmatter')) {
|
|
const { frontmatter } = obj as any;
|
|
try {
|
|
// ensure frontmatter is JSON-serializable
|
|
JSON.stringify(frontmatter);
|
|
} catch {
|
|
return false;
|
|
}
|
|
return typeof frontmatter === 'object' && frontmatter !== null;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Copied from markdown utils
|
|
* @see "vite-plugin-utils"
|
|
*/
|
|
function safelyGetAstroData(vfileData: Data): MarkdownAstroData {
|
|
const { astro } = vfileData;
|
|
|
|
if (!astro) return { frontmatter: {} };
|
|
if (!isValidAstroData(astro)) {
|
|
throw Error(
|
|
`[MDX] A remark or rehype plugin tried to add invalid frontmatter. Ensure "astro.frontmatter" is a JSON object!`
|
|
);
|
|
}
|
|
|
|
return astro;
|
|
}
|
|
|
|
/**
|
|
* Check if estree entry is "import.meta.env.VARIABLE"
|
|
* If it is, return the variable name (i.e. "VARIABLE")
|
|
*/
|
|
function getImportMetaEnvVariableName(node: MemberExpression): string | Error {
|
|
try {
|
|
// check for ".[ANYTHING]"
|
|
if (node.object.type !== 'MemberExpression' || node.property.type !== 'Identifier')
|
|
return new Error();
|
|
|
|
const nestedExpression = node.object;
|
|
// check for ".env"
|
|
if (nestedExpression.property.type !== 'Identifier' || nestedExpression.property.name !== 'env')
|
|
return new Error();
|
|
|
|
const envExpression = nestedExpression.object;
|
|
// check for ".meta"
|
|
if (
|
|
envExpression.type !== 'MetaProperty' ||
|
|
envExpression.property.type !== 'Identifier' ||
|
|
envExpression.property.name !== 'meta'
|
|
)
|
|
return new Error();
|
|
|
|
// check for "import"
|
|
if (envExpression.meta.name !== 'import') return new Error();
|
|
|
|
return node.property.name;
|
|
} catch (e) {
|
|
if (e instanceof Error) {
|
|
return e;
|
|
}
|
|
return new Error('Unknown parsing error');
|
|
}
|
|
}
|