Skip to content

Commit 75a8cbf

Browse files
committed
[Fiber] support hydration when rendering Suspense anywhere
stacked on #32163 This continues the work of making Suspense workable anywhere in a react-dom tree. See the prior PRs for how we handle server rendering and client rendering. In this change we update the hydration implementation to be able to locate expected nodes. In particular this means hydration understands now that the default hydration context is the document body when the container is above the body. One case that is unique to hydration is clearing Suspense boundaries. When hydration fails or when the server instructs the client to recover an errored boundary it's possible that the html, head, and body tags in the initial document were written from a fallback or a different primary content on the server and need to be replaced by the client render. However these tags (and in the case of head, their content) won't be inside the comment nodes that identify the bounds of the Suspense boundary. And when client rendering you may not even render the same singletons that were server rendered. So when server rendering a boudnary which contributes to the preamble (the html, head, and body tag openings plus the head contents) we emit a special marker comment just before closing the boundary out. This marker encodes which parts of the preamble this boundary owned. If we need to clear the suspense boundary on the client we read this marker and use it to reset the appropriate singleton state.
1 parent 6cd02da commit 75a8cbf

11 files changed

+863
-40
lines changed

packages/react-dom-bindings/src/client/ReactFiberConfigDOM.js

+106-2
Original file line numberDiff line numberDiff line change
@@ -206,6 +206,11 @@ const SUSPENSE_START_DATA = '$';
206206
const SUSPENSE_END_DATA = '/$';
207207
const SUSPENSE_PENDING_START_DATA = '$?';
208208
const SUSPENSE_FALLBACK_START_DATA = '$!';
209+
const PREAMBLE_CONTRIBUTION_MARKER = 'P';
210+
const PREAMBLE_CONTRIBUTION_INDICATOR = '|';
211+
const HTML_CONTRIBUTION_INDEX = 1;
212+
const BODY_CONTRIBUTION_INDEX = 2;
213+
const HEAD_CONTRIBUTION_INDEX = 3;
209214
const FORM_STATE_IS_MATCHING = 'F!';
210215
const FORM_STATE_IS_NOT_MATCHING = 'F';
211216

@@ -986,6 +991,37 @@ export function clearSuspenseBoundary(
986991
data === SUSPENSE_FALLBACK_START_DATA
987992
) {
988993
depth++;
994+
} else if (data[0] === PREAMBLE_CONTRIBUTION_MARKER) {
995+
const ownerDocument = parentInstance.ownerDocument;
996+
997+
// If this comment is too short or if this slot is an empty space (the indicator)
998+
// Then this boundary DID contribute the singleton component. It is constructed this way
999+
// to minimize the contribution data size for common contributions such as None (no comment)
1000+
// and All (<!--P-->). Less common contributions such as Body only would be (<!--P | -->)
1001+
if (
1002+
(data[HTML_CONTRIBUTION_INDEX] || PREAMBLE_CONTRIBUTION_INDICATOR) ===
1003+
PREAMBLE_CONTRIBUTION_INDICATOR
1004+
) {
1005+
const documentElement: Element = (ownerDocument.documentElement: any);
1006+
releaseSingletonInstance(documentElement);
1007+
}
1008+
if (
1009+
(data[BODY_CONTRIBUTION_INDEX] || PREAMBLE_CONTRIBUTION_INDICATOR) ===
1010+
PREAMBLE_CONTRIBUTION_INDICATOR
1011+
) {
1012+
const body: Element = (ownerDocument.body: any);
1013+
releaseSingletonInstance(body);
1014+
}
1015+
if (
1016+
(data[HEAD_CONTRIBUTION_INDEX] || PREAMBLE_CONTRIBUTION_INDICATOR) ===
1017+
PREAMBLE_CONTRIBUTION_INDICATOR
1018+
) {
1019+
const head: Element = (ownerDocument.head: any);
1020+
releaseSingletonInstance(head);
1021+
// We need to clear the head because this is the only singleton that can have children that
1022+
// were part of this boundary but are not inside this boundary.
1023+
clearHead(head);
1024+
}
9891025
}
9901026
}
9911027
// $FlowFixMe[incompatible-type] we bail out when we get a null
@@ -1499,7 +1535,7 @@ function clearContainerSparingly(container: Node) {
14991535
case 'STYLE': {
15001536
continue;
15011537
}
1502-
// Stylesheet tags are retained because tehy may likely come from 3rd party scripts and extensions
1538+
// Stylesheet tags are retained because they may likely come from 3rd party scripts and extensions
15031539
case 'LINK': {
15041540
if (((node: any): HTMLLinkElement).rel.toLowerCase() === 'stylesheet') {
15051541
continue;
@@ -1511,6 +1547,27 @@ function clearContainerSparingly(container: Node) {
15111547
return;
15121548
}
15131549

1550+
function clearHead(head: Element): void {
1551+
let node = head.firstChild;
1552+
while (node) {
1553+
const nextNode = node.nextSibling;
1554+
const nodeName = node.nodeName;
1555+
if (
1556+
isMarkedHoistable(node) ||
1557+
nodeName === 'SCRIPT' ||
1558+
nodeName === 'STYLE' ||
1559+
(nodeName === 'LINK' &&
1560+
((node: any): HTMLLinkElement).rel.toLowerCase() === 'stylesheet')
1561+
) {
1562+
// retain these nodes
1563+
} else {
1564+
head.removeChild(node);
1565+
}
1566+
node = nextNode;
1567+
}
1568+
return;
1569+
}
1570+
15141571
// Making this so we can eventually move all of the instance caching to the commit phase.
15151572
// Currently this is only used to associate fiber and props to instances for hydrating
15161573
// HostSingletons. The reason we need it here is we only want to make this binding on commit
@@ -1872,7 +1929,20 @@ export function getFirstHydratableChild(
18721929
export function getFirstHydratableChildWithinContainer(
18731930
parentContainer: Container,
18741931
): null | HydratableInstance {
1875-
return getNextHydratable(parentContainer.firstChild);
1932+
let parentElement: Element;
1933+
switch (parentContainer.nodeType) {
1934+
case DOCUMENT_NODE:
1935+
parentElement = (parentContainer: any).body;
1936+
break;
1937+
default: {
1938+
if (parentContainer.nodeName === 'HTML') {
1939+
parentElement = (parentContainer: any).ownerDocument.body;
1940+
} else {
1941+
parentElement = (parentContainer: any);
1942+
}
1943+
}
1944+
}
1945+
return getNextHydratable(parentElement.firstChild);
18761946
}
18771947

18781948
export function getFirstHydratableChildWithinSuspenseInstance(
@@ -1881,6 +1951,40 @@ export function getFirstHydratableChildWithinSuspenseInstance(
18811951
return getNextHydratable(parentInstance.nextSibling);
18821952
}
18831953

1954+
// If it were possible to have more than one scope singleton in a DOM tree
1955+
// we would need to model this as a stack but since you can only have one <head>
1956+
// and head is the only singleton that is a scope in DOM we can get away with
1957+
// tracking this as a single value.
1958+
let previousHydratableOnEnteringScopedSingleton: null | HydratableInstance =
1959+
null;
1960+
1961+
export function getFirstHydratableChildWithinSingleton(
1962+
type: string,
1963+
singletonInstance: Instance,
1964+
currentHydratableInstance: null | HydratableInstance,
1965+
): null | HydratableInstance {
1966+
if (isSingletonScope(type)) {
1967+
previousHydratableOnEnteringScopedSingleton = currentHydratableInstance;
1968+
return getNextHydratable(singletonInstance.firstChild);
1969+
} else {
1970+
return currentHydratableInstance;
1971+
}
1972+
}
1973+
1974+
export function getNextHydratableSiblingAfterSingleton(
1975+
type: string,
1976+
currentHydratableInstance: null | HydratableInstance,
1977+
): null | HydratableInstance {
1978+
if (isSingletonScope(type)) {
1979+
const previousHydratableInstance =
1980+
previousHydratableOnEnteringScopedSingleton;
1981+
previousHydratableOnEnteringScopedSingleton = null;
1982+
return previousHydratableInstance;
1983+
} else {
1984+
return currentHydratableInstance;
1985+
}
1986+
}
1987+
18841988
export function describeHydratableInstanceForDevWarnings(
18851989
instance: HydratableInstance,
18861990
): string | {type: string, props: $ReadOnly<Props>} {

packages/react-dom-bindings/src/server/ReactFizzConfigDOM.js

+106-6
Original file line numberDiff line numberDiff line change
@@ -684,16 +684,25 @@ export function completeResumableState(resumableState: ResumableState): void {
684684
resumableState.bootstrapModules = undefined;
685685
}
686686

687+
const NoContribution /* */ = 0b000;
688+
const HTMLContribution /* */ = 0b001;
689+
const BodyContribution /* */ = 0b010;
690+
const HeadContribution /* */ = 0b100;
691+
const TotalContribution =
692+
HTMLContribution | HeadContribution | BodyContribution;
693+
687694
export type PreambleState = {
688695
htmlChunks: null | Array<Chunk | PrecomputedChunk>,
689696
headChunks: null | Array<Chunk | PrecomputedChunk>,
690697
bodyChunks: null | Array<Chunk | PrecomputedChunk>,
698+
contribution: number,
691699
};
692700
export function createPreambleState(): PreambleState {
693701
return {
694702
htmlChunks: null,
695703
headChunks: null,
696704
bodyChunks: null,
705+
contribution: NoContribution,
697706
};
698707
}
699708

@@ -3227,7 +3236,7 @@ function pushStartHead(
32273236
throw new Error(`The ${'`<head>`'} tag may only be rendered once.`);
32283237
}
32293238
preamble.headChunks = [];
3230-
return pushStartGenericElement(preamble.headChunks, props, 'head');
3239+
return pushStartSingletonElement(preamble.headChunks, props, 'head');
32313240
} else {
32323241
// This <head> is deep and is likely just an error. we emit it inline though.
32333242
// Validation should warn that this tag is the the wrong spot.
@@ -3251,7 +3260,7 @@ function pushStartBody(
32513260
}
32523261

32533262
preamble.bodyChunks = [];
3254-
return pushStartGenericElement(preamble.bodyChunks, props, 'body');
3263+
return pushStartSingletonElement(preamble.bodyChunks, props, 'body');
32553264
} else {
32563265
// This <head> is deep and is likely just an error. we emit it inline though.
32573266
// Validation should warn that this tag is the the wrong spot.
@@ -3275,7 +3284,7 @@ function pushStartHtml(
32753284
}
32763285

32773286
preamble.htmlChunks = [DOCTYPE];
3278-
return pushStartGenericElement(preamble.htmlChunks, props, 'html');
3287+
return pushStartSingletonElement(preamble.htmlChunks, props, 'html');
32793288
} else {
32803289
// This <html> is deep and is likely just an error. we emit it inline though.
32813290
// Validation should warn that this tag is the the wrong spot.
@@ -3416,6 +3425,43 @@ function pushScriptImpl(
34163425
return null;
34173426
}
34183427

3428+
// This is a fork of pushStartGenericElement because we don't ever want to do
3429+
// the children as strign optimization on that path when rendering singletons.
3430+
// When we eliminate that special path we can delete this fork and unify it again
3431+
function pushStartSingletonElement(
3432+
target: Array<Chunk | PrecomputedChunk>,
3433+
props: Object,
3434+
tag: string,
3435+
): ReactNodeList {
3436+
target.push(startChunkForTag(tag));
3437+
3438+
let children = null;
3439+
let innerHTML = null;
3440+
for (const propKey in props) {
3441+
if (hasOwnProperty.call(props, propKey)) {
3442+
const propValue = props[propKey];
3443+
if (propValue == null) {
3444+
continue;
3445+
}
3446+
switch (propKey) {
3447+
case 'children':
3448+
children = propValue;
3449+
break;
3450+
case 'dangerouslySetInnerHTML':
3451+
innerHTML = propValue;
3452+
break;
3453+
default:
3454+
pushAttribute(target, propKey, propValue);
3455+
break;
3456+
}
3457+
}
3458+
}
3459+
3460+
target.push(endOfStartTag);
3461+
pushInnerHTML(target, innerHTML, children);
3462+
return children;
3463+
}
3464+
34193465
function pushStartGenericElement(
34203466
target: Array<Chunk | PrecomputedChunk>,
34213467
props: Object,
@@ -3907,14 +3953,17 @@ export function hoistPreambleState(
39073953
preambleState: PreambleState,
39083954
) {
39093955
const rootPreamble = renderState.preamble;
3910-
if (rootPreamble.htmlChunks === null) {
3956+
if (rootPreamble.htmlChunks === null && preambleState.htmlChunks) {
39113957
rootPreamble.htmlChunks = preambleState.htmlChunks;
3958+
preambleState.contribution |= HTMLContribution;
39123959
}
3913-
if (rootPreamble.headChunks === null) {
3960+
if (rootPreamble.headChunks === null && preambleState.headChunks) {
39143961
rootPreamble.headChunks = preambleState.headChunks;
3962+
preambleState.contribution |= HeadContribution;
39153963
}
3916-
if (rootPreamble.bodyChunks === null) {
3964+
if (rootPreamble.bodyChunks === null && preambleState.bodyChunks) {
39173965
rootPreamble.bodyChunks = preambleState.bodyChunks;
3966+
preambleState.contribution |= BodyContribution;
39183967
}
39193968
}
39203969

@@ -4005,6 +4054,21 @@ const clientRenderedSuspenseBoundaryError1D =
40054054
const clientRenderedSuspenseBoundaryError2 =
40064055
stringToPrecomputedChunk('></template>');
40074056

4057+
const boundaryPreambleContributionChunkTotal =
4058+
stringToPrecomputedChunk('<!--P-->');
4059+
const boundaryPreambleContributionChunkHTMLOnly =
4060+
stringToPrecomputedChunk('<!--P| -->');
4061+
const boundaryPreambleContributionChunkBodyOnly =
4062+
stringToPrecomputedChunk('<!--P | -->');
4063+
const boundaryPreambleContributionChunkHeadOnly =
4064+
stringToPrecomputedChunk('<!--P -->');
4065+
const boundaryPreambleContributionChunkHTMLAndBody =
4066+
stringToPrecomputedChunk('<!--P|| -->');
4067+
const boundaryPreambleContributionChunkHTMLAndHead =
4068+
stringToPrecomputedChunk('<!--P| -->');
4069+
const boundaryPreambleContributionChunkHeadAndBody =
4070+
stringToPrecomputedChunk('<!--P -->');
4071+
40084072
export function writeStartCompletedSuspenseBoundary(
40094073
destination: Destination,
40104074
renderState: RenderState,
@@ -4091,7 +4155,11 @@ export function writeStartClientRenderedSuspenseBoundary(
40914155
export function writeEndCompletedSuspenseBoundary(
40924156
destination: Destination,
40934157
renderState: RenderState,
4158+
preambleState: null | PreambleState,
40944159
): boolean {
4160+
if (preambleState) {
4161+
writePreambleContribution(destination, preambleState);
4162+
}
40954163
return writeChunkAndReturn(destination, endSuspenseBoundary);
40964164
}
40974165
export function writeEndPendingSuspenseBoundary(
@@ -4103,9 +4171,41 @@ export function writeEndPendingSuspenseBoundary(
41034171
export function writeEndClientRenderedSuspenseBoundary(
41044172
destination: Destination,
41054173
renderState: RenderState,
4174+
preambleState: null | PreambleState,
41064175
): boolean {
4176+
if (preambleState) {
4177+
writePreambleContribution(destination, preambleState);
4178+
}
41074179
return writeChunkAndReturn(destination, endSuspenseBoundary);
41084180
}
4181+
function writePreambleContribution(
4182+
destination: Destination,
4183+
preambleState: PreambleState,
4184+
) {
4185+
const contribution = preambleState.contribution;
4186+
switch (contribution) {
4187+
case TotalContribution:
4188+
writeChunk(destination, boundaryPreambleContributionChunkTotal);
4189+
break;
4190+
case HeadContribution | BodyContribution:
4191+
writeChunk(destination, boundaryPreambleContributionChunkHeadAndBody);
4192+
break;
4193+
case HTMLContribution | HeadContribution:
4194+
writeChunk(destination, boundaryPreambleContributionChunkHTMLAndHead);
4195+
break;
4196+
case HTMLContribution | BodyContribution:
4197+
writeChunk(destination, boundaryPreambleContributionChunkHTMLAndBody);
4198+
break;
4199+
case HeadContribution:
4200+
writeChunk(destination, boundaryPreambleContributionChunkHeadOnly);
4201+
break;
4202+
case BodyContribution:
4203+
writeChunk(destination, boundaryPreambleContributionChunkBodyOnly);
4204+
break;
4205+
case HTMLContribution:
4206+
writeChunk(destination, boundaryPreambleContributionChunkHTMLOnly);
4207+
}
4208+
}
41094209

41104210
const startSegmentHTML = stringToPrecomputedChunk('<div hidden id="');
41114211
const startSegmentHTML2 = stringToPrecomputedChunk('">');

packages/react-dom-bindings/src/server/ReactFizzConfigDOMLegacy.js

+12-2
Original file line numberDiff line numberDiff line change
@@ -244,20 +244,30 @@ export function writeStartClientRenderedSuspenseBoundary(
244244
export function writeEndCompletedSuspenseBoundary(
245245
destination: Destination,
246246
renderState: RenderState,
247+
preambleState: null | PreambleState,
247248
): boolean {
248249
if (renderState.generateStaticMarkup) {
249250
return true;
250251
}
251-
return writeEndCompletedSuspenseBoundaryImpl(destination, renderState);
252+
return writeEndCompletedSuspenseBoundaryImpl(
253+
destination,
254+
renderState,
255+
preambleState,
256+
);
252257
}
253258
export function writeEndClientRenderedSuspenseBoundary(
254259
destination: Destination,
255260
renderState: RenderState,
261+
preambleState: null | PreambleState,
256262
): boolean {
257263
if (renderState.generateStaticMarkup) {
258264
return true;
259265
}
260-
return writeEndClientRenderedSuspenseBoundaryImpl(destination, renderState);
266+
return writeEndClientRenderedSuspenseBoundaryImpl(
267+
destination,
268+
renderState,
269+
preambleState,
270+
);
261271
}
262272

263273
export type TransitionStatus = FormStatus;

0 commit comments

Comments
 (0)