From 5ce16dfb36f172b730d2aa63db76d397c1bd1851 Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Mon, 24 Oct 2022 18:20:10 -0700 Subject: [PATCH] Don't unwind stack while suspended, when possible Initial draft. I need to test this more. If a promise is passed to `use`, but the I/O isn't cached, we should still be able to unwrap it. This already worked in Server Components, and during SSR. For Fiber (in the browser), before this fix the state would get lost between attempts unless the promise resolved immediately in a microtask, which requires IO to be cached. This was due to an implementation quirk of Fiber where the state is reset as soon as the stack unwinds. The workaround is to suspend the entire Fiber work loop until the promise resolves. The Server Components and SSR runtimes don't require a workaround: they can maintain multiple parallel child tasks and reuse the state indefinitely across attempts. That's ideally how Fiber should work, too, but it will require larger refactor. The downside of our approach in Fiber is that it won't "warm up" the siblings while you're suspended, but to avoid waterfalls you're supposed to hoist data fetches higher in the tree regardless. But we have other ideas for how we can add this back in the future. (Though again, this doesn't affect Server Components, which already have the ideal behavior.) --- .../src/ReactFiberWorkLoop.new.js | 127 ++++++++++++-- .../src/ReactFiberWorkLoop.old.js | 127 ++++++++++++-- .../src/__tests__/ReactThenable-test.js | 162 ++++++++++++++++++ 3 files changed, 386 insertions(+), 30 deletions(-) diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.new.js b/packages/react-reconciler/src/ReactFiberWorkLoop.new.js index ce03757caa4e7..d2da78c8fbb8c 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.new.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.new.js @@ -9,10 +9,13 @@ import {REACT_STRICT_MODE_TYPE} from 'shared/ReactSymbols'; -import type {Wakeable} from 'shared/ReactTypes'; +import type {Wakeable, Thenable} from 'shared/ReactTypes'; import type {Fiber, FiberRoot} from './ReactInternalTypes'; import type {Lanes, Lane} from './ReactFiberLane.new'; -import type {SuspenseState} from './ReactFiberSuspenseComponent.new'; +import type { + SuspenseProps, + SuspenseState, +} from './ReactFiberSuspenseComponent.new'; import type {FunctionComponentUpdateQueue} from './ReactFiberHooks.new'; import type {EventPriority} from './ReactEventPriorities.new'; import type { @@ -272,6 +275,10 @@ import { isThenableStateResolved, } from './ReactFiberThenable.new'; import {schedulePostPaintCallback} from './ReactPostPaintCallback'; +import { + getSuspenseHandler, + isBadSuspenseFallback, +} from './ReactFiberSuspenseContext.new'; const ceil = Math.ceil; @@ -313,7 +320,7 @@ let workInProgressRootRenderLanes: Lanes = NoLanes; opaque type SuspendedReason = 0 | 1 | 2 | 3 | 4; const NotSuspended: SuspendedReason = 0; const SuspendedOnError: SuspendedReason = 1; -// const SuspendedOnData: SuspendedReason = 2; +const SuspendedOnData: SuspendedReason = 2; const SuspendedOnImmediate: SuspendedReason = 3; const SuspendedAndReadyToUnwind: SuspendedReason = 4; @@ -707,6 +714,18 @@ export function scheduleUpdateOnFiber( } } + // Check if the work loop is currently suspended and waiting for data to + // finish loading. + if ( + workInProgressSuspendedReason === SuspendedOnData && + root === workInProgressRoot + ) { + // The incoming update might unblock the current render. Interrupt the + // current attempt and restart from the top. + prepareFreshStack(root, NoLanes); + markRootSuspended(root, workInProgressRootRenderLanes); + } + // Mark that the root has a pending update. markRootUpdated(root, lane, eventTime); @@ -1131,6 +1150,17 @@ function performConcurrentWorkOnRoot(root, didTimeout) { if (root.callbackNode === originalCallbackNode) { // The task node scheduled for this root is the same one that's // currently executed. Need to return a continuation. + if (workInProgressSuspendedReason === SuspendedOnData) { + // Special case: The work loop is currently suspended and waiting for + // data to resolve. Unschedule the current task. + // + // TODO: The factoring is a little weird. Arguably this should be checked + // in ensureRootIsScheduled instead. I went back and forth, not totally + // sure yet. + root.callbackPriority = NoLane; + root.callbackNode = null; + return null; + } return performConcurrentWorkOnRoot.bind(null, root); } return null; @@ -1740,7 +1770,9 @@ function handleThrow(root, thrownValue): void { // deprecate the old API in favor of `use`. thrownValue = getSuspendedThenable(); workInProgressSuspendedThenableState = getThenableStateAfterSuspending(); - workInProgressSuspendedReason = SuspendedOnImmediate; + workInProgressSuspendedReason = shouldAttemptToSuspendUntilDataResolves() + ? SuspendedOnData + : SuspendedOnImmediate; } else { // This is a regular error. If something earlier in the component already // suspended, we must clear the thenable state to unblock the work loop. @@ -1797,6 +1829,48 @@ function handleThrow(root, thrownValue): void { } } +function shouldAttemptToSuspendUntilDataResolves() { + // TODO: We should be able to move the + // renderDidSuspend/renderDidSuspendWithDelay logic into this function, + // instead of repeating it in the complete phase. Or something to that effect. + + if (includesOnlyRetries(workInProgressRootRenderLanes)) { + // We can always wait during a retry. + return true; + } + + // TODO: We should be able to remove the equivalent check in + // finishConcurrentRender, and rely just on this one. + if (includesOnlyTransitions(workInProgressRootRenderLanes)) { + const suspenseHandler = getSuspenseHandler(); + if (suspenseHandler !== null && suspenseHandler.tag === SuspenseComponent) { + const currentSuspenseHandler = suspenseHandler.alternate; + const nextProps: SuspenseProps = suspenseHandler.memoizedProps; + if (isBadSuspenseFallback(currentSuspenseHandler, nextProps)) { + // The nearest Suspense boundary is already showing content. We should + // avoid replacing it with a fallback, and instead wait until the + // data finishes loading. + return true; + } else { + // This is not a bad fallback condition. We should show a fallback + // immediately instead of waiting for the data to resolve. This includes + // when suspending inside new trees. + return false; + } + } + + // During a transition, if there is no Suspense boundary (i.e. suspending in + // the "shell" of an application), or if we're inside a hidden tree, then + // we should wait until the data finishes loading. + return true; + } + + // For all other Lanes besides Transitions and Retries, we should not wait + // for the data to load. + // TODO: We should wait during Offscreen prerendering, too. + return false; +} + function pushDispatcher(container) { prepareRendererToRender(container); const prevDispatcher = ReactCurrentDispatcher.current; @@ -2061,7 +2135,7 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { markRenderStarted(lanes); } - do { + outer: do { try { if ( workInProgressSuspendedReason !== NotSuspended && @@ -2071,19 +2145,48 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { // replay the suspended component. const unitOfWork = workInProgress; const thrownValue = workInProgressThrownValue; - workInProgressSuspendedReason = NotSuspended; - workInProgressThrownValue = null; switch (workInProgressSuspendedReason) { case SuspendedOnError: { // Unwind then continue with the normal work loop. + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; unwindSuspendedUnitOfWork(unitOfWork, thrownValue); break; } + case SuspendedOnData: { + const didResolve = + workInProgressSuspendedThenableState !== null && + isThenableStateResolved(workInProgressSuspendedThenableState); + if (didResolve) { + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; + replaySuspendedUnitOfWork(unitOfWork, thrownValue); + } else { + // The work loop is suspended on data. We should wait for it to + // resolve before continuing to render. + const thenable: Thenable = (workInProgressThrownValue: any); + const onResolution = () => { + ensureRootIsScheduled(root, now()); + }; + thenable.then(onResolution, onResolution); + break outer; + } + break; + } + case SuspendedOnImmediate: { + // If this fiber just suspended, it's possible the data is already + // cached. Yield to the main thread to give it a chance to ping. If + // it does, we can retry immediately without unwinding the stack. + workInProgressSuspendedReason = SuspendedAndReadyToUnwind; + break outer; + } default: { - const wasPinged = + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; + const didResolve = workInProgressSuspendedThenableState !== null && isThenableStateResolved(workInProgressSuspendedThenableState); - if (wasPinged) { + if (didResolve) { replaySuspendedUnitOfWork(unitOfWork, thrownValue); } else { unwindSuspendedUnitOfWork(unitOfWork, thrownValue); @@ -2097,12 +2200,6 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { break; } catch (thrownValue) { handleThrow(root, thrownValue); - if (workInProgressSuspendedThenableState !== null) { - // If this fiber just suspended, it's possible the data is already - // cached. Yield to the main thread to give it a chance to ping. If - // it does, we can retry immediately without unwinding the stack. - break; - } } } while (true); resetContextDependencies(); diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.old.js b/packages/react-reconciler/src/ReactFiberWorkLoop.old.js index 444b8129c0db5..f0e1b734c300f 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.old.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.old.js @@ -9,10 +9,13 @@ import {REACT_STRICT_MODE_TYPE} from 'shared/ReactSymbols'; -import type {Wakeable} from 'shared/ReactTypes'; +import type {Wakeable, Thenable} from 'shared/ReactTypes'; import type {Fiber, FiberRoot} from './ReactInternalTypes'; import type {Lanes, Lane} from './ReactFiberLane.old'; -import type {SuspenseState} from './ReactFiberSuspenseComponent.old'; +import type { + SuspenseProps, + SuspenseState, +} from './ReactFiberSuspenseComponent.old'; import type {FunctionComponentUpdateQueue} from './ReactFiberHooks.old'; import type {EventPriority} from './ReactEventPriorities.old'; import type { @@ -272,6 +275,10 @@ import { isThenableStateResolved, } from './ReactFiberThenable.old'; import {schedulePostPaintCallback} from './ReactPostPaintCallback'; +import { + getSuspenseHandler, + isBadSuspenseFallback, +} from './ReactFiberSuspenseContext.old'; const ceil = Math.ceil; @@ -313,7 +320,7 @@ let workInProgressRootRenderLanes: Lanes = NoLanes; opaque type SuspendedReason = 0 | 1 | 2 | 3 | 4; const NotSuspended: SuspendedReason = 0; const SuspendedOnError: SuspendedReason = 1; -// const SuspendedOnData: SuspendedReason = 2; +const SuspendedOnData: SuspendedReason = 2; const SuspendedOnImmediate: SuspendedReason = 3; const SuspendedAndReadyToUnwind: SuspendedReason = 4; @@ -707,6 +714,18 @@ export function scheduleUpdateOnFiber( } } + // Check if the work loop is currently suspended and waiting for data to + // finish loading. + if ( + workInProgressSuspendedReason === SuspendedOnData && + root === workInProgressRoot + ) { + // The incoming update might unblock the current render. Interrupt the + // current attempt and restart from the top. + prepareFreshStack(root, NoLanes); + markRootSuspended(root, workInProgressRootRenderLanes); + } + // Mark that the root has a pending update. markRootUpdated(root, lane, eventTime); @@ -1131,6 +1150,17 @@ function performConcurrentWorkOnRoot(root, didTimeout) { if (root.callbackNode === originalCallbackNode) { // The task node scheduled for this root is the same one that's // currently executed. Need to return a continuation. + if (workInProgressSuspendedReason === SuspendedOnData) { + // Special case: The work loop is currently suspended and waiting for + // data to resolve. Unschedule the current task. + // + // TODO: The factoring is a little weird. Arguably this should be checked + // in ensureRootIsScheduled instead. I went back and forth, not totally + // sure yet. + root.callbackPriority = NoLane; + root.callbackNode = null; + return null; + } return performConcurrentWorkOnRoot.bind(null, root); } return null; @@ -1740,7 +1770,9 @@ function handleThrow(root, thrownValue): void { // deprecate the old API in favor of `use`. thrownValue = getSuspendedThenable(); workInProgressSuspendedThenableState = getThenableStateAfterSuspending(); - workInProgressSuspendedReason = SuspendedOnImmediate; + workInProgressSuspendedReason = shouldAttemptToSuspendUntilDataResolves() + ? SuspendedOnData + : SuspendedOnImmediate; } else { // This is a regular error. If something earlier in the component already // suspended, we must clear the thenable state to unblock the work loop. @@ -1797,6 +1829,48 @@ function handleThrow(root, thrownValue): void { } } +function shouldAttemptToSuspendUntilDataResolves() { + // TODO: We should be able to move the + // renderDidSuspend/renderDidSuspendWithDelay logic into this function, + // instead of repeating it in the complete phase. Or something to that effect. + + if (includesOnlyRetries(workInProgressRootRenderLanes)) { + // We can always wait during a retry. + return true; + } + + // TODO: We should be able to remove the equivalent check in + // finishConcurrentRender, and rely just on this one. + if (includesOnlyTransitions(workInProgressRootRenderLanes)) { + const suspenseHandler = getSuspenseHandler(); + if (suspenseHandler !== null && suspenseHandler.tag === SuspenseComponent) { + const currentSuspenseHandler = suspenseHandler.alternate; + const nextProps: SuspenseProps = suspenseHandler.memoizedProps; + if (isBadSuspenseFallback(currentSuspenseHandler, nextProps)) { + // The nearest Suspense boundary is already showing content. We should + // avoid replacing it with a fallback, and instead wait until the + // data finishes loading. + return true; + } else { + // This is not a bad fallback condition. We should show a fallback + // immediately instead of waiting for the data to resolve. This includes + // when suspending inside new trees. + return false; + } + } + + // During a transition, if there is no Suspense boundary (i.e. suspending in + // the "shell" of an application), or if we're inside a hidden tree, then + // we should wait until the data finishes loading. + return true; + } + + // For all other Lanes besides Transitions and Retries, we should not wait + // for the data to load. + // TODO: We should wait during Offscreen prerendering, too. + return false; +} + function pushDispatcher(container) { prepareRendererToRender(container); const prevDispatcher = ReactCurrentDispatcher.current; @@ -2061,7 +2135,7 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { markRenderStarted(lanes); } - do { + outer: do { try { if ( workInProgressSuspendedReason !== NotSuspended && @@ -2071,19 +2145,48 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { // replay the suspended component. const unitOfWork = workInProgress; const thrownValue = workInProgressThrownValue; - workInProgressSuspendedReason = NotSuspended; - workInProgressThrownValue = null; switch (workInProgressSuspendedReason) { case SuspendedOnError: { // Unwind then continue with the normal work loop. + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; unwindSuspendedUnitOfWork(unitOfWork, thrownValue); break; } + case SuspendedOnData: { + const didResolve = + workInProgressSuspendedThenableState !== null && + isThenableStateResolved(workInProgressSuspendedThenableState); + if (didResolve) { + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; + replaySuspendedUnitOfWork(unitOfWork, thrownValue); + } else { + // The work loop is suspended on data. We should wait for it to + // resolve before continuing to render. + const thenable: Thenable = (workInProgressThrownValue: any); + const onResolution = () => { + ensureRootIsScheduled(root, now()); + }; + thenable.then(onResolution, onResolution); + break outer; + } + break; + } + case SuspendedOnImmediate: { + // If this fiber just suspended, it's possible the data is already + // cached. Yield to the main thread to give it a chance to ping. If + // it does, we can retry immediately without unwinding the stack. + workInProgressSuspendedReason = SuspendedAndReadyToUnwind; + break outer; + } default: { - const wasPinged = + workInProgressSuspendedReason = NotSuspended; + workInProgressThrownValue = null; + const didResolve = workInProgressSuspendedThenableState !== null && isThenableStateResolved(workInProgressSuspendedThenableState); - if (wasPinged) { + if (didResolve) { replaySuspendedUnitOfWork(unitOfWork, thrownValue); } else { unwindSuspendedUnitOfWork(unitOfWork, thrownValue); @@ -2097,12 +2200,6 @@ function renderRootConcurrent(root: FiberRoot, lanes: Lanes) { break; } catch (thrownValue) { handleThrow(root, thrownValue); - if (workInProgressSuspendedThenableState !== null) { - // If this fiber just suspended, it's possible the data is already - // cached. Yield to the main thread to give it a chance to ping. If - // it does, we can retry immediately without unwinding the stack. - break; - } } } while (true); resetContextDependencies(); diff --git a/packages/react-reconciler/src/__tests__/ReactThenable-test.js b/packages/react-reconciler/src/__tests__/ReactThenable-test.js index fd95a43bfab8f..251b54433851f 100644 --- a/packages/react-reconciler/src/__tests__/ReactThenable-test.js +++ b/packages/react-reconciler/src/__tests__/ReactThenable-test.js @@ -497,4 +497,166 @@ describe('ReactThenable', () => { ); } }); + + // @gate enableUseHook + test('during a transition, can unwrap async operations even if nothing is cached', async () => { + let listeners = []; + + function resolvePending() { + const old = listeners; + listeners = []; + old.forEach(resolve => { + resolve(); + }); + } + + // This function is completely uncached — it performs a new async operation + // every time it's called. During a transition, React should be able to + // unwrap it anyway. + function getAsyncText(text) { + Scheduler.unstable_yieldValue('getAsyncText'); + return { + then(resolve) { + listeners.push(() => resolve(text)); + }, + }; + } + + function App() { + return ; + } + + const root = ReactNoop.createRoot(); + await act(async () => { + root.render( + }> + + , + ); + }); + expect(Scheduler).toHaveYielded(['(empty)']); + expect(root).toMatchRenderedOutput('(empty)'); + + await act(async () => { + startTransition(() => { + root.render( + }> + + , + ); + }); + }); + expect(Scheduler).toHaveYielded(['getAsyncText']); + expect(root).toMatchRenderedOutput('(empty)'); + + await act(async () => { + resolvePending(); + }); + expect(Scheduler).toHaveYielded(['getAsyncText', 'Async']); + expect(root).toMatchRenderedOutput('Async'); + }); + + // @gate enableUseHook + test("does not prevent a Suspense fallback from showing if it's a new boundary, even during a transition", async () => { + let listeners = []; + + function resolvePending() { + const old = listeners; + listeners = []; + old.forEach(resolve => { + resolve(); + }); + } + + // This function is completely uncached — it performs a new async operation + // every time it's called. During a transition, React should be able to + // unwrap it anyway. + function getAsyncText(text) { + Scheduler.unstable_yieldValue('getAsyncText'); + return { + then(resolve) { + listeners.push(() => resolve(text)); + }, + }; + } + + function App() { + return ; + } + + const root = ReactNoop.createRoot(); + await act(async () => { + startTransition(() => { + root.render( + }> + + , + ); + }); + }); + // Even though the initial render was a transition, it shows a fallback. + expect(Scheduler).toHaveYielded(['getAsyncText', 'Loading...']); + expect(root).toMatchRenderedOutput('Loading...'); + + // Resolve the original data + await act(async () => { + resolvePending(); + }); + // During the retry, a fresh request is initiated. Now we must wait for this + // one to finish. + // TODO: This is awkward. Intuitively, you might expect for `act` to wait + // until the new request has finished loading. But if it's mock IO, as in + // this test, how would the developer be able to imperatively flush it if it + // wasn't initiated until the current `act` call? Can't think of a better + // strategy at the moment. + expect(Scheduler).toHaveYielded(['getAsyncText']); + expect(root).toMatchRenderedOutput('Loading...'); + + // Flush the second request. + await act(async () => { + resolvePending(); + }); + // This time it finishes because it was during a retry. + expect(Scheduler).toHaveYielded(['getAsyncText', 'Async']); + expect(root).toMatchRenderedOutput('Async'); + }); + + // @gate enableUseHook + test('when waiting for data to resolve, a fresh update will trigger a restart', async () => { + function getAsyncDataThatNeverResolves(text) { + Scheduler.unstable_yieldValue('getAsyncDataThatNeverResolves'); + return { + then(resolve) {}, + }; + } + + function App() { + return ; + } + + const root = ReactNoop.createRoot(); + await act(async () => { + root.render(} />); + }); + + await act(async () => { + startTransition(() => { + root.render( + }> + + , + ); + }); + }); + expect(Scheduler).toHaveYielded(['getAsyncDataThatNeverResolves']); + + await act(async () => { + root.render( + }> + + , + ); + }); + expect(Scheduler).toHaveYielded(['Something different']); + }); });