Skip to content

Commit

Permalink
Track Owner for Server Components in DEV (#28753)
Browse files Browse the repository at this point in the history
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.

DiffTrain build for commit e0455fe.
  • Loading branch information
kassens committed Apr 11, 2024
1 parent 47691a3 commit 812db07
Show file tree
Hide file tree
Showing 13 changed files with 361 additions and 434 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<3151ca79177b1e7b421f14db372bfd97>>
* @generated SignedSource<<568cb702cdaad263cf51f5b74157b2ba>>
*/

"use strict";
Expand Down Expand Up @@ -349,6 +349,17 @@ if (__DEV__) {
return type.displayName || "Context";
}

function getComponentNameFromOwner(owner) {
if (typeof owner.tag === "number") {
return getComponentNameFromFiber(owner);
}

if (typeof owner.name === "string") {
return owner.name;
}

return null;
}
function getComponentNameFromFiber(fiber) {
var tag = fiber.tag,
type = fiber.type;
Expand Down Expand Up @@ -2881,21 +2892,14 @@ if (__DEV__) {

var objectIs = typeof Object.is === "function" ? Object.is : is; // $FlowFixMe[method-unbinding]

function describeBuiltInComponentFrame(name, ownerFn) {
function describeBuiltInComponentFrame(name) {
{
var ownerName = null;

if (ownerFn) {
ownerName = ownerFn.displayName || ownerFn.name || null;
}

return describeComponentFrame(name, ownerName);
return describeComponentFrame(name);
}
}
function describeDebugInfoFrame(name, env) {
return describeBuiltInComponentFrame(
name + (env ? " (" + env + ")" : ""),
null
name + (env ? " (" + env + ")" : "")
);
}

Expand All @@ -2904,65 +2908,51 @@ if (__DEV__) {
new PossiblyWeakMap$1();
}

function describeComponentFrame(name, ownerName) {
var sourceInfo = "";

if (ownerName) {
sourceInfo = " (created by " + ownerName + ")";
}

return "\n in " + (name || "Unknown") + sourceInfo;
function describeComponentFrame(name) {
return "\n in " + (name || "Unknown");
}

function describeClassComponentFrame(ctor, ownerFn) {
function describeClassComponentFrame(ctor) {
{
return describeFunctionComponentFrame(ctor, ownerFn);
return describeFunctionComponentFrame(ctor);
}
}
function describeFunctionComponentFrame(fn, ownerFn) {
function describeFunctionComponentFrame(fn) {
{
if (!fn) {
return "";
}

var name = fn.displayName || fn.name || null;
var ownerName = null;

if (ownerFn) {
ownerName = ownerFn.displayName || ownerFn.name || null;
}

return describeComponentFrame(name, ownerName);
return describeComponentFrame(name);
}
}

function describeFiber(fiber) {
var owner = fiber._debugOwner ? fiber._debugOwner.type : null;

switch (fiber.tag) {
case HostHoistable:
case HostSingleton:
case HostComponent:
return describeBuiltInComponentFrame(fiber.type, owner);
return describeBuiltInComponentFrame(fiber.type);

case LazyComponent:
return describeBuiltInComponentFrame("Lazy", owner);
return describeBuiltInComponentFrame("Lazy");

case SuspenseComponent:
return describeBuiltInComponentFrame("Suspense", owner);
return describeBuiltInComponentFrame("Suspense");

case SuspenseListComponent:
return describeBuiltInComponentFrame("SuspenseList", owner);
return describeBuiltInComponentFrame("SuspenseList");

case FunctionComponent:
case SimpleMemoComponent:
return describeFunctionComponentFrame(fiber.type, owner);
return describeFunctionComponentFrame(fiber.type);

case ForwardRef:
return describeFunctionComponentFrame(fiber.type.render, owner);
return describeFunctionComponentFrame(fiber.type.render);

case ClassComponent:
return describeClassComponentFrame(fiber.type, owner);
return describeClassComponentFrame(fiber.type);

default:
return "";
Expand Down Expand Up @@ -5253,8 +5243,8 @@ if (__DEV__) {

var owner = current._debugOwner;

if (owner !== null && typeof owner !== "undefined") {
return getComponentNameFromFiber(owner);
if (owner != null) {
return getComponentNameFromOwner(owner);
}
}

Expand Down Expand Up @@ -26560,7 +26550,7 @@ if (__DEV__) {
"named imports.";
}

var ownerName = owner ? getComponentNameFromFiber(owner) : null;
var ownerName = owner ? getComponentNameFromOwner(owner) : null;

if (ownerName) {
info += "\n\nCheck the render method of `" + ownerName + "`.";
Expand Down Expand Up @@ -26823,7 +26813,7 @@ if (__DEV__) {
return root;
}

var ReactVersion = "19.0.0-canary-29bd6113";
var ReactVersion = "19.0.0-canary-6077cfa0";

// Might add PROFILE later.

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<432e9d2dc4a6b45da0162dd89f63ecfc>>
* @generated SignedSource<<edf13283385b2e5a6dba5a299a226ed0>>
*/

"use strict";
Expand Down Expand Up @@ -654,28 +654,24 @@ function is(x, y) {
return (x === y && (0 !== x || 1 / x === 1 / y)) || (x !== x && y !== y);
}
var objectIs = "function" === typeof Object.is ? Object.is : is;
function describeComponentFrame(name, ownerName) {
var sourceInfo = "";
ownerName && (sourceInfo = " (created by " + ownerName + ")");
return "\n in " + (name || "Unknown") + sourceInfo;
function describeBuiltInComponentFrame(name) {
return "\n in " + (name || "Unknown");
}
function describeFunctionComponentFrame(fn) {
return fn
? describeComponentFrame(fn.displayName || fn.name || null, null)
: "";
return fn ? "\n in " + (fn.displayName || fn.name || "Unknown") : "";
}
function describeFiber(fiber) {
switch (fiber.tag) {
case 26:
case 27:
case 5:
return describeComponentFrame(fiber.type, null);
return describeBuiltInComponentFrame(fiber.type);
case 16:
return describeComponentFrame("Lazy", null);
return describeBuiltInComponentFrame("Lazy");
case 13:
return describeComponentFrame("Suspense", null);
return describeBuiltInComponentFrame("Suspense");
case 19:
return describeComponentFrame("SuspenseList", null);
return describeBuiltInComponentFrame("SuspenseList");
case 0:
case 15:
return describeFunctionComponentFrame(fiber.type);
Expand Down Expand Up @@ -9211,19 +9207,19 @@ function wrapFiber(fiber) {
fiberToWrapper.set(fiber, wrapper));
return wrapper;
}
var devToolsConfig$jscomp$inline_995 = {
var devToolsConfig$jscomp$inline_994 = {
findFiberByHostInstance: function () {
throw Error("TestRenderer does not support findFiberByHostInstance()");
},
bundleType: 0,
version: "19.0.0-canary-0689007c",
version: "19.0.0-canary-51953c35",
rendererPackageName: "react-test-renderer"
};
var internals$jscomp$inline_1185 = {
bundleType: devToolsConfig$jscomp$inline_995.bundleType,
version: devToolsConfig$jscomp$inline_995.version,
rendererPackageName: devToolsConfig$jscomp$inline_995.rendererPackageName,
rendererConfig: devToolsConfig$jscomp$inline_995.rendererConfig,
var internals$jscomp$inline_1184 = {
bundleType: devToolsConfig$jscomp$inline_994.bundleType,
version: devToolsConfig$jscomp$inline_994.version,
rendererPackageName: devToolsConfig$jscomp$inline_994.rendererPackageName,
rendererConfig: devToolsConfig$jscomp$inline_994.rendererConfig,
overrideHookState: null,
overrideHookStateDeletePath: null,
overrideHookStateRenamePath: null,
Expand All @@ -9240,26 +9236,26 @@ var internals$jscomp$inline_1185 = {
return null === fiber ? null : fiber.stateNode;
},
findFiberByHostInstance:
devToolsConfig$jscomp$inline_995.findFiberByHostInstance ||
devToolsConfig$jscomp$inline_994.findFiberByHostInstance ||
emptyFindFiberByHostInstance,
findHostInstancesForRefresh: null,
scheduleRefresh: null,
scheduleRoot: null,
setRefreshHandler: null,
getCurrentFiber: null,
reconcilerVersion: "19.0.0-canary-0689007c"
reconcilerVersion: "19.0.0-canary-51953c35"
};
if ("undefined" !== typeof __REACT_DEVTOOLS_GLOBAL_HOOK__) {
var hook$jscomp$inline_1186 = __REACT_DEVTOOLS_GLOBAL_HOOK__;
var hook$jscomp$inline_1185 = __REACT_DEVTOOLS_GLOBAL_HOOK__;
if (
!hook$jscomp$inline_1186.isDisabled &&
hook$jscomp$inline_1186.supportsFiber
!hook$jscomp$inline_1185.isDisabled &&
hook$jscomp$inline_1185.supportsFiber
)
try {
(rendererID = hook$jscomp$inline_1186.inject(
internals$jscomp$inline_1185
(rendererID = hook$jscomp$inline_1185.inject(
internals$jscomp$inline_1184
)),
(injectedHook = hook$jscomp$inline_1186);
(injectedHook = hook$jscomp$inline_1185);
} catch (err) {}
}
exports._Scheduler = Scheduler;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<c03b4f2580a3f28e5204dac43fdc8e39>>
* @generated SignedSource<<ab4a7efab0297a08768bdea5ec49f164>>
*/

"use strict";
Expand Down Expand Up @@ -742,28 +742,24 @@ function is(x, y) {
return (x === y && (0 !== x || 1 / x === 1 / y)) || (x !== x && y !== y);
}
var objectIs = "function" === typeof Object.is ? Object.is : is;
function describeComponentFrame(name, ownerName) {
var sourceInfo = "";
ownerName && (sourceInfo = " (created by " + ownerName + ")");
return "\n in " + (name || "Unknown") + sourceInfo;
function describeBuiltInComponentFrame(name) {
return "\n in " + (name || "Unknown");
}
function describeFunctionComponentFrame(fn) {
return fn
? describeComponentFrame(fn.displayName || fn.name || null, null)
: "";
return fn ? "\n in " + (fn.displayName || fn.name || "Unknown") : "";
}
function describeFiber(fiber) {
switch (fiber.tag) {
case 26:
case 27:
case 5:
return describeComponentFrame(fiber.type, null);
return describeBuiltInComponentFrame(fiber.type);
case 16:
return describeComponentFrame("Lazy", null);
return describeBuiltInComponentFrame("Lazy");
case 13:
return describeComponentFrame("Suspense", null);
return describeBuiltInComponentFrame("Suspense");
case 19:
return describeComponentFrame("SuspenseList", null);
return describeBuiltInComponentFrame("SuspenseList");
case 0:
case 15:
return describeFunctionComponentFrame(fiber.type);
Expand Down Expand Up @@ -9827,12 +9823,12 @@ function wrapFiber(fiber) {
fiberToWrapper.set(fiber, wrapper));
return wrapper;
}
var devToolsConfig$jscomp$inline_1079 = {
var devToolsConfig$jscomp$inline_1078 = {
findFiberByHostInstance: function () {
throw Error("TestRenderer does not support findFiberByHostInstance()");
},
bundleType: 0,
version: "19.0.0-canary-3aa3c53d",
version: "19.0.0-canary-c0ff0c26",
rendererPackageName: "react-test-renderer"
};
(function (internals) {
Expand All @@ -9849,10 +9845,10 @@ var devToolsConfig$jscomp$inline_1079 = {
} catch (err) {}
return hook.checkDCE ? !0 : !1;
})({
bundleType: devToolsConfig$jscomp$inline_1079.bundleType,
version: devToolsConfig$jscomp$inline_1079.version,
rendererPackageName: devToolsConfig$jscomp$inline_1079.rendererPackageName,
rendererConfig: devToolsConfig$jscomp$inline_1079.rendererConfig,
bundleType: devToolsConfig$jscomp$inline_1078.bundleType,
version: devToolsConfig$jscomp$inline_1078.version,
rendererPackageName: devToolsConfig$jscomp$inline_1078.rendererPackageName,
rendererConfig: devToolsConfig$jscomp$inline_1078.rendererConfig,
overrideHookState: null,
overrideHookStateDeletePath: null,
overrideHookStateRenamePath: null,
Expand All @@ -9869,14 +9865,14 @@ var devToolsConfig$jscomp$inline_1079 = {
return null === fiber ? null : fiber.stateNode;
},
findFiberByHostInstance:
devToolsConfig$jscomp$inline_1079.findFiberByHostInstance ||
devToolsConfig$jscomp$inline_1078.findFiberByHostInstance ||
emptyFindFiberByHostInstance,
findHostInstancesForRefresh: null,
scheduleRefresh: null,
scheduleRoot: null,
setRefreshHandler: null,
getCurrentFiber: null,
reconcilerVersion: "19.0.0-canary-3aa3c53d"
reconcilerVersion: "19.0.0-canary-c0ff0c26"
});
exports._Scheduler = Scheduler;
exports.act = act;
Expand Down
Loading

0 comments on commit 812db07

Please sign in to comment.