Vike has first-class support for HTML streaming (aka SSR streaming) and gives you full control over it.

See API > stream if you merely want to enable/disable HTML streaming.

This page documents how to manually integrate HTML streaming.

If you use vike-react/vike-vue/vike-solid then HTML streaming is already integrated and you can skip reading this page.

Examples

Examples of manually integrating HTML streaming (without using a Vike extension).

React

Vue

Basics

// /renderer/+onRenderHtml.js
 
export { onRenderHtml }
 
import { escapeInject } from 'vike/server'
import { renderToStream } from 'some-ui-framework' // React, Vue, ...
 
async function onRenderHtml(pageContext) {
  const { Page } = pageContext
 
  const stream = renderToStream(Page)
 
  return escapeInject`<!DOCTYPE html>
    <html>
      <body>
        <div id="page-view">${stream}</div>
      </body>
    </html>`
}

Node.js(-like) environments (Node.js/Bun/Deno server, Vercel, AWS EC2, AWS Lambda, ...):

// server.js
 
import { renderPage } from 'vike/server'
 
app.get("*", async (req, res) => {
  const pageContextInit = { urlOriginal: req.url }
  const pageContext = await renderPage(pageContextInit)
  const { httpResponse } = pageContext
  httpResponse.pipe(res)
})

For Node.js(-like) environments we recommend using a Node.js Writable Stream with pageContext.httpResponse.pipe() because it allows the stream to be flushed. In other words: your UI framework can say "now is a good time to flush the stream buffer and send it to the user".

Web Streams (both Writables and Readables) and Node.js Readable Streams don't have this flushing capability.

As far as we know, only React leverages the flushing capability. Thus, this recommendation may be irrelevant if you use a UI framework other than React.

If you don't follow this recommendation, then you can use pageContext.httpResponse.getReadableNodeStream() instead.

Edge platforms (e.g. Cloudflare Workers):

// worker.js
 
import { renderPage } from 'vike/server'
 
addEventListener('fetch', (event) => {
  event.respondWith(handleFetchEvent(event))
})
 
async function handleFetchEvent(event) {
  const pageContextInit = { urlOriginal: event.request.url }
  const pageContext = await renderPage(pageContextInit)
  const { httpResponse } = pageContext
  const readable = httpResponse.getReadableWebStream()
  const { statusCode: status, headers } = httpResponse
  return new Response(readable, { headers, status })
}

If you have a Web Writable Stream then use httpResponse.pipe() (it also works with Web Streams).

API

enableEagerStreaming

By default, the HTML stream isn't immediately sent to the user. Instead, Vike awaits for your UI framework to start its stream.

// /renderer/+onRenderHtml.js
 
export { onRenderHtml }
 
import { renderToStream } from 'some-ui-framework' // React, Vue, ...
 
async function onRenderHtml(pageContext) {
  const { Page } = pageContext
 
  const stream = renderToStream(Page)
 
  // The HTML template below isn't immediately sent to the user.
  // Instead, Vike awaits for `stream` to start emitting.
  return escapeInject`<!DOCTYPE html>
    <html>
      <head>
        <title>Hello</title>
      </head>
      <body>
        <div id="page-view">${stream}</div>
      </body>
    </html>`
}

If you set pageContext.enableEagerStreaming to true then Vike starts emitting the HTML template right away.

async function onRenderHtml(pageContext) {
  // The HTML template below is immediately sent to the user.
  const documentHtml = escapeInject`<!DOCTYPE html>
    <html>
      <head>
        <title>Hello</title>
      </head>
      <body>
        <div id="page-view">${renderToStream(pageContext.Page)}</div>
      </body>
    </html>`
 
  return {
    documentHtml,
    pageContext: {
      enableEagerStreaming: true
    }
  }
}

Make sure your server (or any proxy between your server and the user) doesn't buffer the stream, otherwise you may still notice a delay.

stampPipe()

If your UI framework provides a stream pipe, then you need to use stampPipe().

// /renderer/+onRenderHtml.js
 
export { onRenderHtml }
 
import { renderToStreamPipe } from 'some-ui-framework' // React, Vue, ...
import { escapeInject, stampPipe } from 'vike/server'
 
async function onRenderHtml(pageContext) {
  const { Page } = pageContext
 
  const pipe = renderToStreamPipe(Page)
 
  // If `pipe(writable)` expects `writable` to be a Writable Node.js Stream
  stampPipe(pipe, 'node-stream')
  // If `pipe(writable)` expects `writable` to be a Writable Web Stream
  stampPipe(pipe, 'web-stream')
 
  return escapeInject`<!DOCTYPE html>
    <html>
      <body>
        <div id="page-view">${pipe}</div>
      </body>
    </html>`
}

For Node.js:

// server.js
 
const pageContext = await renderPage(pageContextInit)
const { httpResponse } = pageContext
// Using pageContext.httpResponse.pipe() as usual
httpResponse.pipe(res)

If your server expects a readable stream (e.g. Cloudflare Workers) you can use new TransformStream():

// worker.js
 
const { readable, writable } = new TransformStream()
httpResponse.pipe(writable)
const resp = new Response(readable)

For some UI frameworks, such as Vue, you need a pipe wrapper:

// /renderer/+onRenderHtml.js
 
export { onRenderHtml }
 
import { pipePageToWritable } from 'some-ui-framework'
import { stampPipe, escapeInject } from 'vike/server'
 
async function onRenderHtml(pageContext) {
  // Using a pipe wrapper so that pipePageToWritable() can access pageContext.Page
  const pipeWrapper = (writable) => {
    pipePageToWritable(pageContext.Page, writable)
  }
  stampPipe(pipeWrapper, 'node-stream')
 
  return escapeInject`<!DOCTYPE html>
    <html>
      <body>
        <div id="page-view">${pipeWrapper}</div>
      </body>
    </html>`
}

See /examples/cloudflare-workers-vue for an example of using a pipe wrapper with Vue's pipeToWebWritable()/pipeToNodeWritable(), as well as using new TransformStream() for Cloudflare Workers.

pageContext.httpResponse.getBody()

You can convert the stream to a string:

/* This won't work: (a stream cannot be consumed synchronously)
const { body } = httpResponse
res.send(body)
*/
 
// But this works:
const body = await httpResponse.getBody()
assert(typeof body === 'string')
res.send(body)

Initial data after stream end

Some data fetching tools, such as Relay and Vue's onServerPrefetch(), collect data during the stream.

Consequently, you can determine the initial data (which needs to be passed to the client-side) only after the stream has ended.

In such situations, you can return a pageContext async function in your onRenderHtml() hook:

// /renderer/+onRenderHtml.js
 
export { onRenderHtml }
 
import { escapeInject } from 'vike/server'
import { renderToStream } from 'some-ui-framework' // React, Vue, ...
 
async function onRenderHtml(pageContext) {
  const { Page } = pageContext
 
  const stream = renderToStream(Page)
 
  const documentHtml = escapeInject`<!DOCTYPE html>
    <html>
      <body>
        <div id="page-view">${stream}</div>
      </body>
    </html>`
 
  const pageContextPromise = async () => {
     // I'm called after the stream has ended
     return {
       initialData,
     }
  }
 
  return {
    documentHtml,
    pageContext: pageContextPromise
  }
}

Progressive Rendering

Some UI frameworks, such as React, support progressive rendering: while some parts of the UI are being loaded, other parts are already rendered (and already hydrated).

Instead of using HTML streaming, an easy alternative is to use a stateful component. But the issue with that approach is that the content isn't rendered to HTML. For example, a product page fetching its content from a database won't get the SEO and performance advantages of SSR.

With HTML streaming, all content is rendered to HTML.

Vike has first-class support for HTML streaming and progressive rendering.

Example: vike-react > examples/full/.

See also: React > React Server Components.

See also