fix: markdown issues

This commit is contained in:
Nate Moore 2021-05-21 13:15:38 -05:00
parent f329001af0
commit 405fe86f85
14 changed files with 327 additions and 204 deletions

View file

@ -8,7 +8,15 @@ export default function text(parser: Parser) {
let data = ''; let data = '';
while (parser.index < parser.template.length && !parser.match('---') && !parser.match('<') && !parser.match('{') && !parser.match('`')) { const shouldContinue = () => {
// Special case 'code' content to avoid tripping up on user code
if (parser.current().name === 'code') {
return !parser.match('<') && !parser.match('{');
}
return !parser.match('---') && !parser.match('<') && !parser.match('{') && !parser.match('`');
}
while (parser.index < parser.template.length && shouldContinue()) {
data += parser.template[parser.index++]; data += parser.template[parser.index++];
} }

View file

@ -8,12 +8,20 @@ export interface AstroConfigRaw {
export type ValidExtensionPlugins = 'astro' | 'react' | 'preact' | 'svelte' | 'vue'; export type ValidExtensionPlugins = 'astro' | 'react' | 'preact' | 'svelte' | 'vue';
export interface AstroMarkdownOptions {
/** Enable or disable footnotes syntax extension */
footnotes: boolean;
/** Enable or disable GitHub-flavored Markdown syntax extension */
gfm: boolean;
}
export interface AstroConfig { export interface AstroConfig {
dist: string; dist: string;
projectRoot: URL; projectRoot: URL;
astroRoot: URL; astroRoot: URL;
public: URL; public: URL;
extensions?: Record<string, ValidExtensionPlugins>; extensions?: Record<string, ValidExtensionPlugins>;
/** Options for rendering markdown content */
markdownOptions?: Partial<AstroMarkdownOptions>;
/** Options specific to `astro build` */ /** Options specific to `astro build` */
buildOptions: { buildOptions: {
/** Your public domain, e.g.: https://my-site.dev/. Used to generate sitemaps and canonical URLs. */ /** Your public domain, e.g.: https://my-site.dev/. Used to generate sitemaps and canonical URLs. */

View file

@ -1,12 +1,13 @@
import type { Ast, Script, Style, TemplateNode } from 'astro-parser'; import type { Ast, Script, Style, TemplateNode } from 'astro-parser';
import type { CompileOptions } from '../../@types/compiler'; import type { CompileOptions } from '../../@types/compiler';
import type { AstroConfig, TransformResult, ValidExtensionPlugins } from '../../@types/astro'; import type { AstroConfig, AstroMarkdownOptions, TransformResult, ValidExtensionPlugins } from '../../@types/astro';
import 'source-map-support/register.js'; import 'source-map-support/register.js';
import eslexer from 'es-module-lexer'; import eslexer from 'es-module-lexer';
import esbuild from 'esbuild'; import esbuild from 'esbuild';
import path from 'path'; import path from 'path';
import { walk } from 'estree-walker'; import { parse } from 'astro-parser';
import { walk, asyncWalk } from 'estree-walker';
import _babelGenerator from '@babel/generator'; import _babelGenerator from '@babel/generator';
import babelParser from '@babel/parser'; import babelParser from '@babel/parser';
import { codeFrameColumns } from '@babel/code-frame'; import { codeFrameColumns } from '@babel/code-frame';
@ -16,6 +17,7 @@ import { error, warn } from '../../logger.js';
import { fetchContent } from './content.js'; import { fetchContent } from './content.js';
import { isFetchContent } from './utils.js'; import { isFetchContent } from './utils.js';
import { yellow } from 'kleur/colors'; import { yellow } from 'kleur/colors';
import { MarkdownRenderingOptions, renderMarkdown } from '../utils';
const traverse: typeof babelTraverse.default = (babelTraverse.default as any).default; const traverse: typeof babelTraverse.default = (babelTraverse.default as any).default;
@ -306,7 +308,7 @@ interface CodegenState {
components: Components; components: Components;
css: string[]; css: string[];
markers: { markers: {
insideMarkdown: boolean | string; insideMarkdown: boolean | Record<string, any>;
}; };
importExportStatements: Set<string>; importExportStatements: Set<string>;
dynamicImports: DynamicImportMap; dynamicImports: DynamicImportMap;
@ -538,160 +540,213 @@ function compileCss(style: Style, state: CodegenState) {
}); });
} }
/** dedent markdown */
function dedent(str: string) {
let arr = str.match(/^[ \t]*(?=\S)/gm);
let first = !!arr && arr.find((x) => x.length > 0)?.length;
return !arr || !first ? str : str.replace(new RegExp(`^[ \\t]{0,${first}}`, 'gm'), '');
}
/** Compile page markup */ /** Compile page markup */
function compileHtml(enterNode: TemplateNode, state: CodegenState, compileOptions: CompileOptions) { async function compileHtml(enterNode: TemplateNode, state: CodegenState, compileOptions: CompileOptions): Promise<string> {
const { components, css, importExportStatements, dynamicImports, filename } = state; return new Promise((resolve) => {
const { astroConfig } = compileOptions; const { components, css, importExportStatements, dynamicImports, filename } = state;
const { astroConfig } = compileOptions;
let outSource = ''; let paren = -1;
walk(enterNode, { let buffers = {
enter(node: TemplateNode, parent: TemplateNode) { out: '',
switch (node.type) { markdown: '',
case 'Expression': { };
let children: string[] = []; let curr: keyof typeof buffers = 'out';
for (const child of node.children || []) {
children.push(compileHtml(child, state, compileOptions));
}
let raw = '';
let nextChildIndex = 0;
for (const chunk of node.codeChunks) {
raw += chunk;
if (nextChildIndex < children.length) {
raw += children[nextChildIndex++];
}
}
// TODO Do we need to compile this now, or should we compile the entire module at the end?
let code = compileExpressionSafe(raw).trim().replace(/\;$/, '');
outSource += `,(${code})`;
this.skip();
break;
}
case 'MustacheTag':
case 'Comment':
return;
case 'Fragment':
break;
case 'Slot':
case 'Head':
case 'InlineComponent':
case 'Title':
case 'Element': {
const name: string = node.name;
if (!name) {
throw new Error('AHHHH');
}
try {
const attributes = getAttributes(node.attributes);
outSource += outSource === '' ? '' : ','; /** renders markdown stored in `buffers.markdown` to JSX and pushes that to `buffers.out` */
if (node.type === 'Slot') { async function pushMarkdownToBuffer() {
outSource += `(children`; const md = buffers.markdown;
return; const { markdownOptions = {} } = astroConfig;
const { $scope: scopedClassName } = state.markers.insideMarkdown as Record<'$scope', any>;
let { content: rendered } = await renderMarkdown(dedent(md), { ...markdownOptions as AstroMarkdownOptions, mode: 'astro-md', $: { scopedClassName: scopedClassName.slice(1, -1) } });
const ast = parse(rendered);
const result = await compileHtml(ast.html, {...state, markers: {...state.markers, insideMarkdown: false }}, compileOptions);
buffers.out += ',' + result;
buffers.markdown = '';
curr = 'out';
}
asyncWalk(enterNode, {
async enter(node: TemplateNode, parent: TemplateNode) {
switch (node.type) {
case 'Expression': {
let children: string[] = [];
for (const child of node.children || []) {
children.push(await compileHtml(child, state, compileOptions));
} }
const COMPONENT_NAME_SCANNER = /^[A-Z]/; let raw = '';
if (!COMPONENT_NAME_SCANNER.test(name)) { let nextChildIndex = 0;
outSource += `h("${name}", ${attributes ? generateAttributes(attributes) : 'null'}`; for (const chunk of node.codeChunks) {
if (state.markers.insideMarkdown) { raw += chunk;
outSource += `,h(__astroMarkdownRender, null`; if (nextChildIndex < children.length) {
raw += children[nextChildIndex++];
} }
return;
} }
const [componentName, componentKind] = name.split(':'); // TODO Do we need to compile this now, or should we compile the entire module at the end?
const componentImportData = components[componentName]; let code = compileExpressionSafe(raw).trim().replace(/\;$/, '');
if (!componentImportData) { if (state.markers.insideMarkdown) {
throw new Error(`Unknown Component: ${componentName}`); buffers[curr] += `{${code}}`;
} else {
buffers[curr] += `,(${code})`;
} }
if (componentImportData.type === '.astro') { this.skip();
if (componentName === 'Markdown') { break;
const attributeStr = attributes ? generateAttributes(attributes) : 'null'; }
state.markers.insideMarkdown = attributeStr; case 'MustacheTag':
outSource += `h(__astroMarkdownRender, ${attributeStr}`; case 'Comment':
return;
case 'Fragment':
break;
case 'Slot':
case 'Head':
case 'InlineComponent':
case 'Title':
case 'Element': {
const name: string = node.name;
if (!name) {
throw new Error('AHHHH');
}
try {
const attributes = getAttributes(node.attributes);
buffers.out += buffers.out === '' ? '' : ',';
if (node.type === 'Slot') {
buffers[curr] += `(children`;
paren++;
return; return;
} }
} const COMPONENT_NAME_SCANNER = /^[A-Z]/;
const { wrapper, wrapperImport } = getComponentWrapper(name, components[componentName], { astroConfig, dynamicImports, filename }); if (!COMPONENT_NAME_SCANNER.test(name)) {
if (wrapperImport) { if (curr === 'markdown') {
importExportStatements.add(wrapperImport); await pushMarkdownToBuffer();
} }
buffers[curr] += `h("${name}", ${attributes ? generateAttributes(attributes) : 'null'}`;
paren++;
return;
}
const [componentName, componentKind] = name.split(':');
const componentImportData = components[componentName];
if (!componentImportData) {
throw new Error(`Unknown Component: ${componentName}`);
}
if (componentImportData.type === '.astro') {
if (componentName === 'Markdown') {
const { $scope } = attributes ?? {};
state.markers.insideMarkdown = { $scope };
curr = 'markdown';
return;
}
}
const { wrapper, wrapperImport } = getComponentWrapper(name, components[componentName], { astroConfig, dynamicImports, filename });
if (wrapperImport) {
importExportStatements.add(wrapperImport);
}
if (curr === 'markdown') {
await pushMarkdownToBuffer();
}
outSource += `h(${wrapper}, ${attributes ? generateAttributes(attributes) : 'null'}`; paren++;
if (state.markers.insideMarkdown) { buffers[curr] += `h(${wrapper}, ${attributes ? generateAttributes(attributes) : 'null'}`;
const attributeStr = state.markers.insideMarkdown; } catch (err) {
outSource += `,h(__astroMarkdownRender, ${attributeStr}`; // handle errors in scope with filename
const rel = filename.replace(astroConfig.projectRoot.pathname, '');
// TODO: return actual codeframe here
error(compileOptions.logging, rel, err.toString());
} }
} catch (err) {
// handle errors in scope with filename
const rel = filename.replace(astroConfig.projectRoot.pathname, '');
// TODO: return actual codeframe here
error(compileOptions.logging, rel, err.toString());
}
return;
}
case 'Attribute': {
this.skip();
return;
}
case 'Style': {
css.push(node.content.styles); // if multiple <style> tags, combine together
this.skip();
return;
}
case 'CodeSpan':
case 'CodeFence': {
outSource += ',' + JSON.stringify(node.raw);
return;
}
case 'Text': {
const text = getTextFromAttribute(node);
// Whitespace is significant if we are immediately inside of <Markdown>,
// but not if we're inside of another component in <Markdown>
if (parent.name !== 'Markdown' && !text.trim()) {
return; return;
} }
outSource += ',' + JSON.stringify(text); case 'Attribute': {
return; this.skip();
} return;
default:
throw new Error('Unexpected (enter) node type: ' + node.type);
}
},
leave(node, parent, prop, index) {
switch (node.type) {
case 'Text':
case 'CodeSpan':
case 'CodeFence':
case 'Attribute':
case 'Comment':
case 'Fragment':
case 'Expression':
case 'MustacheTag':
return;
case 'Slot':
case 'Head':
case 'Body':
case 'Title':
case 'Element':
case 'InlineComponent': {
if (node.type === 'InlineComponent' && node.name === 'Markdown') {
state.markers.insideMarkdown = false;
} }
if (state.markers.insideMarkdown) { case 'Style': {
outSource += ')'; css.push(node.content.styles); // if multiple <style> tags, combine together
this.skip();
return;
} }
outSource += ')'; case 'CodeSpan':
return; case 'CodeFence': {
if (state.markers.insideMarkdown) {
if (curr === 'out') curr = 'markdown';
buffers[curr] += node.raw;
return;
}
buffers[curr] += ',' + JSON.stringify(node.data);
return;
}
case 'Text': {
let text = getTextFromAttribute(node);
if (state.markers.insideMarkdown) {
if (curr === 'out') curr = 'markdown';
buffers[curr] += text;
return;
}
if (parent.name !== 'Markdown' && !text.trim()) {
return;
}
if (parent.name === 'code') {
// Special case, escaped { characters from markdown content
text = node.raw.replace(/&#x26;#123;/g, '{');
}
buffers[curr] += ',' + JSON.stringify(text);
return;
}
default:
throw new Error('Unexpected (enter) node type: ' + node.type);
} }
case 'Style': { },
this.remove(); // this will be optimized in a global CSS file; remove so its not accidentally inlined async leave(node, parent, prop, index) {
return; switch (node.type) {
case 'Text':
case 'Attribute':
case 'Comment':
case 'Fragment':
case 'Expression':
case 'MustacheTag':
return;
case 'CodeSpan':
case 'CodeFence':
return;
case 'Slot':
case 'Head':
case 'Body':
case 'Title':
case 'Element':
case 'InlineComponent': {
if (node.type === 'InlineComponent' && curr === 'markdown' && buffers.markdown !== '') {
await pushMarkdownToBuffer();
}
if (paren !== -1) {
buffers.out += ')';
paren--;
}
return;
}
case 'Style': {
this.remove(); // this will be optimized in a global CSS file; remove so its not accidentally inlined
return;
}
default:
throw new Error('Unexpected (leave) node type: ' + node.type);
} }
default: },
throw new Error('Unexpected (leave) node type: ' + node.type); }).then(() => {
} const content = buffers.out.replace(/^\,/, '').replace(/\,\)/g, ')').replace(/\,+/g, ',').replace(/\)h/g, '),h');
}, buffers.out = '';
buffers.markdown = '';
return resolve(content);
});
}); });
return outSource;
} }
/** /**
@ -721,7 +776,7 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
compileCss(ast.css, state); compileCss(ast.css, state);
const html = compileHtml(ast.html, state, compileOptions); const html = await compileHtml(ast.html, state, compileOptions);
return { return {
script: script, script: script,

View file

@ -29,7 +29,7 @@ interface ConvertAstroOptions {
* 2. Transform * 2. Transform
* 3. Codegen * 3. Codegen
*/ */
async function convertAstroToJsx(template: string, opts: ConvertAstroOptions): Promise<TransformResult> { export async function convertAstroToJsx(template: string, opts: ConvertAstroOptions): Promise<TransformResult> {
const { filename } = opts; const { filename } = opts;
// 1. Parse // 1. Parse
@ -48,11 +48,12 @@ async function convertAstroToJsx(template: string, opts: ConvertAstroOptions): P
* .md -> .astro source * .md -> .astro source
*/ */
export async function convertMdToAstroSource(contents: string, { filename }: { filename: string }): Promise<string> { export async function convertMdToAstroSource(contents: string, { filename }: { filename: string }): Promise<string> {
const { let {
content, content,
frontmatter: { layout, ...frontmatter }, frontmatter: { layout, ...frontmatter },
...data ...data
} = await renderMarkdownWithFrontmatter(contents); } = await renderMarkdownWithFrontmatter(contents);
if (frontmatter['astro'] !== undefined) { if (frontmatter['astro'] !== undefined) {
throw new Error(`"astro" is a reserved word but was used as a frontmatter value!\n\tat ${filename}`); throw new Error(`"astro" is a reserved word but was used as a frontmatter value!\n\tat ${filename}`);
} }
@ -109,7 +110,6 @@ export async function compileComponent(
): Promise<CompileResult> { ): Promise<CompileResult> {
const result = await transformFromSource(source, { compileOptions, filename, projectRoot }); const result = await transformFromSource(source, { compileOptions, filename, projectRoot });
const site = compileOptions.astroConfig.buildOptions.site || `http://localhost:${compileOptions.astroConfig.devOptions.port}`; const site = compileOptions.astroConfig.buildOptions.site || `http://localhost:${compileOptions.astroConfig.devOptions.port}`;
const usesMarkdown = !!result.imports.find((spec) => spec.indexOf('Markdown') > -1);
// return template // return template
let modJsx = ` let modJsx = `
@ -120,7 +120,6 @@ ${result.imports.join('\n')}
// \`__render()\`: Render the contents of the Astro module. // \`__render()\`: Render the contents of the Astro module.
import { h, Fragment } from '${internalImport('h.js')}'; import { h, Fragment } from '${internalImport('h.js')}';
${usesMarkdown ? `import __astroMarkdownRender from '${internalImport('markdown.js')}';` : ''};
const __astroRequestSymbol = Symbol('astro.request'); const __astroRequestSymbol = Symbol('astro.request');
async function __render(props, ...children) { async function __render(props, ...children) {
const Astro = { const Astro = {

View file

@ -0,0 +1,41 @@
import { visit } from 'unist-util-visit';
/** */
export function remarkCodeBlock() {
const visitor = (node: any) => {
const { data, lang, meta } = node;
let currentClassName = data?.hProperties?.class ?? '';
node.data = node.data || {};
node.data.hProperties = node.data.hProperties || {};
node.data.hProperties = { ...node.data.hProperties, class: `language-${lang} ${currentClassName}`.trim(), lang, meta }
return node;
};
return () => (tree: any) => visit(tree, 'code', visitor);
}
/** */
export function rehypeCodeBlock() {
const escapeCode = (code: any) => {
code.children = code.children.map((child: any) => {
if (child.type === 'text') {
return { ...child, value: child.value.replace(/\{/g, '&#123;') };
}
return child;
})
}
const visitor = (node: any) => {
if (node.tagName === 'code') {
escapeCode(node);
return;
}
if (node.tagName !== 'pre') return;
const code = node.children[0];
if (code.tagName !== 'code') return;
node.properties = { ...code.properties };
return node;
};
return () => (tree: any) => visit(tree, 'element', visitor);
}

View file

@ -2,6 +2,9 @@ declare module '@silvenon/remark-smartypants' {
export default function (): any; export default function (): any;
} }
declare module 'mdast-util-mdx';
declare module 'micromark-extension-mdxjs';
declare module 'mdast-util-mdx/from-markdown.js' { declare module 'mdast-util-mdx/from-markdown.js' {
export default function (): any; export default function (): any;
} }

View file

@ -1,26 +1,31 @@
import fromMarkdown from 'mdast-util-mdx/from-markdown.js';
import toMarkdown from 'mdast-util-mdx/to-markdown.js';
/** See https://github.com/micromark/micromark-extension-mdx-md */ import syntaxMdxjs from 'micromark-extension-mdxjs'
const syntax = { disable: { null: ['autolink', 'codeIndented'] } }; import {fromMarkdown, toMarkdown} from 'mdast-util-mdx'
/** /**
* Lite version of https://github.com/mdx-js/mdx/tree/main/packages/remark-mdx * Add the micromark and mdast extensions for MDX.js (JS aware MDX).
* We don't need all the features MDX does because all components are precompiled *
* to HTML. We just want to disable a few MD features that cause issues. * @this {Processor}
* @param {MdxOptions} [options]
* @return {void}
*/ */
function mdxLite(this: any) { export function remarkMdx(this: any, options: any) {
let data = this.data(); let data = this.data()
add('micromarkExtensions', syntax); add('micromarkExtensions', syntaxMdxjs(options))
add('fromMarkdownExtensions', fromMarkdown); add('fromMarkdownExtensions', fromMarkdown)
add('toMarkdownExtensions', toMarkdown); add('toMarkdownExtensions', toMarkdown)
/** Adds remark plugin */ /**
* @param {string} field
* @param {unknown} value
*/
function add(field: string, value: any) { function add(field: string, value: any) {
if (data[field]) data[field].push(value); // Other extensions defined before this.
else data[field] = [value]; // Useful when externalizing.
/* c8 ignore next 2 */
// @ts-ignore Assume its an array.
if (data[field]) data[field].push(value)
else data[field] = [value]
} }
} }
export default mdxLite;

View file

@ -7,10 +7,10 @@ export default function scopedStyles(className: string) {
if (noVisit.has(node.type)) return; if (noVisit.has(node.type)) return;
const { data } = node; const { data } = node;
const currentClassName = data?.hProperties?.class ?? ''; let currentClassName = data?.hProperties?.class ?? '';
node.data = node.data || {}; node.data = node.data || {};
node.data.hProperties = node.data.hProperties || {}; node.data.hProperties = node.data.hProperties || {};
node.data.hProperties.className = `${className} ${currentClassName}`.trim(); node.data.hProperties.class = `${className} ${currentClassName}`.trim();
return node; return node;
}; };

View file

@ -1,5 +1,5 @@
import type { Transformer } from '../../@types/transformer'; import type { Transformer } from '../../@types/transformer';
import type { Script } from 'astro-parser'; import type { Script, TemplateNode } from 'astro-parser';
import { getAttrValue } from '../../ast.js'; import { getAttrValue } from '../../ast.js';
const PRISM_IMPORT = `import Prism from 'astro/components/Prism.astro';\n`; const PRISM_IMPORT = `import Prism from 'astro/components/Prism.astro';\n`;
@ -8,7 +8,17 @@ const prismImportExp = /import Prism from ['"]astro\/components\/Prism.astro['"]
function escape(code: string) { function escape(code: string) {
return code.replace(/[`$]/g, (match) => { return code.replace(/[`$]/g, (match) => {
return '\\' + match; return '\\' + match;
}); }).replace(/&#123;/g, '{');
}
/** Unescape { characters transformed by Markdown generation */
function unescapeCode(code: TemplateNode) {
code.children = code.children?.map(child => {
if (child.type === 'Text') {
return { ...child, raw: child.raw.replace(/&#x26;#123;/g, '{') }
}
return child;
})
} }
/** default export - Transform prism */ /** default export - Transform prism */
export default function (module: Script): Transformer { export default function (module: Script): Transformer {
@ -19,6 +29,11 @@ export default function (module: Script): Transformer {
html: { html: {
Element: { Element: {
enter(node) { enter(node) {
if (node.name === 'code') {
unescapeCode(node);
return;
}
if (node.name !== 'pre') return; if (node.name !== 'pre') return;
const codeEl = node.children && node.children[0]; const codeEl = node.children && node.children[0];
if (!codeEl || codeEl.name !== 'code') return; if (!codeEl || codeEl.name !== 'code') return;

View file

@ -1,20 +1,20 @@
import mdxLite from './markdown/remark-mdx-lite.js'; import type { AstroMarkdownOptions } from '../@types/astro';
import createCollectHeaders from './markdown/rehype-collect-headers.js'; import createCollectHeaders from './markdown/rehype-collect-headers.js';
import scopedStyles from './markdown/remark-scoped-styles.js'; import scopedStyles from './markdown/remark-scoped-styles.js';
import { remarkCodeBlock, rehypeCodeBlock } from './markdown/codeblock.js';
import raw from 'rehype-raw'; import raw from 'rehype-raw';
import unified from 'unified'; import unified from 'unified';
import markdown from 'remark-parse'; import markdown from 'remark-parse';
import markdownToHtml from 'remark-rehype'; import markdownToHtml from 'remark-rehype';
import smartypants from '@silvenon/remark-smartypants'; // import smartypants from '@silvenon/remark-smartypants';
import stringify from 'rehype-stringify'; import rehypeStringify from 'rehype-stringify';
export interface MarkdownRenderingOptions { export interface MarkdownRenderingOptions extends Partial<AstroMarkdownOptions> {
$?: { $?: {
scopedClassName: string | null; scopedClassName: string | null;
}; };
footnotes?: boolean; mode: 'md'|'astro-md';
gfm?: boolean;
plugins?: any[];
} }
/** Internal utility for rendering a full markdown file and extracting Frontmatter data */ /** Internal utility for rendering a full markdown file and extracting Frontmatter data */
@ -22,16 +22,16 @@ export async function renderMarkdownWithFrontmatter(contents: string, opts?: Mar
// Dynamic import to ensure that "gray-matter" isn't built by Snowpack // Dynamic import to ensure that "gray-matter" isn't built by Snowpack
const { default: matter } = await import('gray-matter'); const { default: matter } = await import('gray-matter');
const { data: frontmatter, content } = matter(contents); const { data: frontmatter, content } = matter(contents);
const value = await renderMarkdown(content, opts); const value = await renderMarkdown(content, { ...opts, mode: 'md' });
return { ...value, frontmatter }; return { ...value, frontmatter };
} }
/** Shared utility for rendering markdown */ /** Shared utility for rendering markdown */
export async function renderMarkdown(content: string, opts?: MarkdownRenderingOptions | null) { export async function renderMarkdown(content: string, opts?: MarkdownRenderingOptions | null) {
const { $: { scopedClassName = null } = {}, footnotes: useFootnotes = true, gfm: useGfm = true, plugins = [] } = opts ?? {}; const { $: { scopedClassName = null } = {}, mode = 'astro-md', footnotes: useFootnotes = true, gfm: useGfm = true } = opts ?? {};
const { headers, rehypeCollectHeaders } = createCollectHeaders(); const { headers, rehypeCollectHeaders } = createCollectHeaders();
let parser = unified().use(markdown).use(mdxLite).use(smartypants); let parser = unified().use(markdown).use(remarkCodeBlock());
if (scopedClassName) { if (scopedClassName) {
parser = parser.use(scopedStyles(scopedClassName)); parser = parser.use(scopedStyles(scopedClassName));
@ -53,7 +53,8 @@ export async function renderMarkdown(content: string, opts?: MarkdownRenderingOp
.use(markdownToHtml, { allowDangerousHtml: true, passThrough: ['raw'] }) .use(markdownToHtml, { allowDangerousHtml: true, passThrough: ['raw'] })
.use(raw) .use(raw)
.use(rehypeCollectHeaders) .use(rehypeCollectHeaders)
.use(stringify) .use(rehypeCodeBlock())
.use(rehypeStringify)
.process(content); .process(content);
result = vfile.contents.toString(); result = vfile.contents.toString();
} catch (err) { } catch (err) {

View file

@ -54,6 +54,7 @@ function configDefaults(userConfig?: any): any {
if (!config.devOptions) config.devOptions = {}; if (!config.devOptions) config.devOptions = {};
if (!config.devOptions.port) config.devOptions.port = 3000; if (!config.devOptions.port) config.devOptions.port = 3000;
if (!config.buildOptions) config.buildOptions = {}; if (!config.buildOptions) config.buildOptions = {};
if (!config.markdownOptions) config.markdownOptions = {};
if (typeof config.buildOptions.sitemap === 'undefined') config.buildOptions.sitemap = true; if (typeof config.buildOptions.sitemap === 'undefined') config.buildOptions.sitemap = true;
return config; return config;

View file

@ -1,26 +0,0 @@
import { renderMarkdown } from '../compiler/utils.js';
/**
* Functional component which uses Astro's built-in Markdown rendering
* to render out its children.
*
* Note: the children have already been properly escaped/rendered
* by the parser and Astro, so at this point we're just rendering
* out plain markdown, no need for JSX support
*/
export default async function Markdown(props: { $scope: string | null }, ...children: string[]): Promise<string> {
const { $scope = null } = props ?? {};
const text = dedent(children.join('').trimEnd());
let { content } = await renderMarkdown(text, { $: { scopedClassName: $scope } });
if (content.split('<p>').length === 2) {
content = content.replace(/^\<p\>/i, '').replace(/\<\/p\>$/i, '');
}
return content;
}
/** Remove leading indentation based on first line */
function dedent(str: string) {
let arr = str.match(/^[ \t]*(?=\S)/gm);
let first = !!arr && arr.find((x) => x.length > 0)?.length;
return !arr || !first ? str : str.replace(new RegExp(`^[ \\t]{0,${first}}`, 'gm'), '');
}

View file

@ -314,7 +314,12 @@ async function createSnowpack(astroConfig: AstroConfig, options: CreateSnowpackO
}, },
packageOptions: { packageOptions: {
knownEntrypoints: ['preact-render-to-string'], knownEntrypoints: ['preact-render-to-string'],
external: ['@vue/server-renderer', 'node-fetch', 'prismjs/components/index.js'], external: [
'@vue/server-renderer',
'node-fetch',
'prismjs/components/index.js',
'gray-matter',
],
}, },
}); });

View file

@ -2218,6 +2218,14 @@ astral-regex@^2.0.0:
"astro-languageserver@file:tools/astro-languageserver": "astro-languageserver@file:tools/astro-languageserver":
version "0.4.0" version "0.4.0"
dependencies:
source-map "^0.7.3"
typescript "^4.3.1-rc"
vscode-css-languageservice "^5.1.1"
vscode-emmet-helper "2.1.2"
vscode-html-languageservice "^3.0.3"
vscode-languageserver "6.1.1"
vscode-languageserver-textdocument "^1.0.1"
"astro-scripts@file:scripts": "astro-scripts@file:scripts":
version "0.0.1" version "0.0.1"