Skip to content

Commit f33a6b6

Browse files
authored
Track Owner for Server Components in DEV (#28753)
This implements the concept of a DEV-only "owner" for Server Components. The owner concept isn't really super useful. We barely use it anymore, but we do have it as a concept in DevTools in a couple of cases so this adds it for parity. However, this is mainly interesting because it could be used to wire up future owner-based stacks. I do this by outlining the DebugInfo for a Server Component (ReactComponentInfo). Then I just rely on Flight deduping to refer to that. I refer to the same thing by referential equality so that we can associate a Server Component parent in DebugInfo with an owner. If you suspend and replay a Server Component, we have to restore the same owner. To do that, I did a little ugly hack and stashed it on the thenable state object. Felt unnecessarily complicated to add a stateful wrapper for this one dev-only case. The owner could really be anything since it could be coming from a different implementation. Because this is the first time we have an owner other than Fiber, I have to fix up a bunch of places that assumes Fiber. I mainly did the `typeof owner.tag === 'number'` to assume it's a Fiber for now. This also doesn't actually add it to DevTools / RN Inspector yet. I just ignore them there for now. Because Server Components can be async the owner isn't tracked after an await. We need per-component AsyncLocalStorage for that. This can be done in a follow up.
1 parent e3ebcd5 commit f33a6b6

20 files changed

+291
-158
lines changed

packages/react-client/src/ReactFlightClient.js

+18-7
Original file line numberDiff line numberDiff line change
@@ -484,6 +484,7 @@ function createElement(
484484
type: mixed,
485485
key: mixed,
486486
props: mixed,
487+
owner: null | ReactComponentInfo, // DEV-only
487488
): React$Element<any> {
488489
let element: any;
489490
if (__DEV__ && enableRefAsProp) {
@@ -493,7 +494,7 @@ function createElement(
493494
type,
494495
key,
495496
props,
496-
_owner: null,
497+
_owner: owner,
497498
}: any);
498499
Object.defineProperty(element, 'ref', {
499500
enumerable: false,
@@ -520,7 +521,7 @@ function createElement(
520521
props,
521522

522523
// Record the component responsible for creating this element.
523-
_owner: null,
524+
_owner: owner,
524525
}: any);
525526
}
526527

@@ -854,7 +855,12 @@ function parseModelTuple(
854855
if (tuple[0] === REACT_ELEMENT_TYPE) {
855856
// TODO: Consider having React just directly accept these arrays as elements.
856857
// Or even change the ReactElement type to be an array.
857-
return createElement(tuple[1], tuple[2], tuple[3]);
858+
return createElement(
859+
tuple[1],
860+
tuple[2],
861+
tuple[3],
862+
__DEV__ ? (tuple: any)[4] : null,
863+
);
858864
}
859865
return value;
860866
}
@@ -1132,12 +1138,14 @@ function resolveConsoleEntry(
11321138
);
11331139
}
11341140

1135-
const payload: [string, string, string, mixed] = parseModel(response, value);
1141+
const payload: [string, string, null | ReactComponentInfo, string, mixed] =
1142+
parseModel(response, value);
11361143
const methodName = payload[0];
11371144
// TODO: Restore the fake stack before logging.
11381145
// const stackTrace = payload[1];
1139-
const env = payload[2];
1140-
const args = payload.slice(3);
1146+
// const owner = payload[2];
1147+
const env = payload[3];
1148+
const args = payload.slice(4);
11411149
printToConsole(methodName, args, env);
11421150
}
11431151

@@ -1286,7 +1294,10 @@ function processFullRow(
12861294
}
12871295
case 68 /* "D" */: {
12881296
if (__DEV__) {
1289-
const debugInfo = JSON.parse(row);
1297+
const debugInfo: ReactComponentInfo | ReactAsyncInfo = parseModel(
1298+
response,
1299+
row,
1300+
);
12901301
resolveDebugInfo(response, id, debugInfo);
12911302
return;
12921303
}

packages/react-client/src/__tests__/ReactFlight-test.js

+53-5
Original file line numberDiff line numberDiff line change
@@ -214,7 +214,7 @@ describe('ReactFlight', () => {
214214
const rootModel = await ReactNoopFlightClient.read(transport);
215215
const greeting = rootModel.greeting;
216216
expect(greeting._debugInfo).toEqual(
217-
__DEV__ ? [{name: 'Greeting', env: 'Server'}] : undefined,
217+
__DEV__ ? [{name: 'Greeting', env: 'Server', owner: null}] : undefined,
218218
);
219219
ReactNoop.render(greeting);
220220
});
@@ -241,7 +241,7 @@ describe('ReactFlight', () => {
241241
await act(async () => {
242242
const promise = ReactNoopFlightClient.read(transport);
243243
expect(promise._debugInfo).toEqual(
244-
__DEV__ ? [{name: 'Greeting', env: 'Server'}] : undefined,
244+
__DEV__ ? [{name: 'Greeting', env: 'Server', owner: null}] : undefined,
245245
);
246246
ReactNoop.render(await promise);
247247
});
@@ -2072,19 +2072,21 @@ describe('ReactFlight', () => {
20722072
await act(async () => {
20732073
const promise = ReactNoopFlightClient.read(transport);
20742074
expect(promise._debugInfo).toEqual(
2075-
__DEV__ ? [{name: 'ServerComponent', env: 'Server'}] : undefined,
2075+
__DEV__
2076+
? [{name: 'ServerComponent', env: 'Server', owner: null}]
2077+
: undefined,
20762078
);
20772079
const result = await promise;
20782080
const thirdPartyChildren = await result.props.children[1];
20792081
// We expect the debug info to be transferred from the inner stream to the outer.
20802082
expect(thirdPartyChildren[0]._debugInfo).toEqual(
20812083
__DEV__
2082-
? [{name: 'ThirdPartyComponent', env: 'third-party'}]
2084+
? [{name: 'ThirdPartyComponent', env: 'third-party', owner: null}]
20832085
: undefined,
20842086
);
20852087
expect(thirdPartyChildren[1]._debugInfo).toEqual(
20862088
__DEV__
2087-
? [{name: 'ThirdPartyLazyComponent', env: 'third-party'}]
2089+
? [{name: 'ThirdPartyLazyComponent', env: 'third-party', owner: null}]
20882090
: undefined,
20892091
);
20902092
ReactNoop.render(result);
@@ -2145,4 +2147,50 @@ describe('ReactFlight', () => {
21452147
expect(loggedFn).not.toBe(foo);
21462148
expect(loggedFn.toString()).toBe(foo.toString());
21472149
});
2150+
2151+
it('uses the server component debug info as the element owner in DEV', async () => {
2152+
function Container({children}) {
2153+
return children;
2154+
}
2155+
2156+
function Greeting({firstName}) {
2157+
// We can't use JSX here because it'll use the Client React.
2158+
return ReactServer.createElement(
2159+
Container,
2160+
null,
2161+
ReactServer.createElement('span', null, 'Hello, ', firstName),
2162+
);
2163+
}
2164+
2165+
const model = {
2166+
greeting: ReactServer.createElement(Greeting, {firstName: 'Seb'}),
2167+
};
2168+
2169+
const transport = ReactNoopFlightServer.render(model);
2170+
2171+
await act(async () => {
2172+
const rootModel = await ReactNoopFlightClient.read(transport);
2173+
const greeting = rootModel.greeting;
2174+
// We've rendered down to the span.
2175+
expect(greeting.type).toBe('span');
2176+
if (__DEV__) {
2177+
const greetInfo = {name: 'Greeting', env: 'Server', owner: null};
2178+
expect(greeting._debugInfo).toEqual([
2179+
greetInfo,
2180+
{name: 'Container', env: 'Server', owner: greetInfo},
2181+
]);
2182+
// The owner that created the span was the outer server component.
2183+
// We expect the debug info to be referentially equal to the owner.
2184+
expect(greeting._owner).toBe(greeting._debugInfo[0]);
2185+
} else {
2186+
expect(greeting._debugInfo).toBe(undefined);
2187+
expect(greeting._owner).toBe(
2188+
gate(flags => flags.disableStringRefs) ? undefined : null,
2189+
);
2190+
}
2191+
ReactNoop.render(greeting);
2192+
});
2193+
2194+
expect(ReactNoop).toMatchRenderedOutput(<span>Hello, Seb</span>);
2195+
});
21482196
});

packages/react-devtools-shared/src/__tests__/componentStacks-test.js

+1
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,7 @@ describe('component stack', () => {
101101
{
102102
name: 'ServerComponent',
103103
env: 'Server',
104+
owner: null,
104105
},
105106
];
106107
const Parent = () => ChildPromise;

packages/react-devtools-shared/src/backend/DevToolsComponentStackFrame.js

+5-17
Original file line numberDiff line numberDiff line change
@@ -33,10 +33,7 @@ import {
3333
import {disableLogs, reenableLogs} from './DevToolsConsolePatching';
3434

3535
let prefix;
36-
export function describeBuiltInComponentFrame(
37-
name: string,
38-
ownerFn: void | null | Function,
39-
): string {
36+
export function describeBuiltInComponentFrame(name: string): string {
4037
if (prefix === undefined) {
4138
// Extract the VM specific prefix used by each line.
4239
try {
@@ -51,10 +48,7 @@ export function describeBuiltInComponentFrame(
5148
}
5249

5350
export function describeDebugInfoFrame(name: string, env: ?string): string {
54-
return describeBuiltInComponentFrame(
55-
name + (env ? ' (' + env + ')' : ''),
56-
null,
57-
);
51+
return describeBuiltInComponentFrame(name + (env ? ' (' + env + ')' : ''));
5852
}
5953

6054
let reentry = false;
@@ -292,15 +286,13 @@ export function describeNativeComponentFrame(
292286

293287
export function describeClassComponentFrame(
294288
ctor: Function,
295-
ownerFn: void | null | Function,
296289
currentDispatcherRef: CurrentDispatcherRef,
297290
): string {
298291
return describeNativeComponentFrame(ctor, true, currentDispatcherRef);
299292
}
300293

301294
export function describeFunctionComponentFrame(
302295
fn: Function,
303-
ownerFn: void | null | Function,
304296
currentDispatcherRef: CurrentDispatcherRef,
305297
): string {
306298
return describeNativeComponentFrame(fn, false, currentDispatcherRef);
@@ -313,7 +305,6 @@ function shouldConstruct(Component: Function) {
313305

314306
export function describeUnknownElementTypeFrameInDEV(
315307
type: any,
316-
ownerFn: void | null | Function,
317308
currentDispatcherRef: CurrentDispatcherRef,
318309
): string {
319310
if (!__DEV__) {
@@ -330,31 +321,29 @@ export function describeUnknownElementTypeFrameInDEV(
330321
);
331322
}
332323
if (typeof type === 'string') {
333-
return describeBuiltInComponentFrame(type, ownerFn);
324+
return describeBuiltInComponentFrame(type);
334325
}
335326
switch (type) {
336327
case SUSPENSE_NUMBER:
337328
case SUSPENSE_SYMBOL_STRING:
338-
return describeBuiltInComponentFrame('Suspense', ownerFn);
329+
return describeBuiltInComponentFrame('Suspense');
339330
case SUSPENSE_LIST_NUMBER:
340331
case SUSPENSE_LIST_SYMBOL_STRING:
341-
return describeBuiltInComponentFrame('SuspenseList', ownerFn);
332+
return describeBuiltInComponentFrame('SuspenseList');
342333
}
343334
if (typeof type === 'object') {
344335
switch (type.$$typeof) {
345336
case FORWARD_REF_NUMBER:
346337
case FORWARD_REF_SYMBOL_STRING:
347338
return describeFunctionComponentFrame(
348339
type.render,
349-
ownerFn,
350340
currentDispatcherRef,
351341
);
352342
case MEMO_NUMBER:
353343
case MEMO_SYMBOL_STRING:
354344
// Memo may contain any component type so we recursively resolve it.
355345
return describeUnknownElementTypeFrameInDEV(
356346
type.type,
357-
ownerFn,
358347
currentDispatcherRef,
359348
);
360349
case LAZY_NUMBER:
@@ -366,7 +355,6 @@ export function describeUnknownElementTypeFrameInDEV(
366355
// Lazy may contain any component type so we recursively resolve it.
367356
return describeUnknownElementTypeFrameInDEV(
368357
init(payload),
369-
ownerFn,
370358
currentDispatcherRef,
371359
);
372360
} catch (x) {}

packages/react-devtools-shared/src/backend/DevToolsFiberComponentStack.js

+4-12
Original file line numberDiff line numberDiff line change
@@ -39,38 +39,30 @@ export function describeFiber(
3939
ClassComponent,
4040
} = workTagMap;
4141

42-
const owner: null | Function = __DEV__
43-
? workInProgress._debugOwner
44-
? workInProgress._debugOwner.type
45-
: null
46-
: null;
4742
switch (workInProgress.tag) {
4843
case HostComponent:
49-
return describeBuiltInComponentFrame(workInProgress.type, owner);
44+
return describeBuiltInComponentFrame(workInProgress.type);
5045
case LazyComponent:
51-
return describeBuiltInComponentFrame('Lazy', owner);
46+
return describeBuiltInComponentFrame('Lazy');
5247
case SuspenseComponent:
53-
return describeBuiltInComponentFrame('Suspense', owner);
48+
return describeBuiltInComponentFrame('Suspense');
5449
case SuspenseListComponent:
55-
return describeBuiltInComponentFrame('SuspenseList', owner);
50+
return describeBuiltInComponentFrame('SuspenseList');
5651
case FunctionComponent:
5752
case IndeterminateComponent:
5853
case SimpleMemoComponent:
5954
return describeFunctionComponentFrame(
6055
workInProgress.type,
61-
owner,
6256
currentDispatcherRef,
6357
);
6458
case ForwardRef:
6559
return describeFunctionComponentFrame(
6660
workInProgress.type.render,
67-
owner,
6861
currentDispatcherRef,
6962
);
7063
case ClassComponent:
7164
return describeClassComponentFrame(
7265
workInProgress.type,
73-
owner,
7466
currentDispatcherRef,
7567
);
7668
default:

packages/react-devtools-shared/src/backend/renderer.js

+35-18
Original file line numberDiff line numberDiff line change
@@ -1952,15 +1952,24 @@ export function attach(
19521952
const {key} = fiber;
19531953
const displayName = getDisplayNameForFiber(fiber);
19541954
const elementType = getElementTypeForFiber(fiber);
1955-
const {_debugOwner} = fiber;
1955+
const debugOwner = fiber._debugOwner;
19561956

19571957
// Ideally we should call getFiberIDThrows() for _debugOwner,
19581958
// since owners are almost always higher in the tree (and so have already been processed),
19591959
// but in some (rare) instances reported in open source, a descendant mounts before an owner.
19601960
// Since this is a DEV only field it's probably okay to also just lazily generate and ID here if needed.
19611961
// See https://github.com/facebook/react/issues/21445
1962-
const ownerID =
1963-
_debugOwner != null ? getOrGenerateFiberID(_debugOwner) : 0;
1962+
let ownerID: number;
1963+
if (debugOwner != null) {
1964+
if (typeof debugOwner.tag === 'number') {
1965+
ownerID = getOrGenerateFiberID((debugOwner: any));
1966+
} else {
1967+
// TODO: Track Server Component Owners.
1968+
ownerID = 0;
1969+
}
1970+
} else {
1971+
ownerID = 0;
1972+
}
19641973
const parentID = parentFiber ? getFiberIDThrows(parentFiber) : 0;
19651974

19661975
const displayNameStringID = getStringID(displayName);
@@ -3104,15 +3113,17 @@ export function attach(
31043113
return null;
31053114
}
31063115

3107-
const {_debugOwner} = fiber;
3108-
31093116
const owners: Array<SerializedElement> = [fiberToSerializedElement(fiber)];
31103117

3111-
if (_debugOwner) {
3112-
let owner: null | Fiber = _debugOwner;
3113-
while (owner !== null) {
3114-
owners.unshift(fiberToSerializedElement(owner));
3115-
owner = owner._debugOwner || null;
3118+
let owner = fiber._debugOwner;
3119+
while (owner != null) {
3120+
if (typeof owner.tag === 'number') {
3121+
const ownerFiber: Fiber = (owner: any); // Refined
3122+
owners.unshift(fiberToSerializedElement(ownerFiber));
3123+
owner = ownerFiber._debugOwner;
3124+
} else {
3125+
// TODO: Track Server Component Owners.
3126+
break;
31163127
}
31173128
}
31183129

@@ -3173,7 +3184,7 @@ export function attach(
31733184
}
31743185

31753186
const {
3176-
_debugOwner,
3187+
_debugOwner: debugOwner,
31773188
stateNode,
31783189
key,
31793190
memoizedProps,
@@ -3300,13 +3311,19 @@ export function attach(
33003311
context = {value: context};
33013312
}
33023313

3303-
let owners = null;
3304-
if (_debugOwner) {
3305-
owners = ([]: Array<SerializedElement>);
3306-
let owner: null | Fiber = _debugOwner;
3307-
while (owner !== null) {
3308-
owners.push(fiberToSerializedElement(owner));
3309-
owner = owner._debugOwner || null;
3314+
let owners: null | Array<SerializedElement> = null;
3315+
let owner = debugOwner;
3316+
while (owner != null) {
3317+
if (typeof owner.tag === 'number') {
3318+
const ownerFiber: Fiber = (owner: any); // Refined
3319+
if (owners === null) {
3320+
owners = [];
3321+
}
3322+
owners.push(fiberToSerializedElement(ownerFiber));
3323+
owner = ownerFiber._debugOwner;
3324+
} else {
3325+
// TODO: Track Server Component Owners.
3326+
break;
33103327
}
33113328
}
33123329

0 commit comments

Comments
 (0)