server.js 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. import Vue from 'vue'
  2. import { joinURL, normalizeURL, withQuery } from 'ufo'
  3. import fetch from 'node-fetch-native'
  4. import middleware from './middleware.js'
  5. import {
  6. applyAsyncData,
  7. middlewareSeries,
  8. sanitizeComponent,
  9. getMatchedComponents,
  10. promisify
  11. } from './utils.js'
  12. import fetchMixin from './mixins/fetch.server'
  13. import { createApp, NuxtError } from './index.js'
  14. import NuxtLink from './components/nuxt-link.server.js' // should be included after ./index.js
  15. // Update serverPrefetch strategy
  16. Vue.config.optionMergeStrategies.serverPrefetch = Vue.config.optionMergeStrategies.created
  17. // Fetch mixin
  18. if (!Vue.__nuxt__fetch__mixin__) {
  19. Vue.mixin(fetchMixin)
  20. Vue.__nuxt__fetch__mixin__ = true
  21. }
  22. // Component: <NuxtLink>
  23. Vue.component(NuxtLink.name, NuxtLink)
  24. Vue.component('NLink', NuxtLink)
  25. if (!global.fetch) { global.fetch = fetch }
  26. const noopApp = () => new Vue({ render: h => h('div', { domProps: { id: '__nuxt' } }) })
  27. const createNext = ssrContext => (opts) => {
  28. // If static target, render on client-side
  29. ssrContext.redirected = opts
  30. if (ssrContext.target === 'static' || !ssrContext.res) {
  31. ssrContext.nuxt.serverRendered = false
  32. return
  33. }
  34. let fullPath = withQuery(opts.path, opts.query)
  35. const $config = ssrContext.nuxt.config || {}
  36. const routerBase = ($config._app && $config._app.basePath) || '/'
  37. if (!fullPath.startsWith('http') && (routerBase !== '/' && !fullPath.startsWith(routerBase))) {
  38. fullPath = joinURL(routerBase, fullPath)
  39. }
  40. // Avoid loop redirect
  41. if (decodeURI(fullPath) === decodeURI(ssrContext.url)) {
  42. ssrContext.redirected = false
  43. return
  44. }
  45. ssrContext.res.writeHead(opts.status, {
  46. Location: normalizeURL(fullPath)
  47. })
  48. ssrContext.res.end()
  49. }
  50. // This exported function will be called by `bundleRenderer`.
  51. // This is where we perform data-prefetching to determine the
  52. // state of our application before actually rendering it.
  53. // Since data fetching is async, this function is expected to
  54. // return a Promise that resolves to the app instance.
  55. export default async (ssrContext) => {
  56. // Create ssrContext.next for simulate next() of beforeEach() when wanted to redirect
  57. ssrContext.redirected = false
  58. ssrContext.next = createNext(ssrContext)
  59. // Used for beforeNuxtRender({ Components, nuxtState })
  60. ssrContext.beforeRenderFns = []
  61. // for beforeSerialize(nuxtState)
  62. ssrContext.beforeSerializeFns = []
  63. // Nuxt object (window.{{globals.context}}, defaults to window.__NUXT__)
  64. ssrContext.nuxt = { layout: 'default', data: [], fetch: { }, error: null , state: null, serverRendered: true, routePath: ''
  65. }
  66. ssrContext.fetchCounters = { }
  67. // Remove query from url is static target
  68. // Public runtime config
  69. ssrContext.nuxt.config = ssrContext.runtimeConfig.public
  70. if (ssrContext.nuxt.config._app) {
  71. __webpack_public_path__ = joinURL(ssrContext.nuxt.config._app.cdnURL, ssrContext.nuxt.config._app.assetsPath)
  72. }
  73. // Create the app definition and the instance (created for each request)
  74. const { app, router, store } = await createApp(ssrContext, ssrContext.runtimeConfig.private)
  75. const _app = new Vue(app)
  76. // Add ssr route path to nuxt context so we can account for page navigation between ssr and csr
  77. ssrContext.nuxt.routePath = app.context.route.path
  78. // Add meta infos (used in renderer.js)
  79. ssrContext.meta = _app.$meta()
  80. // Keep asyncData for each matched component in ssrContext (used in app/utils.js via this.$ssrContext)
  81. ssrContext.asyncData = { }
  82. const beforeRender = async () => {
  83. // Call beforeNuxtRender() methods
  84. await Promise.all(ssrContext.beforeRenderFns.map(fn => promisify(fn, { Components, nuxtState: ssrContext.nuxt })))
  85. ssrContext.rendered = () => {
  86. // Call beforeSerialize() hooks
  87. ssrContext.beforeSerializeFns.forEach(fn => fn(ssrContext.nuxt))
  88. // Add the state from the vuex store
  89. ssrContext.nuxt.state = store.state
  90. }
  91. }
  92. const renderErrorPage = async () => {
  93. // Don't server-render the page in static target
  94. if (ssrContext.target === 'static') {
  95. ssrContext.nuxt.serverRendered = false
  96. }
  97. // Load layout for error page
  98. const layout = (NuxtError.options || NuxtError).layout
  99. const errLayout = typeof layout === 'function' ? layout.call(NuxtError, app.context) : layout
  100. ssrContext.nuxt.layout = errLayout || 'default'
  101. await _app.loadLayout(errLayout)
  102. _app.setLayout(errLayout)
  103. await beforeRender()
  104. return _app
  105. }
  106. const render404Page = () => {
  107. app.context.error({ statusCode: 404, path: ssrContext.url, message: 'This page could not be found' })
  108. return renderErrorPage()
  109. }
  110. // Components are already resolved by setContext -> getRouteData (app/utils.js)
  111. const Components = getMatchedComponents(app.context.route)
  112. /*
  113. ** Dispatch store nuxtServerInit
  114. */
  115. if (store._actions && store._actions.nuxtServerInit) {
  116. try {
  117. await store.dispatch('nuxtServerInit', app.context)
  118. } catch (err) {
  119. console.debug('Error occurred when calling nuxtServerInit: ', err.message)
  120. throw err
  121. }
  122. }
  123. // ...If there is a redirect or an error, stop the process
  124. if (ssrContext.redirected) {
  125. return noopApp()
  126. }
  127. if (ssrContext.nuxt.error) {
  128. return renderErrorPage()
  129. }
  130. /*
  131. ** Call global middleware (nuxt.config.js)
  132. */
  133. let midd = ["redirect"]
  134. midd = midd.map((name) => {
  135. if (typeof name === 'function') {
  136. return name
  137. }
  138. if (typeof middleware[name] !== 'function') {
  139. app.context.error({ statusCode: 500, message: 'Unknown middleware ' + name })
  140. }
  141. return middleware[name]
  142. })
  143. await middlewareSeries(midd, app.context)
  144. // ...If there is a redirect or an error, stop the process
  145. if (ssrContext.redirected) {
  146. return noopApp()
  147. }
  148. if (ssrContext.nuxt.error) {
  149. return renderErrorPage()
  150. }
  151. /*
  152. ** Set layout
  153. */
  154. let layout = Components.length ? Components[0].options.layout : NuxtError.layout
  155. if (typeof layout === 'function') {
  156. layout = layout(app.context)
  157. }
  158. await _app.loadLayout(layout)
  159. if (ssrContext.nuxt.error) {
  160. return renderErrorPage()
  161. }
  162. layout = _app.setLayout(layout)
  163. ssrContext.nuxt.layout = _app.layoutName
  164. /*
  165. ** Call middleware (layout + pages)
  166. */
  167. midd =[]
  168. layout = sanitizeComponent(layout)
  169. if (layout.options.middleware) {
  170. midd = midd.concat(layout.options.middleware)
  171. }
  172. Components.forEach((Component) => {
  173. if (Component.options.middleware) {
  174. midd = midd.concat(Component.options.middleware)
  175. }
  176. })
  177. midd = midd.map((name) => {
  178. if (typeof name === 'function') {
  179. return name
  180. }
  181. if (typeof middleware[name] !== 'function') {
  182. app.context.error({ statusCode: 500, message: 'Unknown middleware ' + name })
  183. }
  184. return middleware[name]
  185. })
  186. await middlewareSeries(midd, app.context)
  187. // ...If there is a redirect or an error, stop the process
  188. if (ssrContext.redirected) {
  189. return noopApp()
  190. }
  191. if (ssrContext.nuxt.error) {
  192. return renderErrorPage()
  193. }
  194. /*
  195. ** Call .validate()
  196. */
  197. let isValid = true
  198. try {
  199. for (const Component of Components) {
  200. if (typeof Component.options.validate !== 'function') {
  201. continue
  202. }
  203. isValid = await Component.options.validate(app.context)
  204. if (!isValid) {
  205. break
  206. }
  207. }
  208. } catch (validationError) {
  209. // ...If .validate() threw an error
  210. app.context.error({
  211. statusCode: validationError.statusCode || '500',
  212. message: validationError.message
  213. })
  214. return renderErrorPage()
  215. }
  216. // ...If .validate() returned false
  217. if (!isValid) {
  218. // Render a 404 error page
  219. return render404Page()
  220. }
  221. // If no Components found, returns 404
  222. if (!Components.length) {
  223. return render404Page()
  224. }
  225. // Call asyncData & fetch hooks on components matched by the route.
  226. const asyncDatas = await Promise.all(Components.map((Component) => {
  227. const promises = []
  228. // Call asyncData(context)
  229. if (Component.options.asyncData && typeof Component.options.asyncData === 'function') {
  230. const promise = promisify(Component.options.asyncData, app.context)
  231. .then((asyncDataResult) => {
  232. ssrContext.asyncData[Component.cid] = asyncDataResult
  233. applyAsyncData(Component)
  234. return asyncDataResult
  235. })
  236. promises.push(promise)
  237. } else {
  238. promises.push(null)
  239. }
  240. // Call fetch(context)
  241. if (Component.options.fetch && Component.options.fetch.length) {
  242. promises.push(Component.options.fetch(app.context))
  243. } else {
  244. promises.push(null)
  245. }
  246. return Promise.all(promises)
  247. }))
  248. // datas are the first row of each
  249. ssrContext.nuxt.data = asyncDatas.map(r => r[0] || {})
  250. // ...If there is a redirect or an error, stop the process
  251. if (ssrContext.redirected) {
  252. return noopApp()
  253. }
  254. if (ssrContext.nuxt.error) {
  255. return renderErrorPage()
  256. }
  257. // Call beforeNuxtRender methods & add store state
  258. await beforeRender()
  259. return _app
  260. }