From 3b597c0576977773910c77e075cc6d6308decb04 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Markb=C3=A5ge?= Date: Tue, 10 Dec 2024 11:24:59 -0500 Subject: [PATCH 1/2] Clean up findFiberByHostInstance from DevTools Hook (#31711) The need for this was removed in https://github.com/facebook/react/pull/30831 Since the new DevTools version has been released for a while and we expect people to more or less auto-update. Future versions of React don't need this. Once we remove the remaining uses of `getInstanceFromNode` e.g. in the deprecated internal `findDOMNode`/`findNodeHandle` and the event system, we can completely remove the tagging of DOM nodes. --- packages/react-reconciler/src/ReactFiberReconciler.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/packages/react-reconciler/src/ReactFiberReconciler.js b/packages/react-reconciler/src/ReactFiberReconciler.js index 94f1397eb129e..31a7fbeccf36b 100644 --- a/packages/react-reconciler/src/ReactFiberReconciler.js +++ b/packages/react-reconciler/src/ReactFiberReconciler.js @@ -42,7 +42,6 @@ import {enableSchedulingProfiler} from 'shared/ReactFeatureFlags'; import ReactSharedInternals from 'shared/ReactSharedInternals'; import { getPublicInstance, - getInstanceFromNode, rendererVersion, rendererPackageName, extraDevToolsConfig, @@ -847,7 +846,6 @@ export function injectIntoDevTools(): boolean { version: rendererVersion, rendererPackageName: rendererPackageName, currentDispatcherRef: ReactSharedInternals, - findFiberByHostInstance: getInstanceFromNode, // Enables DevTools to detect reconciler version rather than renderer version // which may not match for third party renderers. reconcilerVersion: ReactVersion, From 4a8fc0f92e0f75257962522b51a938bf4dfda77a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Sebastian=20Markb=C3=A5ge?= Date: Tue, 10 Dec 2024 11:59:50 -0500 Subject: [PATCH 2/2] [Flight] Don't call onError/onPostpone when halting and unify error branches (#31715) We shouldn't call onError/onPostpone when we halt a stream because that node didn't error yet. Its digest would also get lost. We also have a lot of error branches now for thenables and streams. This unifies them under erroredTask. I'm not yet unifying the cases that don't allocate a task for the error when those are outlined. --- .../src/__tests__/ReactFlightDOM-test.js | 4 +- .../__tests__/ReactFlightDOMBrowser-test.js | 2 +- .../src/__tests__/ReactFlightDOMEdge-test.js | 2 +- .../src/__tests__/ReactFlightDOMNode-test.js | 2 +- .../react-server/src/ReactFlightServer.js | 236 ++++++------------ 5 files changed, 78 insertions(+), 168 deletions(-) diff --git a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOM-test.js b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOM-test.js index febb5faf4ccec..d81a4c2f0149d 100644 --- a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOM-test.js +++ b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOM-test.js @@ -2870,7 +2870,7 @@ describe('ReactFlightDOM', () => { resolveGreeting(); const {prelude} = await pendingResult; - expect(errors).toEqual(['boom']); + expect(errors).toEqual([]); const preludeWeb = Readable.toWeb(prelude); const response = ReactServerDOMClient.createFromReadableStream(preludeWeb); @@ -3032,7 +3032,7 @@ describe('ReactFlightDOM', () => { const {prelude} = await pendingResult; - expect(errors).toEqual(['boom']); + expect(errors).toEqual([]); const preludeWeb = Readable.toWeb(prelude); const response = ReactServerDOMClient.createFromReadableStream(preludeWeb); diff --git a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMBrowser-test.js b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMBrowser-test.js index 882c0bbb01969..6803dfcfe2226 100644 --- a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMBrowser-test.js +++ b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMBrowser-test.js @@ -2559,7 +2559,7 @@ describe('ReactFlightDOMBrowser', () => { controller.abort('boom'); resolveGreeting(); const {prelude} = await pendingResult; - expect(errors).toEqual(['boom']); + expect(errors).toEqual([]); function ClientRoot({response}) { return use(response); diff --git a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMEdge-test.js b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMEdge-test.js index 1a097ffbda564..ad936391b7f4e 100644 --- a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMEdge-test.js +++ b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMEdge-test.js @@ -1209,7 +1209,7 @@ describe('ReactFlightDOMEdge', () => { resolveGreeting(); const {prelude} = await pendingResult; - expect(errors).toEqual(['boom']); + expect(errors).toEqual([]); function ClientRoot({response}) { return use(response); diff --git a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMNode-test.js b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMNode-test.js index 6593044005083..d7e947345546f 100644 --- a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMNode-test.js +++ b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMNode-test.js @@ -485,7 +485,7 @@ describe('ReactFlightDOMNode', () => { controller.abort('boom'); resolveGreeting(); const {prelude} = await pendingResult; - expect(errors).toEqual(['boom']); + expect(errors).toEqual([]); function ClientRoot({response}) { return use(response); diff --git a/packages/react-server/src/ReactFlightServer.js b/packages/react-server/src/ReactFlightServer.js index 0c330d0b8ce3e..2f25fe318aade 100644 --- a/packages/react-server/src/ReactFlightServer.js +++ b/packages/react-server/src/ReactFlightServer.js @@ -629,21 +629,7 @@ function serializeThenable( } case 'rejected': { const x = thenable.reason; - if ( - enablePostpone && - typeof x === 'object' && - x !== null && - (x: any).$$typeof === REACT_POSTPONE_TYPE - ) { - const postponeInstance: Postpone = (x: any); - logPostpone(request, postponeInstance.message, newTask); - emitPostponeChunk(request, newTask.id, postponeInstance); - } else { - const digest = logRecoverableError(request, x, null); - emitErrorChunk(request, newTask.id, digest, x); - } - newTask.status = ERRORED; - request.abortableTasks.delete(newTask); + erroredTask(request, newTask, x); return newTask.id; } default: { @@ -698,21 +684,7 @@ function serializeThenable( // We expect that the only status it might be otherwise is ABORTED. // When we abort we emit chunks in each pending task slot and don't need // to do so again here. - if ( - enablePostpone && - typeof reason === 'object' && - reason !== null && - (reason: any).$$typeof === REACT_POSTPONE_TYPE - ) { - const postponeInstance: Postpone = (reason: any); - logPostpone(request, postponeInstance.message, newTask); - emitPostponeChunk(request, newTask.id, postponeInstance); - } else { - const digest = logRecoverableError(request, reason, newTask); - emitErrorChunk(request, newTask.id, digest, reason); - } - newTask.status = ERRORED; - request.abortableTasks.delete(newTask); + erroredTask(request, newTask, reason); enqueueFlush(request); } }, @@ -795,8 +767,7 @@ function serializeReadableStream( } aborted = true; request.abortListeners.delete(abortStream); - const digest = logRecoverableError(request, reason, streamTask); - emitErrorChunk(request, streamTask.id, digest, reason); + erroredTask(request, streamTask, reason); enqueueFlush(request); // $FlowFixMe should be able to pass mixed @@ -808,30 +779,12 @@ function serializeReadableStream( } aborted = true; request.abortListeners.delete(abortStream); - if ( - enablePostpone && - typeof reason === 'object' && - reason !== null && - (reason: any).$$typeof === REACT_POSTPONE_TYPE - ) { - const postponeInstance: Postpone = (reason: any); - logPostpone(request, postponeInstance.message, streamTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitPostponeChunk(request, streamTask.id, postponeInstance); - enqueueFlush(request); - } + if (enableHalt && request.type === PRERENDER) { + request.pendingChunks--; } else { - const digest = logRecoverableError(request, reason, streamTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitErrorChunk(request, streamTask.id, digest, reason); - enqueueFlush(request); - } + erroredTask(request, streamTask, reason); + enqueueFlush(request); } - // $FlowFixMe should be able to pass mixed reader.cancel(reason).then(error, error); } @@ -937,8 +890,7 @@ function serializeAsyncIterable( } aborted = true; request.abortListeners.delete(abortIterable); - const digest = logRecoverableError(request, reason, streamTask); - emitErrorChunk(request, streamTask.id, digest, reason); + erroredTask(request, streamTask, reason); enqueueFlush(request); if (typeof (iterator: any).throw === 'function') { // The iterator protocol doesn't necessarily include this but a generator do. @@ -952,28 +904,11 @@ function serializeAsyncIterable( } aborted = true; request.abortListeners.delete(abortIterable); - if ( - enablePostpone && - typeof reason === 'object' && - reason !== null && - (reason: any).$$typeof === REACT_POSTPONE_TYPE - ) { - const postponeInstance: Postpone = (reason: any); - logPostpone(request, postponeInstance.message, streamTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitPostponeChunk(request, streamTask.id, postponeInstance); - enqueueFlush(request); - } + if (enableHalt && request.type === PRERENDER) { + request.pendingChunks--; } else { - const digest = logRecoverableError(request, reason, streamTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitErrorChunk(request, streamTask.id, digest, reason); - enqueueFlush(request); - } + erroredTask(request, streamTask, reason); + enqueueFlush(request); } if (typeof (iterator: any).throw === 'function') { // The iterator protocol doesn't necessarily include this but a generator do. @@ -2281,8 +2216,7 @@ function serializeBlob(request: Request, blob: Blob): string { } aborted = true; request.abortListeners.delete(abortBlob); - const digest = logRecoverableError(request, reason, newTask); - emitErrorChunk(request, newTask.id, digest, reason); + erroredTask(request, newTask, reason); enqueueFlush(request); // $FlowFixMe should be able to pass mixed reader.cancel(reason).then(error, error); @@ -2293,28 +2227,11 @@ function serializeBlob(request: Request, blob: Blob): string { } aborted = true; request.abortListeners.delete(abortBlob); - if ( - enablePostpone && - typeof reason === 'object' && - reason !== null && - (reason: any).$$typeof === REACT_POSTPONE_TYPE - ) { - const postponeInstance: Postpone = (reason: any); - logPostpone(request, postponeInstance.message, newTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitPostponeChunk(request, newTask.id, postponeInstance); - enqueueFlush(request); - } + if (enableHalt && request.type === PRERENDER) { + request.pendingChunks--; } else { - const digest = logRecoverableError(request, reason, newTask); - if (enableHalt && request.type === PRERENDER) { - request.pendingChunks--; - } else { - emitErrorChunk(request, newTask.id, digest, reason); - enqueueFlush(request); - } + erroredTask(request, newTask, reason); + enqueueFlush(request); } // $FlowFixMe should be able to pass mixed reader.cancel(reason).then(error, error); @@ -2414,24 +2331,6 @@ function renderModel( return serializeLazyID(newTask.id); } return serializeByValueID(newTask.id); - } else if (enablePostpone && x.$$typeof === REACT_POSTPONE_TYPE) { - // Something postponed. We'll still send everything we have up until this point. - // We'll replace this element with a lazy reference that postpones on the client. - const postponeInstance: Postpone = (x: any); - request.pendingChunks++; - const postponeId = request.nextChunkId++; - logPostpone(request, postponeInstance.message, task); - emitPostponeChunk(request, postponeId, postponeInstance); - - // Restore the context. We assume that this will be restored by the inner - // functions in case nothing throws so we don't use "finally" here. - task.keyPath = prevKeyPath; - task.implicitSlot = prevImplicitSlot; - - if (wasReactNode) { - return serializeLazyID(postponeId); - } - return serializeByValueID(postponeId); } } @@ -2443,8 +2342,21 @@ function renderModel( // Something errored. We'll still send everything we have up until this point. request.pendingChunks++; const errorId = request.nextChunkId++; - const digest = logRecoverableError(request, x, task); - emitErrorChunk(request, errorId, digest, x); + if ( + enablePostpone && + typeof x === 'object' && + x !== null && + x.$$typeof === REACT_POSTPONE_TYPE + ) { + // Something postponed. We'll still send everything we have up until this point. + // We'll replace this element with a lazy reference that postpones on the client. + const postponeInstance: Postpone = (x: any); + logPostpone(request, postponeInstance.message, task); + emitPostponeChunk(request, errorId, postponeInstance); + } else { + const digest = logRecoverableError(request, x, task); + emitErrorChunk(request, errorId, digest, x); + } if (wasReactNode) { // We'll replace this element with a lazy reference that throws on the client // once it gets rendered. @@ -3964,6 +3876,24 @@ function emitChunk( emitModelChunk(request, task.id, json); } +function erroredTask(request: Request, task: Task, error: mixed): void { + request.abortableTasks.delete(task); + task.status = ERRORED; + if ( + enablePostpone && + typeof error === 'object' && + error !== null && + error.$$typeof === REACT_POSTPONE_TYPE + ) { + const postponeInstance: Postpone = (error: any); + logPostpone(request, postponeInstance.message, task); + emitPostponeChunk(request, task.id, postponeInstance); + } else { + const digest = logRecoverableError(request, error, task); + emitErrorChunk(request, task.id, digest, error); + } +} + const emptyRoot = {}; function retryTask(request: Request, task: Task): void { @@ -4083,20 +4013,9 @@ function retryTask(request: Request, task: Task): void { const ping = task.ping; x.then(ping, ping); return; - } else if (enablePostpone && x.$$typeof === REACT_POSTPONE_TYPE) { - request.abortableTasks.delete(task); - task.status = ERRORED; - const postponeInstance: Postpone = (x: any); - logPostpone(request, postponeInstance.message, task); - emitPostponeChunk(request, task.id, postponeInstance); - return; } } - - request.abortableTasks.delete(task); - task.status = ERRORED; - const digest = logRecoverableError(request, x, task); - emitErrorChunk(request, task.id, digest, x); + erroredTask(request, task, x); } finally { if (__DEV__) { debugID = prevDebugID; @@ -4336,7 +4255,12 @@ export function abort(request: Request, reason: mixed): void { } const abortableTasks = request.abortableTasks; if (abortableTasks.size > 0) { - if ( + if (enableHalt && request.type === PRERENDER) { + // When prerendering with halt semantics we simply halt the task + // and leave the reference unfulfilled. + abortableTasks.forEach(task => haltTask(task, request)); + abortableTasks.clear(); + } else if ( enablePostpone && typeof reason === 'object' && reason !== null && @@ -4344,21 +4268,14 @@ export function abort(request: Request, reason: mixed): void { ) { const postponeInstance: Postpone = (reason: any); logPostpone(request, postponeInstance.message, null); - if (enableHalt && request.type === PRERENDER) { - // When prerendering with halt semantics we simply halt the task - // and leave the reference unfulfilled. - abortableTasks.forEach(task => haltTask(task, request)); - abortableTasks.clear(); - } else { - // When rendering we produce a shared postpone chunk and then - // fulfill each task with a reference to that chunk. - const errorId = request.nextChunkId++; - request.fatalError = errorId; - request.pendingChunks++; - emitPostponeChunk(request, errorId, postponeInstance); - abortableTasks.forEach(task => abortTask(task, request, errorId)); - abortableTasks.clear(); - } + // When rendering we produce a shared postpone chunk and then + // fulfill each task with a reference to that chunk. + const errorId = request.nextChunkId++; + request.fatalError = errorId; + request.pendingChunks++; + emitPostponeChunk(request, errorId, postponeInstance); + abortableTasks.forEach(task => abortTask(task, request, errorId)); + abortableTasks.clear(); } else { const error = reason === undefined @@ -4373,21 +4290,14 @@ export function abort(request: Request, reason: mixed): void { ) : reason; const digest = logRecoverableError(request, error, null); - if (enableHalt && request.type === PRERENDER) { - // When prerendering with halt semantics we simply halt the task - // and leave the reference unfulfilled. - abortableTasks.forEach(task => haltTask(task, request)); - abortableTasks.clear(); - } else { - // When rendering we produce a shared error chunk and then - // fulfill each task with a reference to that chunk. - const errorId = request.nextChunkId++; - request.fatalError = errorId; - request.pendingChunks++; - emitErrorChunk(request, errorId, digest, error); - abortableTasks.forEach(task => abortTask(task, request, errorId)); - abortableTasks.clear(); - } + // When rendering we produce a shared error chunk and then + // fulfill each task with a reference to that chunk. + const errorId = request.nextChunkId++; + request.fatalError = errorId; + request.pendingChunks++; + emitErrorChunk(request, errorId, digest, error); + abortableTasks.forEach(task => abortTask(task, request, errorId)); + abortableTasks.clear(); } const onAllReady = request.onAllReady; onAllReady();