Skip to content

Commit 561447d

Browse files
committed
feat: support multi-chunk bundles in ssr bundle renderer
1 parent d57f942 commit 561447d

File tree

4 files changed

+86
-29
lines changed

4 files changed

+86
-29
lines changed

src/server/create-bundle-renderer.js

+37-6
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,53 @@
1+
/* @flow */
2+
13
import runInVm from './run-in-vm'
24
import { PassThrough } from 'stream'
5+
import type { Renderer, RenderOptions } from './create-renderer'
6+
7+
const INVALID_MSG =
8+
'Invalid server-rendering bundle format. Should be a string of bundled code ' +
9+
'or an Object of type { entry: string; chunks: { [filename: string]: string }}.'
10+
11+
// The render bundle can either be a string (single bundled file)
12+
// or an object containing a chunks hash of filename:code pairs with the
13+
// name of the entry file. The object format is used in conjunction with
14+
// Webpack's compilation output so that code-split chunks can also be loaded.
15+
type RenderBundle = string | {
16+
entry: string;
17+
chunks: {
18+
[filename: string]: string;
19+
};
20+
};
321

4-
export function createBundleRendererCreator (createRenderer) {
5-
return (code, rendererOptions) => {
22+
export function createBundleRendererCreator (createRenderer: () => Renderer) {
23+
return (bundle: RenderBundle, rendererOptions?: RenderOptions) => {
624
const renderer = createRenderer(rendererOptions)
25+
let chunks, entry
26+
if (typeof bundle === 'object') {
27+
entry = bundle.entry
28+
chunks = bundle.chunks
29+
if (typeof entry !== 'string' || typeof chunks !== 'object') {
30+
throw new Error(INVALID_MSG)
31+
}
32+
} else if (typeof bundle === 'string') {
33+
entry = '__vue_ssr_bundle__'
34+
chunks = { '__vue_ssr_bundle__': bundle }
35+
} else {
36+
throw new Error(INVALID_MSG)
37+
}
738
return {
8-
renderToString: (context, cb) => {
39+
renderToString: (context?: Object, cb: (err: ?Error, res: ?string) => void) => {
940
if (typeof context === 'function') {
1041
cb = context
1142
context = {}
1243
}
13-
runInVm(code, context).then(app => {
44+
runInVm(entry, chunks, context).then(app => {
1445
renderer.renderToString(app, cb)
1546
}).catch(cb)
1647
},
17-
renderToStream: (context) => {
48+
renderToStream: (context?: Object) => {
1849
const res = new PassThrough()
19-
runInVm(code, context).then(app => {
50+
runInVm(entry, chunks, context).then(app => {
2051
const renderStream = renderer.renderToStream(app)
2152
renderStream.on('error', err => {
2253
res.emit('error', err)

src/server/create-renderer.js

+13-9
Original file line numberDiff line numberDiff line change
@@ -4,20 +4,24 @@ import RenderStream from './render-stream'
44
import { createWriteFunction } from './write'
55
import { createRenderFunction } from './render'
66

7+
export type Renderer = {
8+
renderToString: (component: Component, cb: (err: ?Error, res: ?string) => void) => void;
9+
renderToStream: (component: Component) => RenderStream;
10+
};
11+
12+
export type RenderOptions = {
13+
modules: Array<Function>,
14+
directives: Object,
15+
isUnaryTag: Function,
16+
cache: ?Object
17+
};
18+
719
export function createRenderer ({
820
modules = [],
921
directives = {},
1022
isUnaryTag = (() => false),
1123
cache
12-
}: {
13-
modules: Array<Function>,
14-
directives: Object,
15-
isUnaryTag: Function,
16-
cache: ?Object
17-
} = {}): {
18-
renderToString: Function,
19-
renderToStream: Function
20-
} {
24+
}: RenderOptions = {}): Renderer {
2125
const render = createRenderFunction(modules, directives, isUnaryTag, cache)
2226

2327
return {

src/server/render-stream.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,8 @@
88
* Modified by Evan You (@yyx990803)
99
*/
1010

11-
import stream from 'stream'
11+
const stream = require('stream')
12+
1213
import { createWriteFunction } from './write'
1314

1415
export default class RenderStream extends stream.Readable {

src/server/run-in-vm.js

+34-13
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
1-
import NativeModule from 'module'
2-
import vm from 'vm'
1+
const NativeModule = require('module')
2+
const vm = require('vm')
3+
const path = require('path')
34

45
function createContext (context) {
56
const sandbox = {
@@ -18,19 +19,39 @@ function createContext (context) {
1819
return sandbox
1920
}
2021

21-
export default function runInVm (code, _context = {}) {
22+
function evaluateModule (filename, chunks, context, evaluatedModules) {
23+
if (evaluatedModules[filename]) {
24+
return evaluatedModules[filename]
25+
}
26+
27+
const code = chunks[filename]
28+
const wrapper = NativeModule.wrap(code)
29+
const compiledWrapper = vm.runInNewContext(wrapper, context, {
30+
filename,
31+
displayErrors: true
32+
})
33+
const m = { exports: {}}
34+
const r = file => {
35+
file = path.join('.', file)
36+
if (chunks[file]) {
37+
return evaluateModule(file, chunks, context, evaluatedModules)
38+
} else {
39+
return require(file)
40+
}
41+
}
42+
compiledWrapper.call(m.exports, m.exports, r, m)
43+
44+
const res = Object.prototype.hasOwnProperty.call(m.exports, 'default')
45+
? m.exports.default
46+
: m.exports
47+
evaluatedModules[filename] = res
48+
return res
49+
}
50+
51+
export default function runInVm (entry, chunks, _context = {}) {
2252
return new Promise((resolve, reject) => {
23-
const wrapper = NativeModule.wrap(code)
2453
const context = createContext(_context)
25-
const compiledWrapper = vm.runInNewContext(wrapper, context, {
26-
filename: '__vue_ssr_bundle__',
27-
displayErrors: true
28-
})
29-
const m = { exports: {}}
30-
compiledWrapper.call(m.exports, m.exports, require, m)
31-
const res = Object.prototype.hasOwnProperty.call(m.exports, 'default')
32-
? m.exports.default
33-
: m.exports
54+
const res = evaluateModule(entry, chunks, context, {})
3455
resolve(typeof res === 'function' ? res(_context) : res)
3556
})
3657
}

0 commit comments

Comments
 (0)