@tanstack/start-server-core
Version:
Modern and scalable routing for React applications
798 lines (710 loc) • 24.7 kB
text/typescript
import { createMemoryHistory } from '@tanstack/history'
import {
createNullProtoObject,
flattenMiddlewares,
mergeHeaders,
safeObjectMerge,
} from '@tanstack/start-client-core'
import {
executeRewriteInput,
isRedirect,
isResolvedRedirect,
} from '@tanstack/router-core'
import {
attachRouterServerSsrUtils,
getNormalizedURL,
getOrigin,
} from '@tanstack/router-core/ssr/server'
import { runWithStartContext } from '@tanstack/start-storage-context'
import { requestHandler } from './request-response'
import { getStartManifest } from './router-manifest'
import { handleServerAction } from './server-functions-handler'
import {
buildManifestWithClientEntry,
resolveTransformConfig,
transformManifestUrls,
} from './transformAssetUrls'
import { HEADERS } from './constants'
import { ServerFunctionSerializationAdapter } from './serializer/ServerFunctionSerializationAdapter'
import type {
AnyFunctionMiddleware,
AnyRequestMiddleware,
AnyStartInstanceOptions,
RouteMethod,
RouteMethodHandlerFn,
RouterEntry,
StartEntry,
} from '@tanstack/start-client-core'
import type { RequestHandler } from './request-handler'
import type {
AnyRoute,
AnyRouter,
Manifest,
Register,
} from '@tanstack/router-core'
import type { HandlerCallback } from '@tanstack/router-core/ssr/server'
import type {
StartManifestWithClientEntry,
TransformAssetUrls,
TransformAssetUrlsFn,
} from './transformAssetUrls'
type TODO = any
type AnyMiddlewareServerFn =
| AnyRequestMiddleware['options']['server']
| AnyFunctionMiddleware['options']['server']
export interface CreateStartHandlerOptions {
handler: HandlerCallback<AnyRouter>
/**
* Transform asset URLs at runtime, e.g. to prepend a CDN prefix.
*
* **String** — a URL prefix prepended to every asset URL (cached by default):
* ```ts
* createStartHandler({
* handler: defaultStreamHandler,
* transformAssetUrls: 'https://cdn.example.com',
* })
* ```
*
* **Callback** — receives `{ url, type }` and returns a new URL
* (cached by default — runs once on first request):
* ```ts
* createStartHandler({
* handler: defaultStreamHandler,
* transformAssetUrls: ({ url, type }) => {
* return `https://cdn.example.com${url}`
* },
* })
* ```
*
* **Object** — for explicit cache control:
* ```ts
* createStartHandler({
* handler: defaultStreamHandler,
* transformAssetUrls: {
* transform: ({ url }) => {
* const region = getRequest().headers.get('x-region') || 'us'
* return `https://cdn-${region}.example.com${url}`
* },
* cache: false, // transform per-request
* },
* })
* ```
*
* `type` is one of `'modulepreload' | 'stylesheet' | 'clientEntry'`.
*
* By default, the transformed manifest is cached after the first request
* (`cache: true`). Set `cache: false` for per-request transforms.
*
* If you're using a cached transform, you can optionally set `warmup: true`
* (object form only) to compute the transformed manifest in the background at
* server startup.
*
* Note: This only transforms URLs managed by TanStack Start's manifest
* (JS preloads, CSS links, and the client entry script). For asset imports
* used directly in components (e.g. `import logo from './logo.svg'`),
* configure Vite's `experimental.renderBuiltUrl` in your vite.config.ts.
*/
transformAssetUrls?: TransformAssetUrls
}
function getStartResponseHeaders(opts: { router: AnyRouter }) {
const headers = mergeHeaders(
{
'Content-Type': 'text/html; charset=utf-8',
},
...opts.router.state.matches.map((match) => {
return match.headers
}),
)
return headers
}
// Cached entries - promises stored immediately to prevent concurrent imports
// that can cause race conditions during module initialization
let entriesPromise:
| Promise<{
startEntry: StartEntry
routerEntry: RouterEntry
}>
| undefined
let baseManifestPromise: Promise<StartManifestWithClientEntry> | undefined
/**
* Cached final manifest (with client entry script tag). In production,
* this is computed once and reused for every request when caching is enabled.
*/
let cachedFinalManifestPromise: Promise<Manifest> | undefined
async function loadEntries() {
// @ts-ignore when building, we currently don't respect tsconfig.ts' `include` so we are not picking up the .d.ts from start-client-core
const routerEntry = (await import('#tanstack-router-entry')) as RouterEntry
// @ts-ignore when building, we currently don't respect tsconfig.ts' `include` so we are not picking up the .d.ts from start-client-core
const startEntry = (await import('#tanstack-start-entry')) as StartEntry
return { startEntry, routerEntry }
}
function getEntries() {
if (!entriesPromise) {
entriesPromise = loadEntries()
}
return entriesPromise
}
/**
* Returns the raw manifest data (without client entry script tag baked in).
* In dev mode, always returns fresh data. In prod, cached.
*/
function getBaseManifest(
matchedRoutes?: ReadonlyArray<AnyRoute>,
): Promise<StartManifestWithClientEntry> {
// In dev mode, always get fresh manifest (no caching) to include route-specific dev styles
if (process.env.TSS_DEV_SERVER === 'true') {
return getStartManifest(matchedRoutes)
}
// In prod, cache the base manifest
if (!baseManifestPromise) {
baseManifestPromise = getStartManifest()
}
return baseManifestPromise
}
/**
* Resolves a final Manifest for a given request.
*
* - No transform: builds client entry script tag and returns (cached in prod).
* - Cached transform: transforms all URLs + builds script tag, caches result.
* - Per-request transform: deep-clones base manifest, transforms per-request.
*/
async function resolveManifest(
matchedRoutes: ReadonlyArray<AnyRoute> | undefined,
transformFn: TransformAssetUrlsFn | undefined,
cache: boolean,
): Promise<Manifest> {
const base = await getBaseManifest(matchedRoutes)
const computeFinalManifest = async () => {
return transformFn
? await transformManifestUrls(base, transformFn, { clone: !cache })
: buildManifestWithClientEntry(base)
}
// In dev, always compute fresh to include route-specific dev styles.
if (process.env.TSS_DEV_SERVER === 'true') {
return computeFinalManifest()
}
// In prod, cache unless we're explicitly doing per-request transforms.
if (!transformFn || cache) {
if (!cachedFinalManifestPromise) {
cachedFinalManifestPromise = computeFinalManifest()
}
return cachedFinalManifestPromise
}
// Per-request transform — deep-clone and transform every time.
return computeFinalManifest()
}
// Pre-computed constants
const ROUTER_BASEPATH = process.env.TSS_ROUTER_BASEPATH || '/'
const SERVER_FN_BASE = process.env.TSS_SERVER_FN_BASE
const IS_PRERENDERING = process.env.TSS_PRERENDERING === 'true'
const IS_SHELL_ENV = process.env.TSS_SHELL === 'true'
const IS_DEV = process.env.NODE_ENV === 'development'
// Reusable error messages
const ERR_NO_RESPONSE = IS_DEV
? `It looks like you forgot to return a response from your server route handler. If you want to defer to the app router, make sure to have a component set in this route.`
: 'Internal Server Error'
const ERR_NO_DEFER = IS_DEV
? `You cannot defer to the app router if there is no component defined on this route.`
: 'Internal Server Error'
function throwRouteHandlerError(): never {
throw new Error(ERR_NO_RESPONSE)
}
function throwIfMayNotDefer(): never {
throw new Error(ERR_NO_DEFER)
}
/**
* Check if a value is a special response (Response or Redirect)
*/
function isSpecialResponse(value: unknown): value is Response {
return value instanceof Response || isRedirect(value)
}
/**
* Normalize middleware result to context shape
*/
function handleCtxResult(result: TODO) {
if (isSpecialResponse(result)) {
return { response: result }
}
return result
}
/**
* Execute a middleware chain
*/
function executeMiddleware(middlewares: Array<TODO>, ctx: TODO): Promise<TODO> {
let index = -1
const next = async (nextCtx?: TODO): Promise<TODO> => {
// Merge context if provided using safeObjectMerge for prototype pollution prevention
if (nextCtx) {
if (nextCtx.context) {
ctx.context = safeObjectMerge(ctx.context, nextCtx.context)
}
// Copy own properties except context (Object.keys returns only own enumerable properties)
for (const key of Object.keys(nextCtx)) {
if (key !== 'context') {
ctx[key] = nextCtx[key]
}
}
}
index++
const middleware = middlewares[index]
if (!middleware) return ctx
let result: TODO
try {
result = await middleware({ ...ctx, next })
} catch (err) {
if (isSpecialResponse(err)) {
ctx.response = err
return ctx
}
throw err
}
const normalized = handleCtxResult(result)
if (normalized) {
if (normalized.response !== undefined) {
ctx.response = normalized.response
}
if (normalized.context) {
ctx.context = safeObjectMerge(ctx.context, normalized.context)
}
}
return ctx
}
return next()
}
/**
* Wrap a route handler as middleware
*/
function handlerToMiddleware(
handler: RouteMethodHandlerFn<any, AnyRoute, any, any, any, any, any>,
mayDefer: boolean = false,
): TODO {
if (mayDefer) {
return handler
}
return async (ctx: TODO) => {
const response = await handler({ ...ctx, next: throwIfMayNotDefer })
if (!response) {
throwRouteHandlerError()
}
return response
}
}
/**
* Creates the TanStack Start request handler.
*
* @example Backwards-compatible usage (handler callback only):
* ```ts
* export default createStartHandler(defaultStreamHandler)
* ```
*
* @example With CDN URL rewriting:
* ```ts
* export default createStartHandler({
* handler: defaultStreamHandler,
* transformAssetUrls: 'https://cdn.example.com',
* })
* ```
*
* @example With per-request URL rewriting:
* ```ts
* export default createStartHandler({
* handler: defaultStreamHandler,
* transformAssetUrls: {
* transform: ({ url }) => {
* const cdnBase = getRequest().headers.get('x-cdn-base') || ''
* return `${cdnBase}${url}`
* },
* cache: false,
* },
* })
* ```
*/
export function createStartHandler<TRegister = Register>(
cbOrOptions: HandlerCallback<AnyRouter> | CreateStartHandlerOptions,
): RequestHandler<TRegister> {
// Normalize the overloaded argument
const cb: HandlerCallback<AnyRouter> =
typeof cbOrOptions === 'function' ? cbOrOptions : cbOrOptions.handler
const transformAssetUrlsOption: TransformAssetUrls | undefined =
typeof cbOrOptions === 'function'
? undefined
: cbOrOptions.transformAssetUrls
const warmupTransformManifest =
!!transformAssetUrlsOption &&
typeof transformAssetUrlsOption === 'object' &&
transformAssetUrlsOption.warmup === true
// Pre-resolve the transform function and cache flag
const resolvedTransformConfig = transformAssetUrlsOption
? resolveTransformConfig(transformAssetUrlsOption)
: undefined
const cache = resolvedTransformConfig ? resolvedTransformConfig.cache : true
// Memoize a single createTransform() result when caching is enabled.
let cachedCreateTransformPromise: Promise<TransformAssetUrlsFn> | undefined
const getTransformFn = async (
opts: { warmup: true } | { warmup: false; request: Request },
): Promise<TransformAssetUrlsFn | undefined> => {
if (!resolvedTransformConfig) return undefined
if (resolvedTransformConfig.type === 'createTransform') {
if (cache) {
if (!cachedCreateTransformPromise) {
cachedCreateTransformPromise = Promise.resolve(
resolvedTransformConfig.createTransform(opts),
)
}
return cachedCreateTransformPromise
}
return resolvedTransformConfig.createTransform(opts)
}
return resolvedTransformConfig.transformFn
}
// Background warmup for cached transforms (production only)
if (
warmupTransformManifest &&
cache &&
process.env.TSS_DEV_SERVER !== 'true' &&
!cachedFinalManifestPromise
) {
// NOTE: Do not call resolveManifest() here.
// resolveManifest() reads from cachedFinalManifestPromise, and since we set
// cachedFinalManifestPromise to this warmup promise, that would create a
// self-referential promise and hang forever.
const warmupPromise = (async () => {
const base = await getBaseManifest(undefined)
const transformFn = await getTransformFn({ warmup: true })
return transformFn
? await transformManifestUrls(base, transformFn, { clone: false })
: buildManifestWithClientEntry(base)
})()
cachedFinalManifestPromise = warmupPromise
warmupPromise.catch(() => {
// If warmup fails, allow the next request to retry.
if (cachedFinalManifestPromise === warmupPromise) {
cachedFinalManifestPromise = undefined
}
cachedCreateTransformPromise = undefined
})
}
const startRequestResolver: RequestHandler<Register> = async (
request,
requestOpts,
) => {
let router: AnyRouter | null = null as AnyRouter | null
let cbWillCleanup = false as boolean
try {
// normalizing and sanitizing the pathname here for server, so we always deal with the same format during SSR.
// during normalization paths like '//posts' are flattened to '/posts'.
// in these cases we would prefer to redirect to the new path
const { url, handledProtocolRelativeURL } = getNormalizedURL(request.url)
const href = url.pathname + url.search + url.hash
const origin = getOrigin(request)
if (handledProtocolRelativeURL) {
return Response.redirect(url, 308)
}
const entries = await getEntries()
const startOptions: AnyStartInstanceOptions =
(await entries.startEntry.startInstance?.getOptions()) ||
({} as AnyStartInstanceOptions)
const serializationAdapters = [
...(startOptions.serializationAdapters || []),
ServerFunctionSerializationAdapter,
]
const requestStartOptions = {
...startOptions,
serializationAdapters,
}
// Flatten request middlewares once
const flattenedRequestMiddlewares = startOptions.requestMiddleware
? flattenMiddlewares(startOptions.requestMiddleware)
: []
// Create set for deduplication
const executedRequestMiddlewares = new Set<TODO>(
flattenedRequestMiddlewares,
)
// Memoized router getter
const getRouter = async (): Promise<AnyRouter> => {
if (router) return router
router = await entries.routerEntry.getRouter()
let isShell = IS_SHELL_ENV
if (IS_PRERENDERING && !isShell) {
isShell = request.headers.get(HEADERS.TSS_SHELL) === 'true'
}
const history = createMemoryHistory({
initialEntries: [href],
})
router.update({
history,
isShell,
isPrerendering: IS_PRERENDERING,
origin: router.options.origin ?? origin,
...{
defaultSsr: requestStartOptions.defaultSsr,
serializationAdapters: [
...requestStartOptions.serializationAdapters,
...(router.options.serializationAdapters || []),
],
},
basepath: ROUTER_BASEPATH,
})
return router
}
// Check for server function requests first (early exit)
if (SERVER_FN_BASE && url.pathname.startsWith(SERVER_FN_BASE)) {
const serverFnId = url.pathname
.slice(SERVER_FN_BASE.length)
.split('/')[0]
if (!serverFnId) {
throw new Error('Invalid server action param for serverFnId')
}
const serverFnHandler = async ({ context }: TODO) => {
return runWithStartContext(
{
getRouter,
startOptions: requestStartOptions,
contextAfterGlobalMiddlewares: context,
request,
executedRequestMiddlewares,
},
() =>
handleServerAction({
request,
context: requestOpts?.context,
serverFnId,
}),
)
}
const middlewares = flattenedRequestMiddlewares.map(
(d) => d.options.server,
)
const ctx = await executeMiddleware([...middlewares, serverFnHandler], {
request,
pathname: url.pathname,
context: createNullProtoObject(requestOpts?.context),
})
return handleRedirectResponse(ctx.response, request, getRouter)
}
// Router execution function
const executeRouter = async (
serverContext: TODO,
matchedRoutes?: ReadonlyArray<AnyRoute>,
): Promise<Response> => {
const acceptHeader = request.headers.get('Accept') || '*/*'
const acceptParts = acceptHeader.split(',')
const supportedMimeTypes = ['*/*', 'text/html']
const isSupported = supportedMimeTypes.some((mimeType) =>
acceptParts.some((part) => part.trim().startsWith(mimeType)),
)
if (!isSupported) {
return Response.json(
{ error: 'Only HTML requests are supported here' },
{ status: 500 },
)
}
const manifest = await resolveManifest(
matchedRoutes,
await getTransformFn({ warmup: false, request }),
cache,
)
const routerInstance = await getRouter()
attachRouterServerSsrUtils({
router: routerInstance,
manifest,
})
routerInstance.update({ additionalContext: { serverContext } })
await routerInstance.load()
if (routerInstance.state.redirect) {
return routerInstance.state.redirect
}
await routerInstance.serverSsr!.dehydrate()
const responseHeaders = getStartResponseHeaders({
router: routerInstance,
})
cbWillCleanup = true
return cb({
request,
router: routerInstance,
responseHeaders,
})
}
// Main request handler
const requestHandlerMiddleware = async ({ context }: TODO) => {
return runWithStartContext(
{
getRouter,
startOptions: requestStartOptions,
contextAfterGlobalMiddlewares: context,
request,
executedRequestMiddlewares,
},
async () => {
try {
return await handleServerRoutes({
getRouter,
request,
url,
executeRouter,
context,
executedRequestMiddlewares,
})
} catch (err) {
if (err instanceof Response) {
return err
}
throw err
}
},
)
}
const middlewares = flattenedRequestMiddlewares.map(
(d) => d.options.server,
)
const ctx = await executeMiddleware(
[...middlewares, requestHandlerMiddleware],
{
request,
pathname: url.pathname,
context: createNullProtoObject(requestOpts?.context),
},
)
return handleRedirectResponse(ctx.response, request, getRouter)
} finally {
if (router && !cbWillCleanup) {
// Clean up router SSR state if it was set up but won't be cleaned up by the callback
// (e.g., in redirect cases or early returns before the callback is invoked).
// When the callback runs, it handles cleanup (either via transformStreamWithRouter
// for streaming, or directly in renderRouterToString for non-streaming).
router.serverSsr?.cleanup()
}
router = null
}
}
return requestHandler(startRequestResolver)
}
async function handleRedirectResponse(
response: Response,
request: Request,
getRouter: () => Promise<AnyRouter>,
): Promise<Response> {
if (!isRedirect(response)) {
return response
}
if (isResolvedRedirect(response)) {
if (request.headers.get('x-tsr-serverFn') === 'true') {
return Response.json(
{ ...response.options, isSerializedRedirect: true },
{ headers: response.headers },
)
}
return response
}
const opts = response.options
if (opts.to && typeof opts.to === 'string' && !opts.to.startsWith('/')) {
throw new Error(
`Server side redirects must use absolute paths via the 'href' or 'to' options. The redirect() method's "to" property accepts an internal path only. Use the "href" property to provide an external URL. Received: ${JSON.stringify(opts)}`,
)
}
if (
['params', 'search', 'hash'].some(
(d) => typeof (opts as TODO)[d] === 'function',
)
) {
throw new Error(
`Server side redirects must use static search, params, and hash values and do not support functional values. Received functional values for: ${Object.keys(
opts,
)
.filter((d) => typeof (opts as TODO)[d] === 'function')
.map((d) => `"${d}"`)
.join(', ')}`,
)
}
const router = await getRouter()
const redirect = router.resolveRedirect(response)
if (request.headers.get('x-tsr-serverFn') === 'true') {
return Response.json(
{ ...response.options, isSerializedRedirect: true },
{ headers: response.headers },
)
}
return redirect
}
async function handleServerRoutes({
getRouter,
request,
url,
executeRouter,
context,
executedRequestMiddlewares,
}: {
getRouter: () => Promise<AnyRouter>
request: Request
url: URL
executeRouter: (
serverContext: any,
matchedRoutes?: ReadonlyArray<AnyRoute>,
) => Promise<Response>
context: any
executedRequestMiddlewares: Set<AnyRequestMiddleware>
}): Promise<Response> {
const router = await getRouter()
const rewrittenUrl = executeRewriteInput(router.rewrite, url)
const pathname = rewrittenUrl.pathname
// this will perform a fuzzy match, however for server routes we need an exact match
// if the route is not an exact match, executeRouter will handle rendering the app router
// the match will be cached internally, so no extra work is done during the app router render
const { matchedRoutes, foundRoute, routeParams } =
router.getMatchedRoutes(pathname)
const isExactMatch = foundRoute && routeParams['**'] === undefined
// Collect and dedupe route middlewares
const routeMiddlewares: Array<AnyMiddlewareServerFn> = []
// Collect middleware from matched routes, filtering out those already executed
// in the request phase
for (const route of matchedRoutes) {
const serverMiddleware = route.options.server?.middleware as
| Array<AnyRequestMiddleware>
| undefined
if (serverMiddleware) {
const flattened = flattenMiddlewares(serverMiddleware)
for (const m of flattened) {
if (!executedRequestMiddlewares.has(m)) {
routeMiddlewares.push(m.options.server)
}
}
}
}
// Add handler middleware if exact match
const server = foundRoute?.options.server
if (server?.handlers && isExactMatch) {
const handlers =
typeof server.handlers === 'function'
? server.handlers({ createHandlers: (d: any) => d })
: server.handlers
const requestMethod = request.method.toUpperCase() as RouteMethod
const handler = handlers[requestMethod] ?? handlers['ANY']
if (handler) {
const mayDefer = !!foundRoute.options.component
if (typeof handler === 'function') {
routeMiddlewares.push(handlerToMiddleware(handler, mayDefer))
} else {
if (handler.middleware?.length) {
const handlerMiddlewares = flattenMiddlewares(handler.middleware)
for (const m of handlerMiddlewares) {
routeMiddlewares.push(m.options.server)
}
}
if (handler.handler) {
routeMiddlewares.push(handlerToMiddleware(handler.handler, mayDefer))
}
}
}
}
// Final middleware: execute router with matched routes for dev styles
routeMiddlewares.push((ctx: TODO) =>
executeRouter(ctx.context, matchedRoutes),
)
const ctx = await executeMiddleware(routeMiddlewares, {
request,
context,
params: routeParams,
pathname,
})
return ctx.response
}