2022-08-04 11:13:12 +00:00
|
|
|
import { performance } from 'node:perf_hooks'
|
2022-09-03 10:02:14 +00:00
|
|
|
import { createError } from 'h3'
|
2022-04-12 10:04:55 +00:00
|
|
|
import { ViteNodeRunner } from 'vite-node/client'
|
|
|
|
import { $fetch } from 'ohmyfetch'
|
2022-08-04 11:13:12 +00:00
|
|
|
import consola from 'consola'
|
2022-04-12 10:04:55 +00:00
|
|
|
import { getViteNodeOptions } from './vite-node-shared.mjs'
|
|
|
|
|
|
|
|
const viteNodeOptions = getViteNodeOptions()
|
2022-09-03 10:02:14 +00:00
|
|
|
const runner = createRunner()
|
2022-04-12 10:04:55 +00:00
|
|
|
let render
|
|
|
|
|
|
|
|
export default async (ssrContext) => {
|
|
|
|
// Workaround for stub mode
|
|
|
|
// https://github.com/nuxt/framework/pull/3983
|
|
|
|
process.server = true
|
2022-08-04 10:03:46 +00:00
|
|
|
|
|
|
|
// Invalidate cache for files changed since last rendering
|
|
|
|
const invalidates = await $fetch('/invalidates', {
|
|
|
|
baseURL: viteNodeOptions.baseURL
|
|
|
|
})
|
2022-08-18 08:05:37 +00:00
|
|
|
const updates = runner.moduleCache.invalidateDepTree(invalidates)
|
2022-08-04 10:03:46 +00:00
|
|
|
|
|
|
|
// Execute SSR bundle on demand
|
2022-08-04 11:13:12 +00:00
|
|
|
const start = performance.now()
|
2022-09-08 14:17:22 +00:00
|
|
|
render = (updates.has(viteNodeOptions.entryPath) || !render) ? (await runner.executeFile(viteNodeOptions.entryPath)).default : render
|
2022-08-04 11:13:12 +00:00
|
|
|
if (updates.size) {
|
|
|
|
const time = Math.round((performance.now() - start) * 1000) / 1000
|
|
|
|
consola.success(`Vite server hmr ${updates.size} files`, time ? `in ${time}ms` : '')
|
|
|
|
}
|
|
|
|
|
2022-04-12 10:04:55 +00:00
|
|
|
const result = await render(ssrContext)
|
|
|
|
return result
|
|
|
|
}
|
2022-09-03 10:02:14 +00:00
|
|
|
|
|
|
|
function createRunner () {
|
|
|
|
return new ViteNodeRunner({
|
|
|
|
root: viteNodeOptions.root, // Equals to Nuxt `srcDir`
|
|
|
|
base: viteNodeOptions.base,
|
|
|
|
async fetchModule (id) {
|
2022-09-09 09:54:20 +00:00
|
|
|
// TODO: fix in vite-node
|
|
|
|
id = id.replace(/\/\//g, '/')
|
2022-09-03 10:02:14 +00:00
|
|
|
return await $fetch('/module/' + encodeURI(id), {
|
|
|
|
baseURL: viteNodeOptions.baseURL
|
|
|
|
}).catch((err) => {
|
|
|
|
const errorData = err?.data?.data
|
|
|
|
if (!errorData) {
|
|
|
|
throw err
|
|
|
|
}
|
2022-09-14 15:58:28 +00:00
|
|
|
let _err
|
2022-09-03 10:02:14 +00:00
|
|
|
try {
|
|
|
|
const { message, stack } = formatViteError(errorData)
|
2022-09-14 15:58:28 +00:00
|
|
|
_err = createError({
|
2022-09-03 10:02:14 +00:00
|
|
|
statusMessage: 'Vite Error',
|
|
|
|
message,
|
|
|
|
stack
|
|
|
|
})
|
|
|
|
} catch (err) {
|
|
|
|
// This should not happen unless there is an internal error with formatViteError!
|
|
|
|
consola.error('Error while formatting vite error:', errorData)
|
|
|
|
throw createError({
|
|
|
|
statusMessage: 'Vite Error',
|
|
|
|
message: errorData.message || 'Vite Error',
|
|
|
|
stack: 'Vite Error\nat [check console]'
|
|
|
|
})
|
|
|
|
}
|
2022-09-14 15:58:28 +00:00
|
|
|
throw _err
|
2022-09-03 10:02:14 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
function formatViteError (errorData) {
|
|
|
|
const errorCode = errorData.name || errorData.reasonCode || errorData.code
|
|
|
|
const frame = errorData.frame || errorData.source || errorData.pluginCode
|
|
|
|
|
|
|
|
const getLocId = (locObj = {}) => locObj.file || locObj.id || locObj.url || ''
|
|
|
|
const getLocPos = (locObj = {}) => locObj.line ? `${locObj.line}:${locObj.column || 0}` : ''
|
|
|
|
const locId = getLocId(errorData.loc) || getLocId(errorData.location) || getLocId(errorData.input) || getLocId(errorData)
|
|
|
|
const locPos = getLocPos(errorData.loc) || getLocPos(errorData.location) || getLocPos(errorData.input) || getLocPos(errorData)
|
|
|
|
const loc = locId.replace(process.cwd(), '.') + (locPos ? `:${locPos}` : '')
|
|
|
|
|
|
|
|
const message = [
|
|
|
|
'[vite-node]',
|
|
|
|
errorData.plugin && `[plugin:${errorData.plugin}]`,
|
|
|
|
errorCode && `[${errorCode}]`,
|
|
|
|
loc,
|
|
|
|
errorData.reason && `: ${errorData.reason}`,
|
|
|
|
frame && `<br><pre>${frame.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')}</pre><br>`
|
|
|
|
].filter(Boolean).join(' ')
|
|
|
|
|
|
|
|
const stack = [
|
|
|
|
message,
|
|
|
|
'at ' + loc,
|
|
|
|
errorData.stack
|
|
|
|
].filter(Boolean).join('\n')
|
|
|
|
|
|
|
|
return {
|
|
|
|
message,
|
|
|
|
stack
|
|
|
|
}
|
|
|
|
}
|