@tanstack/router-core 1.169.1 → 1.169.2
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/dist/cjs/load-matches.cjs.map +1 -1
- package/dist/esm/load-matches.js.map +1 -1
- package/package.json +3 -7
- package/skills/router-core/auth-and-guards/SKILL.md +41 -0
- package/skills/router-core/ssr/SKILL.md +59 -5
- package/skills/router-core/type-safety/SKILL.md +37 -42
- package/src/load-matches.ts +1 -1
- package/bin/intent.js +0 -25
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"load-matches.cjs","names":[],"sources":["../../src/load-matches.ts"],"sourcesContent":["import { isServer } from '@tanstack/router-core/isServer'\nimport { invariant } from './invariant'\nimport { createControlledPromise, isPromise } from './utils'\nimport { isNotFound } from './not-found'\nimport { rootRouteId } from './root'\nimport { isRedirect } from './redirect'\nimport type { NotFoundError } from './not-found'\nimport type { ParsedLocation } from './location'\nimport type {\n AnyRoute,\n BeforeLoadContextOptions,\n LoaderFnContext,\n SsrContextOptions,\n} from './route'\nimport type { AnyRouteMatch, MakeRouteMatch } from './Matches'\nimport type { AnyRouter, SSROption, UpdateMatchFn } from './router'\n\n/**\n * An object of this shape is created when calling `loadMatches`.\n * It contains everything we need for all other functions in this file\n * to work. (It's basically the function's argument, plus a few mutable states)\n */\ntype InnerLoadContext = {\n /** the calling router instance */\n router: AnyRouter\n location: ParsedLocation\n /** mutable state, scoped to a `loadMatches` call */\n firstBadMatchIndex?: number\n /** mutable state, scoped to a `loadMatches` call */\n rendered?: boolean\n serialError?: unknown\n updateMatch: UpdateMatchFn\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n sync?: boolean\n}\n\nconst triggerOnReady = (inner: InnerLoadContext): void | Promise<void> => {\n if (!inner.rendered) {\n inner.rendered = true\n return inner.onReady?.()\n }\n}\n\nconst hasForcePendingActiveMatch = (router: AnyRouter): boolean => {\n return router.stores.matchesId.get().some((matchId) => {\n return router.stores.matchStores.get(matchId)?.get()._forcePending\n })\n}\n\nconst resolvePreload = (inner: InnerLoadContext, matchId: string): boolean => {\n return !!(inner.preload && !inner.router.stores.matchStores.has(matchId))\n}\n\n/**\n * Builds the accumulated context from router options and all matches up to (and optionally including) the given index.\n * Merges __routeContext and __beforeLoadContext from each match.\n */\nconst buildMatchContext = (\n inner: InnerLoadContext,\n index: number,\n includeCurrentMatch: boolean = true,\n): Record<string, unknown> => {\n const context: Record<string, unknown> = {\n ...(inner.router.options.context ?? {}),\n }\n const end = includeCurrentMatch ? index : index - 1\n for (let i = 0; i <= end; i++) {\n const innerMatch = inner.matches[i]\n if (!innerMatch) continue\n const m = inner.router.getMatch(innerMatch.id)\n if (!m) continue\n Object.assign(context, m.__routeContext, m.__beforeLoadContext)\n }\n return context\n}\n\nconst getNotFoundBoundaryIndex = (\n inner: InnerLoadContext,\n err: NotFoundError,\n): number | undefined => {\n if (!inner.matches.length) {\n return undefined\n }\n\n const requestedRouteId = err.routeId\n const matchedRootIndex = inner.matches.findIndex(\n (m) => m.routeId === inner.router.routeTree.id,\n )\n const rootIndex = matchedRootIndex >= 0 ? matchedRootIndex : 0\n\n let startIndex = requestedRouteId\n ? inner.matches.findIndex((match) => match.routeId === requestedRouteId)\n : (inner.firstBadMatchIndex ?? inner.matches.length - 1)\n\n if (startIndex < 0) {\n startIndex = rootIndex\n }\n\n for (let i = startIndex; i >= 0; i--) {\n const match = inner.matches[i]!\n const route = inner.router.looseRoutesById[match.routeId]!\n if (route.options.notFoundComponent) {\n return i\n }\n }\n\n // If no boundary component is found, preserve explicit routeId targeting behavior,\n // otherwise default to root for untargeted notFounds.\n return requestedRouteId ? startIndex : rootIndex\n}\n\nconst handleRedirectAndNotFound = (\n inner: InnerLoadContext,\n match: AnyRouteMatch | undefined,\n err: unknown,\n): void => {\n if (!isRedirect(err) && !isNotFound(err)) return\n\n if (isRedirect(err) && err.redirectHandled && !err.options.reloadDocument) {\n throw err\n }\n\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n match._nonReactive.loaderPromise = undefined\n\n match._nonReactive.error = err\n\n inner.updateMatch(match.id, (prev) => ({\n ...prev,\n status: isRedirect(err)\n ? 'redirected'\n : isNotFound(err)\n ? 'notFound'\n : prev.status === 'pending'\n ? 'success'\n : prev.status,\n context: buildMatchContext(inner, match.index),\n isFetching: false,\n error: err,\n }))\n\n if (isNotFound(err) && !err.routeId) {\n // Stamp the throwing match's routeId so that the finalization step in\n // loadMatches knows where the notFound originated. The actual boundary\n // resolution (walking up to the nearest notFoundComponent) is deferred to\n // the finalization step, where firstBadMatchIndex is stable and\n // headMaxIndex can be capped correctly.\n err.routeId = match.routeId\n }\n\n match._nonReactive.loadPromise?.resolve()\n }\n\n if (isRedirect(err)) {\n inner.rendered = true\n err.options._fromLocation = inner.location\n err.redirectHandled = true\n err = inner.router.resolveRedirect(err)\n }\n\n throw err\n}\n\nconst shouldSkipLoader = (\n inner: InnerLoadContext,\n matchId: string,\n): boolean => {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return true\n }\n // upon hydration, we skip the loader if the match has been dehydrated on the server\n if (!(isServer ?? inner.router.isServer) && match._nonReactive.dehydrated) {\n return true\n }\n\n if ((isServer ?? inner.router.isServer) && match.ssr === false) {\n return true\n }\n\n return false\n}\n\nconst syncMatchContext = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n): void => {\n const nextContext = buildMatchContext(inner, index)\n\n inner.updateMatch(matchId, (prev) => {\n return {\n ...prev,\n context: nextContext,\n }\n })\n}\n\nconst handleSerialError = (\n inner: InnerLoadContext,\n index: number,\n err: any,\n routerCode: string,\n): void => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n // Much like suspense, we use a promise here to know if\n // we've been outdated by a new loadMatches call and\n // should abort the current async operation\n if (err instanceof Promise) {\n throw err\n }\n\n err.routerCode = routerCode\n inner.firstBadMatchIndex ??= index\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n\n try {\n route.options.onError?.(err)\n } catch (errorHandlerErr) {\n err = errorHandlerErr\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n }\n\n inner.updateMatch(matchId, (prev) => {\n prev._nonReactive.beforeLoadPromise?.resolve()\n prev._nonReactive.beforeLoadPromise = undefined\n prev._nonReactive.loadPromise?.resolve()\n\n return {\n ...prev,\n error: err,\n status: 'error',\n isFetching: false,\n updatedAt: Date.now(),\n abortController: new AbortController(),\n }\n })\n\n if (!inner.preload && !isRedirect(err) && !isNotFound(err)) {\n inner.serialError ??= err\n }\n}\n\nconst isBeforeLoadSsr = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n const parentMatchId = inner.matches[index - 1]?.id\n const parentMatch = parentMatchId\n ? inner.router.getMatch(parentMatchId)!\n : undefined\n\n // in SPA mode, only SSR the root route\n if (inner.router.isShell()) {\n existingMatch.ssr = route.id === rootRouteId\n return\n }\n\n if (parentMatch?.ssr === false) {\n existingMatch.ssr = false\n return\n }\n\n const parentOverride = (tempSsr: SSROption) => {\n if (tempSsr === true && parentMatch?.ssr === 'data-only') {\n return 'data-only'\n }\n return tempSsr\n }\n\n const defaultSsr = inner.router.options.defaultSsr ?? true\n\n if (route.options.ssr === undefined) {\n existingMatch.ssr = parentOverride(defaultSsr)\n return\n }\n\n if (typeof route.options.ssr !== 'function') {\n existingMatch.ssr = parentOverride(route.options.ssr)\n return\n }\n const { search, params } = existingMatch\n\n const ssrFnContext: SsrContextOptions<any, any, any> = {\n search: makeMaybe(search, existingMatch.searchError),\n params: makeMaybe(params, existingMatch.paramsError),\n location: inner.location,\n matches: inner.matches.map((match) => ({\n index: match.index,\n pathname: match.pathname,\n fullPath: match.fullPath,\n staticData: match.staticData,\n id: match.id,\n routeId: match.routeId,\n search: makeMaybe(match.search, match.searchError),\n params: makeMaybe(match.params, match.paramsError),\n ssr: match.ssr,\n })),\n }\n\n const tempSsr = route.options.ssr(ssrFnContext)\n if (isPromise(tempSsr)) {\n return tempSsr.then((ssr) => {\n existingMatch.ssr = parentOverride(ssr ?? defaultSsr)\n })\n }\n\n existingMatch.ssr = parentOverride(tempSsr ?? defaultSsr)\n return\n}\n\nconst setupPendingTimeout = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n match: AnyRouteMatch,\n): void => {\n if (match._nonReactive.pendingTimeout !== undefined) return\n\n const pendingMs =\n route.options.pendingMs ?? inner.router.options.defaultPendingMs\n const shouldPending = !!(\n inner.onReady &&\n !(isServer ?? inner.router.isServer) &&\n !resolvePreload(inner, matchId) &&\n (route.options.loader ||\n route.options.beforeLoad ||\n routeNeedsPreload(route)) &&\n typeof pendingMs === 'number' &&\n pendingMs !== Infinity &&\n (route.options.pendingComponent ??\n (inner.router.options as any)?.defaultPendingComponent)\n )\n\n if (shouldPending) {\n const pendingTimeout = setTimeout(() => {\n // Update the match and prematurely resolve the loadMatches promise so that\n // the pending component can start rendering\n triggerOnReady(inner)\n }, pendingMs)\n match._nonReactive.pendingTimeout = pendingTimeout\n }\n}\n\nconst preBeforeLoadSetup = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n\n // If we are in the middle of a load, either of these will be present\n // (not to be confused with `loadPromise`, which is always defined)\n if (\n !existingMatch._nonReactive.beforeLoadPromise &&\n !existingMatch._nonReactive.loaderPromise\n )\n return\n\n setupPendingTimeout(inner, matchId, route, existingMatch)\n\n const then = () => {\n const match = inner.router.getMatch(matchId)!\n if (\n match.preload &&\n (match.status === 'redirected' || match.status === 'notFound')\n ) {\n handleRedirectAndNotFound(inner, match, match.error)\n }\n }\n\n // Wait for the previous beforeLoad to resolve before we continue\n return existingMatch._nonReactive.beforeLoadPromise\n ? existingMatch._nonReactive.beforeLoadPromise.then(then)\n : then()\n}\n\nconst executeBeforeLoad = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const match = inner.router.getMatch(matchId)!\n\n // explicitly capture the previous loadPromise\n let prevLoadPromise = match._nonReactive.loadPromise\n match._nonReactive.loadPromise = createControlledPromise<void>(() => {\n prevLoadPromise?.resolve()\n prevLoadPromise = undefined\n })\n\n const { paramsError, searchError } = match\n\n if (paramsError) {\n handleSerialError(inner, index, paramsError, 'PARSE_PARAMS')\n }\n\n if (searchError) {\n handleSerialError(inner, index, searchError, 'VALIDATE_SEARCH')\n }\n\n setupPendingTimeout(inner, matchId, route, match)\n\n const abortController = new AbortController()\n\n let isPending = false\n const pending = () => {\n if (isPending) return\n isPending = true\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'beforeLoad',\n fetchCount: prev.fetchCount + 1,\n abortController,\n // Note: We intentionally don't update context here.\n // Context should only be updated after beforeLoad resolves to avoid\n // components seeing incomplete context during async beforeLoad execution.\n }))\n }\n\n const resolve = () => {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: false,\n }))\n }\n\n // if there is no `beforeLoad` option, just mark as pending and resolve\n // Context will be updated later in loadRouteMatch after loader completes\n if (!route.options.beforeLoad) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n\n match._nonReactive.beforeLoadPromise = createControlledPromise<void>()\n\n // Build context from all parent matches, excluding current match's __beforeLoadContext\n // (since we're about to execute beforeLoad for this match)\n const context = {\n ...buildMatchContext(inner, index, false),\n ...match.__routeContext,\n }\n const { search, params, cause } = match\n const preload = resolvePreload(inner, matchId)\n const beforeLoadFnContext: BeforeLoadContextOptions<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > = {\n search,\n abortController,\n params,\n preload,\n context,\n location: inner.location,\n navigate: (opts: any) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n buildLocation: inner.router.buildLocation,\n cause: preload ? 'preload' : cause,\n matches: inner.matches,\n routeId: route.id,\n ...inner.router.options.additionalContext,\n }\n\n const updateContext = (beforeLoadContext: any) => {\n if (beforeLoadContext === undefined) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n if (isRedirect(beforeLoadContext) || isNotFound(beforeLoadContext)) {\n pending()\n handleSerialError(inner, index, beforeLoadContext, 'BEFORE_LOAD')\n }\n\n inner.router.batch(() => {\n pending()\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n __beforeLoadContext: beforeLoadContext,\n }))\n resolve()\n })\n }\n\n let beforeLoadContext\n try {\n beforeLoadContext = route.options.beforeLoad(beforeLoadFnContext)\n if (isPromise(beforeLoadContext)) {\n pending()\n return beforeLoadContext\n .catch((err) => {\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n })\n .then(updateContext)\n }\n } catch (err) {\n pending()\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n }\n\n updateContext(beforeLoadContext)\n return\n}\n\nconst handleBeforeLoad = (\n inner: InnerLoadContext,\n index: number,\n): void | Promise<void> => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n const serverSsr = () => {\n // on the server, determine whether SSR the current match or not\n if (isServer ?? inner.router.isServer) {\n const maybePromise = isBeforeLoadSsr(inner, matchId, index, route)\n if (isPromise(maybePromise)) return maybePromise.then(queueExecution)\n }\n return queueExecution()\n }\n\n const execute = () => executeBeforeLoad(inner, matchId, index, route)\n\n const queueExecution = () => {\n if (shouldSkipLoader(inner, matchId)) return\n const result = preBeforeLoadSetup(inner, matchId, route)\n return isPromise(result) ? result.then(execute) : execute()\n }\n\n return serverSsr()\n}\n\nconst executeHead = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<\n Pick<\n AnyRouteMatch,\n 'meta' | 'links' | 'headScripts' | 'headers' | 'scripts' | 'styles'\n >\n> => {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (!match) {\n return\n }\n if (!route.options.head && !route.options.scripts && !route.options.headers) {\n return\n }\n const assetContext = {\n ssr: inner.router.options.ssr,\n matches: inner.matches,\n match,\n params: match.params,\n loaderData: match.loaderData,\n }\n\n return Promise.all([\n route.options.head?.(assetContext),\n route.options.scripts?.(assetContext),\n route.options.headers?.(assetContext),\n ]).then(([headFnContent, scripts, headers]) => {\n const meta = headFnContent?.meta\n const links = headFnContent?.links\n const headScripts = headFnContent?.scripts\n const styles = headFnContent?.styles\n\n return {\n meta,\n links,\n headScripts,\n headers,\n scripts,\n styles,\n }\n })\n}\n\nconst getLoaderContext = (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): LoaderFnContext => {\n const parentMatchPromise = matchPromises[index - 1] as any\n const { params, loaderDeps, abortController, cause } =\n inner.router.getMatch(matchId)!\n\n const context = buildMatchContext(inner, index)\n\n const preload = resolvePreload(inner, matchId)\n\n return {\n params,\n deps: loaderDeps,\n preload: !!preload,\n parentMatchPromise,\n abortController,\n context,\n location: inner.location,\n navigate: (opts) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n cause: preload ? 'preload' : cause,\n route,\n ...inner.router.options.additionalContext,\n }\n}\n\nconst runLoader = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): Promise<void> => {\n try {\n // If the Matches component rendered\n // the pending component and needs to show it for\n // a minimum duration, we''ll wait for it to resolve\n // before committing to the match and resolving\n // the loadPromise\n\n const match = inner.router.getMatch(matchId)!\n\n // Actually run the loader and handle the result\n try {\n if (!(isServer ?? inner.router.isServer) || match.ssr === true) {\n loadRouteChunk(route)\n }\n\n // Kick off the loader!\n const routeLoader = route.options.loader\n const loader =\n typeof routeLoader === 'function' ? routeLoader : routeLoader?.handler\n const loaderResult = loader?.(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n const loaderResultIsPromise = !!loader && isPromise(loaderResult)\n\n const willLoadSomething = !!(\n loaderResultIsPromise ||\n route._lazyPromise ||\n route._componentsPromise ||\n route.options.head ||\n route.options.scripts ||\n route.options.headers ||\n match._nonReactive.minPendingPromise\n )\n\n if (willLoadSomething) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'loader',\n }))\n }\n\n if (loader) {\n const loaderData = loaderResultIsPromise\n ? await loaderResult\n : loaderResult\n\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n loaderData,\n )\n if (loaderData !== undefined) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n loaderData,\n }))\n }\n }\n\n // Lazy option can modify the route options,\n // so we need to wait for it to resolve before\n // we can use the options\n if (route._lazyPromise) await route._lazyPromise\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n // Last but not least, wait for the the components\n // to be preloaded before we resolve the match\n if (route._componentsPromise) await route._componentsPromise\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error: undefined,\n context: buildMatchContext(inner, index),\n status: 'success',\n isFetching: false,\n updatedAt: Date.now(),\n }))\n } catch (e) {\n let error = e\n\n if ((error as any)?.name === 'AbortError') {\n if (match.abortController.signal.aborted) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n return\n }\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n status: prev.status === 'pending' ? 'success' : prev.status,\n isFetching: false,\n context: buildMatchContext(inner, index),\n }))\n return\n }\n\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n if (isNotFound(e)) {\n await (route.options.notFoundComponent as any)?.preload?.()\n }\n\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), e)\n\n try {\n route.options.onError?.(e)\n } catch (onErrorError) {\n error = onErrorError\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n onErrorError,\n )\n }\n if (!isRedirect(error) && !isNotFound(error)) {\n await loadRouteChunk(route, ['errorComponent'])\n }\n\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error,\n context: buildMatchContext(inner, index),\n status: 'error',\n isFetching: false,\n }))\n }\n } catch (err) {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.loaderPromise = undefined\n }\n handleRedirectAndNotFound(inner, match, err)\n }\n}\n\nconst loadRouteMatch = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n index: number,\n): Promise<AnyRouteMatch> => {\n async function handleLoader(\n preload: boolean,\n prevMatch: AnyRouteMatch,\n previousRouteMatchId: string | undefined,\n match: AnyRouteMatch,\n route: AnyRoute,\n ) {\n const age = Date.now() - prevMatch.updatedAt\n\n const staleAge = preload\n ? (route.options.preloadStaleTime ??\n inner.router.options.defaultPreloadStaleTime ??\n 30_000) // 30 seconds for preloads by default\n : (route.options.staleTime ?? inner.router.options.defaultStaleTime ?? 0)\n\n const shouldReloadOption = route.options.shouldReload\n\n // Default to reloading the route all the time\n // Allow shouldReload to get the last say,\n // if provided.\n const shouldReload =\n typeof shouldReloadOption === 'function'\n ? shouldReloadOption(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n : shouldReloadOption\n\n // If the route is successful and still fresh, just resolve\n const { status, invalid } = match\n const staleMatchShouldReload =\n age >= staleAge &&\n (!!inner.forceStaleReload ||\n match.cause === 'enter' ||\n (previousRouteMatchId !== undefined &&\n previousRouteMatchId !== match.id))\n loaderShouldRunAsync =\n status === 'success' &&\n (invalid || (shouldReload ?? staleMatchShouldReload))\n if (preload && route.options.preload === false) {\n // Do nothing\n } else if (\n loaderShouldRunAsync &&\n !inner.sync &&\n shouldReloadInBackground\n ) {\n loaderIsRunningAsync = true\n ;(async () => {\n try {\n await runLoader(inner, matchPromises, matchId, index, route)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n match._nonReactive.loadPromise = undefined\n } catch (err) {\n if (isRedirect(err)) {\n await inner.router.navigate(err.options)\n }\n }\n })()\n } else if (status !== 'success' || loaderShouldRunAsync) {\n await runLoader(inner, matchPromises, matchId, index, route)\n } else {\n syncMatchContext(inner, matchId, index)\n }\n }\n\n const { id: matchId, routeId } = inner.matches[index]!\n let loaderShouldRunAsync = false\n let loaderIsRunningAsync = false\n const route = inner.router.looseRoutesById[routeId]!\n const routeLoader = route.options.loader\n const shouldReloadInBackground =\n ((typeof routeLoader === 'function'\n ? undefined\n : routeLoader?.staleReloadMode) ??\n inner.router.options.defaultStaleReloadMode) !== 'blocking'\n\n if (shouldSkipLoader(inner, matchId)) {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return inner.matches[index]!\n }\n\n syncMatchContext(inner, matchId, index)\n\n if (isServer ?? inner.router.isServer) {\n return inner.router.getMatch(matchId)!\n }\n } else {\n const prevMatch = inner.router.getMatch(matchId)! // This is where all of the stale-while-revalidate magic happens\n const activeIdAtIndex = inner.router.stores.matchesId.get()[index]\n const activeAtIndex =\n (activeIdAtIndex &&\n inner.router.stores.matchStores.get(activeIdAtIndex)) ||\n null\n const previousRouteMatchId =\n activeAtIndex?.routeId === routeId\n ? activeIdAtIndex\n : inner.router.stores.matches.get().find((d) => d.routeId === routeId)\n ?.id\n const preload = resolvePreload(inner, matchId)\n\n // there is a loaderPromise, so we are in the middle of a load\n if (prevMatch._nonReactive.loaderPromise) {\n // do not block if we already have stale data we can show\n // but only if the ongoing load is not a preload since error handling is different for preloads\n // and we don't want to swallow errors\n if (\n prevMatch.status === 'success' &&\n !inner.sync &&\n !prevMatch.preload &&\n shouldReloadInBackground\n ) {\n return prevMatch\n }\n await prevMatch._nonReactive.loaderPromise\n const match = inner.router.getMatch(matchId)!\n const error = match._nonReactive.error || match.error\n if (error) {\n handleRedirectAndNotFound(inner, match, error)\n }\n\n if (match.status === 'pending') {\n await handleLoader(\n preload,\n prevMatch,\n previousRouteMatchId,\n match,\n route,\n )\n }\n } else {\n const nextPreload =\n preload && !inner.router.stores.matchStores.has(matchId)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise = createControlledPromise<void>()\n if (nextPreload !== match.preload) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n preload: nextPreload,\n }))\n }\n\n await handleLoader(preload, prevMatch, previousRouteMatchId, match, route)\n }\n }\n const match = inner.router.getMatch(matchId)!\n if (!loaderIsRunningAsync) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loadPromise = undefined\n }\n\n clearTimeout(match._nonReactive.pendingTimeout)\n match._nonReactive.pendingTimeout = undefined\n if (!loaderIsRunningAsync) match._nonReactive.loaderPromise = undefined\n match._nonReactive.dehydrated = undefined\n\n const nextIsFetching = loaderIsRunningAsync ? match.isFetching : false\n if (nextIsFetching !== match.isFetching || match.invalid !== false) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: nextIsFetching,\n invalid: false,\n }))\n return inner.router.getMatch(matchId)!\n } else {\n return match\n }\n}\n\nexport async function loadMatches(arg: {\n router: AnyRouter\n location: ParsedLocation\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n updateMatch: UpdateMatchFn\n sync?: boolean\n}): Promise<Array<MakeRouteMatch>> {\n const inner: InnerLoadContext = arg\n const matchPromises: Array<Promise<AnyRouteMatch>> = []\n\n // make sure the pending component is immediately rendered when hydrating a match that is not SSRed\n // the pending component was already rendered on the server and we want to keep it shown on the client until minPendingMs is reached\n if (\n !(isServer ?? inner.router.isServer) &&\n hasForcePendingActiveMatch(inner.router)\n ) {\n triggerOnReady(inner)\n }\n\n let beforeLoadNotFound: NotFoundError | undefined\n\n // Execute all beforeLoads one by one\n for (let i = 0; i < inner.matches.length; i++) {\n try {\n const beforeLoad = handleBeforeLoad(inner, i)\n if (isPromise(beforeLoad)) await beforeLoad\n } catch (err) {\n if (isRedirect(err)) {\n throw err\n }\n if (isNotFound(err)) {\n beforeLoadNotFound = err\n } else {\n if (!inner.preload) throw err\n }\n break\n }\n\n if (inner.serialError || inner.firstBadMatchIndex != null) {\n break\n }\n }\n\n // Execute loaders once, with max index adapted for beforeLoad notFound handling.\n const baseMaxIndexExclusive = inner.firstBadMatchIndex ?? inner.matches.length\n\n const boundaryIndex =\n beforeLoadNotFound && !inner.preload\n ? getNotFoundBoundaryIndex(inner, beforeLoadNotFound)\n : undefined\n\n const maxIndexExclusive =\n beforeLoadNotFound && inner.preload\n ? 0\n : boundaryIndex !== undefined\n ? Math.min(boundaryIndex + 1, baseMaxIndexExclusive)\n : baseMaxIndexExclusive\n\n let firstNotFound: NotFoundError | undefined\n let firstUnhandledRejection: unknown\n\n for (let i = 0; i < maxIndexExclusive; i++) {\n matchPromises.push(loadRouteMatch(inner, matchPromises, i))\n }\n\n try {\n await Promise.all(matchPromises)\n } catch {\n const settled = await Promise.allSettled(matchPromises)\n\n for (const result of settled) {\n if (result.status !== 'rejected') continue\n\n const reason = result.reason\n if (isRedirect(reason)) {\n throw reason\n }\n if (isNotFound(reason)) {\n firstNotFound ??= reason\n } else {\n firstUnhandledRejection ??= reason\n }\n }\n\n if (firstUnhandledRejection !== undefined) {\n throw firstUnhandledRejection\n }\n }\n\n const notFoundToThrow =\n firstNotFound ??\n (beforeLoadNotFound && !inner.preload ? beforeLoadNotFound : undefined)\n\n let headMaxIndex =\n inner.firstBadMatchIndex !== undefined\n ? inner.firstBadMatchIndex\n : inner.matches.length - 1\n\n if (!notFoundToThrow && beforeLoadNotFound && inner.preload) {\n return inner.matches\n }\n\n if (notFoundToThrow) {\n // Determine once which matched route will actually render the\n // notFoundComponent, then pass this precomputed index through the remaining\n // finalization steps.\n // This can differ from the throwing route when routeId targets an ancestor\n // boundary (or when bubbling resolves to a parent/root boundary).\n const renderedBoundaryIndex = getNotFoundBoundaryIndex(\n inner,\n notFoundToThrow,\n )\n\n if (renderedBoundaryIndex === undefined) {\n if (process.env.NODE_ENV !== 'production') {\n throw new Error(\n 'Invariant failed: Could not find match for notFound boundary',\n )\n }\n\n invariant()\n }\n const boundaryMatch = inner.matches[renderedBoundaryIndex]!\n\n const boundaryRoute = inner.router.looseRoutesById[boundaryMatch.routeId]!\n const defaultNotFoundComponent = (inner.router.options as any)\n ?.defaultNotFoundComponent\n\n // Ensure a notFoundComponent exists on the boundary route\n if (!boundaryRoute.options.notFoundComponent && defaultNotFoundComponent) {\n boundaryRoute.options.notFoundComponent = defaultNotFoundComponent\n }\n\n notFoundToThrow.routeId = boundaryMatch.routeId\n\n const boundaryIsRoot = boundaryMatch.routeId === inner.router.routeTree.id\n\n inner.updateMatch(boundaryMatch.id, (prev) => ({\n ...prev,\n ...(boundaryIsRoot\n ? // For root boundary, use globalNotFound so the root component's\n // shell still renders and <Outlet> handles the not-found display,\n // instead of replacing the entire root shell via status='notFound'.\n { status: 'success' as const, globalNotFound: true, error: undefined }\n : // For non-root boundaries, set status:'notFound' so MatchInner\n // renders the notFoundComponent directly.\n { status: 'notFound' as const, error: notFoundToThrow }),\n isFetching: false,\n }))\n\n headMaxIndex = renderedBoundaryIndex\n\n // Ensure the rendering boundary route chunk (and its lazy components, including\n // lazy notFoundComponent) is loaded before we continue to head execution/render.\n await loadRouteChunk(boundaryRoute, ['notFoundComponent'])\n } else if (!inner.preload) {\n // Clear stale root global-not-found state on normal navigations that do not\n // throw notFound. This must live here (instead of only in runLoader success)\n // because the root loader may be skipped when data is still fresh.\n const rootMatch = inner.matches[0]!\n // `rootMatch` is the next match for this navigation. If it is not global\n // not-found, then any currently stored root global-not-found is stale.\n if (!rootMatch.globalNotFound) {\n // `currentRootMatch` is the current store state (from the previous\n // navigation/load). Update only when a stale flag is actually present.\n const currentRootMatch = inner.router.getMatch(rootMatch.id)\n if (currentRootMatch?.globalNotFound) {\n inner.updateMatch(rootMatch.id, (prev) => ({\n ...prev,\n globalNotFound: false,\n error: undefined,\n }))\n }\n }\n }\n\n // When a serial error occurred (e.g. beforeLoad threw a regular Error),\n // the erroring route's lazy chunk wasn't loaded because loaders were skipped.\n // We need to load it so the code-split errorComponent is available for rendering.\n if (inner.serialError && inner.firstBadMatchIndex !== undefined) {\n const errorRoute =\n inner.router.looseRoutesById[\n inner.matches[inner.firstBadMatchIndex]!.routeId\n ]!\n await loadRouteChunk(errorRoute, ['errorComponent'])\n }\n\n // serially execute heads once after loaders/notFound handling, ensuring\n // all head functions get a chance even if one throws.\n for (let i = 0; i <= headMaxIndex; i++) {\n const match = inner.matches[i]!\n const { id: matchId, routeId } = match\n const route = inner.router.looseRoutesById[routeId]!\n try {\n const headResult = executeHead(inner, matchId, route)\n if (headResult) {\n const head = await headResult\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n ...head,\n }))\n }\n } catch (err) {\n console.error(`Error executing head for route ${routeId}:`, err)\n }\n }\n\n const readyPromise = triggerOnReady(inner)\n if (isPromise(readyPromise)) {\n await readyPromise\n }\n\n if (notFoundToThrow) {\n throw notFoundToThrow\n }\n\n if (inner.serialError && !inner.preload && !inner.onReady) {\n throw inner.serialError\n }\n\n return inner.matches\n}\n\nexport type RouteComponentType =\n | 'component'\n | 'errorComponent'\n | 'pendingComponent'\n | 'notFoundComponent'\n\nfunction preloadRouteComponents(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType>,\n): Promise<void> | undefined {\n const preloads = componentTypesToLoad\n .map((type) => (route.options[type] as any)?.preload?.())\n .filter(Boolean)\n\n if (preloads.length === 0) return undefined\n\n return Promise.all(preloads) as any as Promise<void>\n}\n\nexport function loadRouteChunk(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType> = componentTypes,\n) {\n if (!route._lazyLoaded && route._lazyPromise === undefined) {\n if (route.lazyFn) {\n route._lazyPromise = route.lazyFn().then((lazyRoute) => {\n // explicitly don't copy over the lazy route's id\n const { id: _id, ...options } = lazyRoute.options\n Object.assign(route.options, options)\n route._lazyLoaded = true\n route._lazyPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._lazyLoaded = true\n }\n }\n\n const runAfterLazy = () =>\n route._componentsLoaded\n ? undefined\n : componentTypesToLoad === componentTypes\n ? (() => {\n if (route._componentsPromise === undefined) {\n const componentsPromise = preloadRouteComponents(\n route,\n componentTypes,\n )\n\n if (componentsPromise) {\n route._componentsPromise = componentsPromise.then(() => {\n route._componentsLoaded = true\n route._componentsPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._componentsLoaded = true\n }\n }\n\n return route._componentsPromise\n })()\n : preloadRouteComponents(route, componentTypesToLoad)\n\n return route._lazyPromise\n ? route._lazyPromise.then(runAfterLazy)\n : runAfterLazy()\n}\n\nfunction makeMaybe<TValue, TError>(\n value: TValue,\n error: TError,\n): { status: 'success'; value: TValue } | { status: 'error'; error: TError } {\n if (error) {\n return { status: 'error' as const, error }\n }\n return { status: 'success' as const, value }\n}\n\nexport function routeNeedsPreload(route: AnyRoute) {\n for (const componentType of componentTypes) {\n if ((route.options[componentType] as any)?.preload) {\n return true\n }\n }\n return false\n}\n\nexport const componentTypes: Array<RouteComponentType> = [\n 'component',\n 'errorComponent',\n 'pendingComponent',\n 'notFoundComponent',\n] as const\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"load-matches.cjs","names":[],"sources":["../../src/load-matches.ts"],"sourcesContent":["import { isServer } from '@tanstack/router-core/isServer'\nimport { invariant } from './invariant'\nimport { createControlledPromise, isPromise } from './utils'\nimport { isNotFound } from './not-found'\nimport { rootRouteId } from './root'\nimport { isRedirect } from './redirect'\nimport type { NotFoundError } from './not-found'\nimport type { ParsedLocation } from './location'\nimport type {\n AnyRoute,\n BeforeLoadContextOptions,\n LoaderFnContext,\n SsrContextOptions,\n} from './route'\nimport type { AnyRouteMatch, MakeRouteMatch } from './Matches'\nimport type { AnyRouter, SSROption, UpdateMatchFn } from './router'\n\n/**\n * An object of this shape is created when calling `loadMatches`.\n * It contains everything we need for all other functions in this file\n * to work. (It's basically the function's argument, plus a few mutable states)\n */\ntype InnerLoadContext = {\n /** the calling router instance */\n router: AnyRouter\n location: ParsedLocation\n /** mutable state, scoped to a `loadMatches` call */\n firstBadMatchIndex?: number\n /** mutable state, scoped to a `loadMatches` call */\n rendered?: boolean\n serialError?: unknown\n updateMatch: UpdateMatchFn\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n sync?: boolean\n}\n\nconst triggerOnReady = (inner: InnerLoadContext): void | Promise<void> => {\n if (!inner.rendered) {\n inner.rendered = true\n return inner.onReady?.()\n }\n}\n\nconst hasForcePendingActiveMatch = (router: AnyRouter): boolean => {\n return router.stores.matchesId.get().some((matchId) => {\n return router.stores.matchStores.get(matchId)?.get()._forcePending\n })\n}\n\nconst resolvePreload = (inner: InnerLoadContext, matchId: string): boolean => {\n return !!(inner.preload && !inner.router.stores.matchStores.has(matchId))\n}\n\n/**\n * Builds the accumulated context from router options and all matches up to (and optionally including) the given index.\n * Merges __routeContext and __beforeLoadContext from each match.\n */\nconst buildMatchContext = (\n inner: InnerLoadContext,\n index: number,\n includeCurrentMatch: boolean = true,\n): Record<string, unknown> => {\n const context: Record<string, unknown> = {\n ...(inner.router.options.context ?? {}),\n }\n const end = includeCurrentMatch ? index : index - 1\n for (let i = 0; i <= end; i++) {\n const innerMatch = inner.matches[i]\n if (!innerMatch) continue\n const m = inner.router.getMatch(innerMatch.id)\n if (!m) continue\n Object.assign(context, m.__routeContext, m.__beforeLoadContext)\n }\n return context\n}\n\nconst getNotFoundBoundaryIndex = (\n inner: InnerLoadContext,\n err: NotFoundError,\n): number | undefined => {\n if (!inner.matches.length) {\n return undefined\n }\n\n const requestedRouteId = err.routeId\n const matchedRootIndex = inner.matches.findIndex(\n (m) => m.routeId === inner.router.routeTree.id,\n )\n const rootIndex = matchedRootIndex >= 0 ? matchedRootIndex : 0\n\n let startIndex = requestedRouteId\n ? inner.matches.findIndex((match) => match.routeId === requestedRouteId)\n : (inner.firstBadMatchIndex ?? inner.matches.length - 1)\n\n if (startIndex < 0) {\n startIndex = rootIndex\n }\n\n for (let i = startIndex; i >= 0; i--) {\n const match = inner.matches[i]!\n const route = inner.router.looseRoutesById[match.routeId]!\n if (route.options.notFoundComponent) {\n return i\n }\n }\n\n // If no boundary component is found, preserve explicit routeId targeting behavior,\n // otherwise default to root for untargeted notFounds.\n return requestedRouteId ? startIndex : rootIndex\n}\n\nconst handleRedirectAndNotFound = (\n inner: InnerLoadContext,\n match: AnyRouteMatch | undefined,\n err: unknown,\n): void => {\n if (!isRedirect(err) && !isNotFound(err)) return\n\n if (isRedirect(err) && err.redirectHandled && !err.options.reloadDocument) {\n throw err\n }\n\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n match._nonReactive.loaderPromise = undefined\n\n match._nonReactive.error = err\n\n inner.updateMatch(match.id, (prev) => ({\n ...prev,\n status: isRedirect(err)\n ? 'redirected'\n : isNotFound(err)\n ? 'notFound'\n : prev.status === 'pending'\n ? 'success'\n : prev.status,\n context: buildMatchContext(inner, match.index),\n isFetching: false,\n error: err,\n }))\n\n if (isNotFound(err) && !err.routeId) {\n // Stamp the throwing match's routeId so that the finalization step in\n // loadMatches knows where the notFound originated. The actual boundary\n // resolution (walking up to the nearest notFoundComponent) is deferred to\n // the finalization step, where firstBadMatchIndex is stable and\n // headMaxIndex can be capped correctly.\n err.routeId = match.routeId\n }\n\n match._nonReactive.loadPromise?.resolve()\n }\n\n if (isRedirect(err)) {\n inner.rendered = true\n err.options._fromLocation = inner.location\n err.redirectHandled = true\n err = inner.router.resolveRedirect(err)\n }\n\n throw err\n}\n\nconst shouldSkipLoader = (\n inner: InnerLoadContext,\n matchId: string,\n): boolean => {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return true\n }\n // upon hydration, we skip the loader if the match has been dehydrated on the server\n if (!(isServer ?? inner.router.isServer) && match._nonReactive.dehydrated) {\n return true\n }\n\n if ((isServer ?? inner.router.isServer) && match.ssr === false) {\n return true\n }\n\n return false\n}\n\nconst syncMatchContext = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n): void => {\n const nextContext = buildMatchContext(inner, index)\n\n inner.updateMatch(matchId, (prev) => {\n return {\n ...prev,\n context: nextContext,\n }\n })\n}\n\nconst handleSerialError = (\n inner: InnerLoadContext,\n index: number,\n err: any,\n routerCode: string,\n): void => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n // Much like suspense, we use a promise here to know if\n // we've been outdated by a new loadMatches call and\n // should abort the current async operation\n if (err instanceof Promise) {\n throw err\n }\n\n err.routerCode = routerCode\n inner.firstBadMatchIndex ??= index\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n\n try {\n route.options.onError?.(err)\n } catch (errorHandlerErr) {\n err = errorHandlerErr\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n }\n\n inner.updateMatch(matchId, (prev) => {\n prev._nonReactive.beforeLoadPromise?.resolve()\n prev._nonReactive.beforeLoadPromise = undefined\n prev._nonReactive.loadPromise?.resolve()\n\n return {\n ...prev,\n error: err,\n status: 'error',\n isFetching: false,\n updatedAt: Date.now(),\n abortController: new AbortController(),\n }\n })\n\n if (!inner.preload && !isRedirect(err) && !isNotFound(err)) {\n inner.serialError ??= err\n }\n}\n\nconst isBeforeLoadSsr = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n const parentMatchId = inner.matches[index - 1]?.id\n const parentMatch = parentMatchId\n ? inner.router.getMatch(parentMatchId)!\n : undefined\n\n // in SPA mode, only SSR the root route\n if (inner.router.isShell()) {\n existingMatch.ssr = route.id === rootRouteId\n return\n }\n\n if (parentMatch?.ssr === false) {\n existingMatch.ssr = false\n return\n }\n\n const parentOverride = (tempSsr: SSROption) => {\n if (tempSsr === true && parentMatch?.ssr === 'data-only') {\n return 'data-only'\n }\n return tempSsr\n }\n\n const defaultSsr = inner.router.options.defaultSsr ?? true\n\n if (route.options.ssr === undefined) {\n existingMatch.ssr = parentOverride(defaultSsr)\n return\n }\n\n if (typeof route.options.ssr !== 'function') {\n existingMatch.ssr = parentOverride(route.options.ssr)\n return\n }\n const { search, params } = existingMatch\n\n const ssrFnContext: SsrContextOptions<any, any, any> = {\n search: makeMaybe(search, existingMatch.searchError),\n params: makeMaybe(params, existingMatch.paramsError),\n location: inner.location,\n matches: inner.matches.map((match) => ({\n index: match.index,\n pathname: match.pathname,\n fullPath: match.fullPath,\n staticData: match.staticData,\n id: match.id,\n routeId: match.routeId,\n search: makeMaybe(match.search, match.searchError),\n params: makeMaybe(match.params, match.paramsError),\n ssr: match.ssr,\n })),\n }\n\n const tempSsr = route.options.ssr(ssrFnContext)\n if (isPromise(tempSsr)) {\n return tempSsr.then((ssr) => {\n existingMatch.ssr = parentOverride(ssr ?? defaultSsr)\n })\n }\n\n existingMatch.ssr = parentOverride(tempSsr ?? defaultSsr)\n return\n}\n\nconst setupPendingTimeout = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n match: AnyRouteMatch,\n): void => {\n if (match._nonReactive.pendingTimeout !== undefined) return\n\n const pendingMs =\n route.options.pendingMs ?? inner.router.options.defaultPendingMs\n const shouldPending = !!(\n inner.onReady &&\n !(isServer ?? inner.router.isServer) &&\n !resolvePreload(inner, matchId) &&\n (route.options.loader ||\n route.options.beforeLoad ||\n routeNeedsPreload(route)) &&\n typeof pendingMs === 'number' &&\n pendingMs !== Infinity &&\n (route.options.pendingComponent ??\n (inner.router.options as any)?.defaultPendingComponent)\n )\n\n if (shouldPending) {\n const pendingTimeout = setTimeout(() => {\n // Update the match and prematurely resolve the loadMatches promise so that\n // the pending component can start rendering\n triggerOnReady(inner)\n }, pendingMs)\n match._nonReactive.pendingTimeout = pendingTimeout\n }\n}\n\nconst preBeforeLoadSetup = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n\n // If we are in the middle of a load, either of these will be present\n // (not to be confused with `loadPromise`, which is always defined)\n if (\n !existingMatch._nonReactive.beforeLoadPromise &&\n !existingMatch._nonReactive.loaderPromise\n )\n return\n\n setupPendingTimeout(inner, matchId, route, existingMatch)\n\n const then = () => {\n const match = inner.router.getMatch(matchId)!\n if (\n match.preload &&\n (match.status === 'redirected' || match.status === 'notFound')\n ) {\n handleRedirectAndNotFound(inner, match, match.error)\n }\n }\n\n // Wait for the previous beforeLoad to resolve before we continue\n return existingMatch._nonReactive.beforeLoadPromise\n ? existingMatch._nonReactive.beforeLoadPromise.then(then)\n : then()\n}\n\nconst executeBeforeLoad = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const match = inner.router.getMatch(matchId)!\n\n // explicitly capture the previous loadPromise\n let prevLoadPromise = match._nonReactive.loadPromise\n match._nonReactive.loadPromise = createControlledPromise<void>(() => {\n prevLoadPromise?.resolve()\n prevLoadPromise = undefined\n })\n\n const { paramsError, searchError } = match\n\n if (paramsError) {\n handleSerialError(inner, index, paramsError, 'PARSE_PARAMS')\n }\n\n if (searchError) {\n handleSerialError(inner, index, searchError, 'VALIDATE_SEARCH')\n }\n\n setupPendingTimeout(inner, matchId, route, match)\n\n const abortController = new AbortController()\n\n let isPending = false\n const pending = () => {\n if (isPending) return\n isPending = true\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'beforeLoad',\n fetchCount: prev.fetchCount + 1,\n abortController,\n // Note: We intentionally don't update context here.\n // Context should only be updated after beforeLoad resolves to avoid\n // components seeing incomplete context during async beforeLoad execution.\n }))\n }\n\n const resolve = () => {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: false,\n }))\n }\n\n // if there is no `beforeLoad` option, just mark as pending and resolve\n // Context will be updated later in loadRouteMatch after loader completes\n if (!route.options.beforeLoad) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n\n match._nonReactive.beforeLoadPromise = createControlledPromise<void>()\n\n // Build context from all parent matches, excluding current match's __beforeLoadContext\n // (since we're about to execute beforeLoad for this match)\n const context = {\n ...buildMatchContext(inner, index, false),\n ...match.__routeContext,\n }\n const { search, params, cause } = match\n const preload = resolvePreload(inner, matchId)\n const beforeLoadFnContext: BeforeLoadContextOptions<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > = {\n search,\n abortController,\n params,\n preload,\n context,\n location: inner.location,\n navigate: (opts: any) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n buildLocation: inner.router.buildLocation,\n cause: preload ? 'preload' : cause,\n matches: inner.matches,\n routeId: route.id,\n ...inner.router.options.additionalContext,\n }\n\n const updateContext = (beforeLoadContext: any) => {\n if (beforeLoadContext === undefined) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n if (isRedirect(beforeLoadContext) || isNotFound(beforeLoadContext)) {\n pending()\n handleSerialError(inner, index, beforeLoadContext, 'BEFORE_LOAD')\n }\n\n inner.router.batch(() => {\n pending()\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n __beforeLoadContext: beforeLoadContext,\n }))\n resolve()\n })\n }\n\n let beforeLoadContext\n try {\n beforeLoadContext = route.options.beforeLoad(beforeLoadFnContext)\n if (isPromise(beforeLoadContext)) {\n pending()\n return beforeLoadContext\n .catch((err) => {\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n })\n .then(updateContext)\n }\n } catch (err) {\n pending()\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n }\n\n updateContext(beforeLoadContext)\n return\n}\n\nconst handleBeforeLoad = (\n inner: InnerLoadContext,\n index: number,\n): void | Promise<void> => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n const serverSsr = () => {\n // on the server, determine whether SSR the current match or not\n if (isServer ?? inner.router.isServer) {\n const maybePromise = isBeforeLoadSsr(inner, matchId, index, route)\n if (isPromise(maybePromise)) return maybePromise.then(queueExecution)\n }\n return queueExecution()\n }\n\n const execute = () => executeBeforeLoad(inner, matchId, index, route)\n\n const queueExecution = () => {\n if (shouldSkipLoader(inner, matchId)) return\n const result = preBeforeLoadSetup(inner, matchId, route)\n return isPromise(result) ? result.then(execute) : execute()\n }\n\n return serverSsr()\n}\n\nconst executeHead = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<\n Pick<\n AnyRouteMatch,\n 'meta' | 'links' | 'headScripts' | 'headers' | 'scripts' | 'styles'\n >\n> => {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (!match) {\n return\n }\n if (!route.options.head && !route.options.scripts && !route.options.headers) {\n return\n }\n const assetContext = {\n ssr: inner.router.options.ssr,\n matches: inner.matches,\n match,\n params: match.params,\n loaderData: match.loaderData,\n }\n\n return Promise.all([\n route.options.head?.(assetContext),\n route.options.scripts?.(assetContext),\n route.options.headers?.(assetContext),\n ]).then(([headFnContent, scripts, headers]) => {\n const meta = headFnContent?.meta\n const links = headFnContent?.links\n const headScripts = headFnContent?.scripts\n const styles = headFnContent?.styles\n\n return {\n meta,\n links,\n headScripts,\n headers,\n scripts,\n styles,\n }\n })\n}\n\nconst getLoaderContext = (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): LoaderFnContext => {\n const parentMatchPromise = matchPromises[index - 1] as any\n const { params, loaderDeps, abortController, cause } =\n inner.router.getMatch(matchId)!\n\n const context = buildMatchContext(inner, index)\n\n const preload = resolvePreload(inner, matchId)\n\n return {\n params,\n deps: loaderDeps,\n preload: !!preload,\n parentMatchPromise,\n abortController,\n context,\n location: inner.location,\n navigate: (opts) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n cause: preload ? 'preload' : cause,\n route,\n ...inner.router.options.additionalContext,\n }\n}\n\nconst runLoader = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): Promise<void> => {\n try {\n // If the Matches component rendered\n // the pending component and needs to show it for\n // a minimum duration, we''ll wait for it to resolve\n // before committing to the match and resolving\n // the loadPromise\n\n const match = inner.router.getMatch(matchId)!\n\n // Actually run the loader and handle the result\n try {\n if (!(isServer ?? inner.router.isServer) || match.ssr === true) {\n loadRouteChunk(route)\n }\n\n // Kick off the loader!\n const routeLoader = route.options.loader\n const loader =\n typeof routeLoader === 'function' ? routeLoader : routeLoader?.handler\n const loaderResult = loader?.(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n const loaderResultIsPromise = !!loader && isPromise(loaderResult)\n\n const willLoadSomething = !!(\n loaderResultIsPromise ||\n route._lazyPromise ||\n route._componentsPromise ||\n route.options.head ||\n route.options.scripts ||\n route.options.headers ||\n match._nonReactive.minPendingPromise\n )\n\n if (willLoadSomething) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'loader',\n }))\n }\n\n if (loader) {\n const loaderData = loaderResultIsPromise\n ? await loaderResult\n : loaderResult\n\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n loaderData,\n )\n if (loaderData !== undefined) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n loaderData,\n }))\n }\n }\n\n // Lazy option can modify the route options,\n // so we need to wait for it to resolve before\n // we can use the options\n if (route._lazyPromise) await route._lazyPromise\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n // Last but not least, wait for the components\n // to be preloaded before we resolve the match\n if (route._componentsPromise) await route._componentsPromise\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error: undefined,\n context: buildMatchContext(inner, index),\n status: 'success',\n isFetching: false,\n updatedAt: Date.now(),\n }))\n } catch (e) {\n let error = e\n\n if ((error as any)?.name === 'AbortError') {\n if (match.abortController.signal.aborted) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n return\n }\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n status: prev.status === 'pending' ? 'success' : prev.status,\n isFetching: false,\n context: buildMatchContext(inner, index),\n }))\n return\n }\n\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n if (isNotFound(e)) {\n await (route.options.notFoundComponent as any)?.preload?.()\n }\n\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), e)\n\n try {\n route.options.onError?.(e)\n } catch (onErrorError) {\n error = onErrorError\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n onErrorError,\n )\n }\n if (!isRedirect(error) && !isNotFound(error)) {\n await loadRouteChunk(route, ['errorComponent'])\n }\n\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error,\n context: buildMatchContext(inner, index),\n status: 'error',\n isFetching: false,\n }))\n }\n } catch (err) {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.loaderPromise = undefined\n }\n handleRedirectAndNotFound(inner, match, err)\n }\n}\n\nconst loadRouteMatch = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n index: number,\n): Promise<AnyRouteMatch> => {\n async function handleLoader(\n preload: boolean,\n prevMatch: AnyRouteMatch,\n previousRouteMatchId: string | undefined,\n match: AnyRouteMatch,\n route: AnyRoute,\n ) {\n const age = Date.now() - prevMatch.updatedAt\n\n const staleAge = preload\n ? (route.options.preloadStaleTime ??\n inner.router.options.defaultPreloadStaleTime ??\n 30_000) // 30 seconds for preloads by default\n : (route.options.staleTime ?? inner.router.options.defaultStaleTime ?? 0)\n\n const shouldReloadOption = route.options.shouldReload\n\n // Default to reloading the route all the time\n // Allow shouldReload to get the last say,\n // if provided.\n const shouldReload =\n typeof shouldReloadOption === 'function'\n ? shouldReloadOption(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n : shouldReloadOption\n\n // If the route is successful and still fresh, just resolve\n const { status, invalid } = match\n const staleMatchShouldReload =\n age >= staleAge &&\n (!!inner.forceStaleReload ||\n match.cause === 'enter' ||\n (previousRouteMatchId !== undefined &&\n previousRouteMatchId !== match.id))\n loaderShouldRunAsync =\n status === 'success' &&\n (invalid || (shouldReload ?? staleMatchShouldReload))\n if (preload && route.options.preload === false) {\n // Do nothing\n } else if (\n loaderShouldRunAsync &&\n !inner.sync &&\n shouldReloadInBackground\n ) {\n loaderIsRunningAsync = true\n ;(async () => {\n try {\n await runLoader(inner, matchPromises, matchId, index, route)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n match._nonReactive.loadPromise = undefined\n } catch (err) {\n if (isRedirect(err)) {\n await inner.router.navigate(err.options)\n }\n }\n })()\n } else if (status !== 'success' || loaderShouldRunAsync) {\n await runLoader(inner, matchPromises, matchId, index, route)\n } else {\n syncMatchContext(inner, matchId, index)\n }\n }\n\n const { id: matchId, routeId } = inner.matches[index]!\n let loaderShouldRunAsync = false\n let loaderIsRunningAsync = false\n const route = inner.router.looseRoutesById[routeId]!\n const routeLoader = route.options.loader\n const shouldReloadInBackground =\n ((typeof routeLoader === 'function'\n ? undefined\n : routeLoader?.staleReloadMode) ??\n inner.router.options.defaultStaleReloadMode) !== 'blocking'\n\n if (shouldSkipLoader(inner, matchId)) {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return inner.matches[index]!\n }\n\n syncMatchContext(inner, matchId, index)\n\n if (isServer ?? inner.router.isServer) {\n return inner.router.getMatch(matchId)!\n }\n } else {\n const prevMatch = inner.router.getMatch(matchId)! // This is where all of the stale-while-revalidate magic happens\n const activeIdAtIndex = inner.router.stores.matchesId.get()[index]\n const activeAtIndex =\n (activeIdAtIndex &&\n inner.router.stores.matchStores.get(activeIdAtIndex)) ||\n null\n const previousRouteMatchId =\n activeAtIndex?.routeId === routeId\n ? activeIdAtIndex\n : inner.router.stores.matches.get().find((d) => d.routeId === routeId)\n ?.id\n const preload = resolvePreload(inner, matchId)\n\n // there is a loaderPromise, so we are in the middle of a load\n if (prevMatch._nonReactive.loaderPromise) {\n // do not block if we already have stale data we can show\n // but only if the ongoing load is not a preload since error handling is different for preloads\n // and we don't want to swallow errors\n if (\n prevMatch.status === 'success' &&\n !inner.sync &&\n !prevMatch.preload &&\n shouldReloadInBackground\n ) {\n return prevMatch\n }\n await prevMatch._nonReactive.loaderPromise\n const match = inner.router.getMatch(matchId)!\n const error = match._nonReactive.error || match.error\n if (error) {\n handleRedirectAndNotFound(inner, match, error)\n }\n\n if (match.status === 'pending') {\n await handleLoader(\n preload,\n prevMatch,\n previousRouteMatchId,\n match,\n route,\n )\n }\n } else {\n const nextPreload =\n preload && !inner.router.stores.matchStores.has(matchId)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise = createControlledPromise<void>()\n if (nextPreload !== match.preload) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n preload: nextPreload,\n }))\n }\n\n await handleLoader(preload, prevMatch, previousRouteMatchId, match, route)\n }\n }\n const match = inner.router.getMatch(matchId)!\n if (!loaderIsRunningAsync) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loadPromise = undefined\n }\n\n clearTimeout(match._nonReactive.pendingTimeout)\n match._nonReactive.pendingTimeout = undefined\n if (!loaderIsRunningAsync) match._nonReactive.loaderPromise = undefined\n match._nonReactive.dehydrated = undefined\n\n const nextIsFetching = loaderIsRunningAsync ? match.isFetching : false\n if (nextIsFetching !== match.isFetching || match.invalid !== false) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: nextIsFetching,\n invalid: false,\n }))\n return inner.router.getMatch(matchId)!\n } else {\n return match\n }\n}\n\nexport async function loadMatches(arg: {\n router: AnyRouter\n location: ParsedLocation\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n updateMatch: UpdateMatchFn\n sync?: boolean\n}): Promise<Array<MakeRouteMatch>> {\n const inner: InnerLoadContext = arg\n const matchPromises: Array<Promise<AnyRouteMatch>> = []\n\n // make sure the pending component is immediately rendered when hydrating a match that is not SSRed\n // the pending component was already rendered on the server and we want to keep it shown on the client until minPendingMs is reached\n if (\n !(isServer ?? inner.router.isServer) &&\n hasForcePendingActiveMatch(inner.router)\n ) {\n triggerOnReady(inner)\n }\n\n let beforeLoadNotFound: NotFoundError | undefined\n\n // Execute all beforeLoads one by one\n for (let i = 0; i < inner.matches.length; i++) {\n try {\n const beforeLoad = handleBeforeLoad(inner, i)\n if (isPromise(beforeLoad)) await beforeLoad\n } catch (err) {\n if (isRedirect(err)) {\n throw err\n }\n if (isNotFound(err)) {\n beforeLoadNotFound = err\n } else {\n if (!inner.preload) throw err\n }\n break\n }\n\n if (inner.serialError || inner.firstBadMatchIndex != null) {\n break\n }\n }\n\n // Execute loaders once, with max index adapted for beforeLoad notFound handling.\n const baseMaxIndexExclusive = inner.firstBadMatchIndex ?? inner.matches.length\n\n const boundaryIndex =\n beforeLoadNotFound && !inner.preload\n ? getNotFoundBoundaryIndex(inner, beforeLoadNotFound)\n : undefined\n\n const maxIndexExclusive =\n beforeLoadNotFound && inner.preload\n ? 0\n : boundaryIndex !== undefined\n ? Math.min(boundaryIndex + 1, baseMaxIndexExclusive)\n : baseMaxIndexExclusive\n\n let firstNotFound: NotFoundError | undefined\n let firstUnhandledRejection: unknown\n\n for (let i = 0; i < maxIndexExclusive; i++) {\n matchPromises.push(loadRouteMatch(inner, matchPromises, i))\n }\n\n try {\n await Promise.all(matchPromises)\n } catch {\n const settled = await Promise.allSettled(matchPromises)\n\n for (const result of settled) {\n if (result.status !== 'rejected') continue\n\n const reason = result.reason\n if (isRedirect(reason)) {\n throw reason\n }\n if (isNotFound(reason)) {\n firstNotFound ??= reason\n } else {\n firstUnhandledRejection ??= reason\n }\n }\n\n if (firstUnhandledRejection !== undefined) {\n throw firstUnhandledRejection\n }\n }\n\n const notFoundToThrow =\n firstNotFound ??\n (beforeLoadNotFound && !inner.preload ? beforeLoadNotFound : undefined)\n\n let headMaxIndex =\n inner.firstBadMatchIndex !== undefined\n ? inner.firstBadMatchIndex\n : inner.matches.length - 1\n\n if (!notFoundToThrow && beforeLoadNotFound && inner.preload) {\n return inner.matches\n }\n\n if (notFoundToThrow) {\n // Determine once which matched route will actually render the\n // notFoundComponent, then pass this precomputed index through the remaining\n // finalization steps.\n // This can differ from the throwing route when routeId targets an ancestor\n // boundary (or when bubbling resolves to a parent/root boundary).\n const renderedBoundaryIndex = getNotFoundBoundaryIndex(\n inner,\n notFoundToThrow,\n )\n\n if (renderedBoundaryIndex === undefined) {\n if (process.env.NODE_ENV !== 'production') {\n throw new Error(\n 'Invariant failed: Could not find match for notFound boundary',\n )\n }\n\n invariant()\n }\n const boundaryMatch = inner.matches[renderedBoundaryIndex]!\n\n const boundaryRoute = inner.router.looseRoutesById[boundaryMatch.routeId]!\n const defaultNotFoundComponent = (inner.router.options as any)\n ?.defaultNotFoundComponent\n\n // Ensure a notFoundComponent exists on the boundary route\n if (!boundaryRoute.options.notFoundComponent && defaultNotFoundComponent) {\n boundaryRoute.options.notFoundComponent = defaultNotFoundComponent\n }\n\n notFoundToThrow.routeId = boundaryMatch.routeId\n\n const boundaryIsRoot = boundaryMatch.routeId === inner.router.routeTree.id\n\n inner.updateMatch(boundaryMatch.id, (prev) => ({\n ...prev,\n ...(boundaryIsRoot\n ? // For root boundary, use globalNotFound so the root component's\n // shell still renders and <Outlet> handles the not-found display,\n // instead of replacing the entire root shell via status='notFound'.\n { status: 'success' as const, globalNotFound: true, error: undefined }\n : // For non-root boundaries, set status:'notFound' so MatchInner\n // renders the notFoundComponent directly.\n { status: 'notFound' as const, error: notFoundToThrow }),\n isFetching: false,\n }))\n\n headMaxIndex = renderedBoundaryIndex\n\n // Ensure the rendering boundary route chunk (and its lazy components, including\n // lazy notFoundComponent) is loaded before we continue to head execution/render.\n await loadRouteChunk(boundaryRoute, ['notFoundComponent'])\n } else if (!inner.preload) {\n // Clear stale root global-not-found state on normal navigations that do not\n // throw notFound. This must live here (instead of only in runLoader success)\n // because the root loader may be skipped when data is still fresh.\n const rootMatch = inner.matches[0]!\n // `rootMatch` is the next match for this navigation. If it is not global\n // not-found, then any currently stored root global-not-found is stale.\n if (!rootMatch.globalNotFound) {\n // `currentRootMatch` is the current store state (from the previous\n // navigation/load). Update only when a stale flag is actually present.\n const currentRootMatch = inner.router.getMatch(rootMatch.id)\n if (currentRootMatch?.globalNotFound) {\n inner.updateMatch(rootMatch.id, (prev) => ({\n ...prev,\n globalNotFound: false,\n error: undefined,\n }))\n }\n }\n }\n\n // When a serial error occurred (e.g. beforeLoad threw a regular Error),\n // the erroring route's lazy chunk wasn't loaded because loaders were skipped.\n // We need to load it so the code-split errorComponent is available for rendering.\n if (inner.serialError && inner.firstBadMatchIndex !== undefined) {\n const errorRoute =\n inner.router.looseRoutesById[\n inner.matches[inner.firstBadMatchIndex]!.routeId\n ]!\n await loadRouteChunk(errorRoute, ['errorComponent'])\n }\n\n // serially execute heads once after loaders/notFound handling, ensuring\n // all head functions get a chance even if one throws.\n for (let i = 0; i <= headMaxIndex; i++) {\n const match = inner.matches[i]!\n const { id: matchId, routeId } = match\n const route = inner.router.looseRoutesById[routeId]!\n try {\n const headResult = executeHead(inner, matchId, route)\n if (headResult) {\n const head = await headResult\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n ...head,\n }))\n }\n } catch (err) {\n console.error(`Error executing head for route ${routeId}:`, err)\n }\n }\n\n const readyPromise = triggerOnReady(inner)\n if (isPromise(readyPromise)) {\n await readyPromise\n }\n\n if (notFoundToThrow) {\n throw notFoundToThrow\n }\n\n if (inner.serialError && !inner.preload && !inner.onReady) {\n throw inner.serialError\n }\n\n return inner.matches\n}\n\nexport type RouteComponentType =\n | 'component'\n | 'errorComponent'\n | 'pendingComponent'\n | 'notFoundComponent'\n\nfunction preloadRouteComponents(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType>,\n): Promise<void> | undefined {\n const preloads = componentTypesToLoad\n .map((type) => (route.options[type] as any)?.preload?.())\n .filter(Boolean)\n\n if (preloads.length === 0) return undefined\n\n return Promise.all(preloads) as any as Promise<void>\n}\n\nexport function loadRouteChunk(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType> = componentTypes,\n) {\n if (!route._lazyLoaded && route._lazyPromise === undefined) {\n if (route.lazyFn) {\n route._lazyPromise = route.lazyFn().then((lazyRoute) => {\n // explicitly don't copy over the lazy route's id\n const { id: _id, ...options } = lazyRoute.options\n Object.assign(route.options, options)\n route._lazyLoaded = true\n route._lazyPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._lazyLoaded = true\n }\n }\n\n const runAfterLazy = () =>\n route._componentsLoaded\n ? undefined\n : componentTypesToLoad === componentTypes\n ? (() => {\n if (route._componentsPromise === undefined) {\n const componentsPromise = preloadRouteComponents(\n route,\n componentTypes,\n )\n\n if (componentsPromise) {\n route._componentsPromise = componentsPromise.then(() => {\n route._componentsLoaded = true\n route._componentsPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._componentsLoaded = true\n }\n }\n\n return route._componentsPromise\n })()\n : preloadRouteComponents(route, componentTypesToLoad)\n\n return route._lazyPromise\n ? route._lazyPromise.then(runAfterLazy)\n : runAfterLazy()\n}\n\nfunction makeMaybe<TValue, TError>(\n value: TValue,\n error: TError,\n): { status: 'success'; value: TValue } | { status: 'error'; error: TError } {\n if (error) {\n return { status: 'error' as const, error }\n }\n return { status: 'success' as const, value }\n}\n\nexport function routeNeedsPreload(route: AnyRoute) {\n for (const componentType of componentTypes) {\n if ((route.options[componentType] as any)?.preload) {\n return true\n }\n }\n return false\n}\n\nexport const componentTypes: Array<RouteComponentType> = [\n 'component',\n 'errorComponent',\n 'pendingComponent',\n 'notFoundComponent',\n] as const\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"load-matches.js","names":[],"sources":["../../src/load-matches.ts"],"sourcesContent":["import { isServer } from '@tanstack/router-core/isServer'\nimport { invariant } from './invariant'\nimport { createControlledPromise, isPromise } from './utils'\nimport { isNotFound } from './not-found'\nimport { rootRouteId } from './root'\nimport { isRedirect } from './redirect'\nimport type { NotFoundError } from './not-found'\nimport type { ParsedLocation } from './location'\nimport type {\n AnyRoute,\n BeforeLoadContextOptions,\n LoaderFnContext,\n SsrContextOptions,\n} from './route'\nimport type { AnyRouteMatch, MakeRouteMatch } from './Matches'\nimport type { AnyRouter, SSROption, UpdateMatchFn } from './router'\n\n/**\n * An object of this shape is created when calling `loadMatches`.\n * It contains everything we need for all other functions in this file\n * to work. (It's basically the function's argument, plus a few mutable states)\n */\ntype InnerLoadContext = {\n /** the calling router instance */\n router: AnyRouter\n location: ParsedLocation\n /** mutable state, scoped to a `loadMatches` call */\n firstBadMatchIndex?: number\n /** mutable state, scoped to a `loadMatches` call */\n rendered?: boolean\n serialError?: unknown\n updateMatch: UpdateMatchFn\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n sync?: boolean\n}\n\nconst triggerOnReady = (inner: InnerLoadContext): void | Promise<void> => {\n if (!inner.rendered) {\n inner.rendered = true\n return inner.onReady?.()\n }\n}\n\nconst hasForcePendingActiveMatch = (router: AnyRouter): boolean => {\n return router.stores.matchesId.get().some((matchId) => {\n return router.stores.matchStores.get(matchId)?.get()._forcePending\n })\n}\n\nconst resolvePreload = (inner: InnerLoadContext, matchId: string): boolean => {\n return !!(inner.preload && !inner.router.stores.matchStores.has(matchId))\n}\n\n/**\n * Builds the accumulated context from router options and all matches up to (and optionally including) the given index.\n * Merges __routeContext and __beforeLoadContext from each match.\n */\nconst buildMatchContext = (\n inner: InnerLoadContext,\n index: number,\n includeCurrentMatch: boolean = true,\n): Record<string, unknown> => {\n const context: Record<string, unknown> = {\n ...(inner.router.options.context ?? {}),\n }\n const end = includeCurrentMatch ? index : index - 1\n for (let i = 0; i <= end; i++) {\n const innerMatch = inner.matches[i]\n if (!innerMatch) continue\n const m = inner.router.getMatch(innerMatch.id)\n if (!m) continue\n Object.assign(context, m.__routeContext, m.__beforeLoadContext)\n }\n return context\n}\n\nconst getNotFoundBoundaryIndex = (\n inner: InnerLoadContext,\n err: NotFoundError,\n): number | undefined => {\n if (!inner.matches.length) {\n return undefined\n }\n\n const requestedRouteId = err.routeId\n const matchedRootIndex = inner.matches.findIndex(\n (m) => m.routeId === inner.router.routeTree.id,\n )\n const rootIndex = matchedRootIndex >= 0 ? matchedRootIndex : 0\n\n let startIndex = requestedRouteId\n ? inner.matches.findIndex((match) => match.routeId === requestedRouteId)\n : (inner.firstBadMatchIndex ?? inner.matches.length - 1)\n\n if (startIndex < 0) {\n startIndex = rootIndex\n }\n\n for (let i = startIndex; i >= 0; i--) {\n const match = inner.matches[i]!\n const route = inner.router.looseRoutesById[match.routeId]!\n if (route.options.notFoundComponent) {\n return i\n }\n }\n\n // If no boundary component is found, preserve explicit routeId targeting behavior,\n // otherwise default to root for untargeted notFounds.\n return requestedRouteId ? startIndex : rootIndex\n}\n\nconst handleRedirectAndNotFound = (\n inner: InnerLoadContext,\n match: AnyRouteMatch | undefined,\n err: unknown,\n): void => {\n if (!isRedirect(err) && !isNotFound(err)) return\n\n if (isRedirect(err) && err.redirectHandled && !err.options.reloadDocument) {\n throw err\n }\n\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n match._nonReactive.loaderPromise = undefined\n\n match._nonReactive.error = err\n\n inner.updateMatch(match.id, (prev) => ({\n ...prev,\n status: isRedirect(err)\n ? 'redirected'\n : isNotFound(err)\n ? 'notFound'\n : prev.status === 'pending'\n ? 'success'\n : prev.status,\n context: buildMatchContext(inner, match.index),\n isFetching: false,\n error: err,\n }))\n\n if (isNotFound(err) && !err.routeId) {\n // Stamp the throwing match's routeId so that the finalization step in\n // loadMatches knows where the notFound originated. The actual boundary\n // resolution (walking up to the nearest notFoundComponent) is deferred to\n // the finalization step, where firstBadMatchIndex is stable and\n // headMaxIndex can be capped correctly.\n err.routeId = match.routeId\n }\n\n match._nonReactive.loadPromise?.resolve()\n }\n\n if (isRedirect(err)) {\n inner.rendered = true\n err.options._fromLocation = inner.location\n err.redirectHandled = true\n err = inner.router.resolveRedirect(err)\n }\n\n throw err\n}\n\nconst shouldSkipLoader = (\n inner: InnerLoadContext,\n matchId: string,\n): boolean => {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return true\n }\n // upon hydration, we skip the loader if the match has been dehydrated on the server\n if (!(isServer ?? inner.router.isServer) && match._nonReactive.dehydrated) {\n return true\n }\n\n if ((isServer ?? inner.router.isServer) && match.ssr === false) {\n return true\n }\n\n return false\n}\n\nconst syncMatchContext = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n): void => {\n const nextContext = buildMatchContext(inner, index)\n\n inner.updateMatch(matchId, (prev) => {\n return {\n ...prev,\n context: nextContext,\n }\n })\n}\n\nconst handleSerialError = (\n inner: InnerLoadContext,\n index: number,\n err: any,\n routerCode: string,\n): void => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n // Much like suspense, we use a promise here to know if\n // we've been outdated by a new loadMatches call and\n // should abort the current async operation\n if (err instanceof Promise) {\n throw err\n }\n\n err.routerCode = routerCode\n inner.firstBadMatchIndex ??= index\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n\n try {\n route.options.onError?.(err)\n } catch (errorHandlerErr) {\n err = errorHandlerErr\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n }\n\n inner.updateMatch(matchId, (prev) => {\n prev._nonReactive.beforeLoadPromise?.resolve()\n prev._nonReactive.beforeLoadPromise = undefined\n prev._nonReactive.loadPromise?.resolve()\n\n return {\n ...prev,\n error: err,\n status: 'error',\n isFetching: false,\n updatedAt: Date.now(),\n abortController: new AbortController(),\n }\n })\n\n if (!inner.preload && !isRedirect(err) && !isNotFound(err)) {\n inner.serialError ??= err\n }\n}\n\nconst isBeforeLoadSsr = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n const parentMatchId = inner.matches[index - 1]?.id\n const parentMatch = parentMatchId\n ? inner.router.getMatch(parentMatchId)!\n : undefined\n\n // in SPA mode, only SSR the root route\n if (inner.router.isShell()) {\n existingMatch.ssr = route.id === rootRouteId\n return\n }\n\n if (parentMatch?.ssr === false) {\n existingMatch.ssr = false\n return\n }\n\n const parentOverride = (tempSsr: SSROption) => {\n if (tempSsr === true && parentMatch?.ssr === 'data-only') {\n return 'data-only'\n }\n return tempSsr\n }\n\n const defaultSsr = inner.router.options.defaultSsr ?? true\n\n if (route.options.ssr === undefined) {\n existingMatch.ssr = parentOverride(defaultSsr)\n return\n }\n\n if (typeof route.options.ssr !== 'function') {\n existingMatch.ssr = parentOverride(route.options.ssr)\n return\n }\n const { search, params } = existingMatch\n\n const ssrFnContext: SsrContextOptions<any, any, any> = {\n search: makeMaybe(search, existingMatch.searchError),\n params: makeMaybe(params, existingMatch.paramsError),\n location: inner.location,\n matches: inner.matches.map((match) => ({\n index: match.index,\n pathname: match.pathname,\n fullPath: match.fullPath,\n staticData: match.staticData,\n id: match.id,\n routeId: match.routeId,\n search: makeMaybe(match.search, match.searchError),\n params: makeMaybe(match.params, match.paramsError),\n ssr: match.ssr,\n })),\n }\n\n const tempSsr = route.options.ssr(ssrFnContext)\n if (isPromise(tempSsr)) {\n return tempSsr.then((ssr) => {\n existingMatch.ssr = parentOverride(ssr ?? defaultSsr)\n })\n }\n\n existingMatch.ssr = parentOverride(tempSsr ?? defaultSsr)\n return\n}\n\nconst setupPendingTimeout = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n match: AnyRouteMatch,\n): void => {\n if (match._nonReactive.pendingTimeout !== undefined) return\n\n const pendingMs =\n route.options.pendingMs ?? inner.router.options.defaultPendingMs\n const shouldPending = !!(\n inner.onReady &&\n !(isServer ?? inner.router.isServer) &&\n !resolvePreload(inner, matchId) &&\n (route.options.loader ||\n route.options.beforeLoad ||\n routeNeedsPreload(route)) &&\n typeof pendingMs === 'number' &&\n pendingMs !== Infinity &&\n (route.options.pendingComponent ??\n (inner.router.options as any)?.defaultPendingComponent)\n )\n\n if (shouldPending) {\n const pendingTimeout = setTimeout(() => {\n // Update the match and prematurely resolve the loadMatches promise so that\n // the pending component can start rendering\n triggerOnReady(inner)\n }, pendingMs)\n match._nonReactive.pendingTimeout = pendingTimeout\n }\n}\n\nconst preBeforeLoadSetup = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n\n // If we are in the middle of a load, either of these will be present\n // (not to be confused with `loadPromise`, which is always defined)\n if (\n !existingMatch._nonReactive.beforeLoadPromise &&\n !existingMatch._nonReactive.loaderPromise\n )\n return\n\n setupPendingTimeout(inner, matchId, route, existingMatch)\n\n const then = () => {\n const match = inner.router.getMatch(matchId)!\n if (\n match.preload &&\n (match.status === 'redirected' || match.status === 'notFound')\n ) {\n handleRedirectAndNotFound(inner, match, match.error)\n }\n }\n\n // Wait for the previous beforeLoad to resolve before we continue\n return existingMatch._nonReactive.beforeLoadPromise\n ? existingMatch._nonReactive.beforeLoadPromise.then(then)\n : then()\n}\n\nconst executeBeforeLoad = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const match = inner.router.getMatch(matchId)!\n\n // explicitly capture the previous loadPromise\n let prevLoadPromise = match._nonReactive.loadPromise\n match._nonReactive.loadPromise = createControlledPromise<void>(() => {\n prevLoadPromise?.resolve()\n prevLoadPromise = undefined\n })\n\n const { paramsError, searchError } = match\n\n if (paramsError) {\n handleSerialError(inner, index, paramsError, 'PARSE_PARAMS')\n }\n\n if (searchError) {\n handleSerialError(inner, index, searchError, 'VALIDATE_SEARCH')\n }\n\n setupPendingTimeout(inner, matchId, route, match)\n\n const abortController = new AbortController()\n\n let isPending = false\n const pending = () => {\n if (isPending) return\n isPending = true\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'beforeLoad',\n fetchCount: prev.fetchCount + 1,\n abortController,\n // Note: We intentionally don't update context here.\n // Context should only be updated after beforeLoad resolves to avoid\n // components seeing incomplete context during async beforeLoad execution.\n }))\n }\n\n const resolve = () => {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: false,\n }))\n }\n\n // if there is no `beforeLoad` option, just mark as pending and resolve\n // Context will be updated later in loadRouteMatch after loader completes\n if (!route.options.beforeLoad) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n\n match._nonReactive.beforeLoadPromise = createControlledPromise<void>()\n\n // Build context from all parent matches, excluding current match's __beforeLoadContext\n // (since we're about to execute beforeLoad for this match)\n const context = {\n ...buildMatchContext(inner, index, false),\n ...match.__routeContext,\n }\n const { search, params, cause } = match\n const preload = resolvePreload(inner, matchId)\n const beforeLoadFnContext: BeforeLoadContextOptions<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > = {\n search,\n abortController,\n params,\n preload,\n context,\n location: inner.location,\n navigate: (opts: any) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n buildLocation: inner.router.buildLocation,\n cause: preload ? 'preload' : cause,\n matches: inner.matches,\n routeId: route.id,\n ...inner.router.options.additionalContext,\n }\n\n const updateContext = (beforeLoadContext: any) => {\n if (beforeLoadContext === undefined) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n if (isRedirect(beforeLoadContext) || isNotFound(beforeLoadContext)) {\n pending()\n handleSerialError(inner, index, beforeLoadContext, 'BEFORE_LOAD')\n }\n\n inner.router.batch(() => {\n pending()\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n __beforeLoadContext: beforeLoadContext,\n }))\n resolve()\n })\n }\n\n let beforeLoadContext\n try {\n beforeLoadContext = route.options.beforeLoad(beforeLoadFnContext)\n if (isPromise(beforeLoadContext)) {\n pending()\n return beforeLoadContext\n .catch((err) => {\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n })\n .then(updateContext)\n }\n } catch (err) {\n pending()\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n }\n\n updateContext(beforeLoadContext)\n return\n}\n\nconst handleBeforeLoad = (\n inner: InnerLoadContext,\n index: number,\n): void | Promise<void> => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n const serverSsr = () => {\n // on the server, determine whether SSR the current match or not\n if (isServer ?? inner.router.isServer) {\n const maybePromise = isBeforeLoadSsr(inner, matchId, index, route)\n if (isPromise(maybePromise)) return maybePromise.then(queueExecution)\n }\n return queueExecution()\n }\n\n const execute = () => executeBeforeLoad(inner, matchId, index, route)\n\n const queueExecution = () => {\n if (shouldSkipLoader(inner, matchId)) return\n const result = preBeforeLoadSetup(inner, matchId, route)\n return isPromise(result) ? result.then(execute) : execute()\n }\n\n return serverSsr()\n}\n\nconst executeHead = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<\n Pick<\n AnyRouteMatch,\n 'meta' | 'links' | 'headScripts' | 'headers' | 'scripts' | 'styles'\n >\n> => {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (!match) {\n return\n }\n if (!route.options.head && !route.options.scripts && !route.options.headers) {\n return\n }\n const assetContext = {\n ssr: inner.router.options.ssr,\n matches: inner.matches,\n match,\n params: match.params,\n loaderData: match.loaderData,\n }\n\n return Promise.all([\n route.options.head?.(assetContext),\n route.options.scripts?.(assetContext),\n route.options.headers?.(assetContext),\n ]).then(([headFnContent, scripts, headers]) => {\n const meta = headFnContent?.meta\n const links = headFnContent?.links\n const headScripts = headFnContent?.scripts\n const styles = headFnContent?.styles\n\n return {\n meta,\n links,\n headScripts,\n headers,\n scripts,\n styles,\n }\n })\n}\n\nconst getLoaderContext = (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): LoaderFnContext => {\n const parentMatchPromise = matchPromises[index - 1] as any\n const { params, loaderDeps, abortController, cause } =\n inner.router.getMatch(matchId)!\n\n const context = buildMatchContext(inner, index)\n\n const preload = resolvePreload(inner, matchId)\n\n return {\n params,\n deps: loaderDeps,\n preload: !!preload,\n parentMatchPromise,\n abortController,\n context,\n location: inner.location,\n navigate: (opts) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n cause: preload ? 'preload' : cause,\n route,\n ...inner.router.options.additionalContext,\n }\n}\n\nconst runLoader = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): Promise<void> => {\n try {\n // If the Matches component rendered\n // the pending component and needs to show it for\n // a minimum duration, we''ll wait for it to resolve\n // before committing to the match and resolving\n // the loadPromise\n\n const match = inner.router.getMatch(matchId)!\n\n // Actually run the loader and handle the result\n try {\n if (!(isServer ?? inner.router.isServer) || match.ssr === true) {\n loadRouteChunk(route)\n }\n\n // Kick off the loader!\n const routeLoader = route.options.loader\n const loader =\n typeof routeLoader === 'function' ? routeLoader : routeLoader?.handler\n const loaderResult = loader?.(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n const loaderResultIsPromise = !!loader && isPromise(loaderResult)\n\n const willLoadSomething = !!(\n loaderResultIsPromise ||\n route._lazyPromise ||\n route._componentsPromise ||\n route.options.head ||\n route.options.scripts ||\n route.options.headers ||\n match._nonReactive.minPendingPromise\n )\n\n if (willLoadSomething) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'loader',\n }))\n }\n\n if (loader) {\n const loaderData = loaderResultIsPromise\n ? await loaderResult\n : loaderResult\n\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n loaderData,\n )\n if (loaderData !== undefined) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n loaderData,\n }))\n }\n }\n\n // Lazy option can modify the route options,\n // so we need to wait for it to resolve before\n // we can use the options\n if (route._lazyPromise) await route._lazyPromise\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n // Last but not least, wait for the the components\n // to be preloaded before we resolve the match\n if (route._componentsPromise) await route._componentsPromise\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error: undefined,\n context: buildMatchContext(inner, index),\n status: 'success',\n isFetching: false,\n updatedAt: Date.now(),\n }))\n } catch (e) {\n let error = e\n\n if ((error as any)?.name === 'AbortError') {\n if (match.abortController.signal.aborted) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n return\n }\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n status: prev.status === 'pending' ? 'success' : prev.status,\n isFetching: false,\n context: buildMatchContext(inner, index),\n }))\n return\n }\n\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n if (isNotFound(e)) {\n await (route.options.notFoundComponent as any)?.preload?.()\n }\n\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), e)\n\n try {\n route.options.onError?.(e)\n } catch (onErrorError) {\n error = onErrorError\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n onErrorError,\n )\n }\n if (!isRedirect(error) && !isNotFound(error)) {\n await loadRouteChunk(route, ['errorComponent'])\n }\n\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error,\n context: buildMatchContext(inner, index),\n status: 'error',\n isFetching: false,\n }))\n }\n } catch (err) {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.loaderPromise = undefined\n }\n handleRedirectAndNotFound(inner, match, err)\n }\n}\n\nconst loadRouteMatch = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n index: number,\n): Promise<AnyRouteMatch> => {\n async function handleLoader(\n preload: boolean,\n prevMatch: AnyRouteMatch,\n previousRouteMatchId: string | undefined,\n match: AnyRouteMatch,\n route: AnyRoute,\n ) {\n const age = Date.now() - prevMatch.updatedAt\n\n const staleAge = preload\n ? (route.options.preloadStaleTime ??\n inner.router.options.defaultPreloadStaleTime ??\n 30_000) // 30 seconds for preloads by default\n : (route.options.staleTime ?? inner.router.options.defaultStaleTime ?? 0)\n\n const shouldReloadOption = route.options.shouldReload\n\n // Default to reloading the route all the time\n // Allow shouldReload to get the last say,\n // if provided.\n const shouldReload =\n typeof shouldReloadOption === 'function'\n ? shouldReloadOption(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n : shouldReloadOption\n\n // If the route is successful and still fresh, just resolve\n const { status, invalid } = match\n const staleMatchShouldReload =\n age >= staleAge &&\n (!!inner.forceStaleReload ||\n match.cause === 'enter' ||\n (previousRouteMatchId !== undefined &&\n previousRouteMatchId !== match.id))\n loaderShouldRunAsync =\n status === 'success' &&\n (invalid || (shouldReload ?? staleMatchShouldReload))\n if (preload && route.options.preload === false) {\n // Do nothing\n } else if (\n loaderShouldRunAsync &&\n !inner.sync &&\n shouldReloadInBackground\n ) {\n loaderIsRunningAsync = true\n ;(async () => {\n try {\n await runLoader(inner, matchPromises, matchId, index, route)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n match._nonReactive.loadPromise = undefined\n } catch (err) {\n if (isRedirect(err)) {\n await inner.router.navigate(err.options)\n }\n }\n })()\n } else if (status !== 'success' || loaderShouldRunAsync) {\n await runLoader(inner, matchPromises, matchId, index, route)\n } else {\n syncMatchContext(inner, matchId, index)\n }\n }\n\n const { id: matchId, routeId } = inner.matches[index]!\n let loaderShouldRunAsync = false\n let loaderIsRunningAsync = false\n const route = inner.router.looseRoutesById[routeId]!\n const routeLoader = route.options.loader\n const shouldReloadInBackground =\n ((typeof routeLoader === 'function'\n ? undefined\n : routeLoader?.staleReloadMode) ??\n inner.router.options.defaultStaleReloadMode) !== 'blocking'\n\n if (shouldSkipLoader(inner, matchId)) {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return inner.matches[index]!\n }\n\n syncMatchContext(inner, matchId, index)\n\n if (isServer ?? inner.router.isServer) {\n return inner.router.getMatch(matchId)!\n }\n } else {\n const prevMatch = inner.router.getMatch(matchId)! // This is where all of the stale-while-revalidate magic happens\n const activeIdAtIndex = inner.router.stores.matchesId.get()[index]\n const activeAtIndex =\n (activeIdAtIndex &&\n inner.router.stores.matchStores.get(activeIdAtIndex)) ||\n null\n const previousRouteMatchId =\n activeAtIndex?.routeId === routeId\n ? activeIdAtIndex\n : inner.router.stores.matches.get().find((d) => d.routeId === routeId)\n ?.id\n const preload = resolvePreload(inner, matchId)\n\n // there is a loaderPromise, so we are in the middle of a load\n if (prevMatch._nonReactive.loaderPromise) {\n // do not block if we already have stale data we can show\n // but only if the ongoing load is not a preload since error handling is different for preloads\n // and we don't want to swallow errors\n if (\n prevMatch.status === 'success' &&\n !inner.sync &&\n !prevMatch.preload &&\n shouldReloadInBackground\n ) {\n return prevMatch\n }\n await prevMatch._nonReactive.loaderPromise\n const match = inner.router.getMatch(matchId)!\n const error = match._nonReactive.error || match.error\n if (error) {\n handleRedirectAndNotFound(inner, match, error)\n }\n\n if (match.status === 'pending') {\n await handleLoader(\n preload,\n prevMatch,\n previousRouteMatchId,\n match,\n route,\n )\n }\n } else {\n const nextPreload =\n preload && !inner.router.stores.matchStores.has(matchId)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise = createControlledPromise<void>()\n if (nextPreload !== match.preload) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n preload: nextPreload,\n }))\n }\n\n await handleLoader(preload, prevMatch, previousRouteMatchId, match, route)\n }\n }\n const match = inner.router.getMatch(matchId)!\n if (!loaderIsRunningAsync) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loadPromise = undefined\n }\n\n clearTimeout(match._nonReactive.pendingTimeout)\n match._nonReactive.pendingTimeout = undefined\n if (!loaderIsRunningAsync) match._nonReactive.loaderPromise = undefined\n match._nonReactive.dehydrated = undefined\n\n const nextIsFetching = loaderIsRunningAsync ? match.isFetching : false\n if (nextIsFetching !== match.isFetching || match.invalid !== false) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: nextIsFetching,\n invalid: false,\n }))\n return inner.router.getMatch(matchId)!\n } else {\n return match\n }\n}\n\nexport async function loadMatches(arg: {\n router: AnyRouter\n location: ParsedLocation\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n updateMatch: UpdateMatchFn\n sync?: boolean\n}): Promise<Array<MakeRouteMatch>> {\n const inner: InnerLoadContext = arg\n const matchPromises: Array<Promise<AnyRouteMatch>> = []\n\n // make sure the pending component is immediately rendered when hydrating a match that is not SSRed\n // the pending component was already rendered on the server and we want to keep it shown on the client until minPendingMs is reached\n if (\n !(isServer ?? inner.router.isServer) &&\n hasForcePendingActiveMatch(inner.router)\n ) {\n triggerOnReady(inner)\n }\n\n let beforeLoadNotFound: NotFoundError | undefined\n\n // Execute all beforeLoads one by one\n for (let i = 0; i < inner.matches.length; i++) {\n try {\n const beforeLoad = handleBeforeLoad(inner, i)\n if (isPromise(beforeLoad)) await beforeLoad\n } catch (err) {\n if (isRedirect(err)) {\n throw err\n }\n if (isNotFound(err)) {\n beforeLoadNotFound = err\n } else {\n if (!inner.preload) throw err\n }\n break\n }\n\n if (inner.serialError || inner.firstBadMatchIndex != null) {\n break\n }\n }\n\n // Execute loaders once, with max index adapted for beforeLoad notFound handling.\n const baseMaxIndexExclusive = inner.firstBadMatchIndex ?? inner.matches.length\n\n const boundaryIndex =\n beforeLoadNotFound && !inner.preload\n ? getNotFoundBoundaryIndex(inner, beforeLoadNotFound)\n : undefined\n\n const maxIndexExclusive =\n beforeLoadNotFound && inner.preload\n ? 0\n : boundaryIndex !== undefined\n ? Math.min(boundaryIndex + 1, baseMaxIndexExclusive)\n : baseMaxIndexExclusive\n\n let firstNotFound: NotFoundError | undefined\n let firstUnhandledRejection: unknown\n\n for (let i = 0; i < maxIndexExclusive; i++) {\n matchPromises.push(loadRouteMatch(inner, matchPromises, i))\n }\n\n try {\n await Promise.all(matchPromises)\n } catch {\n const settled = await Promise.allSettled(matchPromises)\n\n for (const result of settled) {\n if (result.status !== 'rejected') continue\n\n const reason = result.reason\n if (isRedirect(reason)) {\n throw reason\n }\n if (isNotFound(reason)) {\n firstNotFound ??= reason\n } else {\n firstUnhandledRejection ??= reason\n }\n }\n\n if (firstUnhandledRejection !== undefined) {\n throw firstUnhandledRejection\n }\n }\n\n const notFoundToThrow =\n firstNotFound ??\n (beforeLoadNotFound && !inner.preload ? beforeLoadNotFound : undefined)\n\n let headMaxIndex =\n inner.firstBadMatchIndex !== undefined\n ? inner.firstBadMatchIndex\n : inner.matches.length - 1\n\n if (!notFoundToThrow && beforeLoadNotFound && inner.preload) {\n return inner.matches\n }\n\n if (notFoundToThrow) {\n // Determine once which matched route will actually render the\n // notFoundComponent, then pass this precomputed index through the remaining\n // finalization steps.\n // This can differ from the throwing route when routeId targets an ancestor\n // boundary (or when bubbling resolves to a parent/root boundary).\n const renderedBoundaryIndex = getNotFoundBoundaryIndex(\n inner,\n notFoundToThrow,\n )\n\n if (renderedBoundaryIndex === undefined) {\n if (process.env.NODE_ENV !== 'production') {\n throw new Error(\n 'Invariant failed: Could not find match for notFound boundary',\n )\n }\n\n invariant()\n }\n const boundaryMatch = inner.matches[renderedBoundaryIndex]!\n\n const boundaryRoute = inner.router.looseRoutesById[boundaryMatch.routeId]!\n const defaultNotFoundComponent = (inner.router.options as any)\n ?.defaultNotFoundComponent\n\n // Ensure a notFoundComponent exists on the boundary route\n if (!boundaryRoute.options.notFoundComponent && defaultNotFoundComponent) {\n boundaryRoute.options.notFoundComponent = defaultNotFoundComponent\n }\n\n notFoundToThrow.routeId = boundaryMatch.routeId\n\n const boundaryIsRoot = boundaryMatch.routeId === inner.router.routeTree.id\n\n inner.updateMatch(boundaryMatch.id, (prev) => ({\n ...prev,\n ...(boundaryIsRoot\n ? // For root boundary, use globalNotFound so the root component's\n // shell still renders and <Outlet> handles the not-found display,\n // instead of replacing the entire root shell via status='notFound'.\n { status: 'success' as const, globalNotFound: true, error: undefined }\n : // For non-root boundaries, set status:'notFound' so MatchInner\n // renders the notFoundComponent directly.\n { status: 'notFound' as const, error: notFoundToThrow }),\n isFetching: false,\n }))\n\n headMaxIndex = renderedBoundaryIndex\n\n // Ensure the rendering boundary route chunk (and its lazy components, including\n // lazy notFoundComponent) is loaded before we continue to head execution/render.\n await loadRouteChunk(boundaryRoute, ['notFoundComponent'])\n } else if (!inner.preload) {\n // Clear stale root global-not-found state on normal navigations that do not\n // throw notFound. This must live here (instead of only in runLoader success)\n // because the root loader may be skipped when data is still fresh.\n const rootMatch = inner.matches[0]!\n // `rootMatch` is the next match for this navigation. If it is not global\n // not-found, then any currently stored root global-not-found is stale.\n if (!rootMatch.globalNotFound) {\n // `currentRootMatch` is the current store state (from the previous\n // navigation/load). Update only when a stale flag is actually present.\n const currentRootMatch = inner.router.getMatch(rootMatch.id)\n if (currentRootMatch?.globalNotFound) {\n inner.updateMatch(rootMatch.id, (prev) => ({\n ...prev,\n globalNotFound: false,\n error: undefined,\n }))\n }\n }\n }\n\n // When a serial error occurred (e.g. beforeLoad threw a regular Error),\n // the erroring route's lazy chunk wasn't loaded because loaders were skipped.\n // We need to load it so the code-split errorComponent is available for rendering.\n if (inner.serialError && inner.firstBadMatchIndex !== undefined) {\n const errorRoute =\n inner.router.looseRoutesById[\n inner.matches[inner.firstBadMatchIndex]!.routeId\n ]!\n await loadRouteChunk(errorRoute, ['errorComponent'])\n }\n\n // serially execute heads once after loaders/notFound handling, ensuring\n // all head functions get a chance even if one throws.\n for (let i = 0; i <= headMaxIndex; i++) {\n const match = inner.matches[i]!\n const { id: matchId, routeId } = match\n const route = inner.router.looseRoutesById[routeId]!\n try {\n const headResult = executeHead(inner, matchId, route)\n if (headResult) {\n const head = await headResult\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n ...head,\n }))\n }\n } catch (err) {\n console.error(`Error executing head for route ${routeId}:`, err)\n }\n }\n\n const readyPromise = triggerOnReady(inner)\n if (isPromise(readyPromise)) {\n await readyPromise\n }\n\n if (notFoundToThrow) {\n throw notFoundToThrow\n }\n\n if (inner.serialError && !inner.preload && !inner.onReady) {\n throw inner.serialError\n }\n\n return inner.matches\n}\n\nexport type RouteComponentType =\n | 'component'\n | 'errorComponent'\n | 'pendingComponent'\n | 'notFoundComponent'\n\nfunction preloadRouteComponents(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType>,\n): Promise<void> | undefined {\n const preloads = componentTypesToLoad\n .map((type) => (route.options[type] as any)?.preload?.())\n .filter(Boolean)\n\n if (preloads.length === 0) return undefined\n\n return Promise.all(preloads) as any as Promise<void>\n}\n\nexport function loadRouteChunk(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType> = componentTypes,\n) {\n if (!route._lazyLoaded && route._lazyPromise === undefined) {\n if (route.lazyFn) {\n route._lazyPromise = route.lazyFn().then((lazyRoute) => {\n // explicitly don't copy over the lazy route's id\n const { id: _id, ...options } = lazyRoute.options\n Object.assign(route.options, options)\n route._lazyLoaded = true\n route._lazyPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._lazyLoaded = true\n }\n }\n\n const runAfterLazy = () =>\n route._componentsLoaded\n ? undefined\n : componentTypesToLoad === componentTypes\n ? (() => {\n if (route._componentsPromise === undefined) {\n const componentsPromise = preloadRouteComponents(\n route,\n componentTypes,\n )\n\n if (componentsPromise) {\n route._componentsPromise = componentsPromise.then(() => {\n route._componentsLoaded = true\n route._componentsPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._componentsLoaded = true\n }\n }\n\n return route._componentsPromise\n })()\n : preloadRouteComponents(route, componentTypesToLoad)\n\n return route._lazyPromise\n ? route._lazyPromise.then(runAfterLazy)\n : runAfterLazy()\n}\n\nfunction makeMaybe<TValue, TError>(\n value: TValue,\n error: TError,\n): { status: 'success'; value: TValue } | { status: 'error'; error: TError } {\n if (error) {\n return { status: 'error' as const, error }\n }\n return { status: 'success' as const, value }\n}\n\nexport function routeNeedsPreload(route: AnyRoute) {\n for (const componentType of componentTypes) {\n if ((route.options[componentType] as any)?.preload) {\n return true\n }\n }\n return false\n}\n\nexport const componentTypes: Array<RouteComponentType> = [\n 'component',\n 'errorComponent',\n 'pendingComponent',\n 'notFoundComponent',\n] as const\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"load-matches.js","names":[],"sources":["../../src/load-matches.ts"],"sourcesContent":["import { isServer } from '@tanstack/router-core/isServer'\nimport { invariant } from './invariant'\nimport { createControlledPromise, isPromise } from './utils'\nimport { isNotFound } from './not-found'\nimport { rootRouteId } from './root'\nimport { isRedirect } from './redirect'\nimport type { NotFoundError } from './not-found'\nimport type { ParsedLocation } from './location'\nimport type {\n AnyRoute,\n BeforeLoadContextOptions,\n LoaderFnContext,\n SsrContextOptions,\n} from './route'\nimport type { AnyRouteMatch, MakeRouteMatch } from './Matches'\nimport type { AnyRouter, SSROption, UpdateMatchFn } from './router'\n\n/**\n * An object of this shape is created when calling `loadMatches`.\n * It contains everything we need for all other functions in this file\n * to work. (It's basically the function's argument, plus a few mutable states)\n */\ntype InnerLoadContext = {\n /** the calling router instance */\n router: AnyRouter\n location: ParsedLocation\n /** mutable state, scoped to a `loadMatches` call */\n firstBadMatchIndex?: number\n /** mutable state, scoped to a `loadMatches` call */\n rendered?: boolean\n serialError?: unknown\n updateMatch: UpdateMatchFn\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n sync?: boolean\n}\n\nconst triggerOnReady = (inner: InnerLoadContext): void | Promise<void> => {\n if (!inner.rendered) {\n inner.rendered = true\n return inner.onReady?.()\n }\n}\n\nconst hasForcePendingActiveMatch = (router: AnyRouter): boolean => {\n return router.stores.matchesId.get().some((matchId) => {\n return router.stores.matchStores.get(matchId)?.get()._forcePending\n })\n}\n\nconst resolvePreload = (inner: InnerLoadContext, matchId: string): boolean => {\n return !!(inner.preload && !inner.router.stores.matchStores.has(matchId))\n}\n\n/**\n * Builds the accumulated context from router options and all matches up to (and optionally including) the given index.\n * Merges __routeContext and __beforeLoadContext from each match.\n */\nconst buildMatchContext = (\n inner: InnerLoadContext,\n index: number,\n includeCurrentMatch: boolean = true,\n): Record<string, unknown> => {\n const context: Record<string, unknown> = {\n ...(inner.router.options.context ?? {}),\n }\n const end = includeCurrentMatch ? index : index - 1\n for (let i = 0; i <= end; i++) {\n const innerMatch = inner.matches[i]\n if (!innerMatch) continue\n const m = inner.router.getMatch(innerMatch.id)\n if (!m) continue\n Object.assign(context, m.__routeContext, m.__beforeLoadContext)\n }\n return context\n}\n\nconst getNotFoundBoundaryIndex = (\n inner: InnerLoadContext,\n err: NotFoundError,\n): number | undefined => {\n if (!inner.matches.length) {\n return undefined\n }\n\n const requestedRouteId = err.routeId\n const matchedRootIndex = inner.matches.findIndex(\n (m) => m.routeId === inner.router.routeTree.id,\n )\n const rootIndex = matchedRootIndex >= 0 ? matchedRootIndex : 0\n\n let startIndex = requestedRouteId\n ? inner.matches.findIndex((match) => match.routeId === requestedRouteId)\n : (inner.firstBadMatchIndex ?? inner.matches.length - 1)\n\n if (startIndex < 0) {\n startIndex = rootIndex\n }\n\n for (let i = startIndex; i >= 0; i--) {\n const match = inner.matches[i]!\n const route = inner.router.looseRoutesById[match.routeId]!\n if (route.options.notFoundComponent) {\n return i\n }\n }\n\n // If no boundary component is found, preserve explicit routeId targeting behavior,\n // otherwise default to root for untargeted notFounds.\n return requestedRouteId ? startIndex : rootIndex\n}\n\nconst handleRedirectAndNotFound = (\n inner: InnerLoadContext,\n match: AnyRouteMatch | undefined,\n err: unknown,\n): void => {\n if (!isRedirect(err) && !isNotFound(err)) return\n\n if (isRedirect(err) && err.redirectHandled && !err.options.reloadDocument) {\n throw err\n }\n\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n match._nonReactive.loaderPromise = undefined\n\n match._nonReactive.error = err\n\n inner.updateMatch(match.id, (prev) => ({\n ...prev,\n status: isRedirect(err)\n ? 'redirected'\n : isNotFound(err)\n ? 'notFound'\n : prev.status === 'pending'\n ? 'success'\n : prev.status,\n context: buildMatchContext(inner, match.index),\n isFetching: false,\n error: err,\n }))\n\n if (isNotFound(err) && !err.routeId) {\n // Stamp the throwing match's routeId so that the finalization step in\n // loadMatches knows where the notFound originated. The actual boundary\n // resolution (walking up to the nearest notFoundComponent) is deferred to\n // the finalization step, where firstBadMatchIndex is stable and\n // headMaxIndex can be capped correctly.\n err.routeId = match.routeId\n }\n\n match._nonReactive.loadPromise?.resolve()\n }\n\n if (isRedirect(err)) {\n inner.rendered = true\n err.options._fromLocation = inner.location\n err.redirectHandled = true\n err = inner.router.resolveRedirect(err)\n }\n\n throw err\n}\n\nconst shouldSkipLoader = (\n inner: InnerLoadContext,\n matchId: string,\n): boolean => {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return true\n }\n // upon hydration, we skip the loader if the match has been dehydrated on the server\n if (!(isServer ?? inner.router.isServer) && match._nonReactive.dehydrated) {\n return true\n }\n\n if ((isServer ?? inner.router.isServer) && match.ssr === false) {\n return true\n }\n\n return false\n}\n\nconst syncMatchContext = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n): void => {\n const nextContext = buildMatchContext(inner, index)\n\n inner.updateMatch(matchId, (prev) => {\n return {\n ...prev,\n context: nextContext,\n }\n })\n}\n\nconst handleSerialError = (\n inner: InnerLoadContext,\n index: number,\n err: any,\n routerCode: string,\n): void => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n // Much like suspense, we use a promise here to know if\n // we've been outdated by a new loadMatches call and\n // should abort the current async operation\n if (err instanceof Promise) {\n throw err\n }\n\n err.routerCode = routerCode\n inner.firstBadMatchIndex ??= index\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n\n try {\n route.options.onError?.(err)\n } catch (errorHandlerErr) {\n err = errorHandlerErr\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), err)\n }\n\n inner.updateMatch(matchId, (prev) => {\n prev._nonReactive.beforeLoadPromise?.resolve()\n prev._nonReactive.beforeLoadPromise = undefined\n prev._nonReactive.loadPromise?.resolve()\n\n return {\n ...prev,\n error: err,\n status: 'error',\n isFetching: false,\n updatedAt: Date.now(),\n abortController: new AbortController(),\n }\n })\n\n if (!inner.preload && !isRedirect(err) && !isNotFound(err)) {\n inner.serialError ??= err\n }\n}\n\nconst isBeforeLoadSsr = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n const parentMatchId = inner.matches[index - 1]?.id\n const parentMatch = parentMatchId\n ? inner.router.getMatch(parentMatchId)!\n : undefined\n\n // in SPA mode, only SSR the root route\n if (inner.router.isShell()) {\n existingMatch.ssr = route.id === rootRouteId\n return\n }\n\n if (parentMatch?.ssr === false) {\n existingMatch.ssr = false\n return\n }\n\n const parentOverride = (tempSsr: SSROption) => {\n if (tempSsr === true && parentMatch?.ssr === 'data-only') {\n return 'data-only'\n }\n return tempSsr\n }\n\n const defaultSsr = inner.router.options.defaultSsr ?? true\n\n if (route.options.ssr === undefined) {\n existingMatch.ssr = parentOverride(defaultSsr)\n return\n }\n\n if (typeof route.options.ssr !== 'function') {\n existingMatch.ssr = parentOverride(route.options.ssr)\n return\n }\n const { search, params } = existingMatch\n\n const ssrFnContext: SsrContextOptions<any, any, any> = {\n search: makeMaybe(search, existingMatch.searchError),\n params: makeMaybe(params, existingMatch.paramsError),\n location: inner.location,\n matches: inner.matches.map((match) => ({\n index: match.index,\n pathname: match.pathname,\n fullPath: match.fullPath,\n staticData: match.staticData,\n id: match.id,\n routeId: match.routeId,\n search: makeMaybe(match.search, match.searchError),\n params: makeMaybe(match.params, match.paramsError),\n ssr: match.ssr,\n })),\n }\n\n const tempSsr = route.options.ssr(ssrFnContext)\n if (isPromise(tempSsr)) {\n return tempSsr.then((ssr) => {\n existingMatch.ssr = parentOverride(ssr ?? defaultSsr)\n })\n }\n\n existingMatch.ssr = parentOverride(tempSsr ?? defaultSsr)\n return\n}\n\nconst setupPendingTimeout = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n match: AnyRouteMatch,\n): void => {\n if (match._nonReactive.pendingTimeout !== undefined) return\n\n const pendingMs =\n route.options.pendingMs ?? inner.router.options.defaultPendingMs\n const shouldPending = !!(\n inner.onReady &&\n !(isServer ?? inner.router.isServer) &&\n !resolvePreload(inner, matchId) &&\n (route.options.loader ||\n route.options.beforeLoad ||\n routeNeedsPreload(route)) &&\n typeof pendingMs === 'number' &&\n pendingMs !== Infinity &&\n (route.options.pendingComponent ??\n (inner.router.options as any)?.defaultPendingComponent)\n )\n\n if (shouldPending) {\n const pendingTimeout = setTimeout(() => {\n // Update the match and prematurely resolve the loadMatches promise so that\n // the pending component can start rendering\n triggerOnReady(inner)\n }, pendingMs)\n match._nonReactive.pendingTimeout = pendingTimeout\n }\n}\n\nconst preBeforeLoadSetup = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<void> => {\n const existingMatch = inner.router.getMatch(matchId)!\n\n // If we are in the middle of a load, either of these will be present\n // (not to be confused with `loadPromise`, which is always defined)\n if (\n !existingMatch._nonReactive.beforeLoadPromise &&\n !existingMatch._nonReactive.loaderPromise\n )\n return\n\n setupPendingTimeout(inner, matchId, route, existingMatch)\n\n const then = () => {\n const match = inner.router.getMatch(matchId)!\n if (\n match.preload &&\n (match.status === 'redirected' || match.status === 'notFound')\n ) {\n handleRedirectAndNotFound(inner, match, match.error)\n }\n }\n\n // Wait for the previous beforeLoad to resolve before we continue\n return existingMatch._nonReactive.beforeLoadPromise\n ? existingMatch._nonReactive.beforeLoadPromise.then(then)\n : then()\n}\n\nconst executeBeforeLoad = (\n inner: InnerLoadContext,\n matchId: string,\n index: number,\n route: AnyRoute,\n): void | Promise<void> => {\n const match = inner.router.getMatch(matchId)!\n\n // explicitly capture the previous loadPromise\n let prevLoadPromise = match._nonReactive.loadPromise\n match._nonReactive.loadPromise = createControlledPromise<void>(() => {\n prevLoadPromise?.resolve()\n prevLoadPromise = undefined\n })\n\n const { paramsError, searchError } = match\n\n if (paramsError) {\n handleSerialError(inner, index, paramsError, 'PARSE_PARAMS')\n }\n\n if (searchError) {\n handleSerialError(inner, index, searchError, 'VALIDATE_SEARCH')\n }\n\n setupPendingTimeout(inner, matchId, route, match)\n\n const abortController = new AbortController()\n\n let isPending = false\n const pending = () => {\n if (isPending) return\n isPending = true\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'beforeLoad',\n fetchCount: prev.fetchCount + 1,\n abortController,\n // Note: We intentionally don't update context here.\n // Context should only be updated after beforeLoad resolves to avoid\n // components seeing incomplete context during async beforeLoad execution.\n }))\n }\n\n const resolve = () => {\n match._nonReactive.beforeLoadPromise?.resolve()\n match._nonReactive.beforeLoadPromise = undefined\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: false,\n }))\n }\n\n // if there is no `beforeLoad` option, just mark as pending and resolve\n // Context will be updated later in loadRouteMatch after loader completes\n if (!route.options.beforeLoad) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n\n match._nonReactive.beforeLoadPromise = createControlledPromise<void>()\n\n // Build context from all parent matches, excluding current match's __beforeLoadContext\n // (since we're about to execute beforeLoad for this match)\n const context = {\n ...buildMatchContext(inner, index, false),\n ...match.__routeContext,\n }\n const { search, params, cause } = match\n const preload = resolvePreload(inner, matchId)\n const beforeLoadFnContext: BeforeLoadContextOptions<\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any,\n any\n > = {\n search,\n abortController,\n params,\n preload,\n context,\n location: inner.location,\n navigate: (opts: any) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n buildLocation: inner.router.buildLocation,\n cause: preload ? 'preload' : cause,\n matches: inner.matches,\n routeId: route.id,\n ...inner.router.options.additionalContext,\n }\n\n const updateContext = (beforeLoadContext: any) => {\n if (beforeLoadContext === undefined) {\n inner.router.batch(() => {\n pending()\n resolve()\n })\n return\n }\n if (isRedirect(beforeLoadContext) || isNotFound(beforeLoadContext)) {\n pending()\n handleSerialError(inner, index, beforeLoadContext, 'BEFORE_LOAD')\n }\n\n inner.router.batch(() => {\n pending()\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n __beforeLoadContext: beforeLoadContext,\n }))\n resolve()\n })\n }\n\n let beforeLoadContext\n try {\n beforeLoadContext = route.options.beforeLoad(beforeLoadFnContext)\n if (isPromise(beforeLoadContext)) {\n pending()\n return beforeLoadContext\n .catch((err) => {\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n })\n .then(updateContext)\n }\n } catch (err) {\n pending()\n handleSerialError(inner, index, err, 'BEFORE_LOAD')\n }\n\n updateContext(beforeLoadContext)\n return\n}\n\nconst handleBeforeLoad = (\n inner: InnerLoadContext,\n index: number,\n): void | Promise<void> => {\n const { id: matchId, routeId } = inner.matches[index]!\n const route = inner.router.looseRoutesById[routeId]!\n\n const serverSsr = () => {\n // on the server, determine whether SSR the current match or not\n if (isServer ?? inner.router.isServer) {\n const maybePromise = isBeforeLoadSsr(inner, matchId, index, route)\n if (isPromise(maybePromise)) return maybePromise.then(queueExecution)\n }\n return queueExecution()\n }\n\n const execute = () => executeBeforeLoad(inner, matchId, index, route)\n\n const queueExecution = () => {\n if (shouldSkipLoader(inner, matchId)) return\n const result = preBeforeLoadSetup(inner, matchId, route)\n return isPromise(result) ? result.then(execute) : execute()\n }\n\n return serverSsr()\n}\n\nconst executeHead = (\n inner: InnerLoadContext,\n matchId: string,\n route: AnyRoute,\n): void | Promise<\n Pick<\n AnyRouteMatch,\n 'meta' | 'links' | 'headScripts' | 'headers' | 'scripts' | 'styles'\n >\n> => {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (!match) {\n return\n }\n if (!route.options.head && !route.options.scripts && !route.options.headers) {\n return\n }\n const assetContext = {\n ssr: inner.router.options.ssr,\n matches: inner.matches,\n match,\n params: match.params,\n loaderData: match.loaderData,\n }\n\n return Promise.all([\n route.options.head?.(assetContext),\n route.options.scripts?.(assetContext),\n route.options.headers?.(assetContext),\n ]).then(([headFnContent, scripts, headers]) => {\n const meta = headFnContent?.meta\n const links = headFnContent?.links\n const headScripts = headFnContent?.scripts\n const styles = headFnContent?.styles\n\n return {\n meta,\n links,\n headScripts,\n headers,\n scripts,\n styles,\n }\n })\n}\n\nconst getLoaderContext = (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): LoaderFnContext => {\n const parentMatchPromise = matchPromises[index - 1] as any\n const { params, loaderDeps, abortController, cause } =\n inner.router.getMatch(matchId)!\n\n const context = buildMatchContext(inner, index)\n\n const preload = resolvePreload(inner, matchId)\n\n return {\n params,\n deps: loaderDeps,\n preload: !!preload,\n parentMatchPromise,\n abortController,\n context,\n location: inner.location,\n navigate: (opts) =>\n inner.router.navigate({\n ...opts,\n _fromLocation: inner.location,\n }),\n cause: preload ? 'preload' : cause,\n route,\n ...inner.router.options.additionalContext,\n }\n}\n\nconst runLoader = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n matchId: string,\n index: number,\n route: AnyRoute,\n): Promise<void> => {\n try {\n // If the Matches component rendered\n // the pending component and needs to show it for\n // a minimum duration, we''ll wait for it to resolve\n // before committing to the match and resolving\n // the loadPromise\n\n const match = inner.router.getMatch(matchId)!\n\n // Actually run the loader and handle the result\n try {\n if (!(isServer ?? inner.router.isServer) || match.ssr === true) {\n loadRouteChunk(route)\n }\n\n // Kick off the loader!\n const routeLoader = route.options.loader\n const loader =\n typeof routeLoader === 'function' ? routeLoader : routeLoader?.handler\n const loaderResult = loader?.(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n const loaderResultIsPromise = !!loader && isPromise(loaderResult)\n\n const willLoadSomething = !!(\n loaderResultIsPromise ||\n route._lazyPromise ||\n route._componentsPromise ||\n route.options.head ||\n route.options.scripts ||\n route.options.headers ||\n match._nonReactive.minPendingPromise\n )\n\n if (willLoadSomething) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: 'loader',\n }))\n }\n\n if (loader) {\n const loaderData = loaderResultIsPromise\n ? await loaderResult\n : loaderResult\n\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n loaderData,\n )\n if (loaderData !== undefined) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n loaderData,\n }))\n }\n }\n\n // Lazy option can modify the route options,\n // so we need to wait for it to resolve before\n // we can use the options\n if (route._lazyPromise) await route._lazyPromise\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n // Last but not least, wait for the components\n // to be preloaded before we resolve the match\n if (route._componentsPromise) await route._componentsPromise\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error: undefined,\n context: buildMatchContext(inner, index),\n status: 'success',\n isFetching: false,\n updatedAt: Date.now(),\n }))\n } catch (e) {\n let error = e\n\n if ((error as any)?.name === 'AbortError') {\n if (match.abortController.signal.aborted) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n return\n }\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n status: prev.status === 'pending' ? 'success' : prev.status,\n isFetching: false,\n context: buildMatchContext(inner, index),\n }))\n return\n }\n\n const pendingPromise = match._nonReactive.minPendingPromise\n if (pendingPromise) await pendingPromise\n\n if (isNotFound(e)) {\n await (route.options.notFoundComponent as any)?.preload?.()\n }\n\n handleRedirectAndNotFound(inner, inner.router.getMatch(matchId), e)\n\n try {\n route.options.onError?.(e)\n } catch (onErrorError) {\n error = onErrorError\n handleRedirectAndNotFound(\n inner,\n inner.router.getMatch(matchId),\n onErrorError,\n )\n }\n if (!isRedirect(error) && !isNotFound(error)) {\n await loadRouteChunk(route, ['errorComponent'])\n }\n\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n error,\n context: buildMatchContext(inner, index),\n status: 'error',\n isFetching: false,\n }))\n }\n } catch (err) {\n const match = inner.router.getMatch(matchId)\n // in case of a redirecting match during preload, the match does not exist\n if (match) {\n match._nonReactive.loaderPromise = undefined\n }\n handleRedirectAndNotFound(inner, match, err)\n }\n}\n\nconst loadRouteMatch = async (\n inner: InnerLoadContext,\n matchPromises: Array<Promise<AnyRouteMatch>>,\n index: number,\n): Promise<AnyRouteMatch> => {\n async function handleLoader(\n preload: boolean,\n prevMatch: AnyRouteMatch,\n previousRouteMatchId: string | undefined,\n match: AnyRouteMatch,\n route: AnyRoute,\n ) {\n const age = Date.now() - prevMatch.updatedAt\n\n const staleAge = preload\n ? (route.options.preloadStaleTime ??\n inner.router.options.defaultPreloadStaleTime ??\n 30_000) // 30 seconds for preloads by default\n : (route.options.staleTime ?? inner.router.options.defaultStaleTime ?? 0)\n\n const shouldReloadOption = route.options.shouldReload\n\n // Default to reloading the route all the time\n // Allow shouldReload to get the last say,\n // if provided.\n const shouldReload =\n typeof shouldReloadOption === 'function'\n ? shouldReloadOption(\n getLoaderContext(inner, matchPromises, matchId, index, route),\n )\n : shouldReloadOption\n\n // If the route is successful and still fresh, just resolve\n const { status, invalid } = match\n const staleMatchShouldReload =\n age >= staleAge &&\n (!!inner.forceStaleReload ||\n match.cause === 'enter' ||\n (previousRouteMatchId !== undefined &&\n previousRouteMatchId !== match.id))\n loaderShouldRunAsync =\n status === 'success' &&\n (invalid || (shouldReload ?? staleMatchShouldReload))\n if (preload && route.options.preload === false) {\n // Do nothing\n } else if (\n loaderShouldRunAsync &&\n !inner.sync &&\n shouldReloadInBackground\n ) {\n loaderIsRunningAsync = true\n ;(async () => {\n try {\n await runLoader(inner, matchPromises, matchId, index, route)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loaderPromise = undefined\n match._nonReactive.loadPromise = undefined\n } catch (err) {\n if (isRedirect(err)) {\n await inner.router.navigate(err.options)\n }\n }\n })()\n } else if (status !== 'success' || loaderShouldRunAsync) {\n await runLoader(inner, matchPromises, matchId, index, route)\n } else {\n syncMatchContext(inner, matchId, index)\n }\n }\n\n const { id: matchId, routeId } = inner.matches[index]!\n let loaderShouldRunAsync = false\n let loaderIsRunningAsync = false\n const route = inner.router.looseRoutesById[routeId]!\n const routeLoader = route.options.loader\n const shouldReloadInBackground =\n ((typeof routeLoader === 'function'\n ? undefined\n : routeLoader?.staleReloadMode) ??\n inner.router.options.defaultStaleReloadMode) !== 'blocking'\n\n if (shouldSkipLoader(inner, matchId)) {\n const match = inner.router.getMatch(matchId)\n if (!match) {\n return inner.matches[index]!\n }\n\n syncMatchContext(inner, matchId, index)\n\n if (isServer ?? inner.router.isServer) {\n return inner.router.getMatch(matchId)!\n }\n } else {\n const prevMatch = inner.router.getMatch(matchId)! // This is where all of the stale-while-revalidate magic happens\n const activeIdAtIndex = inner.router.stores.matchesId.get()[index]\n const activeAtIndex =\n (activeIdAtIndex &&\n inner.router.stores.matchStores.get(activeIdAtIndex)) ||\n null\n const previousRouteMatchId =\n activeAtIndex?.routeId === routeId\n ? activeIdAtIndex\n : inner.router.stores.matches.get().find((d) => d.routeId === routeId)\n ?.id\n const preload = resolvePreload(inner, matchId)\n\n // there is a loaderPromise, so we are in the middle of a load\n if (prevMatch._nonReactive.loaderPromise) {\n // do not block if we already have stale data we can show\n // but only if the ongoing load is not a preload since error handling is different for preloads\n // and we don't want to swallow errors\n if (\n prevMatch.status === 'success' &&\n !inner.sync &&\n !prevMatch.preload &&\n shouldReloadInBackground\n ) {\n return prevMatch\n }\n await prevMatch._nonReactive.loaderPromise\n const match = inner.router.getMatch(matchId)!\n const error = match._nonReactive.error || match.error\n if (error) {\n handleRedirectAndNotFound(inner, match, error)\n }\n\n if (match.status === 'pending') {\n await handleLoader(\n preload,\n prevMatch,\n previousRouteMatchId,\n match,\n route,\n )\n }\n } else {\n const nextPreload =\n preload && !inner.router.stores.matchStores.has(matchId)\n const match = inner.router.getMatch(matchId)!\n match._nonReactive.loaderPromise = createControlledPromise<void>()\n if (nextPreload !== match.preload) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n preload: nextPreload,\n }))\n }\n\n await handleLoader(preload, prevMatch, previousRouteMatchId, match, route)\n }\n }\n const match = inner.router.getMatch(matchId)!\n if (!loaderIsRunningAsync) {\n match._nonReactive.loaderPromise?.resolve()\n match._nonReactive.loadPromise?.resolve()\n match._nonReactive.loadPromise = undefined\n }\n\n clearTimeout(match._nonReactive.pendingTimeout)\n match._nonReactive.pendingTimeout = undefined\n if (!loaderIsRunningAsync) match._nonReactive.loaderPromise = undefined\n match._nonReactive.dehydrated = undefined\n\n const nextIsFetching = loaderIsRunningAsync ? match.isFetching : false\n if (nextIsFetching !== match.isFetching || match.invalid !== false) {\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n isFetching: nextIsFetching,\n invalid: false,\n }))\n return inner.router.getMatch(matchId)!\n } else {\n return match\n }\n}\n\nexport async function loadMatches(arg: {\n router: AnyRouter\n location: ParsedLocation\n matches: Array<AnyRouteMatch>\n preload?: boolean\n forceStaleReload?: boolean\n onReady?: () => Promise<void>\n updateMatch: UpdateMatchFn\n sync?: boolean\n}): Promise<Array<MakeRouteMatch>> {\n const inner: InnerLoadContext = arg\n const matchPromises: Array<Promise<AnyRouteMatch>> = []\n\n // make sure the pending component is immediately rendered when hydrating a match that is not SSRed\n // the pending component was already rendered on the server and we want to keep it shown on the client until minPendingMs is reached\n if (\n !(isServer ?? inner.router.isServer) &&\n hasForcePendingActiveMatch(inner.router)\n ) {\n triggerOnReady(inner)\n }\n\n let beforeLoadNotFound: NotFoundError | undefined\n\n // Execute all beforeLoads one by one\n for (let i = 0; i < inner.matches.length; i++) {\n try {\n const beforeLoad = handleBeforeLoad(inner, i)\n if (isPromise(beforeLoad)) await beforeLoad\n } catch (err) {\n if (isRedirect(err)) {\n throw err\n }\n if (isNotFound(err)) {\n beforeLoadNotFound = err\n } else {\n if (!inner.preload) throw err\n }\n break\n }\n\n if (inner.serialError || inner.firstBadMatchIndex != null) {\n break\n }\n }\n\n // Execute loaders once, with max index adapted for beforeLoad notFound handling.\n const baseMaxIndexExclusive = inner.firstBadMatchIndex ?? inner.matches.length\n\n const boundaryIndex =\n beforeLoadNotFound && !inner.preload\n ? getNotFoundBoundaryIndex(inner, beforeLoadNotFound)\n : undefined\n\n const maxIndexExclusive =\n beforeLoadNotFound && inner.preload\n ? 0\n : boundaryIndex !== undefined\n ? Math.min(boundaryIndex + 1, baseMaxIndexExclusive)\n : baseMaxIndexExclusive\n\n let firstNotFound: NotFoundError | undefined\n let firstUnhandledRejection: unknown\n\n for (let i = 0; i < maxIndexExclusive; i++) {\n matchPromises.push(loadRouteMatch(inner, matchPromises, i))\n }\n\n try {\n await Promise.all(matchPromises)\n } catch {\n const settled = await Promise.allSettled(matchPromises)\n\n for (const result of settled) {\n if (result.status !== 'rejected') continue\n\n const reason = result.reason\n if (isRedirect(reason)) {\n throw reason\n }\n if (isNotFound(reason)) {\n firstNotFound ??= reason\n } else {\n firstUnhandledRejection ??= reason\n }\n }\n\n if (firstUnhandledRejection !== undefined) {\n throw firstUnhandledRejection\n }\n }\n\n const notFoundToThrow =\n firstNotFound ??\n (beforeLoadNotFound && !inner.preload ? beforeLoadNotFound : undefined)\n\n let headMaxIndex =\n inner.firstBadMatchIndex !== undefined\n ? inner.firstBadMatchIndex\n : inner.matches.length - 1\n\n if (!notFoundToThrow && beforeLoadNotFound && inner.preload) {\n return inner.matches\n }\n\n if (notFoundToThrow) {\n // Determine once which matched route will actually render the\n // notFoundComponent, then pass this precomputed index through the remaining\n // finalization steps.\n // This can differ from the throwing route when routeId targets an ancestor\n // boundary (or when bubbling resolves to a parent/root boundary).\n const renderedBoundaryIndex = getNotFoundBoundaryIndex(\n inner,\n notFoundToThrow,\n )\n\n if (renderedBoundaryIndex === undefined) {\n if (process.env.NODE_ENV !== 'production') {\n throw new Error(\n 'Invariant failed: Could not find match for notFound boundary',\n )\n }\n\n invariant()\n }\n const boundaryMatch = inner.matches[renderedBoundaryIndex]!\n\n const boundaryRoute = inner.router.looseRoutesById[boundaryMatch.routeId]!\n const defaultNotFoundComponent = (inner.router.options as any)\n ?.defaultNotFoundComponent\n\n // Ensure a notFoundComponent exists on the boundary route\n if (!boundaryRoute.options.notFoundComponent && defaultNotFoundComponent) {\n boundaryRoute.options.notFoundComponent = defaultNotFoundComponent\n }\n\n notFoundToThrow.routeId = boundaryMatch.routeId\n\n const boundaryIsRoot = boundaryMatch.routeId === inner.router.routeTree.id\n\n inner.updateMatch(boundaryMatch.id, (prev) => ({\n ...prev,\n ...(boundaryIsRoot\n ? // For root boundary, use globalNotFound so the root component's\n // shell still renders and <Outlet> handles the not-found display,\n // instead of replacing the entire root shell via status='notFound'.\n { status: 'success' as const, globalNotFound: true, error: undefined }\n : // For non-root boundaries, set status:'notFound' so MatchInner\n // renders the notFoundComponent directly.\n { status: 'notFound' as const, error: notFoundToThrow }),\n isFetching: false,\n }))\n\n headMaxIndex = renderedBoundaryIndex\n\n // Ensure the rendering boundary route chunk (and its lazy components, including\n // lazy notFoundComponent) is loaded before we continue to head execution/render.\n await loadRouteChunk(boundaryRoute, ['notFoundComponent'])\n } else if (!inner.preload) {\n // Clear stale root global-not-found state on normal navigations that do not\n // throw notFound. This must live here (instead of only in runLoader success)\n // because the root loader may be skipped when data is still fresh.\n const rootMatch = inner.matches[0]!\n // `rootMatch` is the next match for this navigation. If it is not global\n // not-found, then any currently stored root global-not-found is stale.\n if (!rootMatch.globalNotFound) {\n // `currentRootMatch` is the current store state (from the previous\n // navigation/load). Update only when a stale flag is actually present.\n const currentRootMatch = inner.router.getMatch(rootMatch.id)\n if (currentRootMatch?.globalNotFound) {\n inner.updateMatch(rootMatch.id, (prev) => ({\n ...prev,\n globalNotFound: false,\n error: undefined,\n }))\n }\n }\n }\n\n // When a serial error occurred (e.g. beforeLoad threw a regular Error),\n // the erroring route's lazy chunk wasn't loaded because loaders were skipped.\n // We need to load it so the code-split errorComponent is available for rendering.\n if (inner.serialError && inner.firstBadMatchIndex !== undefined) {\n const errorRoute =\n inner.router.looseRoutesById[\n inner.matches[inner.firstBadMatchIndex]!.routeId\n ]!\n await loadRouteChunk(errorRoute, ['errorComponent'])\n }\n\n // serially execute heads once after loaders/notFound handling, ensuring\n // all head functions get a chance even if one throws.\n for (let i = 0; i <= headMaxIndex; i++) {\n const match = inner.matches[i]!\n const { id: matchId, routeId } = match\n const route = inner.router.looseRoutesById[routeId]!\n try {\n const headResult = executeHead(inner, matchId, route)\n if (headResult) {\n const head = await headResult\n inner.updateMatch(matchId, (prev) => ({\n ...prev,\n ...head,\n }))\n }\n } catch (err) {\n console.error(`Error executing head for route ${routeId}:`, err)\n }\n }\n\n const readyPromise = triggerOnReady(inner)\n if (isPromise(readyPromise)) {\n await readyPromise\n }\n\n if (notFoundToThrow) {\n throw notFoundToThrow\n }\n\n if (inner.serialError && !inner.preload && !inner.onReady) {\n throw inner.serialError\n }\n\n return inner.matches\n}\n\nexport type RouteComponentType =\n | 'component'\n | 'errorComponent'\n | 'pendingComponent'\n | 'notFoundComponent'\n\nfunction preloadRouteComponents(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType>,\n): Promise<void> | undefined {\n const preloads = componentTypesToLoad\n .map((type) => (route.options[type] as any)?.preload?.())\n .filter(Boolean)\n\n if (preloads.length === 0) return undefined\n\n return Promise.all(preloads) as any as Promise<void>\n}\n\nexport function loadRouteChunk(\n route: AnyRoute,\n componentTypesToLoad: Array<RouteComponentType> = componentTypes,\n) {\n if (!route._lazyLoaded && route._lazyPromise === undefined) {\n if (route.lazyFn) {\n route._lazyPromise = route.lazyFn().then((lazyRoute) => {\n // explicitly don't copy over the lazy route's id\n const { id: _id, ...options } = lazyRoute.options\n Object.assign(route.options, options)\n route._lazyLoaded = true\n route._lazyPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._lazyLoaded = true\n }\n }\n\n const runAfterLazy = () =>\n route._componentsLoaded\n ? undefined\n : componentTypesToLoad === componentTypes\n ? (() => {\n if (route._componentsPromise === undefined) {\n const componentsPromise = preloadRouteComponents(\n route,\n componentTypes,\n )\n\n if (componentsPromise) {\n route._componentsPromise = componentsPromise.then(() => {\n route._componentsLoaded = true\n route._componentsPromise = undefined // gc promise, we won't need it anymore\n })\n } else {\n route._componentsLoaded = true\n }\n }\n\n return route._componentsPromise\n })()\n : preloadRouteComponents(route, componentTypesToLoad)\n\n return route._lazyPromise\n ? route._lazyPromise.then(runAfterLazy)\n : runAfterLazy()\n}\n\nfunction makeMaybe<TValue, TError>(\n value: TValue,\n error: TError,\n): { status: 'success'; value: TValue } | { status: 'error'; error: TError } {\n if (error) {\n return { status: 'error' as const, error }\n }\n return { status: 'success' as const, value }\n}\n\nexport function routeNeedsPreload(route: AnyRoute) {\n for (const componentType of componentTypes) {\n if ((route.options[componentType] as any)?.preload) {\n return true\n }\n }\n return false\n}\n\nexport const componentTypes: Array<RouteComponentType> = [\n 'component',\n 'errorComponent',\n 'pendingComponent',\n 'notFoundComponent',\n] as const\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tanstack/router-core",
|
|
3
|
-
"version": "1.169.
|
|
3
|
+
"version": "1.169.2",
|
|
4
4
|
"description": "Modern and scalable routing for React applications",
|
|
5
5
|
"author": "Tanner Linsley",
|
|
6
6
|
"license": "MIT",
|
|
@@ -168,20 +168,16 @@
|
|
|
168
168
|
},
|
|
169
169
|
"dependencies": {
|
|
170
170
|
"cookie-es": "^3.0.0",
|
|
171
|
-
"seroval": "^1.5.
|
|
172
|
-
"seroval-plugins": "^1.5.
|
|
171
|
+
"seroval": "^1.5.4",
|
|
172
|
+
"seroval-plugins": "^1.5.4",
|
|
173
173
|
"@tanstack/history": "1.161.6"
|
|
174
174
|
},
|
|
175
175
|
"devDependencies": {
|
|
176
|
-
"@tanstack/intent": "^0.0.14",
|
|
177
176
|
"@tanstack/store": "^0.9.3",
|
|
178
177
|
"@types/node": "25.0.9",
|
|
179
178
|
"esbuild": "^0.27.4",
|
|
180
179
|
"vite": "*"
|
|
181
180
|
},
|
|
182
|
-
"bin": {
|
|
183
|
-
"intent": "./bin/intent.js"
|
|
184
|
-
},
|
|
185
181
|
"scripts": {
|
|
186
182
|
"clean": "rimraf ./dist && rimraf ./coverage",
|
|
187
183
|
"test:eslint": "eslint ./src",
|
|
@@ -21,6 +21,10 @@ sources:
|
|
|
21
21
|
|
|
22
22
|
# Auth and Guards
|
|
23
23
|
|
|
24
|
+
> **This skill covers the routing side of auth.** For the **server-side primitives** — session cookies (`HttpOnly`/`Secure`/`SameSite`), `useSession`-style helpers, OAuth `state` + PKCE, password-reset enumeration defense, CSRF, rate limiting — see [start-core/auth-server-primitives](../../../../start-client-core/skills/start-core/auth-server-primitives/SKILL.md). The two skills are designed to be used together.
|
|
25
|
+
>
|
|
26
|
+
> **CRITICAL**: A route guard (`beforeLoad`) does NOT protect a `createServerFn` declared on that route. Server functions are RPC endpoints reachable by direct POST regardless of which route renders them. See "Route guards do not protect server functions" below.
|
|
27
|
+
|
|
24
28
|
## Setup
|
|
25
29
|
|
|
26
30
|
Protect routes with `beforeLoad` + `redirect()` in a pathless layout route (`_authenticated`):
|
|
@@ -371,6 +375,41 @@ export const Route = createFileRoute('/_authenticated')({
|
|
|
371
375
|
|
|
372
376
|
## Common Mistakes
|
|
373
377
|
|
|
378
|
+
### CRITICAL: Route guards do not protect server functions
|
|
379
|
+
|
|
380
|
+
A `beforeLoad` redirect protects the **route's UI**, not the **server functions** declared on it. `createServerFn` produces an RPC endpoint reachable by direct POST regardless of which route renders the calling UI. An attacker doesn't have to load `/_authenticated/orders` — they can curl the RPC endpoint directly.
|
|
381
|
+
|
|
382
|
+
```tsx
|
|
383
|
+
// WRONG — handler has no auth check; the route guard doesn't help
|
|
384
|
+
import { createServerFn } from '@tanstack/react-start'
|
|
385
|
+
import { createFileRoute, redirect } from '@tanstack/react-router'
|
|
386
|
+
|
|
387
|
+
const getMyOrders = createServerFn({ method: 'GET' }).handler(async () => {
|
|
388
|
+
return db.orders.findMany() // ← anyone can hit the RPC
|
|
389
|
+
})
|
|
390
|
+
|
|
391
|
+
export const Route = createFileRoute('/_authenticated/orders')({
|
|
392
|
+
beforeLoad: ({ context }) => {
|
|
393
|
+
if (!context.auth.isAuthenticated) throw redirect({ to: '/login' })
|
|
394
|
+
},
|
|
395
|
+
loader: () => getMyOrders(),
|
|
396
|
+
})
|
|
397
|
+
```
|
|
398
|
+
|
|
399
|
+
```tsx
|
|
400
|
+
// CORRECT — auth enforced on the handler itself, via middleware
|
|
401
|
+
import { createServerFn } from '@tanstack/react-start'
|
|
402
|
+
import { authMiddleware } from '~/server/auth-middleware'
|
|
403
|
+
|
|
404
|
+
const getMyOrders = createServerFn({ method: 'GET' })
|
|
405
|
+
.middleware([authMiddleware])
|
|
406
|
+
.handler(async ({ context }) => {
|
|
407
|
+
return db.orders.findMany({ where: { userId: context.session.userId } })
|
|
408
|
+
})
|
|
409
|
+
```
|
|
410
|
+
|
|
411
|
+
Rule of thumb: every `createServerFn` that touches user data needs `authMiddleware` (or an equivalent in-handler check). The route guard is for the page experience; the RPC guard is for the data. See [start-core/auth-server-primitives](../../../../start-client-core/skills/start-core/auth-server-primitives/SKILL.md) for the full session/middleware pattern.
|
|
412
|
+
|
|
374
413
|
### HIGH: Auth check in component instead of beforeLoad
|
|
375
414
|
|
|
376
415
|
Component-level auth checks cause a **flash of protected content** before the redirect:
|
|
@@ -456,3 +495,5 @@ Place protected routes as children of the `_authenticated` layout route. Public
|
|
|
456
495
|
## Cross-References
|
|
457
496
|
|
|
458
497
|
- See also: **router-core/data-loading/SKILL.md** — `beforeLoad` runs before `loader`; auth context flows into loader via route context
|
|
498
|
+
- See also: **start-core/auth-server-primitives/SKILL.md** — server-side session cookies, OAuth state + PKCE, CSRF, password-reset hardening, rate limiting (the server half of authentication)
|
|
499
|
+
- See also: **start-core/middleware/SKILL.md** — `authMiddleware` factory pattern for protecting individual `createServerFn` calls
|
|
@@ -391,22 +391,74 @@ component: () => {
|
|
|
391
391
|
}
|
|
392
392
|
```
|
|
393
393
|
|
|
394
|
-
### 3. CRITICAL: Generating Next.js or
|
|
394
|
+
### 3. CRITICAL: Generating Next.js, Remix, or React Router DOM patterns
|
|
395
395
|
|
|
396
|
-
TanStack Router does NOT use `getServerSideProps`, `getStaticProps`, App Router `page.tsx`,
|
|
396
|
+
TanStack Router does NOT use `getServerSideProps`, `getStaticProps`, App Router `page.tsx`, Remix-style server-only `loader` exports, or anything from `react-router-dom`.
|
|
397
|
+
|
|
398
|
+
#### Wrong file structures
|
|
399
|
+
|
|
400
|
+
```text
|
|
401
|
+
WRONG (Next.js Pages Router):
|
|
402
|
+
src/pages/index.tsx
|
|
403
|
+
src/pages/_app.tsx
|
|
404
|
+
src/pages/posts/[id].tsx
|
|
405
|
+
|
|
406
|
+
WRONG (Next.js App Router):
|
|
407
|
+
app/layout.tsx
|
|
408
|
+
app/page.tsx
|
|
409
|
+
app/posts/[id]/page.tsx
|
|
410
|
+
|
|
411
|
+
WRONG (Next.js custom App):
|
|
412
|
+
_app/index.tsx
|
|
413
|
+
pages/_app.tsx, pages/_document.tsx
|
|
414
|
+
|
|
415
|
+
CORRECT (TanStack Router file-based routing):
|
|
416
|
+
src/routes/__root.tsx
|
|
417
|
+
src/routes/index.tsx
|
|
418
|
+
src/routes/posts/$postId.tsx
|
|
419
|
+
```
|
|
420
|
+
|
|
421
|
+
#### Wrong imports
|
|
397
422
|
|
|
398
423
|
```tsx
|
|
399
|
-
// WRONG —
|
|
424
|
+
// WRONG — react-router-dom is a different library
|
|
425
|
+
import {
|
|
426
|
+
Link,
|
|
427
|
+
useNavigate,
|
|
428
|
+
BrowserRouter,
|
|
429
|
+
Route,
|
|
430
|
+
Routes,
|
|
431
|
+
} from 'react-router-dom'
|
|
432
|
+
|
|
433
|
+
// WRONG — Next.js Link/router
|
|
434
|
+
import Link from 'next/link'
|
|
435
|
+
import { useRouter } from 'next/router' // Pages Router
|
|
436
|
+
import { useRouter } from 'next/navigation' // App Router
|
|
437
|
+
|
|
438
|
+
// CORRECT — everything routing-related lives in @tanstack/react-router
|
|
439
|
+
import {
|
|
440
|
+
Link,
|
|
441
|
+
useNavigate,
|
|
442
|
+
useRouter,
|
|
443
|
+
useLocation,
|
|
444
|
+
redirect,
|
|
445
|
+
} from '@tanstack/react-router'
|
|
446
|
+
```
|
|
447
|
+
|
|
448
|
+
#### Wrong loader/data-fetching patterns
|
|
449
|
+
|
|
450
|
+
```tsx
|
|
451
|
+
// WRONG — Next.js Pages Router
|
|
400
452
|
export async function getServerSideProps() {
|
|
401
453
|
return { props: { data: await fetchData() } }
|
|
402
454
|
}
|
|
403
455
|
|
|
404
|
-
// WRONG — Remix
|
|
456
|
+
// WRONG — Remix
|
|
405
457
|
export async function loader({ request }: LoaderFunctionArgs) {
|
|
406
458
|
return json({ data: await fetchData() })
|
|
407
459
|
}
|
|
408
460
|
|
|
409
|
-
// CORRECT — TanStack Router
|
|
461
|
+
// CORRECT — TanStack Router
|
|
410
462
|
export const Route = createFileRoute('/data')({
|
|
411
463
|
loader: async () => {
|
|
412
464
|
const data = await fetchData()
|
|
@@ -421,6 +473,8 @@ function DataPage() {
|
|
|
421
473
|
}
|
|
422
474
|
```
|
|
423
475
|
|
|
476
|
+
If you see `src/pages/`, `app/layout.tsx`, `react-router-dom`, or any of the above in agent output, the agent is generating for the wrong framework. The build will either fail or produce duplicate `/` routes that conflict at runtime.
|
|
477
|
+
|
|
424
478
|
## Tension: Client-First Loaders vs SSR
|
|
425
479
|
|
|
426
480
|
TanStack Router loaders are client-first by design. When SSR is enabled, they run in both environments. This means:
|
|
@@ -313,7 +313,9 @@ export function NavItem(props: NavItemProps): React.ReactNode {
|
|
|
313
313
|
<NavItem label="Post" linkOptions={{ to: '/posts/$postId', params: { postId: '1' } }} />
|
|
314
314
|
```
|
|
315
315
|
|
|
316
|
-
### `ValidateNavigateOptions`
|
|
316
|
+
### `ValidateNavigateOptions` and `ValidateRedirectOptions`
|
|
317
|
+
|
|
318
|
+
Same pattern as `ValidateLinkOptions` above, for `useNavigate` and `redirect`. Declare a generic public overload plus a non-generic implementation signature so the call site stays narrowed and the body works without casts:
|
|
317
319
|
|
|
318
320
|
```tsx
|
|
319
321
|
import {
|
|
@@ -332,45 +334,15 @@ export function useDelayedNavigate<
|
|
|
332
334
|
export function useDelayedNavigate(
|
|
333
335
|
options: ValidateNavigateOptions,
|
|
334
336
|
delayMs: number,
|
|
335
|
-
)
|
|
337
|
+
) {
|
|
336
338
|
const navigate = useNavigate()
|
|
337
339
|
return () => {
|
|
338
340
|
setTimeout(() => navigate(options), delayMs)
|
|
339
341
|
}
|
|
340
342
|
}
|
|
341
|
-
|
|
342
|
-
// Usage — type-safe
|
|
343
|
-
const go = useDelayedNavigate(
|
|
344
|
-
{ to: '/posts/$postId', params: { postId: '1' } },
|
|
345
|
-
500,
|
|
346
|
-
)
|
|
347
343
|
```
|
|
348
344
|
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
```tsx
|
|
352
|
-
import {
|
|
353
|
-
redirect,
|
|
354
|
-
type RegisteredRouter,
|
|
355
|
-
type ValidateRedirectOptions,
|
|
356
|
-
} from '@tanstack/react-router'
|
|
357
|
-
|
|
358
|
-
export async function fetchOrRedirect<
|
|
359
|
-
TRouter extends RegisteredRouter = RegisteredRouter,
|
|
360
|
-
TOptions = unknown,
|
|
361
|
-
>(
|
|
362
|
-
url: string,
|
|
363
|
-
redirectOptions: ValidateRedirectOptions<TRouter, TOptions>,
|
|
364
|
-
): Promise<unknown>
|
|
365
|
-
export async function fetchOrRedirect(
|
|
366
|
-
url: string,
|
|
367
|
-
redirectOptions: ValidateRedirectOptions,
|
|
368
|
-
): Promise<unknown> {
|
|
369
|
-
const response = await fetch(url)
|
|
370
|
-
if (!response.ok && response.status === 401) throw redirect(redirectOptions)
|
|
371
|
-
return response.json()
|
|
372
|
-
}
|
|
373
|
-
```
|
|
345
|
+
`ValidateRedirectOptions` works identically — declare a generic overload accepting `ValidateRedirectOptions<TRouter, TOptions>` and an impl signature accepting `ValidateRedirectOptions`, then call `redirect(options)` in the body.
|
|
374
346
|
|
|
375
347
|
### Render Props for Maximum Performance
|
|
376
348
|
|
|
@@ -477,21 +449,44 @@ declare module '@tanstack/react-router' {
|
|
|
477
449
|
}
|
|
478
450
|
```
|
|
479
451
|
|
|
480
|
-
### 5. CRITICAL (cross-skill):
|
|
452
|
+
### 5. CRITICAL (cross-skill): Wrong-framework imports and file structure
|
|
453
|
+
|
|
454
|
+
Wrong-framework code looks plausible (it's React) but breaks the build or produces conflicting `/` routes at runtime.
|
|
481
455
|
|
|
482
456
|
```tsx
|
|
483
|
-
// WRONG —
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
457
|
+
// WRONG — react-router-dom and next/* are different libraries
|
|
458
|
+
import { Link, useNavigate, useSearchParams } from 'react-router-dom'
|
|
459
|
+
import Link from 'next/link'
|
|
460
|
+
import { useRouter, useParams } from 'next/navigation'
|
|
487
461
|
|
|
488
|
-
// CORRECT —
|
|
462
|
+
// CORRECT — all routing exports come from @tanstack/react-router
|
|
463
|
+
import {
|
|
464
|
+
Link,
|
|
465
|
+
Outlet,
|
|
466
|
+
useNavigate,
|
|
467
|
+
useRouter,
|
|
468
|
+
useLocation,
|
|
469
|
+
useParams,
|
|
470
|
+
redirect,
|
|
471
|
+
} from '@tanstack/react-router'
|
|
472
|
+
```
|
|
473
|
+
|
|
474
|
+
```tsx
|
|
475
|
+
// WRONG file structures + APIs:
|
|
476
|
+
// src/pages/*.tsx with getServerSideProps / getStaticProps (Next.js Pages Router)
|
|
477
|
+
// app/layout.tsx + app/page.tsx (Next.js App Router)
|
|
478
|
+
// _app/index.tsx, pages/_app.tsx, pages/_document.tsx (Next.js custom App)
|
|
479
|
+
// loader/action exports (Remix)
|
|
480
|
+
|
|
481
|
+
// CORRECT — TanStack file-based routing at src/routes/*.tsx
|
|
489
482
|
export const Route = createFileRoute('/posts')({
|
|
490
|
-
loader: async () => { ... },
|
|
491
|
-
validateSearch: zodValidator(schema),
|
|
483
|
+
loader: async () => { ... },
|
|
484
|
+
validateSearch: zodValidator(schema),
|
|
492
485
|
component: PostsComponent,
|
|
493
486
|
})
|
|
494
|
-
const search = Route.useSearch()
|
|
487
|
+
const search = Route.useSearch()
|
|
495
488
|
```
|
|
496
489
|
|
|
490
|
+
If a build error mentions `react-router-dom`, `next/`, `pages/_app`, or duplicate `/` routes, fix the import — don't paper over with type assertions.
|
|
491
|
+
|
|
497
492
|
See also: router-core (Register setup), router-core/navigation (from narrowing), router-core/code-splitting (getRouteApi).
|
package/src/load-matches.ts
CHANGED
|
@@ -713,7 +713,7 @@ const runLoader = async (
|
|
|
713
713
|
const pendingPromise = match._nonReactive.minPendingPromise
|
|
714
714
|
if (pendingPromise) await pendingPromise
|
|
715
715
|
|
|
716
|
-
// Last but not least, wait for the
|
|
716
|
+
// Last but not least, wait for the components
|
|
717
717
|
// to be preloaded before we resolve the match
|
|
718
718
|
if (route._componentsPromise) await route._componentsPromise
|
|
719
719
|
inner.updateMatch(matchId, (prev) => ({
|
package/bin/intent.js
DELETED
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
// Auto-generated by @tanstack/intent setup
|
|
3
|
-
// Exposes the intent end-user CLI for consumers of this library.
|
|
4
|
-
// Commit this file, then add to your package.json:
|
|
5
|
-
// "bin": { "intent": "./bin/intent.js" }
|
|
6
|
-
try {
|
|
7
|
-
await import('@tanstack/intent/intent-library')
|
|
8
|
-
} catch (e) {
|
|
9
|
-
const isModuleNotFound =
|
|
10
|
-
e?.code === 'ERR_MODULE_NOT_FOUND' || e?.code === 'MODULE_NOT_FOUND'
|
|
11
|
-
const missingIntentLibrary =
|
|
12
|
-
typeof e?.message === 'string' && e.message.includes('@tanstack/intent')
|
|
13
|
-
|
|
14
|
-
if (isModuleNotFound && missingIntentLibrary) {
|
|
15
|
-
console.error('@tanstack/intent is not installed.')
|
|
16
|
-
console.error('')
|
|
17
|
-
console.error('Install it as a dev dependency:')
|
|
18
|
-
console.error(' npm add -D @tanstack/intent')
|
|
19
|
-
console.error('')
|
|
20
|
-
console.error('Or run directly:')
|
|
21
|
-
console.error(' npx @tanstack/intent@latest list')
|
|
22
|
-
process.exit(1)
|
|
23
|
-
}
|
|
24
|
-
throw e
|
|
25
|
-
}
|