Skip to content

Commit e613f0c

Browse files
committed
fix(ssr): handle error during ssr render call
1 parent 21f8d9d commit e613f0c

File tree

2 files changed

+44
-0
lines changed

2 files changed

+44
-0
lines changed

packages/server-renderer/__tests__/render.spec.ts

+40
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
defineComponent,
1313
getCurrentInstance,
1414
h,
15+
nextTick,
1516
onErrorCaptured,
1617
onServerPrefetch,
1718
reactive,
@@ -819,6 +820,9 @@ function testRender(type: string, render: typeof renderToString) {
819820
)
820821
} catch {}
821822
expect(getCurrentInstance()).toBe(prev)
823+
expect(
824+
'[Vue warn]: Unhandled error during execution of render function',
825+
).toHaveBeenWarned()
822826
})
823827

824828
// #7733
@@ -1188,6 +1192,42 @@ function testRender(type: string, render: typeof renderToString) {
11881192
expect((capturedError as unknown as Error).message).toBe('An error')
11891193
})
11901194

1195+
test('async setup throwing error', async () => {
1196+
let capturedError: string[] = []
1197+
1198+
const Child = {
1199+
async setup() {
1200+
await nextTick()
1201+
throw new Error('An error')
1202+
return { foo: { bar: 1 } }
1203+
},
1204+
template: `<span>{{ foo.bar }}</span>`,
1205+
}
1206+
1207+
const app = createApp({
1208+
components: { Child },
1209+
setup() {
1210+
onErrorCaptured(e => {
1211+
capturedError.push(e.message)
1212+
return false
1213+
})
1214+
},
1215+
template: `<Suspense><Child /></Suspense>`,
1216+
})
1217+
1218+
try {
1219+
await render(app)
1220+
} catch (e: any) {}
1221+
expect(capturedError.length).toBe(2)
1222+
expect(capturedError).toStrictEqual([
1223+
'An error',
1224+
"Cannot read properties of undefined (reading 'bar')",
1225+
])
1226+
expect(
1227+
'[Vue warn]: Property "foo" was accessed during render but is not defined on instance',
1228+
).toHaveBeenWarned()
1229+
})
1230+
11911231
test('computed reactivity during SSR with onServerPrefetch', async () => {
11921232
const store = {
11931233
// initial state could be hydrated

packages/server-renderer/src/render.ts

+4
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,15 @@ import {
33
type Component,
44
type ComponentInternalInstance,
55
type DirectiveBinding,
6+
ErrorCodes,
67
Fragment,
78
type FunctionalComponent,
89
Static,
910
Text,
1011
type VNode,
1112
type VNodeArrayChildren,
1213
type VNodeProps,
14+
handleError,
1315
mergeProps,
1416
ssrUtils,
1517
warn,
@@ -200,6 +202,8 @@ function renderComponentSubTree(
200202
instance.data,
201203
instance.ctx,
202204
)
205+
} catch (err) {
206+
handleError(err, instance, ErrorCodes.RENDER_FUNCTION)
203207
} finally {
204208
setCurrentRenderingInstance(prev)
205209
}

0 commit comments

Comments
 (0)