@tanstack/router-core 1.167.2 → 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.
- package/bin/intent.js +25 -0
- package/dist/cjs/load-matches.cjs +4 -1
- package/dist/cjs/load-matches.cjs.map +1 -1
- package/dist/cjs/router.cjs +2 -1
- package/dist/cjs/router.cjs.map +1 -1
- package/dist/esm/load-matches.js +4 -1
- package/dist/esm/load-matches.js.map +1 -1
- package/dist/esm/router.js +2 -1
- package/dist/esm/router.js.map +1 -1
- package/package.json +9 -2
- package/skills/router-core/SKILL.md +139 -0
- package/skills/router-core/auth-and-guards/SKILL.md +458 -0
- package/skills/router-core/code-splitting/SKILL.md +322 -0
- package/skills/router-core/data-loading/SKILL.md +485 -0
- package/skills/router-core/navigation/SKILL.md +448 -0
- package/skills/router-core/not-found-and-errors/SKILL.md +435 -0
- package/skills/router-core/path-params/SKILL.md +382 -0
- package/skills/router-core/search-params/SKILL.md +355 -0
- package/skills/router-core/search-params/references/validation-patterns.md +379 -0
- package/skills/router-core/ssr/SKILL.md +437 -0
- package/skills/router-core/type-safety/SKILL.md +497 -0
- package/src/load-matches.ts +4 -1
- package/src/router.ts +2 -1
|
@@ -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
|