@tanstack/router-core 1.167.3 → 1.167.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,437 @@
1
+ ---
2
+ name: router-core/ssr
3
+ description: >-
4
+ Non-streaming and streaming SSR, RouterClient/RouterServer,
5
+ renderRouterToString/renderRouterToStream, createRequestHandler,
6
+ defaultRenderHandler/defaultStreamHandler, HeadContent/Scripts
7
+ components, head route option (meta/links/styles/scripts),
8
+ ScriptOnce, automatic loader dehydration/hydration, memory
9
+ history on server, data serialization, document head management.
10
+ type: sub-skill
11
+ library: tanstack-router
12
+ library_version: '1.166.2'
13
+ requires:
14
+ - router-core
15
+ - router-core/data-loading
16
+ sources:
17
+ - TanStack/router:docs/router/guide/ssr.md
18
+ - TanStack/router:docs/router/guide/document-head-management.md
19
+ - TanStack/router:docs/router/how-to/setup-ssr.md
20
+ ---
21
+
22
+ # SSR (Server-Side Rendering)
23
+
24
+ > **WARNING**: SSR APIs are experimental. They share internal implementations with TanStack Start and may change. **TanStack Start is the recommended way to do SSR in production** — use manual SSR setup only when integrating with an existing server.
25
+
26
+ > **CRITICAL**: TanStack Router is CLIENT-FIRST. Loaders run on the client by default. With SSR enabled, loaders run on BOTH client AND server. They are NOT server-only like Remix/Next.js loaders. See [router-core/data-loading](../data-loading/SKILL.md).
27
+
28
+ > **CRITICAL**: Do not generate Next.js patterns (`getServerSideProps`, App Router, server components) or Remix patterns (server-only loader exports). TanStack Router has its own SSR API.
29
+
30
+ ## Concepts
31
+
32
+ There are two SSR flavors:
33
+
34
+ - **Non-streaming**: Full page rendered on server, sent as one HTML response, then hydrated on client.
35
+ - **Streaming**: Critical first paint sent immediately; remaining content streamed incrementally as it resolves.
36
+
37
+ Key behaviors:
38
+
39
+ - Memory history is used automatically on the server (no `window`).
40
+ - Loader data is automatically dehydrated on the server and hydrated on the client.
41
+ - Data serialization supports `Date`, `Error`, `FormData`, and `undefined` out of the box.
42
+
43
+ ## Setup: Shared Router Factory
44
+
45
+ The router must be created identically on server and client. Export a factory function from a shared file:
46
+
47
+ ```tsx
48
+ // src/router.tsx
49
+ import { createRouter as createTanstackRouter } from '@tanstack/react-router'
50
+ import { routeTree } from './routeTree.gen'
51
+
52
+ export function createRouter() {
53
+ return createTanstackRouter({ routeTree })
54
+ }
55
+
56
+ declare module '@tanstack/react-router' {
57
+ interface Register {
58
+ router: ReturnType<typeof createRouter>
59
+ }
60
+ }
61
+ ```
62
+
63
+ ## Non-Streaming SSR
64
+
65
+ ### Server Entry (using `defaultRenderHandler`)
66
+
67
+ ```tsx
68
+ // src/entry-server.tsx
69
+ import {
70
+ createRequestHandler,
71
+ defaultRenderHandler,
72
+ } from '@tanstack/react-router/ssr/server'
73
+ import { createRouter } from './router'
74
+
75
+ export async function render({ request }: { request: Request }) {
76
+ const handler = createRequestHandler({ request, createRouter })
77
+ return await handler(defaultRenderHandler)
78
+ }
79
+ ```
80
+
81
+ ### Server Entry (using `renderRouterToString` for custom wrappers)
82
+
83
+ ```tsx
84
+ // src/entry-server.tsx
85
+ import {
86
+ createRequestHandler,
87
+ renderRouterToString,
88
+ RouterServer,
89
+ } from '@tanstack/react-router/ssr/server'
90
+ import { createRouter } from './router'
91
+
92
+ export function render({ request }: { request: Request }) {
93
+ const handler = createRequestHandler({ request, createRouter })
94
+
95
+ return handler(({ responseHeaders, router }) =>
96
+ renderRouterToString({
97
+ responseHeaders,
98
+ router,
99
+ children: <RouterServer router={router} />,
100
+ }),
101
+ )
102
+ }
103
+ ```
104
+
105
+ ### Client Entry
106
+
107
+ ```tsx
108
+ // src/entry-client.tsx
109
+ import { hydrateRoot } from 'react-dom/client'
110
+ import { RouterClient } from '@tanstack/react-router/ssr/client'
111
+ import { createRouter } from './router'
112
+
113
+ const router = createRouter()
114
+
115
+ hydrateRoot(document, <RouterClient router={router} />)
116
+ ```
117
+
118
+ ## Streaming SSR
119
+
120
+ ### Server Entry (using `defaultStreamHandler`)
121
+
122
+ ```tsx
123
+ // src/entry-server.tsx
124
+ import {
125
+ createRequestHandler,
126
+ defaultStreamHandler,
127
+ } from '@tanstack/react-router/ssr/server'
128
+ import { createRouter } from './router'
129
+
130
+ export async function render({ request }: { request: Request }) {
131
+ const handler = createRequestHandler({ request, createRouter })
132
+ return await handler(defaultStreamHandler)
133
+ }
134
+ ```
135
+
136
+ ### Server Entry (using `renderRouterToStream` for custom wrappers)
137
+
138
+ ```tsx
139
+ // src/entry-server.tsx
140
+ import {
141
+ createRequestHandler,
142
+ renderRouterToStream,
143
+ RouterServer,
144
+ } from '@tanstack/react-router/ssr/server'
145
+ import { createRouter } from './router'
146
+
147
+ export function render({ request }: { request: Request }) {
148
+ const handler = createRequestHandler({ request, createRouter })
149
+
150
+ return handler(({ request, responseHeaders, router }) =>
151
+ renderRouterToStream({
152
+ request,
153
+ responseHeaders,
154
+ router,
155
+ children: <RouterServer router={router} />,
156
+ }),
157
+ )
158
+ }
159
+ ```
160
+
161
+ Streaming is automatic — deferred data (unawaited promises from loaders) and streamed markup just work when using `defaultStreamHandler` or `renderRouterToStream`.
162
+
163
+ ## Document Head Management
164
+
165
+ Use the `head` route option to manage `<title>`, `<meta>`, `<link>`, and `<style>` tags. Render `<HeadContent />` in `<head>` and `<Scripts />` in `<body>`.
166
+
167
+ ### Root Route with Head
168
+
169
+ ```tsx
170
+ // src/routes/__root.tsx
171
+ import {
172
+ createRootRoute,
173
+ HeadContent,
174
+ Outlet,
175
+ Scripts,
176
+ } from '@tanstack/react-router'
177
+
178
+ export const Route = createRootRoute({
179
+ head: () => ({
180
+ meta: [
181
+ { charSet: 'UTF-8' },
182
+ { name: 'viewport', content: 'width=device-width, initial-scale=1.0' },
183
+ { title: 'My App' },
184
+ ],
185
+ links: [{ rel: 'icon', href: '/favicon.ico' }],
186
+ }),
187
+ component: RootComponent,
188
+ })
189
+
190
+ function RootComponent() {
191
+ return (
192
+ <html lang="en">
193
+ <head>
194
+ <HeadContent />
195
+ </head>
196
+ <body>
197
+ <Outlet />
198
+ <Scripts />
199
+ </body>
200
+ </html>
201
+ )
202
+ }
203
+ ```
204
+
205
+ ### Per-Route Head (Nested Deduplication)
206
+
207
+ Child route `title` and `meta` tags override parent tags with the same `name`/`property`:
208
+
209
+ ```tsx
210
+ // src/routes/posts/$postId.tsx
211
+ import { createFileRoute } from '@tanstack/react-router'
212
+
213
+ export const Route = createFileRoute('/posts/$postId')({
214
+ loader: async ({ params }) => {
215
+ const post = await fetchPost(params.postId)
216
+ return { post }
217
+ },
218
+ head: ({ loaderData }) => ({
219
+ meta: [
220
+ { title: loaderData.post.title },
221
+ { name: 'description', content: loaderData.post.excerpt },
222
+ ],
223
+ }),
224
+ component: PostPage,
225
+ })
226
+
227
+ function PostPage() {
228
+ const { post } = Route.useLoaderData()
229
+ return <article>{post.content}</article>
230
+ }
231
+ ```
232
+
233
+ ### SPA Head (No Full HTML Control)
234
+
235
+ For SPAs without server-rendered HTML, render `<HeadContent />` at the top of the component tree:
236
+
237
+ ```tsx
238
+ import { createRootRoute, HeadContent, Outlet } from '@tanstack/react-router'
239
+
240
+ const rootRoute = createRootRoute({
241
+ head: () => ({
242
+ meta: [{ title: 'My SPA' }],
243
+ }),
244
+ component: () => (
245
+ <>
246
+ <HeadContent />
247
+ <Outlet />
248
+ </>
249
+ ),
250
+ })
251
+ ```
252
+
253
+ ## Body Scripts
254
+
255
+ Use `scripts` (separate from `head.scripts`) to inject scripts into `<body>` before the app entry point:
256
+
257
+ ```tsx
258
+ export const Route = createRootRoute({
259
+ scripts: () => [{ children: 'console.log("runs before hydration")' }],
260
+ })
261
+ ```
262
+
263
+ The `<Scripts />` component renders these. Place it at the end of `<body>`.
264
+
265
+ ## ScriptOnce for Pre-Hydration Scripts
266
+
267
+ `ScriptOnce` renders a `<script>` during SSR that executes immediately and self-removes. On client navigation, it does nothing (no duplicate execution).
268
+
269
+ ```tsx
270
+ import { ScriptOnce } from '@tanstack/react-router'
271
+
272
+ const themeScript = `(function() {
273
+ try {
274
+ const theme = localStorage.getItem('theme') || 'auto';
275
+ const resolved = theme === 'auto'
276
+ ? (matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light')
277
+ : theme;
278
+ document.documentElement.classList.add(resolved);
279
+ } catch (e) {}
280
+ })();`
281
+
282
+ function ThemeProvider({ children }: { children: React.ReactNode }) {
283
+ return (
284
+ <>
285
+ <ScriptOnce children={themeScript} />
286
+ {children}
287
+ </>
288
+ )
289
+ }
290
+ ```
291
+
292
+ If the script modifies the DOM (e.g., adds a class to `<html>`), use `suppressHydrationWarning` on the element:
293
+
294
+ ```tsx
295
+ <html lang="en" suppressHydrationWarning>
296
+ ```
297
+
298
+ ## Express Integration Example
299
+
300
+ `createRequestHandler` expects a Web API `Request` and returns a Web API `Response`. For Express, convert between formats:
301
+
302
+ ```tsx
303
+ // src/entry-server.tsx
304
+ import { pipeline } from 'node:stream/promises'
305
+ import {
306
+ RouterServer,
307
+ createRequestHandler,
308
+ renderRouterToString,
309
+ } from '@tanstack/react-router/ssr/server'
310
+ import { createRouter } from './router'
311
+ import type express from 'express'
312
+
313
+ export async function render({
314
+ req,
315
+ res,
316
+ }: {
317
+ req: express.Request
318
+ res: express.Response
319
+ }) {
320
+ const protocol = req.get('x-forwarded-proto') ?? req.protocol
321
+ const host = req.get('x-forwarded-host') ?? req.get('host')
322
+ const url = new URL(req.originalUrl || req.url, `${protocol}://${host}`).href
323
+
324
+ const request = new Request(url, {
325
+ method: req.method,
326
+ headers: (() => {
327
+ const headers = new Headers()
328
+ for (const [key, value] of Object.entries(req.headers)) {
329
+ headers.set(key, value as any)
330
+ }
331
+ return headers
332
+ })(),
333
+ })
334
+
335
+ const handler = createRequestHandler({ request, createRouter })
336
+
337
+ const response = await handler(({ responseHeaders, router }) =>
338
+ renderRouterToString({
339
+ responseHeaders,
340
+ router,
341
+ children: <RouterServer router={router} />,
342
+ }),
343
+ )
344
+
345
+ res.status(response.status)
346
+ response.headers.forEach((value, name) => {
347
+ res.setHeader(name, value)
348
+ })
349
+
350
+ return pipeline(response.body as any, res)
351
+ }
352
+ ```
353
+
354
+ ## Common Mistakes
355
+
356
+ ### 1. HIGH: Using browser APIs in loaders without environment check
357
+
358
+ Loaders run on BOTH client and server with SSR. Browser-only APIs (`window`, `document`, `localStorage`) throw on the server.
359
+
360
+ ```tsx
361
+ // WRONG — crashes on server
362
+ loader: async () => {
363
+ const token = localStorage.getItem('token')
364
+ return fetchData(token)
365
+ }
366
+
367
+ // CORRECT — guard with environment check
368
+ loader: async () => {
369
+ const token =
370
+ typeof window !== 'undefined' ? localStorage.getItem('token') : null
371
+ return fetchData(token)
372
+ }
373
+ ```
374
+
375
+ ### 2. MEDIUM: Using hash fragments for server-rendered content
376
+
377
+ Hash fragments (`#section`) are never sent to the server. Conditional rendering based on hash causes hydration mismatches.
378
+
379
+ ```tsx
380
+ // WRONG — server has no hash, client does → mismatch
381
+ component: () => {
382
+ const hash = window.location.hash
383
+ return hash === '#admin' ? <AdminPanel /> : <UserPanel />
384
+ }
385
+
386
+ // CORRECT — use search params for server-visible state
387
+ validateSearch: z.object({ view: fallback(z.enum(['admin', 'user']), 'user') }),
388
+ component: () => {
389
+ const { view } = Route.useSearch()
390
+ return view === 'admin' ? <AdminPanel /> : <UserPanel />
391
+ }
392
+ ```
393
+
394
+ ### 3. CRITICAL: Generating Next.js or Remix SSR patterns
395
+
396
+ TanStack Router does NOT use `getServerSideProps`, `getStaticProps`, App Router `page.tsx`, or Remix-style server-only `loader` exports.
397
+
398
+ ```tsx
399
+ // WRONG — Next.js patterns
400
+ export async function getServerSideProps() {
401
+ return { props: { data: await fetchData() } }
402
+ }
403
+
404
+ // WRONG — Remix patterns
405
+ export async function loader({ request }: LoaderFunctionArgs) {
406
+ return json({ data: await fetchData() })
407
+ }
408
+
409
+ // CORRECT — TanStack Router pattern
410
+ export const Route = createFileRoute('/data')({
411
+ loader: async () => {
412
+ const data = await fetchData()
413
+ return { data }
414
+ },
415
+ component: DataPage,
416
+ })
417
+
418
+ function DataPage() {
419
+ const { data } = Route.useLoaderData()
420
+ return <div>{data}</div>
421
+ }
422
+ ```
423
+
424
+ ## Tension: Client-First Loaders vs SSR
425
+
426
+ TanStack Router loaders are client-first by design. When SSR is enabled, they run in both environments. This means:
427
+
428
+ - Browser APIs work by default (client-only) but break under SSR
429
+ - Database access does NOT belong in loaders (unlike Remix/Next) — use API routes
430
+ - For server-only data logic with SSR, use TanStack Start's server functions
431
+
432
+ See [router-core/data-loading](../data-loading/SKILL.md) for loader fundamentals.
433
+
434
+ ## Cross-References
435
+
436
+ - [router-core/data-loading](../data-loading/SKILL.md) — SSR changes where loaders execute
437
+ - [compositions/router-query](../../../../react-router/skills/compositions/router-query/SKILL.md) — SSR dehydration/hydration with TanStack Query