diff --git a/packages/react-noop-renderer/src/createReactNoop.js b/packages/react-noop-renderer/src/createReactNoop.js index 58cc128685d5e..a452a391f28d5 100644 --- a/packages/react-noop-renderer/src/createReactNoop.js +++ b/packages/react-noop-renderer/src/createReactNoop.js @@ -534,6 +534,13 @@ function createReactNoop(reconciler: Function, useMutation: boolean) { newChildren: Array, ): void { container.pendingChildren = newChildren; + if ( + newChildren.length === 1 && + newChildren[0].text === 'Error when completing root' + ) { + // Trigger an error for testing purposes + throw Error('Error when completing root'); + } }, replaceContainerChildren( diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.js b/packages/react-reconciler/src/ReactFiberWorkLoop.js index a4664613ac224..5d92eb8ad3833 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.js @@ -1291,6 +1291,13 @@ function handleError(root, thrownValue) { // boundary. workInProgressRootExitStatus = RootFatalErrored; workInProgressRootFatalError = thrownValue; + // Set `workInProgress` to null. This represents advancing to the next + // sibling, or the parent if there are no siblings. But since the root + // has no siblings nor a parent, we set it to null. Usually this is + // handled by `completeUnitOfWork` or `unwindWork`, but since we're + // interntionally not calling those, we need set it here. + // TODO: Consider calling `unwindWork` to pop the contexts. + workInProgress = null; return null; } diff --git a/packages/react-reconciler/src/__tests__/ReactIncrementalErrorHandling-test.internal.js b/packages/react-reconciler/src/__tests__/ReactIncrementalErrorHandling-test.internal.js index bcb74b6ea07f8..6bea104b78db5 100644 --- a/packages/react-reconciler/src/__tests__/ReactIncrementalErrorHandling-test.internal.js +++ b/packages/react-reconciler/src/__tests__/ReactIncrementalErrorHandling-test.internal.js @@ -1658,4 +1658,16 @@ describe('ReactIncrementalErrorHandling', () => { 'Please update the following components: Provider', ]); }); + + if (global.__PERSISTENT__) { + it('regression test: should fatal if error is thrown at the root', () => { + const root = ReactNoop.createRoot(); + root.render('Error when completing root'); + expect(Scheduler).toFlushAndThrow('Error when completing root'); + + const blockingRoot = ReactNoop.createBlockingRoot(); + blockingRoot.render('Error when completing root'); + expect(Scheduler).toFlushAndThrow('Error when completing root'); + }); + } });