-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathserver.js
112 lines (90 loc) · 3.08 KB
/
server.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
const fs = require("fs")
const path = require("path")
const compression = require("compression")
const express = require("express")
const app = express()
const favicon = require('serve-favicon')
const resolve = (file) => path.resolve(__dirname, file)
const config = require("./config")
const isProduction = config.isProduction
const template = fs.readFileSync(resolve("./src/index.template.html"), "utf-8")
const createRenderer = (bundle, options) => {
// https://github.com/vuejs/vue/blob/dev/packages/vue-server-renderer/README.md#why-use-bundlerenderer
return require("vue-server-renderer").createBundleRenderer(bundle, Object.assign(options, {
template,
cache: require("lru-cache")({
max: 1000,
maxAge: 1000 * 60 * 15
}),
// this is only needed when vue-server-renderer is npm-linked
basedir: resolve("./dist"),
// recommended for performance
runInNewContext: false
}))
}
const serve = (path, cache) => express.static(resolve(path), {
maxAge: cache && isProduction ? 60 * 60 * 24 * 30 : 0
})
let renderer
let readyPromise
if (isProduction) {
// In production: create server renderer using built server bundle.
// The server bundle is generated by vue-ssr-webpack-plugin.
const bundle = require("./dist/vue-ssr-server-bundle.json")
// The client manifests are optional, but it allows the renderer
// to automatically infer preload/prefetch links and directly add <script>
// tags for any async chunks used during render, avoiding waterfall requests.
const clientManifest = require("./dist/vue-ssr-client-manifest.json")
renderer = createRenderer(bundle, {
clientManifest
})
readyPromise = Promise.resolve()
} else {
// In development: setup the dev server with watch and hot-reload,
// and create a new renderer on bundle / index template update.
readyPromise = require("./build/setup-dev-server")(app, (bundle, options) => {
renderer = createRenderer(bundle, options)
})
}
const render = (req, res, context) => {
const s = Date.now()
console.log(`Rendering: ${req.url}`)
res.setHeader("Content-Type", "text/html")
const errorHandler = (err) => {
// TODO: Render Error Page
console.error(`Fatal error when rendering : ${req.url}`)
console.error(err)
res.status(500)
res.end(`500 | Fatal error: ${err}`)
console.log(`Whole request: ${Date.now() - s}ms`)
}
renderer.renderToString(context, (err, html) => {
if (err) return errorHandler(err)
res.status(context.meta.httpStatusCode || 200)
res.end(html)
console.log(`Whole request: ${Date.now() - s}ms`)
})
}
app.use(compression({ threshold: 0 }))
app.use(favicon('./static/favicon.png'))
app.use("/dist", serve("./dist", true))
app.use("/static", serve("./static", true))
app.use("/service-worker.js", serve("./dist/service-worker.js"))
app.get("*", (req, res) => {
const context = {
url: req.url
}
isProduction ?
render(req, res, context) :
readyPromise.then(() => render(req, res, context))
})
const port = config.server.port
let server = app.listen(port, () => {
console.log(`Server started at localhost:${port}`)
})
module.exports = {
ready: readyPromise,
close: () => {
server.close()
}
}