2019-04-20 12:02:51 +00:00
|
|
|
import path from 'path'
|
|
|
|
import crypto from 'crypto'
|
2019-05-19 18:49:24 +00:00
|
|
|
import { format } from 'util'
|
2019-04-20 12:02:51 +00:00
|
|
|
import fs from 'fs-extra'
|
2019-05-09 07:06:17 +00:00
|
|
|
import consola from 'consola'
|
2019-04-20 12:02:51 +00:00
|
|
|
import devalue from '@nuxt/devalue'
|
|
|
|
import { createBundleRenderer } from 'vue-server-renderer'
|
|
|
|
import BaseRenderer from './base'
|
|
|
|
|
|
|
|
export default class SSRRenderer extends BaseRenderer {
|
2019-07-10 10:45:49 +00:00
|
|
|
get rendererOptions () {
|
2019-04-20 12:02:51 +00:00
|
|
|
const hasModules = fs.existsSync(path.resolve(this.options.rootDir, 'node_modules'))
|
|
|
|
|
|
|
|
return {
|
|
|
|
clientManifest: this.serverContext.resources.clientManifest,
|
|
|
|
// for globally installed nuxt command, search dependencies in global dir
|
|
|
|
basedir: hasModules ? this.options.rootDir : __dirname,
|
|
|
|
...this.options.render.bundleRenderer
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
renderScripts (renderContext) {
|
2020-02-24 22:58:24 +00:00
|
|
|
const scripts = renderContext.renderScripts()
|
|
|
|
const { build: { crossorigin } } = this.options
|
|
|
|
if (!crossorigin) {
|
|
|
|
return scripts
|
|
|
|
}
|
|
|
|
return scripts.replace(
|
|
|
|
/<script/g,
|
|
|
|
`<script crossorigin="${crossorigin}"`
|
|
|
|
)
|
2019-04-20 12:02:51 +00:00
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
getPreloadFiles (renderContext) {
|
2019-04-20 12:02:51 +00:00
|
|
|
return renderContext.getPreloadFiles()
|
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
renderResourceHints (renderContext) {
|
2020-02-24 22:58:24 +00:00
|
|
|
const resourceHints = renderContext.renderResourceHints()
|
|
|
|
const { build: { crossorigin } } = this.options
|
|
|
|
if (!crossorigin) {
|
|
|
|
return resourceHints
|
|
|
|
}
|
|
|
|
return resourceHints.replace(
|
|
|
|
/rel="preload"/g,
|
|
|
|
`rel="preload" crossorigin="${crossorigin}"`
|
|
|
|
)
|
2019-04-20 12:02:51 +00:00
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
createRenderer () {
|
2019-04-20 12:02:51 +00:00
|
|
|
// Create bundle renderer for SSR
|
|
|
|
return createBundleRenderer(
|
|
|
|
this.serverContext.resources.serverManifest,
|
|
|
|
this.rendererOptions
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
useSSRLog () {
|
2019-05-23 09:49:16 +00:00
|
|
|
if (!this.options.render.ssrLog) {
|
|
|
|
return
|
|
|
|
}
|
2019-05-09 07:06:17 +00:00
|
|
|
const logs = []
|
|
|
|
const devReporter = {
|
2019-07-10 10:45:49 +00:00
|
|
|
log (logObj) {
|
2019-05-19 18:49:24 +00:00
|
|
|
logs.push({
|
|
|
|
...logObj,
|
2019-05-20 16:05:17 +00:00
|
|
|
args: logObj.args.map(arg => format(arg))
|
2019-05-19 18:49:24 +00:00
|
|
|
})
|
2019-05-09 07:06:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
consola.addReporter(devReporter)
|
|
|
|
|
2019-05-23 09:49:16 +00:00
|
|
|
return () => {
|
|
|
|
consola.removeReporter(devReporter)
|
|
|
|
return logs
|
|
|
|
}
|
2019-05-09 07:06:17 +00:00
|
|
|
}
|
|
|
|
|
2019-07-10 10:45:49 +00:00
|
|
|
async render (renderContext) {
|
2019-04-20 12:02:51 +00:00
|
|
|
// Call ssr:context hook to extend context from modules
|
|
|
|
await this.serverContext.nuxt.callHook('vue-renderer:ssr:prepareContext', renderContext)
|
|
|
|
|
2019-05-23 09:49:16 +00:00
|
|
|
const getSSRLog = this.useSSRLog()
|
|
|
|
|
2019-04-20 12:02:51 +00:00
|
|
|
// Call Vue renderer renderToString
|
2019-05-23 09:49:16 +00:00
|
|
|
let APP = await this.vueRenderer.renderToString(renderContext)
|
|
|
|
|
|
|
|
if (typeof getSSRLog === 'function') {
|
|
|
|
renderContext.nuxt.logs = getSSRLog()
|
|
|
|
}
|
2019-04-20 12:02:51 +00:00
|
|
|
|
|
|
|
// Call ssr:context hook
|
|
|
|
await this.serverContext.nuxt.callHook('vue-renderer:ssr:context', renderContext)
|
2020-02-25 16:15:40 +00:00
|
|
|
|
|
|
|
// TODO: Remove in next major release (#4722)
|
|
|
|
await this.serverContext.nuxt.callHook('_render:context', renderContext.nuxt)
|
2019-04-20 12:02:51 +00:00
|
|
|
|
|
|
|
// Fallback to empty response
|
|
|
|
if (!renderContext.nuxt.serverRendered) {
|
|
|
|
APP = `<div id="${this.serverContext.globals.id}"></div>`
|
|
|
|
}
|
|
|
|
|
2020-02-16 13:20:08 +00:00
|
|
|
if (renderContext.redirected && !renderContext._generate) {
|
|
|
|
return {
|
|
|
|
html: APP,
|
|
|
|
error: renderContext.nuxt.error,
|
|
|
|
redirected: renderContext.redirected
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-05 15:15:27 +00:00
|
|
|
let HEAD = ''
|
|
|
|
|
2019-04-20 12:02:51 +00:00
|
|
|
// Inject head meta
|
2019-09-05 15:15:27 +00:00
|
|
|
// (this is unset when features.meta is false in server template)
|
|
|
|
const meta = renderContext.meta && renderContext.meta.inject()
|
|
|
|
if (meta) {
|
|
|
|
HEAD += meta.title.text() +
|
|
|
|
meta.meta.text() +
|
|
|
|
meta.link.text() +
|
|
|
|
meta.style.text() +
|
|
|
|
meta.script.text() +
|
|
|
|
meta.noscript.text()
|
|
|
|
}
|
2019-04-20 12:02:51 +00:00
|
|
|
|
2019-05-25 18:19:10 +00:00
|
|
|
// Check if we need to inject scripts and state
|
|
|
|
const shouldInjectScripts = this.options.render.injectScripts !== false
|
|
|
|
|
2019-04-20 12:02:51 +00:00
|
|
|
// Add <base href=""> meta if router base specified
|
|
|
|
if (this.options._routerBaseSpecified) {
|
|
|
|
HEAD += `<base href="${this.options.router.base}">`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inject resource hints
|
2019-05-25 18:19:10 +00:00
|
|
|
if (this.options.render.resourceHints && shouldInjectScripts) {
|
2019-04-20 12:02:51 +00:00
|
|
|
HEAD += this.renderResourceHints(renderContext)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Inject styles
|
|
|
|
HEAD += renderContext.renderStyles()
|
|
|
|
|
2019-09-05 15:15:27 +00:00
|
|
|
if (meta) {
|
|
|
|
const BODY_PREPEND =
|
|
|
|
meta.meta.text({ pbody: true }) +
|
|
|
|
meta.link.text({ pbody: true }) +
|
|
|
|
meta.style.text({ pbody: true }) +
|
|
|
|
meta.script.text({ pbody: true }) +
|
|
|
|
meta.noscript.text({ pbody: true })
|
2019-08-19 18:38:13 +00:00
|
|
|
|
2019-09-05 15:15:27 +00:00
|
|
|
if (BODY_PREPEND) {
|
|
|
|
APP = `${BODY_PREPEND}${APP}`
|
|
|
|
}
|
2019-08-19 18:38:13 +00:00
|
|
|
}
|
|
|
|
|
2020-01-10 20:43:50 +00:00
|
|
|
const { csp } = this.options.render
|
|
|
|
// Only add the hash if 'unsafe-inline' rule isn't present to avoid conflicts (#5387)
|
|
|
|
const containsUnsafeInlineScriptSrc = csp.policies && csp.policies['script-src'] && csp.policies['script-src'].includes('\'unsafe-inline\'')
|
|
|
|
const shouldHashCspScriptSrc = csp && (csp.unsafeInlineCompatibility || !containsUnsafeInlineScriptSrc)
|
|
|
|
let serializedSession = ''
|
|
|
|
|
2019-04-20 12:02:51 +00:00
|
|
|
// Serialize state
|
2020-01-10 20:43:50 +00:00
|
|
|
if (shouldInjectScripts || shouldHashCspScriptSrc) {
|
|
|
|
// Only serialized session if need inject scripts or csp hash
|
|
|
|
serializedSession = `window.${this.serverContext.globals.context}=${devalue(renderContext.nuxt)};`
|
|
|
|
}
|
|
|
|
|
2019-05-25 18:19:10 +00:00
|
|
|
if (shouldInjectScripts) {
|
|
|
|
APP += `<script>${serializedSession}</script>`
|
|
|
|
}
|
2019-04-20 12:02:51 +00:00
|
|
|
|
|
|
|
// Calculate CSP hashes
|
|
|
|
const cspScriptSrcHashes = []
|
|
|
|
if (csp) {
|
2020-01-10 20:43:50 +00:00
|
|
|
if (shouldHashCspScriptSrc) {
|
2019-04-20 12:02:51 +00:00
|
|
|
const hash = crypto.createHash(csp.hashAlgorithm)
|
|
|
|
hash.update(serializedSession)
|
|
|
|
cspScriptSrcHashes.push(`'${csp.hashAlgorithm}-${hash.digest('base64')}'`)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Call ssr:csp hook
|
|
|
|
await this.serverContext.nuxt.callHook('vue-renderer:ssr:csp', cspScriptSrcHashes)
|
|
|
|
|
|
|
|
// Add csp meta tags
|
|
|
|
if (csp.addMeta) {
|
|
|
|
HEAD += `<meta http-equiv="Content-Security-Policy" content="script-src ${cspScriptSrcHashes.join()}">`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prepend scripts
|
2019-05-25 18:19:10 +00:00
|
|
|
if (shouldInjectScripts) {
|
|
|
|
APP += this.renderScripts(renderContext)
|
|
|
|
}
|
2019-08-19 18:38:13 +00:00
|
|
|
|
2019-09-05 15:15:27 +00:00
|
|
|
if (meta) {
|
|
|
|
// Append body scripts
|
|
|
|
APP += meta.meta.text({ body: true })
|
|
|
|
APP += meta.link.text({ body: true })
|
|
|
|
APP += meta.style.text({ body: true })
|
|
|
|
APP += meta.script.text({ body: true })
|
|
|
|
APP += meta.noscript.text({ body: true })
|
|
|
|
}
|
2019-04-20 12:02:51 +00:00
|
|
|
|
|
|
|
// Template params
|
|
|
|
const templateParams = {
|
2019-09-05 15:15:27 +00:00
|
|
|
HTML_ATTRS: meta ? meta.htmlAttrs.text(true /* addSrrAttribute */) : '',
|
|
|
|
HEAD_ATTRS: meta ? meta.headAttrs.text() : '',
|
|
|
|
BODY_ATTRS: meta ? meta.bodyAttrs.text() : '',
|
2019-04-20 12:02:51 +00:00
|
|
|
HEAD,
|
|
|
|
APP,
|
|
|
|
ENV: this.options.env
|
|
|
|
}
|
|
|
|
|
|
|
|
// Call ssr:templateParams hook
|
|
|
|
await this.serverContext.nuxt.callHook('vue-renderer:ssr:templateParams', templateParams)
|
|
|
|
|
|
|
|
// Render with SSR template
|
|
|
|
const html = this.renderTemplate(this.serverContext.resources.ssrTemplate, templateParams)
|
|
|
|
|
|
|
|
let preloadFiles
|
|
|
|
if (this.options.render.http2.push) {
|
|
|
|
preloadFiles = this.getPreloadFiles(renderContext)
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
html,
|
|
|
|
cspScriptSrcHashes,
|
|
|
|
preloadFiles,
|
|
|
|
error: renderContext.nuxt.error,
|
|
|
|
redirected: renderContext.redirected
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|