Serverseitiges Rendern (SSR)
HINWEIS
SSR bezieht sich speziell auf Frontend-Frameworks (zum Beispiel React, Preact, Vue und Svelte), die die Ausführung derselben Anwendung in Node.js unterstützen, sie in HTML vorrendern und schließlich im Client hydratisieren. Wenn Sie nach Integration mit herkömmlichen serverseitigen Frameworks suchen, werfen Sie stattdessen einen Blick auf den Leitfaden zur Backend-Integration.
Der folgende Leitfaden setzt auch voraus, dass Sie bereits Erfahrung im Umgang mit SSR in Ihrem Framework Ihrer Wahl haben und konzentriert sich nur auf Vite-spezifische Integrationsdetails.
Niedrigstufige API
Dies ist eine niedrigstufige API, die für Bibliotheks- und Framework-Autoren gedacht ist. Wenn Ihr Ziel darin besteht, eine Anwendung zu erstellen, sollten Sie zuerst die höherstufigigen SSR-Plugins und Tools im Awesome Vite SSR-Abschnitt überprüfen. Viele Anwendungen werden jedoch erfolgreich direkt auf der nativen niedrigstufigen API von Vite erstellt.
Derzeit arbeitet Vite an einer verbesserten SSR-API mit der Environment API. Unter dem Link finden Sie weitere Details.
HILFE
Bei Fragen steht Ihnen die Community normalerweise im Vite Discord-Kanal für SSR hilfreich zur Seite.
Beispielprojekte
Vite bietet integrierte Unterstützung für serverseitiges Rendering (SSR). create-vite-extra
enthält Beispiel-SSR-Setups, die Sie als Referenz für diesen Leitfaden verwenden können:
Dateistruktur
Eine typische SSR-Anwendung hat die folgende Dateistruktur:
- index.html
- server.js # Hauptanwendungsserver
- src/
- main.js # Exportiert umgebungsunabhängigen (universalen) App-Code
- entry-client.js # Bindet die App an ein DOM-Element
- entry-server.js # Rendert die App mit der SSR-API des Frameworks
Die index.html
muss auf entry-client.js
verweisen und einen Platzhalter enthalten, in den das servergerenderte Markup eingefügt werden soll:
<div id="app"><!--ssr-outlet--></div>
<script type="module" src="/src/entry-client.js"></script>
Sie können jeden gewünschten Platzhalter anstelle von <!--ssr-outlet-->
verwenden, solange er präzise ersetzt werden kann.
Bedingte Logik
Wenn Sie bedingte Logik basierend auf SSR gegenüber Client ausführen müssen, können Sie folgendes verwenden:
if (import.meta.env.SSR) {
// ... serverseitige Logik
}
Dies wird während des Builds statisch ersetzt und ermöglicht das Entfernen von ungenutzten Verzweigungen (tree-shaking).
Einrichten des Entwicklungsservers
Wenn Sie eine SSR-Anwendung erstellen, möchten Sie wahrscheinlich die volle Kontrolle über Ihren Hauptserver haben und Vite von der Produktionsumgebung entkoppeln. Es wird daher empfohlen, Vite im Middleware-Modus zu verwenden. Hier ist ein Beispiel mit Express:
import from 'node:fs'
import from 'node:path'
import { } from 'node:url'
import from 'express'
import { as } from 'vite'
const = .((import.meta.))
async function () {
const = ()
// Erstellen Sie den Vite-Server im Middleware-Modus und konfigurieren Sie den App-Typ als
// 'custom', um die eigene HTML-Bereitstellungslogik von Vite zu deaktivieren, damit der übergeordnete Server
// die Kontrolle übernehmen kann
const = await ({
: { : true },
: 'custom'
})
// Verwenden Sie die Connect-Instanz von vite als Middleware. Wenn Sie Ihren eigenen
// Express-Router (express.Router()) verwenden, sollten Sie router.use
// Wenn der Server neu startet (z.B. nachdem der Benutzer die
// vite.config.js), ist `vite.middlewares` immer noch die gleiche
// Referenz (mit einem neuen internen Stapel von Vite und Plugin-injected
// Middlewares). Das Folgende ist auch nach Neustarts gültig.
.(.)
.('*', async (, ) => {
// Stellen Sie die index.html bereit - wir werden diese als Nächstes angehen
})
.(5173)
}
()
Here vite
is an instance of ViteDevServer. vite.middlewares
is a Connect instance which can be used as a middleware in any connect-compatible Node.js framework.
The next step is implementing the *
handler to serve server-rendered HTML:
.('*', async (, , ) => {
const = .
try {
// 1. Lesen Sie die index.html
let = .(
.(, 'index.html'),
'utf-8',
)
// 2. Wenden Sie Vite-HTML-Transformationen an. Dadurch wird der Vite HMR-Client eingefügt,
// und es werden auch HTML-Transformationen von Vite-Plugins angewendet, z. B. globale
// Präambeln von @vitejs/plugin-react
= await .(, )
// 3. Laden Sie den Server-Einstieg. ssrLoadModule transformiert automatisch
// ESM-Quellcode, damit er in Node.js verwendbar ist! Es ist kein Bündeln erforderlich und bietet
// effiziente Ungültigmachung ähnlich wie HMR.
const { } = await .('/src/entry-server.js')
// 4. Die HTML-Anwendung rendern. Dies setzt voraus, dass entry-server.js exportiert wird.
// Die Funktion `render` ruft entsprechende SSR-Rahmen-APIs auf,
// z.B. ReactDOMServer.renderToString()
const = await ()
// 5. Fügen Sie das vom App gerenderte HTML in die Vor
lage ein.
const = .(`<!--ssr-outlet-->`, )
// 6. Senden Sie das gerenderte HTML zurück.
.(200).({ 'Content-Type': 'text/html' }).()
} catch () {
// Wenn ein Fehler auftritt, lässt Vite den Stapelverfolgung korrigieren, damit er auf Ihren tatsächlichen Quellcode verweist.
.()
()
}
})
Das dev
-Skript in package.json
sollte ebenfalls geändert werden, um stattdessen das Server-Skript zu verwenden:
"scripts": {
- "dev": "vite"
+ "dev": "node server"
}
Erstellung für die Produktion
Um ein SSR-Projekt für die Produktion bereitzustellen, müssen wir Folgendes tun:
- Erstellen Sie wie gewohnt einen Client-Build;
- Erzeugen eines SSR-Builds, der direkt über
import()
geladen werden kann, so dass wir nicht durch dasssrLoadModule
von Vite gehen müssen;
Unsere Skripts in package.json
sehen folgendermaßen aus:
{
"scripts": {
"dev": "node server",
"build:client": "vite build --outDir dist/client",
"build:server": "vite build --outDir dist/server --ssr src/entry-server.js"
}
}
Beachten Sie die --ssr
-Flagge, die darauf hinweist, dass dies ein SSR-Build ist. Sie sollte auch den SSR-Einstieg angeben.
Dann müssen wir in server.js
einige spezifische Logik für die Produktion hinzufügen, indem wir process.env.NODE_ENV
überprüfen:
Anstatt die Stammdatei
index.html
zu lesen, verwenden Sie die Dateidist/client/index.html
als Vorlage, da sie die richtigen Asset-Links zum Client-Build enthält.Statt
await vite.ssrLoadModule('/src/entry-server.js')
, verwenden Sieimport('./dist/server/entry-server.js')
(diese Datei ist das Ergebnis des SSR-Builds).Verschieben Sie die Erstellung und alle Verwendungen des
vite
-Entwicklungsservers hinter bedingte Verzweigungen, die nur für die Entwicklung gelten, und fügen Sie statische Dateiserver-Middleware hinzu, um Dateien ausdist/client
zu servieren.
In den Beispielprojekten finden Sie eine funktionierende Einrichtung.
Generieren von Preload-Direktiven
vite build
unterstützt die --ssrManifest
-Flagge, die .vite/ssr-manifest.json
im Build-Ausgabeverzeichnis generiert:
- "build:client": "vite build --outDir dist/client",
+ "build:client": "vite build --outDir dist/client --ssrManifest",
Das obige Skript generiert jetzt dist/client/.vite/ssr-manifest.json
für den Client-Build (ja, das SSR-Manifest wird aus dem Client-Build generiert, weil wir Modul-IDs auf Client-Dateien abbilden möchten). Das Manifest enthält Zuordnungen von Modul-IDs zu den zugehörigen Chunks und Asset-Dateien.
Um das Manifest zu nutzen, müssen Frameworks eine Möglichkeit bereitstellen, die Modul-IDs der Komponenten zu sammeln, die während eines Server-Rendervorgangs verwendet wurden.
@vitejs/plugin-vue
unterstützt dies von Haus aus und registriert automatisch verwendete Komponentenmodul-IDs im zugehörigen Vue-SSR-Kontext:
const ctx = {}
const html = await vueServerRenderer.renderToString(app, ctx)
// ctx.modules ist jetzt ein Set von Modul-IDs, die während des Renderns verwendet wurden
Im Produktionszweig von server.js
müssen wir das Manifest lesen und an die von src/entry-server.js
exportierte render
-Funktion übergeben. Dadurch erhalten wir genügend Informationen, um Preload-Direktiven für Dateien zu erstellen, die von asynchronen Routen verwendet werden! Siehe Demo-Quellcode für ein vollständiges Beispiel.
Vorabrendnern / SSG
Wenn die Routen und die für bestimmte Routen benötigten Daten im Voraus bekannt sind, können diese Routen mithilfe derselben Logik wie beim Produktions-SSR im Voraus in statisches HTML vorgerendert werden. Dies kann auch als Form der Generierung von statischen Websites (SSG) angesehen werden. Siehe Vorabrender-Skript im Demo für ein funktionierendes Beispiel.
SSR-Externe
Abhängigkeiten werden standardmäßig von Vites SSR-Transformationsmodul-System "externalisiert", wenn SSR ausgeführt wird. Dies beschleunigt sowohl die Entwicklung als auch das Builden.
Wenn eine Abhängigkeit von Vites Pipeline transformiert werden muss, beispielsweise, weil Vite-Funktionen in ihnen untransformiert verwendet werden, können sie zu ssr.noExternal
hinzugefügt werden.
Verknüpfte Abhängigkeiten werden standardmäßig nicht externalisiert, um von Vites HMR zu profitieren. Wenn dies nicht gewünscht ist, beispielsweise, um Abhängigkeiten zu testen, als wären sie nicht verknüpft, können Sie sie zu ssr.external
hinzufügen.
Arbeit mit Aliasen
Wenn Sie Alias-Konfigurationen haben, die eine Paketumleitung von einem auf ein
anderes Paket vornehmen, möchten Sie möglicherweise die tatsächlichen node_modules
-Pakete aliassen, um sie für SSR-externalisierte Abhängigkeiten zu aktivieren. Sowohl Yarn als auch pnpm unterstützen das Aliasieren über das Präfix npm:
.
SSR-spezifische Plugin-Logik
Einige Frameworks wie Vue oder Svelte kompilieren Komponenten in verschiedene Formate, basierend auf Client vs. SSR. Um bedingte Transformationen zu unterstützen, gibt Vite den Plugin-Hooks resolveId
, load
und transform
ein zusätzliches ssr
-Eigenschaft im options
-Objekt mit.
Beispiel:
export function mySSRPlugin() {
return {
name: 'my-ssr',
transform(code, id, options) {
if (options?.ssr) {
// Führen Sie Transformationen spezifisch für SSR aus...
}
},
}
}
Das options
-Objekt in load
und transform
ist optional, Rollup verwendet dieses Objekt derzeit nicht, kann diese Hooks jedoch in Zukunft um zusätzliche Metadaten erweitern.
HINWEIS
Vor Vite 2.7 wurde dies mit einem positionalen ssr
-Parameter an Plugin-Hooks übergeben, anstatt das options
-Objekt zu verwenden. Alle wichtigen Frameworks und Plugins wurden aktualisiert, aber Sie können veraltete Beiträge finden, die die vorherige API verwenden.
SSR-Ziel
Das Standardziel für den SSR-Build ist eine Node-Umgebung, aber Sie können den Server auch in einem Web Worker ausführen. Die Paket-Eintragsauflösung ist für jede Plattform unterschiedlich. Sie können das Ziel auf Web Worker setzen, indem Sie ssr.target
auf 'webworker'
setzen.
SSR-Bundle
In einigen Fällen, wie bei webworker
-Laufzeiten, möchten Sie Ihren SSR-Build möglicherweise in eine einzige JavaScript-Datei bündeln. Dieses Verhalten können Sie aktivieren, indem Sie ssr.noExternal
auf true
setzen. Dies hat zwei Auswirkungen:
- Alle Abhängigkeiten werden als
noExternal
behandelt. - Ein Fehler wird ausgelöst, wenn Node.js-Standardmodule importiert werden.
SSR Resolve Conditions
By default package entry resolution will use the conditions set in resolve.conditions
for the SSR build. You can use ssr.resolve.conditions
and ssr.resolve.externalConditions
to customize this behavior.
Vite CLI
Die CLI-Befehle $ vite dev
und $ vite preview
können auch für SSR-Anwendungen verwendet werden. Sie können Ihre SSR-Middleware dem Entwicklungsserver mit configureServer
und dem Vorschau-Server mit configurePreviewServer
hinzufügen.
HINWEIS
Verwenden Sie einen Post-Hook, damit Ihre SSR-Middleware nach den Mittelwaren von Vite ausgeführt wird.