Ensure hydration scripts inside of slots render ASAP (#4288)

* Ensure hydration scripts inside of slots render ASAP

* Changeset

* fix type errors

* Update packages/astro/src/runtime/server/render/page.ts

Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>

Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
Matthew Phillips 2022-08-12 12:18:41 -04:00 committed by GitHub
parent 467fabd749
commit c218100684
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
16 changed files with 111 additions and 35 deletions

View file

@ -0,0 +1,5 @@
---
'astro': patch
---
Ensure hydration scripts inside of slots render ASAP

View file

@ -1112,11 +1112,25 @@ export interface SSRElement {
children: string; children: string;
} }
export interface HydrationMetadata {
directive: string;
value: string;
componentUrl: string;
componentExport: { value: string };
}
export interface SSRRenderInstruction {
type: 'directive';
result: SSRResult;
hydration: HydrationMetadata;
}
export interface SSRMetadata { export interface SSRMetadata {
renderers: SSRLoadedRenderer[]; renderers: SSRLoadedRenderer[];
pathname: string; pathname: string;
hasHydrationScript: boolean; hasHydrationScript: boolean;
hasDirectives: Set<string>; hasDirectives: Set<string>;
pendingInstructions: Set<SSRRenderInstruction>;
} }
export interface SSRResult { export interface SSRResult {

View file

@ -265,6 +265,7 @@ const canonicalURL = new URL(Astro.url.pathname, Astro.site);
pathname, pathname,
hasHydrationScript: false, hasHydrationScript: false,
hasDirectives: new Set(), hasDirectives: new Set(),
pendingInstructions: new Set(),
}, },
response, response,
}; };

View file

@ -1,5 +1,6 @@
import type { import type {
AstroComponentMetadata, AstroComponentMetadata,
HydrationMetadata,
SSRElement, SSRElement,
SSRLoadedRenderer, SSRLoadedRenderer,
SSRResult, SSRResult,
@ -12,12 +13,7 @@ const HydrationDirectivesRaw = ['load', 'idle', 'media', 'visible', 'only'];
const HydrationDirectives = new Set(HydrationDirectivesRaw); const HydrationDirectives = new Set(HydrationDirectivesRaw);
export const HydrationDirectiveProps = new Set(HydrationDirectivesRaw.map((n) => `client:${n}`)); export const HydrationDirectiveProps = new Set(HydrationDirectivesRaw.map((n) => `client:${n}`));
export interface HydrationMetadata { export { HydrationMetadata };
directive: string;
value: string;
componentUrl: string;
componentExport: { value: string };
}
interface ExtractedProps { interface ExtractedProps {
isPage: boolean; isPage: boolean;

View file

@ -26,7 +26,7 @@ export {
stringifyChunk, stringifyChunk,
voidElementNames, voidElementNames,
} from './render/index.js'; } from './render/index.js';
export type { AstroComponentFactory, RenderInstruction } from './render/index.js'; export type { AstroComponentFactory } from './render/index.js';
import type { AstroComponentFactory } from './render/index.js'; import type { AstroComponentFactory } from './render/index.js';
import { markHTMLString } from './escape.js'; import { markHTMLString } from './escape.js';

View file

@ -1,12 +1,11 @@
/* eslint-disable no-console */ /* eslint-disable no-console */
import { SSRResult } from '../../@types/astro.js'; import { SSRResult, SSRRenderInstruction } from '../../@types/astro.js';
import { AstroJSX, isVNode } from '../../jsx-runtime/index.js'; import { AstroJSX, isVNode } from '../../jsx-runtime/index.js';
import { import {
escapeHTML, escapeHTML,
HTMLString, HTMLString,
markHTMLString, markHTMLString,
renderComponent, renderComponent,
RenderInstruction,
renderToString, renderToString,
spreadAttributes, spreadAttributes,
stringifyChunk, stringifyChunk,
@ -122,7 +121,7 @@ export async function renderJSX(result: SSRResult, vnode: any): Promise<any> {
} }
await Promise.all(slotPromises); await Promise.all(slotPromises);
let output: string | AsyncIterable<string | RenderInstruction>; let output: string | AsyncIterable<string | SSRRenderInstruction>;
if (vnode.type === ClientOnlyPlaceholder && vnode.props['client:only']) { if (vnode.type === ClientOnlyPlaceholder && vnode.props['client:only']) {
output = await renderComponent( output = await renderComponent(
result, result,

View file

@ -1,3 +1,4 @@
import { SSRRenderInstruction, SSRResult } from '../../../@types/astro';
import { escapeHTML, HTMLString, markHTMLString } from '../escape.js'; import { escapeHTML, HTMLString, markHTMLString } from '../escape.js';
import { AstroComponent, renderAstroComponent } from './astro.js'; import { AstroComponent, renderAstroComponent } from './astro.js';
import { stringifyChunk } from './common.js'; import { stringifyChunk } from './common.js';
@ -34,13 +35,15 @@ export async function* renderChild(child: any): AsyncIterable<any> {
} }
} }
export async function renderSlot(result: any, slotted: string, fallback?: any): Promise<string> { export async function renderSlot(result: SSRResult, slotted: string, fallback?: any): Promise<string> {
if (slotted) { if (slotted) {
let iterator = renderChild(slotted); let iterator = renderChild(slotted);
let content = ''; let content = '';
for await (const chunk of iterator) { for await (const chunk of iterator) {
if ((chunk as any).type === 'directive') { if ((chunk as any).type === 'directive') {
content += stringifyChunk(result, chunk); // Do not render directives scripts in a slot, instead mark them as pending
// so that renderPage includes them ASAP.
result._metadata.pendingInstructions.add(chunk as SSRRenderInstruction);
} else { } else {
content += chunk; content += chunk;
} }

View file

@ -1,6 +1,5 @@
import type { SSRResult } from '../../../@types/astro'; import type { SSRResult, SSRRenderInstruction } from '../../../@types/astro';
import type { AstroComponentFactory } from './index'; import type { AstroComponentFactory } from './index';
import type { RenderInstruction } from './types';
import { markHTMLString } from '../escape.js'; import { markHTMLString } from '../escape.js';
import { HydrationDirectiveProps } from '../hydration.js'; import { HydrationDirectiveProps } from '../hydration.js';
@ -58,7 +57,7 @@ export function isAstroComponent(obj: any): obj is AstroComponent {
export async function* renderAstroComponent( export async function* renderAstroComponent(
component: InstanceType<typeof AstroComponent> component: InstanceType<typeof AstroComponent>
): AsyncIterable<string | RenderInstruction> { ): AsyncIterable<string | SSRRenderInstruction> {
for await (const value of component) { for await (const value of component) {
if (value || value === 0) { if (value || value === 0) {
for await (const chunk of renderChild(value)) { for await (const chunk of renderChild(value)) {
@ -104,7 +103,7 @@ export async function renderToIterable(
displayName: string, displayName: string,
props: any, props: any,
children: any children: any
): Promise<AsyncIterable<string | RenderInstruction>> { ): Promise<AsyncIterable<string | SSRRenderInstruction>> {
validateComponentProps(props, displayName); validateComponentProps(props, displayName);
const Component = await componentFactory(result, props, children); const Component = await componentFactory(result, props, children);

View file

@ -1,5 +1,4 @@
import type { SSRResult } from '../../../@types/astro'; import type { SSRResult, SSRRenderInstruction } from '../../../@types/astro';
import type { RenderInstruction } from './types.js';
import { markHTMLString } from '../escape.js'; import { markHTMLString } from '../escape.js';
import { import {
@ -15,10 +14,10 @@ export const Renderer = Symbol.for('astro:renderer');
// Rendering produces either marked strings of HTML or instructions for hydration. // Rendering produces either marked strings of HTML or instructions for hydration.
// These directive instructions bubble all the way up to renderPage so that we // These directive instructions bubble all the way up to renderPage so that we
// can ensure they are added only once, and as soon as possible. // can ensure they are added only once, and as soon as possible.
export function stringifyChunk(result: SSRResult, chunk: string | RenderInstruction) { export function stringifyChunk(result: SSRResult, chunk: string | SSRRenderInstruction) {
switch ((chunk as any).type) { switch ((chunk as any).type) {
case 'directive': { case 'directive': {
const { hydration } = chunk as RenderInstruction; const { hydration } = chunk as SSRRenderInstruction;
let needsHydrationScript = hydration && determineIfNeedsHydrationScript(result); let needsHydrationScript = hydration && determineIfNeedsHydrationScript(result);
let needsDirectiveScript = let needsDirectiveScript =
hydration && determinesIfNeedsDirectiveScript(result, hydration.directive); hydration && determinesIfNeedsDirectiveScript(result, hydration.directive);

View file

@ -1,5 +1,5 @@
import type { AstroComponentMetadata, SSRLoadedRenderer, SSRResult } from '../../../@types/astro'; import type { AstroComponentMetadata, SSRLoadedRenderer, SSRResult, SSRRenderInstruction } from '../../../@types/astro';
import type { RenderInstruction } from './types.js';
import { markHTMLString } from '../escape.js'; import { markHTMLString } from '../escape.js';
import { extractDirectives, generateHydrateScript } from '../hydration.js'; import { extractDirectives, generateHydrateScript } from '../hydration.js';
@ -49,7 +49,7 @@ export async function renderComponent(
Component: unknown, Component: unknown,
_props: Record<string | number, any>, _props: Record<string | number, any>,
slots: any = {} slots: any = {}
): Promise<string | AsyncIterable<string | RenderInstruction>> { ): Promise<string | AsyncIterable<string | SSRRenderInstruction>> {
Component = await Component; Component = await Component;
switch (getComponentType(Component)) { switch (getComponentType(Component)) {
@ -79,7 +79,7 @@ export async function renderComponent(
case 'astro-factory': { case 'astro-factory': {
async function* renderAstroComponentInline(): AsyncGenerator< async function* renderAstroComponentInline(): AsyncGenerator<
string | RenderInstruction, string | SSRRenderInstruction,
void, void,
undefined undefined
> { > {

View file

@ -7,7 +7,6 @@ export { renderComponent } from './component.js';
export { renderHTMLElement } from './dom.js'; export { renderHTMLElement } from './dom.js';
export { maybeRenderHead, renderHead } from './head.js'; export { maybeRenderHead, renderHead } from './head.js';
export { renderPage } from './page.js'; export { renderPage } from './page.js';
export type { RenderInstruction } from './types';
export { addAttribute, defineScriptVars, voidElementNames } from './util.js'; export { addAttribute, defineScriptVars, voidElementNames } from './util.js';
// The callback passed to to $$createComponent // The callback passed to to $$createComponent

View file

@ -49,13 +49,28 @@ export async function renderPage(
let headers = new Headers(init.headers); let headers = new Headers(init.headers);
let body: BodyInit; let body: BodyInit;
// Combines HTML chunks coming from the iterable with rendering instructions
// added to metadata. These instructions need to go out first to ensure
// the scripts exist before the islands that need them.
async function * eachChunk() {
for await (const chunk of iterable) {
if(result._metadata.pendingInstructions.size) {
for(const instr of result._metadata.pendingInstructions) {
result._metadata.pendingInstructions.delete(instr);
yield instr;
}
}
yield chunk;
}
}
if (streaming) { if (streaming) {
body = new ReadableStream({ body = new ReadableStream({
start(controller) { start(controller) {
async function read() { async function read() {
let i = 0; let i = 0;
try { try {
for await (const chunk of iterable) { for await (const chunk of eachChunk()) {
let html = stringifyChunk(result, chunk); let html = stringifyChunk(result, chunk);
if (i === 0) { if (i === 0) {
@ -77,7 +92,7 @@ export async function renderPage(
} else { } else {
body = ''; body = '';
let i = 0; let i = 0;
for await (const chunk of iterable) { for await (const chunk of eachChunk()) {
let html = stringifyChunk(result, chunk); let html = stringifyChunk(result, chunk);
if (i === 0) { if (i === 0) {
if (!/<!doctype html/i.test(html)) { if (!/<!doctype html/i.test(html)) {

View file

@ -1,8 +0,0 @@
import type { SSRResult } from '../../../@types/astro';
import type { HydrationMetadata } from '../hydration.js';
export interface RenderInstruction {
type: 'directive';
result: SSRResult;
hydration: HydrationMetadata;
}

View file

@ -0,0 +1,6 @@
---
import Wrapper from './Wrapper.astro';
---
<Wrapper />
<slot />

View file

@ -0,0 +1,15 @@
---
import Layout from '../components/Layout.astro';
import Wrapper from '../components/Wrapper.astro';
import One from '../components/One.jsx';
---
<html>
<head>
<title>Testing</title>
</head>
<body>
<Layout>
<One name="Three" client:idle />
</Layout>
</body>
</html>

View file

@ -26,4 +26,37 @@ describe('Hydration script ordering', async () => {
expect($('style')).to.have.a.lengthOf(1, 'hydration style added once'); expect($('style')).to.have.a.lengthOf(1, 'hydration style added once');
expect($('script')).to.have.a.lengthOf(1, 'only one hydration script needed'); expect($('script')).to.have.a.lengthOf(1, 'only one hydration script needed');
}); });
it('Hydration placed in correct location when there is slots used', async () => {
let html = await fixture.readFile('/slot/index.html');
let $ = cheerio.load(html);
//console.log(html);
// First, let's make sure all islands rendered (or test is bad)
expect($('astro-island')).to.have.a.lengthOf(3);
// Now let's make sure the hydration script is placed before the first component
let firstIsland = $($('astro-island').get(0));
let el = firstIsland.prev();
let foundScript = false;
// Traverse the DOM going backwards until we find a script, if there is one.
while(el.length) {
let last = el;
while(el.length) {
if(el.prop('tagName') === 'SCRIPT') {
foundScript = true;
}
last = el;
el = el.prev();
}
el = last.parent();
}
expect(foundScript).to.be.true;
// Sanity check that we're only rendering them once.
expect($('style')).to.have.a.lengthOf(1, 'hydration style added once');
expect($('script')).to.have.a.lengthOf(1, 'only one hydration script needed');
});
}); });