Compiler cleanup (#64)

* Compiler cleanup

This is general compiler cleanup, especially around the codegen part. Goals here were too:

1. Make it possible to compile HTML recursively (needed for future astro-in-expressions work) by moving that work into its own function.
1. Get rid of collectionItems and have compiling the HTML return just a source string.

Also not planned, this change gets rid of the different between components and pages. All Astro components compile to the same JavaScript.

* Remove unused node types
This commit is contained in:
Matthew Phillips 2021-04-06 16:14:42 -04:00 committed by GitHub
parent 7240f0d677
commit 72d9ece6db
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 129 additions and 83 deletions

View file

@ -24,7 +24,7 @@ export interface JsxItem {
export interface TransformResult {
script: string;
imports: string[];
items: JsxItem[];
html: string;
css?: string;
}

View file

@ -1,6 +1,6 @@
import type { CompileOptions } from '../@types/compiler';
import type { AstroConfig, ValidExtensionPlugins } from '../@types/astro';
import type { Ast, TemplateNode } from '../parser/interfaces';
import type { Ast, Script, Style, TemplateNode } from '../parser/interfaces';
import type { JsxItem, TransformResult } from '../@types/astro';
import eslexer from 'es-module-lexer';
@ -262,17 +262,18 @@ async function acquireDynamicComponentImports(plugins: Set<ValidExtensionPlugins
return importMap;
}
/**
* Codegen
* Step 3/3 in Astro SSR.
* This is the final pass over a document AST before its converted to an h() function
* and handed off to Snowpack to build.
* @param {Ast} AST The parsed AST to crawl
* @param {object} CodeGenOptions
*/
export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOptions): Promise<TransformResult> {
const { extensions = defaultExtensions, astroConfig } = compileOptions;
await eslexer.init;
type Components = Record<string, { type: string; url: string; plugin: string | undefined }>;
interface CodegenState {
filename: string;
components: Components;
css: string[];
importExportStatements: Set<string>;
dynamicImports: DynamicImportMap;
}
function compileModule(module: Script, state: CodegenState, compileOptions: CompileOptions) {
const { extensions = defaultExtensions } = compileOptions;
const componentImports: ImportDeclaration[] = [];
const componentProps: VariableDeclarator[] = [];
@ -280,12 +281,10 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
let script = '';
let propsStatement = '';
const importExportStatements: Set<string> = new Set();
const components: Record<string, { type: string; url: string; plugin: string | undefined }> = {};
const componentPlugins = new Set<ValidExtensionPlugins>();
if (ast.module) {
const program = babelParser.parse(ast.module.content, {
if (module) {
const program = babelParser.parse(module.content, {
sourceType: 'module',
plugins: ['jsx', 'typescript', 'topLevelAwait'],
}).program;
@ -317,7 +316,7 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
const componentType = path.posix.extname(importUrl);
const componentName = path.posix.basename(importUrl, componentType);
const plugin = extensions[componentType] || defaultExtensions[componentType];
components[componentName] = {
state.components[componentName] = {
type: componentType,
plugin,
url: importUrl,
@ -325,10 +324,10 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
if (plugin) {
componentPlugins.add(plugin);
}
importExportStatements.add(ast.module.content.slice(componentImport.start!, componentImport.end!));
state.importExportStatements.add(module.content.slice(componentImport.start!, componentImport.end!));
}
for (const componentImport of componentExports) {
importExportStatements.add(ast.module.content.slice(componentImport.start!, componentImport.end!));
state.importExportStatements.add(module.content.slice(componentImport.start!, componentImport.end!));
}
if (componentProps.length > 0) {
@ -345,18 +344,14 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
script = propsStatement + babelGenerator(program).code;
}
const dynamicImports = await acquireDynamicComponentImports(componentPlugins, compileOptions.resolve);
return { script, componentPlugins };
}
let items: JsxItem[] = [];
let collectionItem: JsxItem | undefined;
let currentItemName: string | undefined;
let currentDepth = 0;
let css: string[] = [];
walk(ast.css, {
function compileCss(style: Style, state: CodegenState) {
walk(style, {
enter(node: TemplateNode) {
if (node.type === 'Style') {
css.push(node.content.styles); // if multiple <style> tags, combine together
state.css.push(node.content.styles); // if multiple <style> tags, combine together
this.skip();
}
},
@ -366,8 +361,14 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
}
},
});
}
walk(ast.html, {
function compileHtml(enterNode: TemplateNode, state: CodegenState, compileOptions: CompileOptions) {
const { components, css, importExportStatements, dynamicImports, filename } = state;
const { astroConfig } = compileOptions;
let outSource = '';
walk(enterNode, {
enter(node: TemplateNode) {
switch (node.type) {
case 'MustacheTag':
@ -394,19 +395,13 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
code = code.slice(0, astroComponent.index + 2) + wrapper + code.slice(astroComponent.index + astroComponent[0].length - 1);
}
}
collectionItem!.jsx += `,(${code.trim().replace(/\;$/, '')})`;
outSource += `,(${code.trim().replace(/\;$/, '')})`;
this.skip();
return;
case 'Comment':
return;
case 'Fragment':
// Ignore if its the top level fragment
// This should be cleaned up, but right now this is how the old thing worked
if (!collectionItem) {
return;
}
break;
case 'Slot':
case 'Head':
case 'InlineComponent':
@ -417,20 +412,15 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
throw new Error('AHHHH');
}
const attributes = getAttributes(node.attributes);
currentDepth++;
currentItemName = name;
if (!collectionItem) {
collectionItem = { name, jsx: '' };
items.push(collectionItem);
}
collectionItem.jsx += collectionItem.jsx === '' ? '' : ',';
outSource += outSource === '' ? '' : ',';
if (node.type === 'Slot') {
collectionItem.jsx += `(children`;
outSource += `(children`;
return;
}
const COMPONENT_NAME_SCANNER = /^[A-Z]/;
if (!COMPONENT_NAME_SCANNER.test(name)) {
collectionItem.jsx += `h("${name}", ${attributes ? generateAttributes(attributes) : 'null'}`;
outSource += `h("${name}", ${attributes ? generateAttributes(attributes) : 'null'}`;
return;
}
const [componentName, componentKind] = name.split(':');
@ -443,7 +433,7 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
importExportStatements.add(wrapperImport);
}
collectionItem.jsx += `h(${wrapper}, ${attributes ? generateAttributes(attributes) : 'null'}`;
outSource += `h(${wrapper}, ${attributes ? generateAttributes(attributes) : 'null'}`;
return;
}
case 'Attribute': {
@ -460,14 +450,7 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
if (!text.trim()) {
return;
}
if (!collectionItem) {
throw new Error('Not possible! TEXT:' + text);
}
if (currentItemName === 'script' || currentItemName === 'code') {
collectionItem.jsx += ',' + JSON.stringify(text);
return;
}
collectionItem.jsx += ',' + JSON.stringify(text);
outSource += ',' + JSON.stringify(text);
return;
}
default:
@ -482,23 +465,14 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
case 'Comment':
return;
case 'Fragment':
if (!collectionItem) {
return;
}
return;
case 'Slot':
case 'Head':
case 'Body':
case 'Title':
case 'Element':
case 'InlineComponent':
if (!collectionItem) {
throw new Error('Not possible! CLOSE ' + node.name);
}
collectionItem.jsx += ')';
currentDepth--;
if (currentDepth === 0) {
collectionItem = undefined;
}
outSource += ')';
return;
case 'Style': {
this.remove(); // this will be optimized in a global CSS file; remove so its not accidentally inlined
@ -510,10 +484,39 @@ export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOpt
},
});
return outSource;
}
/**
* Codegen
* Step 3/3 in Astro SSR.
* This is the final pass over a document AST before its converted to an h() function
* and handed off to Snowpack to build.
* @param {Ast} AST The parsed AST to crawl
* @param {object} CodeGenOptions
*/
export async function codegen(ast: Ast, { compileOptions, filename }: CodeGenOptions): Promise<TransformResult> {
await eslexer.init;
const state: CodegenState = {
filename,
components: {},
css: [],
importExportStatements: new Set(),
dynamicImports: new Map()
};
const { script, componentPlugins } = compileModule(ast.module, state, compileOptions);
state.dynamicImports = await acquireDynamicComponentImports(componentPlugins, compileOptions.resolve);
compileCss(ast.css, state);
const html = compileHtml(ast.html, state, compileOptions);
return {
script: script,
imports: Array.from(importExportStatements),
items,
css: css.length ? css.join('\n\n') : undefined,
imports: Array.from(state.importExportStatements),
html,
css: state.css.length ? state.css.join('\n\n') : undefined,
};
}

View file

@ -115,27 +115,23 @@ export async function compileComponent(
source: string,
{ compileOptions, filename, projectRoot }: { compileOptions: CompileOptions; filename: string; projectRoot: string }
): Promise<CompileResult> {
const sourceJsx = await transformFromSource(source, { compileOptions, filename, projectRoot });
const isPage = path.extname(filename) === '.md' || sourceJsx.items.some((item) => item.name === 'html');
const result = await transformFromSource(source, { compileOptions, filename, projectRoot });
// return template
let modJsx = `
import fetch from 'node-fetch';
// <script astro></script>
${sourceJsx.imports.join('\n')}
${result.imports.join('\n')}
// \`__render()\`: Render the contents of the Astro module.
import { h, Fragment } from '${internalImport('h.js')}';
async function __render(props, ...children) {
${sourceJsx.script}
return h(Fragment, null, ${sourceJsx.items.map(({ jsx }) => jsx).join(',')});
${result.script}
return h(Fragment, null, ${result.html});
}
export default __render;
`;
if (isPage) {
modJsx += `
// \`__renderPage()\`: Render the contents of the Astro module as a page. This is a special flow,
// triggered by loading a component directly by URL.
export async function __renderPage({request, children, props}) {
@ -159,14 +155,10 @@ export async function __renderPage({request, children, props}) {
return childBodyResult;
};\n`;
} else {
modJsx += `
export async function __renderPage() { throw new Error("No <html> page element found!"); }\n`;
}
return {
result: sourceJsx,
result,
contents: modJsx,
css: sourceJsx.css,
css: result.css,
};
}

18
test/astro-expr.test.js Normal file
View file

@ -0,0 +1,18 @@
import { suite } from 'uvu';
import * as assert from 'uvu/assert';
import { doc } from './test-utils.js';
import { setup } from './helpers.js';
const Expressions = suite('Expressions');
setup(Expressions, './fixtures/astro-expr');
Expressions('Can load page', async ({ runtime }) => {
const result = await runtime.load('/');
console.log(result)
assert.equal(result.statusCode, 200);
console.log(result.contents);
});
Expressions.run();

View file

@ -0,0 +1,6 @@
export default {
extensions: {
'.jsx': 'preact'
}
}

View file

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

View file

@ -0,0 +1,22 @@
---
import Color from '../components/Color.jsx';
let title = 'My Site';
const colors = ['red', 'yellow', 'blue']
---
<html lang="en">
<head>
<title>My site</title>
</head>
<body>
<h1>{title}</h1>
{colors.map(color => (
<div>
<Color name={color} />
</div>
))}
</body>
</html>