Skip to content

Commit d1294c9

Browse files
authored
[Flight] Add global onError handler (#21129)
* Add onError option to Flight Server The callback is called any time an error is generated in a server component. This allows it to be logged on a server if needed. It'll still be rethrown on the client so it can be logged there too but in case it never reaches the client, here's a way to make sure it doesn't get lost. * Add fatal error handling
1 parent e40f0b2 commit d1294c9

File tree

9 files changed

+106
-20
lines changed

9 files changed

+106
-20
lines changed

packages/react-noop-renderer/src/ReactNoopFlightServer.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -54,13 +54,18 @@ const ReactNoopFlightServer = ReactFlightServer({
5454
},
5555
});
5656

57-
function render(model: ReactModel): Destination {
57+
type Options = {
58+
onError?: (error: mixed) => void,
59+
};
60+
61+
function render(model: ReactModel, options?: Options): Destination {
5862
const destination: Destination = [];
5963
const bundlerConfig = undefined;
6064
const request = ReactNoopFlightServer.createRequest(
6165
model,
6266
destination,
6367
bundlerConfig,
68+
options ? options.onError : undefined,
6469
);
6570
ReactNoopFlightServer.startWork(request);
6671
return destination;

packages/react-server-dom-relay/src/ReactFlightDOMRelayServer.js

+11-1
Original file line numberDiff line numberDiff line change
@@ -15,12 +15,22 @@ import type {
1515

1616
import {createRequest, startWork} from 'react-server/src/ReactFlightServer';
1717

18+
type Options = {
19+
onError?: (error: mixed) => void,
20+
};
21+
1822
function render(
1923
model: ReactModel,
2024
destination: Destination,
2125
config: BundlerConfig,
26+
options?: Options,
2227
): void {
23-
const request = createRequest(model, destination, config);
28+
const request = createRequest(
29+
model,
30+
destination,
31+
config,
32+
options ? options.onError : undefined,
33+
);
2434
startWork(request);
2535
}
2636

packages/react-server-dom-relay/src/ReactFlightDOMRelayServerHostConfig.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import {resolveModelToJSON} from 'react-server/src/ReactFlightServer';
2626
import {
2727
emitRow,
2828
resolveModuleMetaData as resolveModuleMetaDataImpl,
29+
close,
2930
} from 'ReactFlightDOMRelayServerIntegration';
3031

3132
export type {
@@ -146,4 +147,8 @@ export function writeChunk(destination: Destination, chunk: Chunk): boolean {
146147

147148
export function completeWriting(destination: Destination) {}
148149

149-
export {close} from 'ReactFlightDOMRelayServerIntegration';
150+
export {close};
151+
152+
export function closeWithError(destination: Destination, error: mixed): void {
153+
close(destination);
154+
}

packages/react-server-dom-webpack/src/ReactFlightDOMServerBrowser.js

+11-1
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,24 @@ import {
1616
startFlowing,
1717
} from 'react-server/src/ReactFlightServer';
1818

19+
type Options = {
20+
onError?: (error: mixed) => void,
21+
};
22+
1923
function renderToReadableStream(
2024
model: ReactModel,
2125
webpackMap: BundlerConfig,
26+
options?: Options,
2227
): ReadableStream {
2328
let request;
2429
return new ReadableStream({
2530
start(controller) {
26-
request = createRequest(model, controller, webpackMap);
31+
request = createRequest(
32+
model,
33+
controller,
34+
webpackMap,
35+
options ? options.onError : undefined,
36+
);
2737
startWork(request);
2838
},
2939
pull(controller) {

packages/react-server-dom-webpack/src/ReactFlightDOMServerNode.js

+11-1
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,22 @@ function createDrainHandler(destination, request) {
2121
return () => startFlowing(request);
2222
}
2323

24+
type Options = {
25+
onError?: (error: mixed) => void,
26+
};
27+
2428
function pipeToNodeWritable(
2529
model: ReactModel,
2630
destination: Writable,
2731
webpackMap: BundlerConfig,
32+
options?: Options,
2833
): void {
29-
const request = createRequest(model, destination, webpackMap);
34+
const request = createRequest(
35+
model,
36+
destination,
37+
webpackMap,
38+
options ? options.onError : undefined,
39+
);
3040
destination.on('drain', createDrainHandler(destination, request));
3141
startWork(request);
3242
}

packages/react-server-dom-webpack/src/__tests__/ReactFlightDOM-test.js

+15-2
Original file line numberDiff line numberDiff line change
@@ -256,6 +256,7 @@ describe('ReactFlightDOM', () => {
256256

257257
// @gate experimental
258258
it('should progressively reveal server components', async () => {
259+
let reportedErrors = [];
259260
const {Suspense} = React;
260261

261262
// Client Components
@@ -374,7 +375,11 @@ describe('ReactFlightDOM', () => {
374375
}
375376

376377
const {writable, readable} = getTestStream();
377-
ReactServerDOMWriter.pipeToNodeWritable(model, writable, webpackMap);
378+
ReactServerDOMWriter.pipeToNodeWritable(model, writable, webpackMap, {
379+
onError(x) {
380+
reportedErrors.push(x);
381+
},
382+
});
378383
const response = ReactServerDOMReader.createFromReadableStream(readable);
379384

380385
const container = document.createElement('div');
@@ -407,9 +412,12 @@ describe('ReactFlightDOM', () => {
407412
'<p>(loading games)</p>',
408413
);
409414

415+
expect(reportedErrors).toEqual([]);
416+
417+
const theError = new Error('Game over');
410418
// Let's *fail* loading games.
411419
await act(async () => {
412-
rejectGames(new Error('Game over'));
420+
rejectGames(theError);
413421
});
414422
expect(container.innerHTML).toBe(
415423
'<div>:name::avatar:</div>' +
@@ -418,6 +426,9 @@ describe('ReactFlightDOM', () => {
418426
'<p>Game over</p>', // TODO: should not have message in prod.
419427
);
420428

429+
expect(reportedErrors).toEqual([theError]);
430+
reportedErrors = [];
431+
421432
// We can now show the sidebar.
422433
await act(async () => {
423434
resolvePhotos();
@@ -439,6 +450,8 @@ describe('ReactFlightDOM', () => {
439450
'<div>:posts:</div>' +
440451
'<p>Game over</p>', // TODO: should not have message in prod.
441452
);
453+
454+
expect(reportedErrors).toEqual([]);
442455
});
443456

444457
// @gate experimental

packages/react-server-native-relay/src/ReactFlightNativeRelayServerHostConfig.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ import {resolveModelToJSON} from 'react-server/src/ReactFlightServer';
2525

2626
import {
2727
emitRow,
28+
close,
2829
resolveModuleMetaData as resolveModuleMetaDataImpl,
2930
} from 'ReactFlightNativeRelayServerIntegration';
3031

@@ -146,4 +147,8 @@ export function writeChunk(destination: Destination, chunk: Chunk): boolean {
146147

147148
export function completeWriting(destination: Destination) {}
148149

149-
export {close} from 'ReactFlightNativeRelayServerIntegration';
150+
export {close};
151+
152+
export function closeWithError(destination: Destination, error: mixed): void {
153+
close(destination);
154+
}

packages/react-server/src/ReactFlightServer.js

+39-12
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ import {
2424
completeWriting,
2525
flushBuffered,
2626
close,
27+
closeWithError,
2728
processModelChunk,
2829
processModuleChunk,
2930
processSymbolChunk,
@@ -83,16 +84,20 @@ export type Request = {
8384
completedErrorChunks: Array<Chunk>,
8485
writtenSymbols: Map<Symbol, number>,
8586
writtenModules: Map<ModuleKey, number>,
87+
onError: (error: mixed) => void,
8688
flowing: boolean,
8789
toJSON: (key: string, value: ReactModel) => ReactJSONValue,
8890
};
8991

9092
const ReactCurrentDispatcher = ReactSharedInternals.ReactCurrentDispatcher;
9193

94+
function defaultErrorHandler() {}
95+
9296
export function createRequest(
9397
model: ReactModel,
9498
destination: Destination,
9599
bundlerConfig: BundlerConfig,
100+
onError: (error: mixed) => void = defaultErrorHandler,
96101
): Request {
97102
const pingedSegments = [];
98103
const request = {
@@ -107,6 +112,7 @@ export function createRequest(
107112
completedErrorChunks: [],
108113
writtenSymbols: new Map(),
109114
writtenModules: new Map(),
115+
onError,
110116
flowing: false,
111117
toJSON: function(key: string, value: ReactModel): ReactJSONValue {
112118
return resolveModelToJSON(request, this, key, value);
@@ -413,6 +419,7 @@ export function resolveModelToJSON(
413419
x.then(ping, ping);
414420
return serializeByRefID(newSegment.id);
415421
} else {
422+
reportError(request, x);
416423
// Something errored. We'll still send everything we have up until this point.
417424
// We'll replace this element with a lazy reference that throws on the client
418425
// once it gets rendered.
@@ -589,6 +596,15 @@ export function resolveModelToJSON(
589596
);
590597
}
591598

599+
function reportError(request: Request, error: mixed): void {
600+
request.onError(error);
601+
}
602+
603+
function fatalError(request: Request, error: mixed): void {
604+
// This is called outside error handling code such as if an error happens in React internals.
605+
closeWithError(request.destination, error);
606+
}
607+
592608
function emitErrorChunk(request: Request, id: number, error: mixed): void {
593609
// TODO: We should not leak error messages to the client in prod.
594610
// Give this an error code instead and log on the server.
@@ -654,6 +670,7 @@ function retrySegment(request: Request, segment: Segment): void {
654670
x.then(ping, ping);
655671
return;
656672
} else {
673+
reportError(request, x);
657674
// This errored, we need to serialize this error to the
658675
emitErrorChunk(request, segment.id, x);
659676
}
@@ -666,18 +683,23 @@ function performWork(request: Request): void {
666683
ReactCurrentDispatcher.current = Dispatcher;
667684
currentCache = request.cache;
668685

669-
const pingedSegments = request.pingedSegments;
670-
request.pingedSegments = [];
671-
for (let i = 0; i < pingedSegments.length; i++) {
672-
const segment = pingedSegments[i];
673-
retrySegment(request, segment);
674-
}
675-
if (request.flowing) {
676-
flushCompletedChunks(request);
686+
try {
687+
const pingedSegments = request.pingedSegments;
688+
request.pingedSegments = [];
689+
for (let i = 0; i < pingedSegments.length; i++) {
690+
const segment = pingedSegments[i];
691+
retrySegment(request, segment);
692+
}
693+
if (request.flowing) {
694+
flushCompletedChunks(request);
695+
}
696+
} catch (error) {
697+
reportError(request, error);
698+
fatalError(request, error);
699+
} finally {
700+
ReactCurrentDispatcher.current = prevDispatcher;
701+
currentCache = prevCache;
677702
}
678-
679-
ReactCurrentDispatcher.current = prevDispatcher;
680-
currentCache = prevCache;
681703
}
682704

683705
let reentrant = false;
@@ -749,7 +771,12 @@ export function startWork(request: Request): void {
749771

750772
export function startFlowing(request: Request): void {
751773
request.flowing = true;
752-
flushCompletedChunks(request);
774+
try {
775+
flushCompletedChunks(request);
776+
} catch (error) {
777+
reportError(request, error);
778+
fatalError(request, error);
779+
}
753780
}
754781

755782
function unsupportedHook(): void {

packages/react-server/src/ReactFlightServerConfigStream.js

+1
Original file line numberDiff line numberDiff line change
@@ -126,4 +126,5 @@ export {
126126
writeChunk,
127127
completeWriting,
128128
close,
129+
closeWithError,
129130
} from './ReactServerStreamConfig';

0 commit comments

Comments
 (0)