Skip to content

Commit 5c899f6

Browse files
committed
feat: Add support for React error handlers
1 parent 7a28fa9 commit 5c899f6

File tree

4 files changed

+222
-3
lines changed

4 files changed

+222
-3
lines changed

src/__tests__/error-handlers.js

+155
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,155 @@
1+
import * as React from 'react'
2+
import {render, renderHook} from '../'
3+
4+
const isReact19 = React.version.startsWith('19.')
5+
6+
const testGateReact19 = isReact19 ? test : test.skip
7+
8+
test('onUncaughtError is not supported in render', () => {
9+
function Thrower() {
10+
throw new Error('Boom!')
11+
}
12+
const onUncaughtError = jest.fn(() => {})
13+
14+
expect(() => {
15+
render(<Thrower />, {
16+
onUncaughtError(error, errorInfo) {
17+
console.log({error, errorInfo})
18+
},
19+
})
20+
}).toThrow(
21+
'onUncaughtError is not supported. The `render` call will already throw on uncaught errors.',
22+
)
23+
24+
expect(onUncaughtError).toHaveBeenCalledTimes(0)
25+
})
26+
27+
testGateReact19('onCaughtError is supported in render', () => {
28+
const thrownError = new Error('Boom!')
29+
const handleComponentDidCatch = jest.fn()
30+
const onCaughtError = jest.fn()
31+
class ErrorBoundary extends React.Component {
32+
state = {error: null}
33+
static getDerivedStateFromError(error) {
34+
return {error}
35+
}
36+
componentDidCatch(error, errorInfo) {
37+
handleComponentDidCatch(error, errorInfo)
38+
}
39+
render() {
40+
if (this.state.error) {
41+
return null
42+
}
43+
return this.props.children
44+
}
45+
}
46+
function Thrower() {
47+
throw thrownError
48+
}
49+
50+
render(
51+
<ErrorBoundary>
52+
<Thrower />
53+
</ErrorBoundary>,
54+
{
55+
onCaughtError,
56+
},
57+
)
58+
59+
expect(onCaughtError).toHaveBeenCalledWith(thrownError, {
60+
componentStack: expect.any(String),
61+
errorBoundary: expect.any(Object),
62+
})
63+
})
64+
65+
test('onRecoverableError is supported in render', () => {
66+
const onRecoverableError = jest.fn()
67+
68+
const container = document.createElement('div')
69+
container.innerHTML = '<div>server</div>'
70+
render(<div>client</div>, {
71+
container,
72+
hydrate: true,
73+
onRecoverableError,
74+
})
75+
76+
expect(onRecoverableError).toHaveBeenCalledWith(
77+
expect.objectContaining({
78+
message: expect.stringContaining('Hydration failed'),
79+
}),
80+
{
81+
componentStack: expect.any(String),
82+
},
83+
undefined,
84+
undefined,
85+
undefined,
86+
)
87+
})
88+
89+
test('onUncaughtError is not supported in renderHook', () => {
90+
function useThrower() {
91+
throw new Error('Boom!')
92+
}
93+
const onUncaughtError = jest.fn(() => {})
94+
95+
expect(() => {
96+
renderHook(useThrower, {
97+
onUncaughtError(error, errorInfo) {
98+
console.log({error, errorInfo})
99+
},
100+
})
101+
}).toThrow(
102+
'onUncaughtError is not supported. The `render` call will already throw on uncaught errors.',
103+
)
104+
105+
expect(onUncaughtError).toHaveBeenCalledTimes(0)
106+
})
107+
108+
testGateReact19('onCaughtError is supported in renderHook', () => {
109+
const thrownError = new Error('Boom!')
110+
const handleComponentDidCatch = jest.fn()
111+
const onCaughtError = jest.fn()
112+
class ErrorBoundary extends React.Component {
113+
state = {error: null}
114+
static getDerivedStateFromError(error) {
115+
return {error}
116+
}
117+
componentDidCatch(error, errorInfo) {
118+
handleComponentDidCatch(error, errorInfo)
119+
}
120+
render() {
121+
if (this.state.error) {
122+
return null
123+
}
124+
return this.props.children
125+
}
126+
}
127+
function useThrower() {
128+
throw thrownError
129+
}
130+
131+
renderHook(useThrower, {
132+
onCaughtError,
133+
wrapper: ErrorBoundary,
134+
})
135+
136+
expect(onCaughtError).toHaveBeenCalledWith(thrownError, {
137+
componentStack: expect.any(String),
138+
errorBoundary: expect.any(Object),
139+
})
140+
})
141+
142+
// Currently, there's no recoverable error without hydration.
143+
// The option is still supported though.
144+
test('onRecoverableError is supported in renderHook', () => {
145+
const onRecoverableError = jest.fn()
146+
147+
renderHook(
148+
() => {
149+
// TODO: trigger recoverable error
150+
},
151+
{
152+
onRecoverableError,
153+
},
154+
)
155+
})

src/pure.js

+21-3
Original file line numberDiff line numberDiff line change
@@ -91,18 +91,22 @@ function wrapUiIfNeeded(innerElement, wrapperComponent) {
9191

9292
function createConcurrentRoot(
9393
container,
94-
{hydrate, ui, wrapper: WrapperComponent},
94+
{hydrate, onCaughtError, onRecoverableError, ui, wrapper: WrapperComponent},
9595
) {
9696
let root
9797
if (hydrate) {
9898
act(() => {
9999
root = ReactDOMClient.hydrateRoot(
100100
container,
101101
strictModeIfNeeded(wrapUiIfNeeded(ui, WrapperComponent)),
102+
{onCaughtError, onRecoverableError},
102103
)
103104
})
104105
} else {
105-
root = ReactDOMClient.createRoot(container)
106+
root = ReactDOMClient.createRoot(container, {
107+
onCaughtError,
108+
onRecoverableError,
109+
})
106110
}
107111

108112
return {
@@ -202,11 +206,19 @@ function render(
202206
container,
203207
baseElement = container,
204208
legacyRoot = false,
209+
onCaughtError,
210+
onUncaughtError,
211+
onRecoverableError,
205212
queries,
206213
hydrate = false,
207214
wrapper,
208215
} = {},
209216
) {
217+
if (onUncaughtError !== undefined) {
218+
throw new Error(
219+
'onUncaughtError is not supported. The `render` call will already throw on uncaught errors.',
220+
)
221+
}
210222
if (legacyRoot && typeof ReactDOM.render !== 'function') {
211223
const error = new Error(
212224
'`legacyRoot: true` is not supported in this version of React. ' +
@@ -230,7 +242,13 @@ function render(
230242
// eslint-disable-next-line no-negated-condition -- we want to map the evolution of this over time. The root is created first. Only later is it re-used so we don't want to read the case that happens later first.
231243
if (!mountedContainers.has(container)) {
232244
const createRootImpl = legacyRoot ? createLegacyRoot : createConcurrentRoot
233-
root = createRootImpl(container, {hydrate, ui, wrapper})
245+
root = createRootImpl(container, {
246+
hydrate,
247+
onCaughtError,
248+
onRecoverableError,
249+
ui,
250+
wrapper,
251+
})
234252

235253
mountedRootEntries.push({container, root})
236254
// we'll add it to the mounted containers regardless of whether it's actually

types/index.d.ts

+24
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,30 @@ export interface RenderOptions<
118118
* Otherwise `render` will default to concurrent React if available.
119119
*/
120120
legacyRoot?: boolean
121+
/**
122+
* Only supported in React 19.
123+
* Callback called when React catches an error in an Error Boundary.
124+
* Called with the error caught by the Error Boundary, and an `errorInfo` object containing the `componentStack`.
125+
*
126+
* @see {@link https://react.dev/reference/react-dom/client/createRoot#parameters createRoot#options}
127+
*/
128+
onCaughtError?: ReactDOMClient.RootOptions extends {
129+
onCaughtError: infer OnCaughtError
130+
}
131+
? OnCaughtError
132+
: never
133+
/**
134+
* Callback called when React automatically recovers from errors.
135+
* Called with an error React throws, and an `errorInfo` object containing the `componentStack`.
136+
* Some recoverable errors may include the original error cause as `error.cause`.
137+
*
138+
* @see {@link https://react.dev/reference/react-dom/client/createRoot#parameters createRoot#options}
139+
*/
140+
onRecoverableError?: ReactDOMClient.RootOptions['onRecoverableError']
141+
/**
142+
* Not supported at the moment
143+
*/
144+
onUncaughtError?: never
121145
/**
122146
* Queries to bind. Overrides the default set from DOM Testing Library unless merged.
123147
*

types/test.tsx

+22
Original file line numberDiff line numberDiff line change
@@ -254,6 +254,28 @@ export function testContainer() {
254254
renderHook(() => null, {container: document, hydrate: true})
255255
}
256256

257+
export function testErrorHandlers() {
258+
// React 19 types are not used in tests. Verify manually if this works with `"@types/react": "npm:types-react@rc"`
259+
render(null, {
260+
// Should work with React 19 types
261+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
262+
// @ts-expect-error
263+
onCaughtError: () => {},
264+
})
265+
render(null, {
266+
// Should never work as it's not supported yet.
267+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
268+
// @ts-expect-error
269+
onUncaughtError: () => {},
270+
})
271+
render(null, {
272+
onRecoverableError: (error, errorInfo) => {
273+
console.error(error)
274+
console.log(errorInfo.componentStack)
275+
},
276+
})
277+
}
278+
257279
/*
258280
eslint
259281
testing-library/prefer-explicit-assert: "off",

0 commit comments

Comments
 (0)