Skip to content

Commit 47d1ad1

Browse files
authored
[Flight] Skip transferReferencedDebugInfo during debug info resolution (facebook#35795)
When the Flight Client resolves chunk references during model parsing, it calls `transferReferencedDebugInfo` to propagate debug info entries from referenced chunks to the parent chunk. Debug info on chunks is later moved to their resolved values, where it is used by React DevTools to show performance tracks and what a component was suspended by. Debug chunks themselves (specifically `ReactComponentInfo`, `ReactAsyncInfo`, `ReactIOInfo`, and their outlined references) are metadata that is never rendered. They don't need debug info attached to them. Without this fix, debug info entries accumulate on outlined debug chunks via their references to other debug chunks (e.g. owner chains and props deduplication paths). Since each outlined chunk's accumulated entries are copied to every chunk that references it, this creates exponential growth in deep component trees, which can cause the dev server to hang and run out of memory. This generalizes the existing skip of `transferReferencedDebugInfo` for Element owner/stack references (which already recognizes that references to debug chunks don't need debug info transferred) to all references resolved during debug info resolution. It adds an `isInitializingDebugInfo` flag set in `initializeDebugChunk` and `resolveIOInfo`, which propagates through all nested `initializeModelChunk` calls within the same synchronous stack. For the async path, `waitForReference` captures the flag at call time into `InitializationReference.isDebug`, so deferred fulfillments also skip the transfer.
1 parent e8c6362 commit 47d1ad1

2 files changed

Lines changed: 77 additions & 19 deletions

File tree

packages/react-client/src/ReactFlightClient.js

Lines changed: 41 additions & 19 deletions
Original file line numberDiff line numberDiff line change
@@ -943,6 +943,7 @@ type InitializationHandler = {
943943
};
944944
let initializingHandler: null | InitializationHandler = null;
945945
let initializingChunk: null | BlockedChunk<any> = null;
946+
let isInitializingDebugInfo: boolean = false;
946947

947948
function initializeDebugChunk(
948949
response: Response,
@@ -951,6 +952,8 @@ function initializeDebugChunk(
951952
const debugChunk = chunk._debugChunk;
952953
if (debugChunk !== null) {
953954
const debugInfo = chunk._debugInfo;
955+
const prevIsInitializingDebugInfo = isInitializingDebugInfo;
956+
isInitializingDebugInfo = true;
954957
try {
955958
if (debugChunk.status === RESOLVED_MODEL) {
956959
// Find the index of this debug info by walking the linked list.
@@ -1015,6 +1018,8 @@ function initializeDebugChunk(
10151018
}
10161019
} catch (error) {
10171020
triggerErrorOnChunk(response, chunk, error);
1021+
} finally {
1022+
isInitializingDebugInfo = prevIsInitializingDebugInfo;
10181023
}
10191024
}
10201025
}
@@ -1632,7 +1637,9 @@ function fulfillReference(
16321637
const element: any = handler.value;
16331638
switch (key) {
16341639
case '3':
1635-
transferReferencedDebugInfo(handler.chunk, fulfilledChunk);
1640+
if (__DEV__) {
1641+
transferReferencedDebugInfo(handler.chunk, fulfilledChunk);
1642+
}
16361643
element.props = mappedValue;
16371644
break;
16381645
case '4':
@@ -1648,7 +1655,9 @@ function fulfillReference(
16481655
}
16491656
break;
16501657
default:
1651-
transferReferencedDebugInfo(handler.chunk, fulfilledChunk);
1658+
if (__DEV__) {
1659+
transferReferencedDebugInfo(handler.chunk, fulfilledChunk);
1660+
}
16521661
break;
16531662
}
16541663
} else if (__DEV__ && !reference.isDebug) {
@@ -2086,7 +2095,7 @@ function getOutlinedModel<T>(
20862095
response,
20872096
map,
20882097
path.slice(i - 1),
2089-
false,
2098+
isInitializingDebugInfo,
20902099
);
20912100
}
20922101
case HALTED: {
@@ -2158,14 +2167,21 @@ function getOutlinedModel<T>(
21582167
}
21592168

21602169
const chunkValue = map(response, value, parentObject, key);
2161-
if (
2162-
parentObject[0] === REACT_ELEMENT_TYPE &&
2163-
(key === '4' || key === '5')
2164-
) {
2165-
// If we're resolving the "owner" or "stack" slot of an Element array, we don't call
2166-
// transferReferencedDebugInfo because this reference is to a debug chunk.
2167-
} else {
2168-
transferReferencedDebugInfo(initializingChunk, chunk);
2170+
if (__DEV__) {
2171+
if (
2172+
parentObject[0] === REACT_ELEMENT_TYPE &&
2173+
(key === '4' || key === '5')
2174+
) {
2175+
// If we're resolving the "owner" or "stack" slot of an Element array,
2176+
// we don't call transferReferencedDebugInfo because this reference is
2177+
// to a debug chunk.
2178+
} else if (isInitializingDebugInfo) {
2179+
// If we're resolving references as part of debug info resolution, we
2180+
// don't call transferReferencedDebugInfo because these references are
2181+
// to debug chunks.
2182+
} else {
2183+
transferReferencedDebugInfo(initializingChunk, chunk);
2184+
}
21692185
}
21702186
return chunkValue;
21712187
case PENDING:
@@ -2177,7 +2193,7 @@ function getOutlinedModel<T>(
21772193
response,
21782194
map,
21792195
path,
2180-
false,
2196+
isInitializingDebugInfo,
21812197
);
21822198
case HALTED: {
21832199
// Add a dependency that will never resolve.
@@ -4264,15 +4280,21 @@ function resolveIOInfo(
42644280
): void {
42654281
const chunks = response._chunks;
42664282
let chunk = chunks.get(id);
4267-
if (!chunk) {
4268-
chunk = createResolvedModelChunk(response, model);
4269-
chunks.set(id, chunk);
4270-
initializeModelChunk(chunk);
4271-
} else {
4272-
resolveModelChunk(response, chunk, model);
4273-
if (chunk.status === RESOLVED_MODEL) {
4283+
const prevIsInitializingDebugInfo = isInitializingDebugInfo;
4284+
isInitializingDebugInfo = true;
4285+
try {
4286+
if (!chunk) {
4287+
chunk = createResolvedModelChunk(response, model);
4288+
chunks.set(id, chunk);
42744289
initializeModelChunk(chunk);
4290+
} else {
4291+
resolveModelChunk(response, chunk, model);
4292+
if (chunk.status === RESOLVED_MODEL) {
4293+
initializeModelChunk(chunk);
4294+
}
42754295
}
4296+
} finally {
4297+
isInitializingDebugInfo = prevIsInitializingDebugInfo;
42764298
}
42774299
if (chunk.status === INITIALIZED) {
42784300
initializeIOInfo(response, chunk.value);

packages/react-server/src/__tests__/ReactFlightAsyncDebugInfo-test.js

Lines changed: 36 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3633,4 +3633,40 @@ describe('ReactFlightAsyncDebugInfo', () => {
36333633
`);
36343634
}
36353635
});
3636+
3637+
it('should not exponentially accumulate debug info on outlined debug chunks', async () => {
3638+
// Regression test: Each Level wraps its received `context` prop in a new
3639+
// object before passing it down. This creates props deduplication
3640+
// references to the parent's outlined chunk alongside the owner reference,
3641+
// giving 2 references per level to the direct parent's chunk. Without
3642+
// skipping transferReferencedDebugInfo during debug info resolution, this
3643+
// test would fail with an infinite loop detection error.
3644+
async function Level({depth, context}) {
3645+
await delay(0);
3646+
if (depth === 0) {
3647+
return <div>Hello, World!</div>;
3648+
}
3649+
const newContext = {prev: context, id: depth};
3650+
return ReactServer.createElement(Level, {
3651+
depth: depth - 1,
3652+
context: newContext,
3653+
});
3654+
}
3655+
3656+
const stream = ReactServerDOMServer.renderToPipeableStream(
3657+
ReactServer.createElement(Level, {depth: 20, context: {root: true}}),
3658+
);
3659+
3660+
const readable = new Stream.PassThrough(streamOptions);
3661+
const result = ReactServerDOMClient.createFromNodeStream(readable, {
3662+
moduleMap: {},
3663+
moduleLoading: {},
3664+
});
3665+
stream.pipe(readable);
3666+
3667+
const resolved = await result;
3668+
expect(resolved.type).toBe('div');
3669+
3670+
await finishLoadingStream(readable);
3671+
});
36363672
});

0 commit comments

Comments
 (0)