Remove post-rendering head injection (#3679)

* Remove post-rendering head injection

* Adds a changeset

* Use a layout component for vue
This commit is contained in:
Matthew Phillips 2022-06-23 15:37:55 -04:00 committed by GitHub
parent 446f8c4f13
commit fa7ed3f3a9
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
22 changed files with 74 additions and 44 deletions

View file

@ -0,0 +1,11 @@
---
'astro': patch
---
Moves head injection to happen during rendering
This change makes it so that head injection; to insert component stylesheets, hoisted scripts, for example, to happen during rendering than as a post-rendering step.
This is to enable streaming. This change will only be noticeable if you are rendering your `<head>` element inside of a framework component. If that is the case then the head items will be injected before the first non-head element in an Astro file instead.
In the future we may offer a `<Astro.Head>` component as a way to control where these scripts/styles are inserted.

View file

@ -0,0 +1,4 @@
<html>
<head><title>Preact component</title></head>
<body><slot></slot></body>
</html>

View file

@ -1,4 +1,5 @@
--- ---
layout: ../components/Layout.astro
setup: | setup: |
import Counter from '../components/Counter.jsx'; import Counter from '../components/Counter.jsx';
import PreactComponent from '../components/JSXComponent.jsx'; import PreactComponent from '../components/JSXComponent.jsx';

View file

@ -0,0 +1,4 @@
<html>
<head><title>React component</title></head>
<body><slot></slot></body>
</html>

View file

@ -1,4 +1,5 @@
--- ---
layout: ../components/Layout.astro
setup: | setup: |
import Counter from '../components/Counter.jsx'; import Counter from '../components/Counter.jsx';
import ReactComponent from '../components/JSXComponent.jsx'; import ReactComponent from '../components/JSXComponent.jsx';

View file

@ -0,0 +1,4 @@
<html>
<head><title>Solid component</title></head>
<body><slot></slot></body>
</html>

View file

@ -1,4 +1,5 @@
--- ---
layout: ../components/Layout.astro
setup: | setup: |
import Counter from '../components/Counter.jsx'; import Counter from '../components/Counter.jsx';
import SolidComponent from '../components/SolidComponent.jsx'; import SolidComponent from '../components/SolidComponent.jsx';

View file

@ -0,0 +1,4 @@
<html>
<head><title>Solid component</title></head>
<body><slot></slot></body>
</html>

View file

@ -1,4 +1,5 @@
--- ---
layout: ../components/Layout.astro
setup: | setup: |
import Counter from '../components/Counter.svelte'; import Counter from '../components/Counter.svelte';
import SvelteComponent from '../components/SvelteComponent.svelte'; import SvelteComponent from '../components/SvelteComponent.svelte';

View file

@ -0,0 +1,4 @@
<html>
<head><title>Vue component</title></head>
<body><slot></slot></body>
</html>

View file

@ -1,4 +1,5 @@
--- ---
layout: ../components/Layout.astro
setup: | setup: |
import Counter from '../components/Counter.vue'; import Counter from '../components/Counter.vue';
import VueComponent from '../components/VueComponent.vue'; import VueComponent from '../components/VueComponent.vue';

View file

@ -78,7 +78,7 @@
"test:e2e:match": "playwright test -g" "test:e2e:match": "playwright test -g"
}, },
"dependencies": { "dependencies": {
"@astrojs/compiler": "^0.16.1", "@astrojs/compiler": "^0.17.0",
"@astrojs/language-server": "^0.13.4", "@astrojs/language-server": "^0.13.4",
"@astrojs/markdown-remark": "^0.11.3", "@astrojs/markdown-remark": "^0.11.3",
"@astrojs/prism": "0.4.1", "@astrojs/prism": "0.4.1",

View file

@ -1004,7 +1004,6 @@ export interface SSRElement {
export interface SSRMetadata { export interface SSRMetadata {
renderers: SSRLoadedRenderer[]; renderers: SSRLoadedRenderer[];
pathname: string; pathname: string;
needsHydrationStyles: boolean;
} }
export interface SSRResult { export interface SSRResult {

View file

@ -161,12 +161,6 @@ export async function render(
} }
let html = page.html; let html = page.html;
// handle final head injection if it hasn't happened already
if (html.indexOf('<!--astro:head:injected-->') == -1) {
html = (await renderHead(result)) + html;
}
// cleanup internal state flags
html = html.replace('<!--astro:head:injected-->', '');
// inject <!doctype html> if missing (TODO: is a more robust check needed for comments, etc.?) // inject <!doctype html> if missing (TODO: is a more robust check needed for comments, etc.?)
if (!/<!doctype html/i.test(html)) { if (!/<!doctype html/i.test(html)) {

View file

@ -221,7 +221,6 @@ ${extra}`
}, },
resolve, resolve,
_metadata: { _metadata: {
needsHydrationStyles: false,
renderers, renderers,
pathname, pathname,
}, },

View file

@ -344,7 +344,6 @@ If you're still stuck, please open an issue on GitHub or join us at https://astr
{ renderer: renderer!, result, astroId, props }, { renderer: renderer!, result, astroId, props },
metadata as Required<AstroComponentMetadata> metadata as Required<AstroComponentMetadata>
); );
result._metadata.needsHydrationStyles = true;
// Render template if not all astro fragments are provided. // Render template if not all astro fragments are provided.
let unrenderedSlots: string[] = []; let unrenderedSlots: string[] = [];
@ -590,16 +589,6 @@ Update your code to remove this warning.`);
return handler.call(mod, proxy, request); return handler.call(mod, proxy, request);
} }
async function replaceHeadInjection(result: SSRResult, html: string): Promise<string> {
let template = html;
// <!--astro:head--> injected by compiler
// Must be handled at the end of the rendering process
if (template.indexOf('<!--astro:head-->') > -1) {
template = template.replace('<!--astro:head-->', await renderHead(result));
}
return template;
}
// Calls a component and renders it into a string of HTML // Calls a component and renders it into a string of HTML
export async function renderToString( export async function renderToString(
result: SSRResult, result: SSRResult,
@ -627,8 +616,7 @@ export async function renderPage(
const response = await componentFactory(result, props, children); const response = await componentFactory(result, props, children);
if (isAstroComponent(response)) { if (isAstroComponent(response)) {
let template = await renderAstroComponent(response); let html = await renderAstroComponent(response);
const html = await replaceHeadInjection(result, template);
return { return {
type: 'html', type: 'html',
html, html,
@ -660,37 +648,36 @@ const uniqueElements = (item: any, index: number, all: any[]) => {
); );
}; };
// Renders a page to completion by first calling the factory callback, waiting for its result, and then appending const alreadyHeadRenderedResults = new WeakSet<SSRResult>();
// styles and scripts into the head.
export async function renderHead(result: SSRResult): Promise<string> { export async function renderHead(result: SSRResult): Promise<string> {
alreadyHeadRenderedResults.add(result);
const styles = Array.from(result.styles) const styles = Array.from(result.styles)
.filter(uniqueElements) .filter(uniqueElements)
.map((style) => renderElement('style', style)); .map((style) => renderElement('style', style));
let needsHydrationStyles = result._metadata.needsHydrationStyles;
const scripts = Array.from(result.scripts) const scripts = Array.from(result.scripts)
.filter(uniqueElements) .filter(uniqueElements)
.map((script, i) => { .map((script, i) => {
if ('data-astro-component-hydration' in script.props) {
needsHydrationStyles = true;
}
return renderElement('script', script); return renderElement('script', script);
}); });
if (needsHydrationStyles) {
styles.push(
renderElement('style', {
props: {},
children: 'astro-island, astro-slot { display: contents; }',
})
);
}
const links = Array.from(result.links) const links = Array.from(result.links)
.filter(uniqueElements) .filter(uniqueElements)
.map((link) => renderElement('link', link, false)); .map((link) => renderElement('link', link, false));
return markHTMLString( return markHTMLString(
links.join('\n') + styles.join('\n') + scripts.join('\n') + '\n' + '<!--astro:head:injected-->' links.join('\n') + styles.join('\n') + scripts.join('\n')
); );
} }
// This function is called by Astro components that do not contain a <head> component
// This accomodates the fact that using a <head> is optional in Astro, so this
// is called before a component's first non-head HTML element. If the head was
// already injected it is a noop.
export function maybeRenderHead(result: SSRResult): string | Promise<string> {
if(alreadyHeadRenderedResults.has(result)) {
return '';
}
return renderHead(result);
}
export async function renderAstroComponent(component: InstanceType<typeof AstroComponent>) { export async function renderAstroComponent(component: InstanceType<typeof AstroComponent>) {
let template = []; let template = [];

View file

@ -59,7 +59,7 @@ export function getPrescripts(type: PrescriptType, directive: string): string {
// deps to be loaded immediately. // deps to be loaded immediately.
switch (type) { switch (type) {
case 'both': case 'both':
return `<script>${getDirectiveScriptText(directive) + islandScript}</script>`; return `<style>astro-island,astro-slot{display:contents}</style><script>${getDirectiveScriptText(directive) + islandScript}</script>`;
case 'directive': case 'directive':
return `<script>${getDirectiveScriptText(directive)}</script>`; return `<script>${getDirectiveScriptText(directive)}</script>`;
} }

View file

@ -65,7 +65,7 @@ describe('CSS', function () {
it('Using hydrated components adds astro-island styles', async () => { it('Using hydrated components adds astro-island styles', async () => {
const inline = $('style').html(); const inline = $('style').html();
expect(inline).to.include('display: contents'); expect(inline).to.include('display:contents');
}); });
it('<style lang="sass">', async () => { it('<style lang="sass">', async () => {

View file

@ -40,4 +40,10 @@ describe('Partial HTML', async () => {
const allInjectedStyles = $('style[data-astro-injected]').text().replace(/\s*/g, ''); const allInjectedStyles = $('style[data-astro-injected]').text().replace(/\s*/g, '');
expect(allInjectedStyles).to.match(/h1{color:red;}/); expect(allInjectedStyles).to.match(/h1{color:red;}/);
}); });
it('pages with a head, injection happens inside', async () => {
const html = await fixture.fetch('/with-head').then((res) => res.text());
const $ = cheerio.load(html);
expect($('style')).to.have.lengthOf(1);
});
}); });

View file

@ -0,0 +1,9 @@
<html>
<head>
<title>testing</title>
<style>body { color: blue; }</style>
</head>
<body>
<h1>testing</h1>
</body>
</html>

View file

@ -1,5 +1,5 @@
export { pathToPosix } from './lib/utils'; export { pathToPosix } from './lib/utils';
export { AbortController, AbortSignal, alert, atob, Blob, btoa, ByteLengthQueuingStrategy, cancelAnimationFrame, cancelIdleCallback, CanvasRenderingContext2D, CharacterData, clearTimeout, Comment, CountQueuingStrategy, CSSStyleSheet, CustomElementRegistry, CustomEvent, Document, DocumentFragment, DOMException, Element, Event, EventTarget, fetch, File, FormData, Headers, HTMLBodyElement, HTMLCanvasElement, HTMLDivElement, HTMLDocument, HTMLElement, HTMLHeadElement, HTMLHtmlElement, HTMLImageElement, HTMLSpanElement, HTMLStyleElement, HTMLTemplateElement, HTMLUnknownElement, Image, ImageData, IntersectionObserver, MediaQueryList, MutationObserver, Node, NodeFilter, NodeIterator, OffscreenCanvas, ReadableByteStreamController, ReadableStream, ReadableStreamBYOBReader, ReadableStreamBYOBRequest, ReadableStreamDefaultController, ReadableStreamDefaultReader, Request, requestAnimationFrame, requestIdleCallback, ResizeObserver, Response, setTimeout, ShadowRoot, structuredClone, StyleSheet, Text, TransformStream, TreeWalker, URLPattern, Window, WritableStream, WritableStreamDefaultController, WritableStreamDefaultWriter } from './mod.js'; export { AbortController, AbortSignal, alert, atob, Blob, btoa, ByteLengthQueuingStrategy, cancelAnimationFrame, cancelIdleCallback, CanvasRenderingContext2D, CharacterData, clearTimeout, Comment, CountQueuingStrategy, CSSStyleSheet, CustomElementRegistry, CustomEvent, Document, DocumentFragment, DOMException, Element, Event, EventTarget, fetch, File, FormData, Headers, HTMLBodyElement, HTMLCanvasElement, HTMLDivElement, HTMLDocument, HTMLElement, HTMLHeadElement, HTMLHtmlElement, HTMLImageElement, HTMLSpanElement, HTMLStyleElement, HTMLTemplateElement, HTMLUnknownElement, Image, ImageData, IntersectionObserver, MediaQueryList, MutationObserver, Node, NodeFilter, NodeIterator, OffscreenCanvas, ReadableByteStreamController, ReadableStream, ReadableStreamBYOBReader, ReadableStreamBYOBRequest, ReadableStreamDefaultController, ReadableStreamDefaultReader, Request, requestAnimationFrame, requestIdleCallback, ResizeObserver, Response, setTimeout, ShadowRoot, structuredClone, StyleSheet, Text, TransformStream, TreeWalker, URLPattern, Window, WritableStream, WritableStreamDefaultController, WritableStreamDefaultWriter, } from './mod.js';
export declare const polyfill: { export declare const polyfill: {
(target: any, options?: PolyfillOptions): any; (target: any, options?: PolyfillOptions): any;
internals(target: any, name: string): any; internals(target: any, name: string): any;

View file

@ -463,7 +463,7 @@ importers:
packages/astro: packages/astro:
specifiers: specifiers:
'@astrojs/compiler': ^0.16.1 '@astrojs/compiler': ^0.17.0
'@astrojs/language-server': ^0.13.4 '@astrojs/language-server': ^0.13.4
'@astrojs/markdown-remark': ^0.11.3 '@astrojs/markdown-remark': ^0.11.3
'@astrojs/prism': 0.4.1 '@astrojs/prism': 0.4.1
@ -547,7 +547,7 @@ importers:
yargs-parser: ^21.0.1 yargs-parser: ^21.0.1
zod: ^3.17.3 zod: ^3.17.3
dependencies: dependencies:
'@astrojs/compiler': 0.16.1 '@astrojs/compiler': 0.17.0
'@astrojs/language-server': 0.13.4 '@astrojs/language-server': 0.13.4
'@astrojs/markdown-remark': link:../markdown/remark '@astrojs/markdown-remark': link:../markdown/remark
'@astrojs/prism': link:../astro-prism '@astrojs/prism': link:../astro-prism
@ -2439,8 +2439,8 @@ packages:
leven: 3.1.0 leven: 3.1.0
dev: true dev: true
/@astrojs/compiler/0.16.1: /@astrojs/compiler/0.17.0:
resolution: {integrity: sha512-6l5j9b/sEdyqRUvwJpp+SmlAkNO5WeISuNEXnyH9aGwzIAdqgLB2boAJef9lWadlOjG8rSPO29WHRa3qS2Okew==} resolution: {integrity: sha512-3q6Yw6CGDfUwheDS29cHjQxn57ql0X98DskU6ym3bw/FdD8RMbGi0Es1Evlh+WHig948LUcYq19EHAMZO3bP3w==}
dev: false dev: false
/@astrojs/language-server/0.13.4: /@astrojs/language-server/0.13.4: