mirror of
https://github.com/nuxt/nuxt.git
synced 2024-11-21 21:25:11 +00:00
wip
This commit is contained in:
parent
0223761847
commit
409e8ebf37
@ -1,21 +1,15 @@
|
|||||||
import { AsyncLocalStorage } from 'node:async_hooks'
|
import { AsyncLocalStorage } from 'node:async_hooks'
|
||||||
import {
|
import { createRenderer, getPrefetchLinks, getPreloadLinks, getRequestDependencies, getResources, renderResourceHeaders } from 'vue-bundle-renderer/runtime'
|
||||||
createRenderer,
|
|
||||||
getPrefetchLinks,
|
|
||||||
getPreloadLinks,
|
|
||||||
getRequestDependencies,
|
|
||||||
renderResourceHeaders,
|
|
||||||
} from 'vue-bundle-renderer/runtime'
|
|
||||||
import type { Manifest as ClientManifest } from 'vue-bundle-renderer'
|
import type { Manifest as ClientManifest } from 'vue-bundle-renderer'
|
||||||
import type { RenderResponse } from 'nitro/types'
|
import type { NitroApp, RenderResponse } from 'nitro/types'
|
||||||
import type { Manifest } from 'vite'
|
import type { Manifest } from 'vite'
|
||||||
import type { H3Event } from 'h3'
|
import type { H3Event } from 'h3'
|
||||||
import { appendResponseHeader, createError, getQuery, getResponseStatus, getResponseStatusText, readBody, writeEarlyHints } from 'h3'
|
import { appendResponseHeader, createError, getHeader, getQuery, getResponseStatus, getResponseStatusText, readBody, writeEarlyHints } from 'h3'
|
||||||
import devalue from '@nuxt/devalue'
|
import devalue from '@nuxt/devalue'
|
||||||
import { stringify, uneval } from 'devalue'
|
import { stringify, uneval } from 'devalue'
|
||||||
import destr from 'destr'
|
import destr from 'destr'
|
||||||
import { getQuery as getURLQuery, joinURL, withoutTrailingSlash } from 'ufo'
|
import { getQuery as getURLQuery, joinURL, withoutTrailingSlash } from 'ufo'
|
||||||
import { renderToString as _renderToString } from 'vue/server-renderer'
|
import { renderToString as _renderToString, renderToWebStream } from 'vue/server-renderer'
|
||||||
import { hash } from 'ohash'
|
import { hash } from 'ohash'
|
||||||
import { propsToString, renderSSRHead } from '@unhead/ssr'
|
import { propsToString, renderSSRHead } from '@unhead/ssr'
|
||||||
import type { HeadEntryOptions } from '@unhead/schema'
|
import type { HeadEntryOptions } from '@unhead/schema'
|
||||||
@ -339,6 +333,10 @@ export default defineRenderHandler(async (event): Promise<Partial<RenderResponse
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!isRenderingIsland && !isRenderingPayload && !isCrawler(event) /* && isWebBrowser(event) */) {
|
||||||
|
return streamedResponse(event, ssrContext, renderer, head, nitroApp)
|
||||||
|
}
|
||||||
|
|
||||||
const _rendered = await renderer.renderToString(ssrContext).catch(async (error) => {
|
const _rendered = await renderer.renderToString(ssrContext).catch(async (error) => {
|
||||||
// We use error to bypass full render if we have an early response we can make
|
// We use error to bypass full render if we have an early response we can make
|
||||||
if (ssrContext._renderResponse && error.message === 'skipping render') { return {} as ReturnType<typeof renderer['renderToString']> }
|
if (ssrContext._renderResponse && error.message === 'skipping render') { return {} as ReturnType<typeof renderer['renderToString']> }
|
||||||
@ -710,3 +708,192 @@ function replaceIslandTeleports (ssrContext: NuxtSSRContext, html: string) {
|
|||||||
}
|
}
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function renderStreamedHTMLDocument (html: NuxtRenderHTMLContext) {
|
||||||
|
return [
|
||||||
|
'<!DOCTYPE html>' +
|
||||||
|
`<html${joinAttrs(html.htmlAttrs)}>` +
|
||||||
|
`<head>${joinTags(html.head)}</head>` +
|
||||||
|
`<body${joinAttrs(html.bodyAttrs)}>${joinTags(html.bodyPrepend)}` +
|
||||||
|
APP_ROOT_OPEN_TAG,
|
||||||
|
// HTML body will be streamed here
|
||||||
|
APP_ROOT_CLOSE_TAG +
|
||||||
|
`</body>` +
|
||||||
|
'</html>',
|
||||||
|
]
|
||||||
|
}
|
||||||
|
async function streamedResponse (event: H3Event, ssrContext: NuxtSSRContext, renderer: Awaited<ReturnType<typeof getSSRRenderer>>, head: ReturnType<typeof createServerHead>, nitroApp: NitroApp) {
|
||||||
|
const createApp = await Promise.resolve(getServerEntry()).then(r => 'default' in r ? r.default : r)
|
||||||
|
|
||||||
|
// Create a ReadableStream
|
||||||
|
const stream = new ReadableStream({
|
||||||
|
start (controller) {
|
||||||
|
const render = async () => {
|
||||||
|
const _PAYLOAD_EXTRACTION = import.meta.prerender && process.env.NUXT_PAYLOAD_EXTRACTION && !ssrContext.noSSR
|
||||||
|
const routeOptions = getRouteRules(event)
|
||||||
|
const NO_SCRIPTS = process.env.NUXT_NO_SCRIPTS || routeOptions.experimentalNoScripts
|
||||||
|
const payloadURL = _PAYLOAD_EXTRACTION ? joinURL(ssrContext.runtimeConfig.app.cdnURL || ssrContext.runtimeConfig.app.baseURL, event.path, process.env.NUXT_JSON_PAYLOADS ? '_payload.json' : '_payload.js') + '?' + ssrContext.runtimeConfig.app.buildId : undefined
|
||||||
|
const inlinedStyles = (process.env.NUXT_INLINE_STYLES) ? await renderInlineStyles(ssrContext.modules ?? []) : []
|
||||||
|
|
||||||
|
// Setup head
|
||||||
|
const { styles, scripts } = getRequestDependencies(ssrContext, renderer.rendererContext)
|
||||||
|
// 1.Extracted payload preloading
|
||||||
|
if (_PAYLOAD_EXTRACTION && !NO_SCRIPTS) {
|
||||||
|
head.push({
|
||||||
|
link: [
|
||||||
|
process.env.NUXT_JSON_PAYLOADS
|
||||||
|
? { rel: 'preload', as: 'fetch', crossorigin: 'anonymous', href: payloadURL }
|
||||||
|
: { rel: 'modulepreload', href: payloadURL },
|
||||||
|
],
|
||||||
|
}, { mode: 'server' })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Styles
|
||||||
|
// cacheable
|
||||||
|
head.push({ style: inlinedStyles })
|
||||||
|
const link: Link[] = []
|
||||||
|
for (const style in styles) {
|
||||||
|
const resource = styles[style]
|
||||||
|
// Do not add links to resources that are inlined (vite v5+)
|
||||||
|
if (import.meta.dev && 'inline' in getURLQuery(resource.file)) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
link.push({ rel: 'stylesheet', href: renderer.rendererContext.buildAssetsURL(resource.file) })
|
||||||
|
}
|
||||||
|
head.push({ link }, { mode: 'server' })
|
||||||
|
|
||||||
|
// 5. Scripts
|
||||||
|
if (!routeOptions.experimentalNoScripts) {
|
||||||
|
head.push({
|
||||||
|
script: Object.values(scripts).map(resource => (<Script> {
|
||||||
|
type: resource.module ? 'module' : null,
|
||||||
|
src: renderer.rendererContext.buildAssetsURL(resource.file),
|
||||||
|
defer: resource.module ? null : true,
|
||||||
|
// if we are rendering script tag payloads that import an async payload
|
||||||
|
// we need to ensure this resolves before executing the Nuxt entry
|
||||||
|
tagPosition: (_PAYLOAD_EXTRACTION && !process.env.NUXT_JSON_PAYLOADS) ? 'bodyClose' : 'head',
|
||||||
|
crossorigin: '',
|
||||||
|
})),
|
||||||
|
}, { mode: 'server' })
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove certain tags for nuxt islands
|
||||||
|
const { headTags, bodyTags, bodyTagsOpen, htmlAttrs, bodyAttrs } = await renderSSRHead(head, renderSSRHeadOptions)
|
||||||
|
|
||||||
|
// Create render context
|
||||||
|
const htmlContext: NuxtRenderHTMLContext = {
|
||||||
|
island: false,
|
||||||
|
htmlAttrs: htmlAttrs ? [htmlAttrs] : [],
|
||||||
|
head: normalizeChunks([headTags]),
|
||||||
|
bodyAttrs: bodyAttrs ? [bodyAttrs] : [],
|
||||||
|
bodyPrepend: normalizeChunks([bodyTagsOpen, ssrContext.teleports?.body]),
|
||||||
|
body: [],
|
||||||
|
bodyAppend: [bodyTags],
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allow hooking into the rendered result
|
||||||
|
await nitroApp.hooks.callHook('render:html', htmlContext, { event })
|
||||||
|
const [open, close] = renderStreamedHTMLDocument(htmlContext)
|
||||||
|
controller.enqueue(open)
|
||||||
|
|
||||||
|
try {
|
||||||
|
const app = await createApp(ssrContext)
|
||||||
|
const vueStream = renderToWebStream(app, ssrContext)
|
||||||
|
const reader = vueStream.getReader()
|
||||||
|
const decoder = new TextDecoder()
|
||||||
|
let result: ReadableStreamReadResult<any>
|
||||||
|
while (result = await reader.read(), !result.done) {
|
||||||
|
controller.enqueue(decoder.decode(result.value))
|
||||||
|
}
|
||||||
|
// TODO: handle type mismatch
|
||||||
|
await ssrContext.nuxt?.hooks.callHook('app:rendered', { ssrContext })
|
||||||
|
controller.enqueue(APP_TELEPORT_OPEN_TAG + (HAS_APP_TELEPORTS ? joinTags([ssrContext.teleports?.[`#${appTeleportAttrs.id}`]]) : '') + APP_TELEPORT_CLOSE_TAG)
|
||||||
|
controller.enqueue(close)
|
||||||
|
|
||||||
|
if (!NO_SCRIPTS) {
|
||||||
|
// 4. Payloads
|
||||||
|
head.push({
|
||||||
|
script: _PAYLOAD_EXTRACTION
|
||||||
|
? process.env.NUXT_JSON_PAYLOADS
|
||||||
|
? renderPayloadJsonScript({ id: '__NUXT_DATA__', ssrContext, data: splitPayload(ssrContext).initial, src: payloadURL })
|
||||||
|
: renderPayloadScript({ ssrContext, data: splitPayload(ssrContext).initial, src: payloadURL })
|
||||||
|
: process.env.NUXT_JSON_PAYLOADS
|
||||||
|
? renderPayloadJsonScript({ id: '__NUXT_DATA__', ssrContext, data: ssrContext.payload })
|
||||||
|
: renderPayloadScript({ ssrContext, data: ssrContext.payload }),
|
||||||
|
}, {
|
||||||
|
mode: 'server',
|
||||||
|
// this should come before another end of body scripts
|
||||||
|
tagPosition: 'bodyClose',
|
||||||
|
tagPriority: 'high',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const { bodyTags } = await renderSSRHead(head, renderSSRHeadOptions)
|
||||||
|
controller.enqueue(joinTags([bodyTags]))
|
||||||
|
controller.close()
|
||||||
|
// console.log(ssrContext.payload?.error)
|
||||||
|
|
||||||
|
if (ssrContext.payload?.error) {
|
||||||
|
// controller.error(ssrContext.payload.error)
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
if (ssrContext._renderResponse && error.message === 'skipping render') {
|
||||||
|
// skip render
|
||||||
|
controller.close()
|
||||||
|
console.log('skipping render')
|
||||||
|
}
|
||||||
|
// // We use error to bypass full render if we have an early response we can make
|
||||||
|
// if (ssrContext._renderResponse && error.message === 'skipping render') { return {} as ReturnType<typeof renderer['renderToString']> }
|
||||||
|
|
||||||
|
// // Use explicitly thrown error in preference to subsequent rendering errors
|
||||||
|
// const _err = (!ssrError && ssrContext.payload?.error) || error
|
||||||
|
// await ssrContext.nuxt?.hooks.callHook('app:error', _err)
|
||||||
|
// throw _err
|
||||||
|
// })
|
||||||
|
// await ssrContext.nuxt?.hooks.callHook('app:rendered', { ssrContext, renderResult: _rendered })
|
||||||
|
|
||||||
|
// if (ssrContext._renderResponse) { return ssrContext._renderResponse }
|
||||||
|
|
||||||
|
// // Handle errors
|
||||||
|
// if (ssrContext.payload?.error && !ssrError) {
|
||||||
|
// throw ssrContext.payload.error
|
||||||
|
// }
|
||||||
|
controller.error(error)
|
||||||
|
console.log('error in stream')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
render()
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const headers: Record<string, string> = {
|
||||||
|
'content-type': 'text/html;charset=utf-8',
|
||||||
|
'x-powered-by': 'Nuxt',
|
||||||
|
}
|
||||||
|
|
||||||
|
const resourceHeaders = renderResourceHeaders(ssrContext, renderer.rendererContext)
|
||||||
|
|
||||||
|
if (resourceHeaders.link) {
|
||||||
|
headers.link = resourceHeaders.link
|
||||||
|
}
|
||||||
|
|
||||||
|
const streamedResponse = {
|
||||||
|
body: stream,
|
||||||
|
statusCode: getResponseStatus(event),
|
||||||
|
statusMessage: getResponseStatusText(event),
|
||||||
|
headers,
|
||||||
|
} satisfies RenderResponse
|
||||||
|
|
||||||
|
return streamedResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
function isWebBrowser (event: H3Event) {
|
||||||
|
const agent = getHeader(event, 'user-agent')
|
||||||
|
return agent && agent.includes('Mozilla')
|
||||||
|
}
|
||||||
|
|
||||||
|
const BOT_RE = /bot\b|index|spider|facebookexternalhit|crawl|wget|slurp|mediapartners-google|whatsapp/i
|
||||||
|
function isCrawler (event: H3Event) {
|
||||||
|
const agent = getHeader(event, 'user-agent')
|
||||||
|
return agent && BOT_RE.test(agent)
|
||||||
|
}
|
||||||
|
@ -1,13 +0,0 @@
|
|||||||
<script setup lang="ts">
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<!-- Edit this file to play around with Nuxt but never commit changes! -->
|
|
||||||
<div>
|
|
||||||
Nuxt Playground
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
|
|
||||||
</style>
|
|
14
playground/components/TestMe.vue
Normal file
14
playground/components/TestMe.vue
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
await new Promise((resolve) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
resolve()
|
||||||
|
}, 2000)
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
test component
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
24
playground/pages/index.vue
Normal file
24
playground/pages/index.vue
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
const colors = ['red', 'blue', 'green']
|
||||||
|
const color = ref('blue')
|
||||||
|
if (import.meta.client) {
|
||||||
|
setInterval(() => {
|
||||||
|
color.value = colors[Math.floor(Math.random() * colors.length)]
|
||||||
|
console.log('updating color', color.value)
|
||||||
|
}, 200)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
Nuxt Playground before long component
|
||||||
|
<TestMe />
|
||||||
|
after long component
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
div {
|
||||||
|
background-color: v-bind(color);
|
||||||
|
}
|
||||||
|
</style>
|
Loading…
Reference in New Issue
Block a user