Allow HMX components in markdown (#19)

* Allow HMX components in markdown

This adds support for HMX components in markdown. The mechanism for importing is via frontmatter. We could do this differently (setup script maybe?) but since this was the easiest to implement I thought it was a good first-pass option.

* Remove node-fetch from snowpack config

* Assert that the runtime is created successfully

* Add back in the micromark extension for encoding entities

* Encode both codeTextData and codeFlowValue

* Install snowpack app's deps
This commit is contained in:
Matthew Phillips 2021-03-23 15:20:03 -04:00 committed by GitHub
parent e0353d50e7
commit ed85702581
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 171 additions and 19 deletions

View file

@ -19,6 +19,10 @@ jobs:
node-version: ${{ matrix.node-version }} node-version: ${{ matrix.node-version }}
- name: npm install, build, and test - name: npm install, build, and test
run: | run: |
cd examples/snowpack
npm ci
cd ../..
npm ci npm ci
npm run build npm run build
npm test npm test

View file

@ -10,11 +10,7 @@ module.exports = {
'@snowpack/plugin-svelte', '@snowpack/plugin-svelte',
'@snowpack/plugin-vue', '@snowpack/plugin-vue',
], ],
packageOptions: { packageOptions: {},
external: [
'node-fetch'
]
},
buildOptions: { buildOptions: {
out: '_site', out: '_site',
}, },

View file

@ -57,6 +57,8 @@ export default async function (astroConfig: AstroConfig) {
break; break;
} }
} }
res.statusCode = 500;
res.end(formatErrorForBrowser(result.error));
break; break;
} }
} }
@ -66,3 +68,8 @@ export default async function (astroConfig: AstroConfig) {
console.log(`Server running at http://${hostname}:${port}/`); console.log(`Server running at http://${hostname}:${port}/`);
}); });
} }
function formatErrorForBrowser(error: Error) {
// TODO make this pretty.
return error.toString();
}

35
src/micromark-encode.ts Normal file
View file

@ -0,0 +1,35 @@
import type { HtmlExtension, Token, Tokenize } from 'micromark/dist/shared-types';
const characterReferences = {
'"': 'quot',
'&': 'amp',
'<': 'lt',
'>': 'gt',
'{': 'lbrace',
'}': 'rbrace',
};
type EncodedChars = '"' | '&' | '<' | '>' | '{' | '}';
function encode(value: string): string {
return value.replace(/["&<>{}]/g, (raw: string) => {
return '&' + characterReferences[raw as EncodedChars] + ';';
});
}
function encodeToken(this: Record<string, () => void>) {
const token: Token = arguments[0];
const serialize = (this.sliceSerialize as unknown) as (t: Token) => string;
const raw = (this.raw as unknown) as (s: string) => void;
const value = serialize(token);
raw(encode(value));
}
const plugin: HtmlExtension = {
exit: {
codeTextData: encodeToken,
codeFlowValue: encodeToken,
},
};
export { plugin as encodeMarkdown };

View file

@ -8,6 +8,7 @@ import gfmHtml from 'micromark-extension-gfm/html.js';
import { CompileResult, TransformResult } from './@types/astro'; import { CompileResult, TransformResult } from './@types/astro';
import { parse } from './compiler/index.js'; import { parse } from './compiler/index.js';
import { createMarkdownHeadersCollector } from './micromark-collect-headers.js'; import { createMarkdownHeadersCollector } from './micromark-collect-headers.js';
import { encodeMarkdown } from './micromark-encode.js';
import { defaultLogOptions } from './logger.js'; import { defaultLogOptions } from './logger.js';
import { optimize } from './optimize/index.js'; import { optimize } from './optimize/index.js';
import { codegen } from './codegen/index.js'; import { codegen } from './codegen/index.js';
@ -54,8 +55,9 @@ async function convertMdToJsx(
const { data: _frontmatterData, content } = matter(contents); const { data: _frontmatterData, content } = matter(contents);
const { headers, headersExtension } = createMarkdownHeadersCollector(); const { headers, headersExtension } = createMarkdownHeadersCollector();
const mdHtml = micromark(content, { const mdHtml = micromark(content, {
allowDangerousHtml: true,
extensions: [gfmSyntax()], extensions: [gfmSyntax()],
htmlExtensions: [gfmHtml, headersExtension], htmlExtensions: [gfmHtml, encodeMarkdown, headersExtension],
}); });
const setupContext = { const setupContext = {
@ -68,19 +70,26 @@ async function convertMdToJsx(
}, },
}; };
let imports = '';
for(let [ComponentName, specifier] of Object.entries(_frontmatterData.import || {})) {
imports += `import ${ComponentName} from '${specifier}';\n`;
}
// </script> can't be anywhere inside of a JS string, otherwise the HTML parser fails. // </script> can't be anywhere inside of a JS string, otherwise the HTML parser fails.
// Break it up here so that the HTML parser won't detect it. // Break it up here so that the HTML parser won't detect it.
const stringifiedSetupContext = JSON.stringify(setupContext).replace(/\<\/script\>/g, `</scrip" + "t>`); const stringifiedSetupContext = JSON.stringify(setupContext).replace(/\<\/script\>/g, `</scrip" + "t>`);
return convertHmxToJsx( const raw = `<script astro>
`<script astro> ${imports}
${_frontmatterData.layout ? `export const layout = ${JSON.stringify(_frontmatterData.layout)};` : ''} ${_frontmatterData.layout ? `export const layout = ${JSON.stringify(_frontmatterData.layout)};` : ''}
export function setup({context}) { export function setup({context}) {
return {context: ${stringifiedSetupContext} }; return {context: ${stringifiedSetupContext} };
} }
</script><section>{${JSON.stringify(mdHtml)}}</section>`, </script><section>${mdHtml}</section>`;
{ compileOptions, filename, fileID }
); const convertOptions = { compileOptions, filename, fileID };
return convertHmxToJsx(raw, convertOptions);
} }
async function transformFromSource( async function transformFromSource(

View file

@ -0,0 +1,9 @@
export default {
projectRoot: '.',
hmxRoot: './astro',
dist: './_site',
extensions: {
'.jsx': 'preact'
}
}

View file

@ -0,0 +1,5 @@
import { h } from 'preact';
export default function() {
return <div id="test">Testing</div>
}

View file

@ -0,0 +1,3 @@
<div class="container">
<slot></slot>
</div>

View file

@ -0,0 +1,13 @@
<script astro>
export function setup() {
return {
props: {}
}
}
</script>
<astro:head>
<!-- Head Stuff -->
</astro:head>
<h1>Hello world!</h1>

View file

@ -0,0 +1,13 @@
---
layout: layouts/content.hmx
title: My Blog Post
description: This is a post about some stuff.
import:
Example: '../components/Example.jsx'
---
## Interesting Topic
<div id="first">Some content</div>
<Example />

View file

@ -0,0 +1,5 @@
export default {
mount: {
}
};

45
test/hmx-markdown.test.js Normal file
View file

@ -0,0 +1,45 @@
import { suite } from 'uvu';
import * as assert from 'uvu/assert';
import { createRuntime } from '../lib/runtime.js';
import { loadConfig } from '../lib/config.js';
import { doc } from './test-utils.js';
const HMXMD = suite('HMX Markdown');
let runtime, setupError;
HMXMD.before(async () => {
const astroConfig = await loadConfig(new URL('./fixtures/hmx-markdown', import.meta.url).pathname);
const logging = {
level: 'error',
dest: process.stderr
};
try {
runtime = await createRuntime(astroConfig, logging);
} catch(err) {
console.error(err);
setupError = err;
}
});
HMXMD.after(async () => {
runtime && runtime.shutdown();
});
HMXMD('No errors creating a runtime', () => {
assert.equal(setupError, undefined);
});
HMXMD('Can load markdown pages with hmx', async () => {
const result = await runtime.load('/post');
assert.equal(result.statusCode, 200);
const $ = doc(result.contents);
assert.ok($('#first').length, 'There is a div added in markdown');
assert.ok($('#test').length, 'There is a div added via a component from markdown');
});
HMXMD.run();

View file

@ -6,7 +6,7 @@ import { doc } from './test-utils.js';
const React = suite('React Components'); const React = suite('React Components');
let runtime; let runtime, setupError;
React.before(async () => { React.before(async () => {
const astroConfig = await loadConfig(new URL('./fixtures/react-component', import.meta.url).pathname); const astroConfig = await loadConfig(new URL('./fixtures/react-component', import.meta.url).pathname);
@ -20,7 +20,7 @@ React.before(async () => {
runtime = await createRuntime(astroConfig, logging); runtime = await createRuntime(astroConfig, logging);
} catch(err) { } catch(err) {
console.error(err); console.error(err);
throw err; setupError = err;
} }
}); });
@ -28,6 +28,10 @@ React.after(async () => {
await runtime.shutdown(); await runtime.shutdown();
}); });
React('No error creating the runtime', () => {
assert.equal(setupError, undefined);
});
React('Can load hmx page', async () => { React('Can load hmx page', async () => {
const result = await runtime.load('/'); const result = await runtime.load('/');

View file

@ -10,7 +10,7 @@ const { readdir, stat } = fsPromises;
const SnowpackDev = suite('snowpack.dev'); const SnowpackDev = suite('snowpack.dev');
let runtime, cwd; let runtime, cwd, setupError;
SnowpackDev.before(async () => { SnowpackDev.before(async () => {
// Bug: Snowpack config is still loaded relative to the current working directory. // Bug: Snowpack config is still loaded relative to the current working directory.
@ -28,7 +28,7 @@ SnowpackDev.before(async () => {
runtime = await createRuntime(astroConfig, logging); runtime = await createRuntime(astroConfig, logging);
} catch(err) { } catch(err) {
console.error(err); console.error(err);
throw err; setupError = err;
} }
}); });
@ -58,6 +58,10 @@ async function* allPages(root) {
} }
} }
SnowpackDev('No error creating the runtime', () => {
assert.equal(setupError, undefined);
});
SnowpackDev('Can load every page', async () => { SnowpackDev('Can load every page', async () => {
const failed = []; const failed = [];