Skip to content

Commit 7142d11

Browse files
authored
Bugfix: Nested useOpaqueIdentifier references (#22553)
* Handle render phase updates explicitly We fire a warning in development if a component is updated during the render phase (with the exception of local hook updates, which have their own defined behavior). Because it's not a supported React pattern, we don't have that many tests that trigger this path. But it is meant to have reasonable semantics when it does happen, so that if it accidentally ships to production, the app doesn't crash unnecessarily. The behavior is not super well-defined, though. There are also some _internal_ React implementation details that intentionally to rely on this behavior. Most prominently, selective hydration and useOpaqueIdentifier. I need to tweak the behavior of render phase updates slightly as part of a fix for useOpaqueIdentifier. This shouldn't cause a user-facing change in behavior outside of useOpaqueIdentifier, but it does require that we explicitly model render phase updates. * Bugfix: Nested useOpaqueIdentifier calls Fixes an issue where multiple useOpaqueIdentifier hooks are upgraded to client ids within the same render. The way the upgrade works is that useOpaqueIdentifier schedules a render phase update then throws an error to trigger React's error recovery mechanism. The normal error recovery mechanism is designed for errors that occur as a result of interleaved mutations, so we usually only retry a single time, synchronously, before giving up. useOpaqueIdentifier is different because the error its throws when upgrading is not caused by an interleaved mutation. Rather, it happens when an ID is referenced for the first time inside a client-rendered tree (i.e. sommething that wasn't part of the initial server render). The fact that it relies on the error recovery mechanism is an implementation detail. And a single recovery attempt may be insufficient. For example, if a parent and a child component may reference different ids, and both are mounted as a result of the same client update, that will trigger two separate error recovery attempts. Because render phase updates are not allowed when triggered from userspace — we log a warning in developement to prevent them — we can assume that if something does update during the render phase, it is one of our "legit" implementation details like useOpaqueIdentifier. So we can keep retrying until we succeed — up to a limit, to protect against inifite loops. I chose 50 since that's the limit we use for commit phase updates.
1 parent 8ee4ff8 commit 7142d11

File tree

4 files changed

+314
-166
lines changed

4 files changed

+314
-166
lines changed

packages/react-dom/src/__tests__/ReactDOMServerIntegrationHooks-test.js

+35
Original file line numberDiff line numberDiff line change
@@ -1975,5 +1975,40 @@ describe('ReactDOMServerHooks', () => {
19751975
container.getElementsByTagName('span')[0].getAttribute('id'),
19761976
).not.toBeNull();
19771977
});
1978+
1979+
it('useOpaqueIdentifier with multiple ids in nested components', async () => {
1980+
function DivWithId({id, children}) {
1981+
return <div id={id}>{children}</div>;
1982+
}
1983+
1984+
let setShowMore;
1985+
function App() {
1986+
const outerId = useOpaqueIdentifier();
1987+
const innerId = useOpaqueIdentifier();
1988+
const [showMore, _setShowMore] = useState(false);
1989+
setShowMore = _setShowMore;
1990+
return showMore ? (
1991+
<DivWithId id={outerId}>
1992+
<DivWithId id={innerId} />
1993+
</DivWithId>
1994+
) : null;
1995+
}
1996+
1997+
const container = document.createElement('div');
1998+
container.innerHTML = ReactDOMServer.renderToString(<App />);
1999+
2000+
await act(async () => {
2001+
ReactDOM.hydrateRoot(container, <App />);
2002+
});
2003+
2004+
// Show additional content that wasn't part of the initial server-
2005+
// rendered repsonse.
2006+
await act(async () => {
2007+
setShowMore(true);
2008+
});
2009+
const [div1, div2] = container.getElementsByTagName('div');
2010+
expect(typeof div1.getAttribute('id')).toBe('string');
2011+
expect(typeof div2.getAttribute('id')).toBe('string');
2012+
});
19782013
});
19792014
});

packages/react-reconciler/src/ReactFiberWorkLoop.new.js

+110-83
Original file line numberDiff line numberDiff line change
@@ -294,7 +294,9 @@ let workInProgressRootIncludedLanes: Lanes = NoLanes;
294294
// includes unprocessed updates, not work in bailed out children.
295295
let workInProgressRootSkippedLanes: Lanes = NoLanes;
296296
// Lanes that were updated (in an interleaved event) during this render.
297-
let workInProgressRootUpdatedLanes: Lanes = NoLanes;
297+
let workInProgressRootInterleavedUpdatedLanes: Lanes = NoLanes;
298+
// Lanes that were updated during the render phase (*not* an interleaved event).
299+
let workInProgressRootRenderPhaseUpdatedLanes: Lanes = NoLanes;
298300
// Lanes that were pinged (in an interleaved event) during this render.
299301
let workInProgressRootPingedLanes: Lanes = NoLanes;
300302

@@ -454,86 +456,105 @@ export function scheduleUpdateOnFiber(
454456
eventTime: number,
455457
): FiberRoot | null {
456458
checkForNestedUpdates();
457-
warnAboutRenderPhaseUpdatesInDEV(fiber);
458459

459460
const root = markUpdateLaneFromFiberToRoot(fiber, lane);
460461
if (root === null) {
461462
return null;
462463
}
463464

464-
if (enableUpdaterTracking) {
465-
if (isDevToolsPresent) {
466-
addFiberToLanesMap(root, fiber, lane);
467-
}
468-
}
469-
470465
// Mark that the root has a pending update.
471466
markRootUpdated(root, lane, eventTime);
472467

473-
if (enableProfilerTimer && enableProfilerNestedUpdateScheduledHook) {
474-
if (
475-
(executionContext & CommitContext) !== NoContext &&
476-
root === rootCommittingMutationOrLayoutEffects
477-
) {
478-
if (fiber.mode & ProfileMode) {
479-
let current = fiber;
480-
while (current !== null) {
481-
if (current.tag === Profiler) {
482-
const {id, onNestedUpdateScheduled} = current.memoizedProps;
483-
if (typeof onNestedUpdateScheduled === 'function') {
484-
onNestedUpdateScheduled(id);
468+
if (
469+
(executionContext & RenderContext) !== NoLanes &&
470+
root === workInProgressRoot
471+
) {
472+
// This update was dispatched during the render phase. This is a mistake
473+
// if the update originates from user space (with the exception of local
474+
// hook updates, which are handled differently and don't reach this
475+
// function), but there are some internal React features that use this as
476+
// an implementation detail, like selective hydration
477+
// and useOpaqueIdentifier.
478+
warnAboutRenderPhaseUpdatesInDEV(fiber);
479+
480+
// Track lanes that were updated during the render phase
481+
workInProgressRootRenderPhaseUpdatedLanes = mergeLanes(
482+
workInProgressRootRenderPhaseUpdatedLanes,
483+
lane,
484+
);
485+
} else {
486+
// This is a normal update, scheduled from outside the render phase. For
487+
// example, during an input event.
488+
if (enableUpdaterTracking) {
489+
if (isDevToolsPresent) {
490+
addFiberToLanesMap(root, fiber, lane);
491+
}
492+
}
493+
494+
if (enableProfilerTimer && enableProfilerNestedUpdateScheduledHook) {
495+
if (
496+
(executionContext & CommitContext) !== NoContext &&
497+
root === rootCommittingMutationOrLayoutEffects
498+
) {
499+
if (fiber.mode & ProfileMode) {
500+
let current = fiber;
501+
while (current !== null) {
502+
if (current.tag === Profiler) {
503+
const {id, onNestedUpdateScheduled} = current.memoizedProps;
504+
if (typeof onNestedUpdateScheduled === 'function') {
505+
onNestedUpdateScheduled(id);
506+
}
485507
}
508+
current = current.return;
486509
}
487-
current = current.return;
488510
}
489511
}
490512
}
491-
}
492513

493-
// TODO: Consolidate with `isInterleavedUpdate` check
494-
if (root === workInProgressRoot) {
495-
// Received an update to a tree that's in the middle of rendering. Mark
496-
// that there was an interleaved update work on this root. Unless the
497-
// `deferRenderPhaseUpdateToNextBatch` flag is off and this is a render
498-
// phase update. In that case, we don't treat render phase updates as if
499-
// they were interleaved, for backwards compat reasons.
514+
// TODO: Consolidate with `isInterleavedUpdate` check
515+
if (root === workInProgressRoot) {
516+
// Received an update to a tree that's in the middle of rendering. Mark
517+
// that there was an interleaved update work on this root. Unless the
518+
// `deferRenderPhaseUpdateToNextBatch` flag is off and this is a render
519+
// phase update. In that case, we don't treat render phase updates as if
520+
// they were interleaved, for backwards compat reasons.
521+
if (
522+
deferRenderPhaseUpdateToNextBatch ||
523+
(executionContext & RenderContext) === NoContext
524+
) {
525+
workInProgressRootInterleavedUpdatedLanes = mergeLanes(
526+
workInProgressRootInterleavedUpdatedLanes,
527+
lane,
528+
);
529+
}
530+
if (workInProgressRootExitStatus === RootSuspendedWithDelay) {
531+
// The root already suspended with a delay, which means this render
532+
// definitely won't finish. Since we have a new update, let's mark it as
533+
// suspended now, right before marking the incoming update. This has the
534+
// effect of interrupting the current render and switching to the update.
535+
// TODO: Make sure this doesn't override pings that happen while we've
536+
// already started rendering.
537+
markRootSuspended(root, workInProgressRootRenderLanes);
538+
}
539+
}
540+
541+
ensureRootIsScheduled(root, eventTime);
500542
if (
501-
deferRenderPhaseUpdateToNextBatch ||
502-
(executionContext & RenderContext) === NoContext
543+
lane === SyncLane &&
544+
executionContext === NoContext &&
545+
(fiber.mode & ConcurrentMode) === NoMode &&
546+
// Treat `act` as if it's inside `batchedUpdates`, even in legacy mode.
547+
!(__DEV__ && ReactCurrentActQueue.isBatchingLegacy)
503548
) {
504-
workInProgressRootUpdatedLanes = mergeLanes(
505-
workInProgressRootUpdatedLanes,
506-
lane,
507-
);
508-
}
509-
if (workInProgressRootExitStatus === RootSuspendedWithDelay) {
510-
// The root already suspended with a delay, which means this render
511-
// definitely won't finish. Since we have a new update, let's mark it as
512-
// suspended now, right before marking the incoming update. This has the
513-
// effect of interrupting the current render and switching to the update.
514-
// TODO: Make sure this doesn't override pings that happen while we've
515-
// already started rendering.
516-
markRootSuspended(root, workInProgressRootRenderLanes);
549+
// Flush the synchronous work now, unless we're already working or inside
550+
// a batch. This is intentionally inside scheduleUpdateOnFiber instead of
551+
// scheduleCallbackForFiber to preserve the ability to schedule a callback
552+
// without immediately flushing it. We only do this for user-initiated
553+
// updates, to preserve historical behavior of legacy mode.
554+
resetRenderTimer();
555+
flushSyncCallbacksOnlyInLegacyMode();
517556
}
518557
}
519-
520-
ensureRootIsScheduled(root, eventTime);
521-
if (
522-
lane === SyncLane &&
523-
executionContext === NoContext &&
524-
(fiber.mode & ConcurrentMode) === NoMode &&
525-
// Treat `act` as if it's inside `batchedUpdates`, even in legacy mode.
526-
!(__DEV__ && ReactCurrentActQueue.isBatchingLegacy)
527-
) {
528-
// Flush the synchronous work now, unless we're already working or inside
529-
// a batch. This is intentionally inside scheduleUpdateOnFiber instead of
530-
// scheduleCallbackForFiber to preserve the ability to schedule a callback
531-
// without immediately flushing it. We only do this for user-initiated
532-
// updates, to preserve historical behavior of legacy mode.
533-
resetRenderTimer();
534-
flushSyncCallbacksOnlyInLegacyMode();
535-
}
536-
537558
return root;
538559
}
539560

@@ -865,7 +886,25 @@ function recoverFromConcurrentError(root, errorRetryLanes) {
865886
clearContainer(root.containerInfo);
866887
}
867888

868-
const exitStatus = renderRootSync(root, errorRetryLanes);
889+
let exitStatus;
890+
891+
const MAX_ERROR_RETRY_ATTEMPTS = 50;
892+
for (let i = 0; i < MAX_ERROR_RETRY_ATTEMPTS; i++) {
893+
exitStatus = renderRootSync(root, errorRetryLanes);
894+
if (
895+
exitStatus === RootErrored &&
896+
workInProgressRootRenderPhaseUpdatedLanes !== NoLanes
897+
) {
898+
// There was a render phase update during this render. This was likely a
899+
// useOpaqueIdentifier hook upgrading itself to a client ID. Try rendering
900+
// again. This time, the component will use a client ID and will proceed
901+
// without throwing. If multiple IDs upgrade as a result of the same
902+
// update, we will have to do multiple render passes. To protect against
903+
// an inifinite loop, eventually we'll give up.
904+
continue;
905+
}
906+
break;
907+
}
869908

870909
executionContext = prevExecutionContext;
871910

@@ -1042,7 +1081,10 @@ function markRootSuspended(root, suspendedLanes) {
10421081
// TODO: Lol maybe there's a better way to factor this besides this
10431082
// obnoxiously named function :)
10441083
suspendedLanes = removeLanes(suspendedLanes, workInProgressRootPingedLanes);
1045-
suspendedLanes = removeLanes(suspendedLanes, workInProgressRootUpdatedLanes);
1084+
suspendedLanes = removeLanes(
1085+
suspendedLanes,
1086+
workInProgressRootInterleavedUpdatedLanes,
1087+
);
10461088
markRootSuspended_dontCallThisOneDirectly(root, suspendedLanes);
10471089
}
10481090

@@ -1068,30 +1110,15 @@ function performSyncWorkOnRoot(root) {
10681110

10691111
let exitStatus = renderRootSync(root, lanes);
10701112
if (root.tag !== LegacyRoot && exitStatus === RootErrored) {
1071-
const prevExecutionContext = executionContext;
1072-
executionContext |= RetryAfterError;
1073-
1074-
// If an error occurred during hydration,
1075-
// discard server response and fall back to client side render.
1076-
if (root.isDehydrated) {
1077-
root.isDehydrated = false;
1078-
if (__DEV__) {
1079-
errorHydratingContainer(root.containerInfo);
1080-
}
1081-
clearContainer(root.containerInfo);
1082-
}
1083-
10841113
// If something threw an error, try rendering one more time. We'll render
10851114
// synchronously to block concurrent data mutations, and we'll includes
10861115
// all pending updates are included. If it still fails after the second
10871116
// attempt, we'll give up and commit the resulting tree.
10881117
const errorRetryLanes = getLanesToRetrySynchronouslyOnError(root);
10891118
if (errorRetryLanes !== NoLanes) {
10901119
lanes = errorRetryLanes;
1091-
exitStatus = renderRootSync(root, lanes);
1120+
exitStatus = recoverFromConcurrentError(root, errorRetryLanes);
10921121
}
1093-
1094-
executionContext = prevExecutionContext;
10951122
}
10961123

10971124
if (exitStatus === RootFatalErrored) {
@@ -1300,7 +1327,8 @@ function prepareFreshStack(root: FiberRoot, lanes: Lanes) {
13001327
workInProgressRootExitStatus = RootIncomplete;
13011328
workInProgressRootFatalError = null;
13021329
workInProgressRootSkippedLanes = NoLanes;
1303-
workInProgressRootUpdatedLanes = NoLanes;
1330+
workInProgressRootInterleavedUpdatedLanes = NoLanes;
1331+
workInProgressRootRenderPhaseUpdatedLanes = NoLanes;
13041332
workInProgressRootPingedLanes = NoLanes;
13051333

13061334
enqueueInterleavedUpdates();
@@ -1443,7 +1471,7 @@ export function renderDidSuspendDelayIfPossible(): void {
14431471
if (
14441472
workInProgressRoot !== null &&
14451473
(includesNonIdleWork(workInProgressRootSkippedLanes) ||
1446-
includesNonIdleWork(workInProgressRootUpdatedLanes))
1474+
includesNonIdleWork(workInProgressRootInterleavedUpdatedLanes))
14471475
) {
14481476
// Mark the current render as suspended so that we switch to working on
14491477
// the updates that were skipped. Usually we only suspend at the end of
@@ -2697,7 +2725,6 @@ function warnAboutRenderPhaseUpdatesInDEV(fiber) {
26972725
if (__DEV__) {
26982726
if (
26992727
ReactCurrentDebugFiberIsRenderingInDEV &&
2700-
(executionContext & RenderContext) !== NoContext &&
27012728
!getIsUpdatingOpaqueValueInRenderPhaseInDEV()
27022729
) {
27032730
switch (fiber.tag) {

0 commit comments

Comments
 (0)