File tree 4 files changed +55
-5
lines changed
4 files changed +55
-5
lines changed Original file line number Diff line number Diff line change @@ -103,6 +103,7 @@ export interface ComponentOptionsBase<
103
103
directives ?: Record < string , Directive >
104
104
inheritAttrs ?: boolean
105
105
emits ?: ( E | EE [ ] ) & ThisType < void >
106
+ serverPrefetch ?( ) : Promise < any >
106
107
107
108
// Internal ------------------------------------------------------------------
108
109
Original file line number Diff line number Diff line change @@ -15,6 +15,7 @@ import { renderToStream as _renderToStream } from '../src/renderToStream'
15
15
import { Readable } from 'stream'
16
16
import { ssrRenderSlot } from '../src/helpers/ssrRenderSlot'
17
17
import { ssrRenderComponent } from '../src/helpers/ssrRenderComponent'
18
+
18
19
const promisifyStream = ( stream : Readable ) => {
19
20
return new Promise ( ( resolve , reject ) => {
20
21
let result = ''
@@ -599,4 +600,23 @@ describe('ssr: renderToStream', () => {
599
600
)
600
601
} )
601
602
} )
603
+
604
+ test ( 'serverPrefetch' , async ( ) => {
605
+ const msg = Promise . resolve ( 'hello' )
606
+ const app = createApp ( {
607
+ data ( ) {
608
+ return {
609
+ msg : ''
610
+ }
611
+ } ,
612
+ async serverPrefetch ( ) {
613
+ this . msg = await msg
614
+ } ,
615
+ render ( ) {
616
+ return h ( 'div' , this . msg )
617
+ }
618
+ } )
619
+ const html = await renderToStream ( app )
620
+ expect ( html ) . toBe ( `<div>hello</div>` )
621
+ } )
602
622
} )
Original file line number Diff line number Diff line change @@ -14,6 +14,7 @@ import { escapeHtml } from '@vue/shared'
14
14
import { renderToString } from '../src/renderToString'
15
15
import { ssrRenderSlot , SSRSlot } from '../src/helpers/ssrRenderSlot'
16
16
import { ssrRenderComponent } from '../src/helpers/ssrRenderComponent'
17
+
17
18
describe ( 'ssr: renderToString' , ( ) => {
18
19
test ( 'should apply app context' , async ( ) => {
19
20
const app = createApp ( {
@@ -580,4 +581,23 @@ describe('ssr: renderToString', () => {
580
581
) . toHaveBeenWarned ( )
581
582
} )
582
583
} )
584
+
585
+ test ( 'serverPrefetch' , async ( ) => {
586
+ const msg = Promise . resolve ( 'hello' )
587
+ const app = createApp ( {
588
+ data ( ) {
589
+ return {
590
+ msg : ''
591
+ }
592
+ } ,
593
+ async serverPrefetch ( ) {
594
+ this . msg = await msg
595
+ } ,
596
+ render ( ) {
597
+ return h ( 'div' , this . msg )
598
+ }
599
+ } )
600
+ const html = await renderToString ( app )
601
+ expect ( html ) . toBe ( `<div>hello</div>` )
602
+ } )
583
603
} )
Original file line number Diff line number Diff line change 2
2
Comment ,
3
3
Component ,
4
4
ComponentInternalInstance ,
5
+ ComponentOptions ,
5
6
DirectiveBinding ,
6
7
Fragment ,
7
8
mergeProps ,
@@ -84,12 +85,20 @@ export function renderComponentVNode(
84
85
) : SSRBuffer | Promise < SSRBuffer > {
85
86
const instance = createComponentInstance ( vnode , parentComponent , null )
86
87
const res = setupComponent ( instance , true /* isSSR */ )
87
- if ( isPromise ( res ) ) {
88
- return res
89
- . catch ( err => {
90
- warn ( `[@vue/server-renderer]: Uncaught error in async setup:\n` , err )
88
+ const hasAsyncSetup = isPromise ( res )
89
+ const prefetch = ( vnode . type as ComponentOptions ) . serverPrefetch
90
+ if ( hasAsyncSetup || prefetch ) {
91
+ let p = hasAsyncSetup
92
+ ? ( res as Promise < void > ) . catch ( err => {
93
+ warn ( `[@vue/server-renderer]: Uncaught error in async setup:\n` , err )
94
+ } )
95
+ : Promise . resolve ( )
96
+ if ( prefetch ) {
97
+ p = p . then ( ( ) => prefetch . call ( instance . proxy ) ) . catch ( err => {
98
+ warn ( `[@vue/server-renderer]: Uncaught error in serverPrefetch:\n` , err )
91
99
} )
92
- . then ( ( ) => renderComponentSubTree ( instance ) )
100
+ }
101
+ return p . then ( ( ) => renderComponentSubTree ( instance ) )
93
102
} else {
94
103
return renderComponentSubTree ( instance )
95
104
}
You can’t perform that action at this time.
0 commit comments