@tanstack/start-plugin-core 1.120.4-alpha.15 → 1.120.4-alpha.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/nitro/nitro-plugin.cjs +2 -36
- package/dist/cjs/nitro/nitro-plugin.cjs.map +1 -1
- package/dist/cjs/plugin.cjs +2 -1
- package/dist/cjs/plugin.cjs.map +1 -1
- package/dist/cjs/plugin.d.cts +0 -622
- package/dist/cjs/prerender.cjs +13 -22
- package/dist/cjs/prerender.cjs.map +1 -1
- package/dist/cjs/schema.cjs +1 -15
- package/dist/cjs/schema.cjs.map +1 -1
- package/dist/cjs/schema.d.cts +542 -2319
- package/dist/cjs/start-server-routes-plugin/plugin.cjs +2 -6
- package/dist/cjs/start-server-routes-plugin/plugin.cjs.map +1 -1
- package/dist/esm/nitro/nitro-plugin.js +2 -36
- package/dist/esm/nitro/nitro-plugin.js.map +1 -1
- package/dist/esm/plugin.d.ts +0 -622
- package/dist/esm/plugin.js +2 -1
- package/dist/esm/plugin.js.map +1 -1
- package/dist/esm/prerender.js +13 -22
- package/dist/esm/prerender.js.map +1 -1
- package/dist/esm/schema.d.ts +542 -2319
- package/dist/esm/schema.js +1 -15
- package/dist/esm/schema.js.map +1 -1
- package/dist/esm/start-server-routes-plugin/plugin.js +2 -6
- package/dist/esm/start-server-routes-plugin/plugin.js.map +1 -1
- package/package.json +5 -10
- package/src/nitro/nitro-plugin.ts +0 -40
- package/src/plugin.ts +1 -0
- package/src/prerender.ts +17 -29
- package/src/schema.ts +0 -18
- package/src/start-server-routes-plugin/plugin.ts +2 -10
package/dist/esm/schema.js
CHANGED
|
@@ -77,8 +77,7 @@ function createTanStackStartOptionsSchema(frameworkPlugin = {}) {
|
|
|
77
77
|
concurrency: z.number().optional(),
|
|
78
78
|
filter: z.function().args(pageSchema).returns(z.any()).optional(),
|
|
79
79
|
failOnError: z.boolean().optional()
|
|
80
|
-
}).and(pagePrerenderOptionsSchema.optional()).optional()
|
|
81
|
-
shell: shellSchema.optional()
|
|
80
|
+
}).and(pagePrerenderOptionsSchema.optional()).optional()
|
|
82
81
|
}).optional().default({});
|
|
83
82
|
}
|
|
84
83
|
const pageSitemapOptionsSchema = z.object({
|
|
@@ -115,7 +114,6 @@ const pageBaseSchema = z.object({
|
|
|
115
114
|
});
|
|
116
115
|
const pagePrerenderOptionsSchema = z.object({
|
|
117
116
|
enabled: z.boolean().optional(),
|
|
118
|
-
outputPath: z.string().optional(),
|
|
119
117
|
autoSubfolderIndex: z.boolean().optional(),
|
|
120
118
|
crawlLinks: z.boolean().optional(),
|
|
121
119
|
retryCount: z.number().optional(),
|
|
@@ -127,18 +125,6 @@ const pagePrerenderOptionsSchema = z.object({
|
|
|
127
125
|
})
|
|
128
126
|
).returns(z.any()).optional()
|
|
129
127
|
});
|
|
130
|
-
const shellSchema = z.object({
|
|
131
|
-
enabled: z.boolean().optional().default(true),
|
|
132
|
-
maskPath: z.string().optional().default("/"),
|
|
133
|
-
autoRedirect: z.boolean().optional().default(true),
|
|
134
|
-
prerender: pagePrerenderOptionsSchema.optional().default({}).transform((opts) => ({
|
|
135
|
-
outputPath: opts.outputPath ?? "/_shell",
|
|
136
|
-
crawlLinks: false,
|
|
137
|
-
retryCount: 0,
|
|
138
|
-
...opts,
|
|
139
|
-
enabled: true
|
|
140
|
-
}))
|
|
141
|
-
});
|
|
142
128
|
const pageSchema = pageBaseSchema.extend({
|
|
143
129
|
prerender: pagePrerenderOptionsSchema.optional()
|
|
144
130
|
});
|
package/dist/esm/schema.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"schema.js","sources":["../../src/schema.ts"],"sourcesContent":["import path from 'node:path'\nimport { existsSync } from 'node:fs'\nimport { z } from 'zod'\nimport { configSchema, getConfig } from '@tanstack/router-generator'\nimport type { NitroConfig } from 'nitropack'\n\nconst tsrConfig = configSchema.partial().extend({\n srcDirectory: z.string().optional().default('src'),\n})\n\nexport function createTanStackConfig<\n TFrameworkPlugin extends Record<string, unknown>,\n>(frameworkPlugin?: TFrameworkPlugin) {\n const schema = createTanStackStartOptionsSchema(frameworkPlugin)\n\n return {\n schema,\n parse: (opts?: z.input<typeof schema>) => {\n const options = schema.parse(opts)\n\n const srcDirectory = options.tsr.srcDirectory\n\n const routesDirectory =\n options.tsr.routesDirectory ?? path.join(srcDirectory, 'routes')\n\n const generatedRouteTree =\n options.tsr.generatedRouteTree ??\n path.join(srcDirectory, 'routeTree.gen.ts')\n\n const clientEntryPath = (() => {\n if (options.client.entry) {\n return path.join(srcDirectory, options.client.entry)\n }\n\n if (existsSync(path.join(srcDirectory, 'client.tsx'))) {\n return path.join(srcDirectory, 'client.tsx')\n }\n\n return '/~start/default-client-entry'\n })()\n\n const serverEntryPath = (() => {\n if (options.server.entry) {\n return path.join(srcDirectory, options.server.entry)\n }\n\n if (existsSync(path.join(srcDirectory, 'server.tsx'))) {\n return path.join(srcDirectory, 'server.tsx')\n }\n\n return '/~start/default-server-entry'\n })()\n\n return {\n ...options,\n tsr: {\n ...options.tsr,\n ...getConfig({\n ...options.tsr,\n routesDirectory,\n generatedRouteTree,\n }),\n },\n clientEntryPath,\n serverEntryPath,\n }\n },\n }\n}\n\nexport function createTanStackStartOptionsSchema(\n frameworkPlugin: Record<string, unknown> = {},\n) {\n return z\n .object({\n root: z.string().optional().default(process.cwd()),\n target: z.custom<NitroConfig['preset']>().optional(),\n ...frameworkPlugin,\n tsr: tsrConfig.optional().default({}),\n client: z\n .object({\n entry: z.string().optional(),\n base: z.string().optional().default('/_build'),\n })\n .optional()\n .default({}),\n server: z\n .object({\n entry: z.string().optional(),\n })\n .optional()\n .default({}),\n serverFns: z\n .object({\n base: z.string().optional().default('/_server'),\n })\n .optional()\n .default({}),\n public: z\n .object({\n dir: z.string().optional().default('public'),\n base: z.string().optional().default('/'),\n })\n .optional()\n .default({}),\n pages: z\n .array(z.union([z.string(), pageSchema]))\n .optional()\n .default([]),\n sitemap: pagePrerenderOptionsSchema\n .extend({\n host: z.string().optional(),\n })\n .optional(),\n prerender: z\n .object({\n enabled: z.boolean().optional(),\n concurrency: z.number().optional(),\n filter: z.function().args(pageSchema).returns(z.any()).optional(),\n failOnError: z.boolean().optional(),\n })\n .and(pagePrerenderOptionsSchema.optional())\n .optional(),\n
|
|
1
|
+
{"version":3,"file":"schema.js","sources":["../../src/schema.ts"],"sourcesContent":["import path from 'node:path'\nimport { existsSync } from 'node:fs'\nimport { z } from 'zod'\nimport { configSchema, getConfig } from '@tanstack/router-generator'\nimport type { NitroConfig } from 'nitropack'\n\nconst tsrConfig = configSchema.partial().extend({\n srcDirectory: z.string().optional().default('src'),\n})\n\nexport function createTanStackConfig<\n TFrameworkPlugin extends Record<string, unknown>,\n>(frameworkPlugin?: TFrameworkPlugin) {\n const schema = createTanStackStartOptionsSchema(frameworkPlugin)\n\n return {\n schema,\n parse: (opts?: z.input<typeof schema>) => {\n const options = schema.parse(opts)\n\n const srcDirectory = options.tsr.srcDirectory\n\n const routesDirectory =\n options.tsr.routesDirectory ?? path.join(srcDirectory, 'routes')\n\n const generatedRouteTree =\n options.tsr.generatedRouteTree ??\n path.join(srcDirectory, 'routeTree.gen.ts')\n\n const clientEntryPath = (() => {\n if (options.client.entry) {\n return path.join(srcDirectory, options.client.entry)\n }\n\n if (existsSync(path.join(srcDirectory, 'client.tsx'))) {\n return path.join(srcDirectory, 'client.tsx')\n }\n\n return '/~start/default-client-entry'\n })()\n\n const serverEntryPath = (() => {\n if (options.server.entry) {\n return path.join(srcDirectory, options.server.entry)\n }\n\n if (existsSync(path.join(srcDirectory, 'server.tsx'))) {\n return path.join(srcDirectory, 'server.tsx')\n }\n\n return '/~start/default-server-entry'\n })()\n\n return {\n ...options,\n tsr: {\n ...options.tsr,\n ...getConfig({\n ...options.tsr,\n routesDirectory,\n generatedRouteTree,\n }),\n },\n clientEntryPath,\n serverEntryPath,\n }\n },\n }\n}\n\nexport function createTanStackStartOptionsSchema(\n frameworkPlugin: Record<string, unknown> = {},\n) {\n return z\n .object({\n root: z.string().optional().default(process.cwd()),\n target: z.custom<NitroConfig['preset']>().optional(),\n ...frameworkPlugin,\n tsr: tsrConfig.optional().default({}),\n client: z\n .object({\n entry: z.string().optional(),\n base: z.string().optional().default('/_build'),\n })\n .optional()\n .default({}),\n server: z\n .object({\n entry: z.string().optional(),\n })\n .optional()\n .default({}),\n serverFns: z\n .object({\n base: z.string().optional().default('/_server'),\n })\n .optional()\n .default({}),\n public: z\n .object({\n dir: z.string().optional().default('public'),\n base: z.string().optional().default('/'),\n })\n .optional()\n .default({}),\n pages: z\n .array(z.union([z.string(), pageSchema]))\n .optional()\n .default([]),\n sitemap: pagePrerenderOptionsSchema\n .extend({\n host: z.string().optional(),\n })\n .optional(),\n prerender: z\n .object({\n enabled: z.boolean().optional(),\n concurrency: z.number().optional(),\n filter: z.function().args(pageSchema).returns(z.any()).optional(),\n failOnError: z.boolean().optional(),\n })\n .and(pagePrerenderOptionsSchema.optional())\n .optional(),\n })\n .optional()\n .default({})\n}\n\nconst pageSitemapOptionsSchema = z.object({\n exclude: z.boolean().optional(),\n priority: z.number().min(0).max(1).optional(),\n changefreq: z\n .enum(['always', 'hourly', 'daily', 'weekly', 'monthly', 'yearly', 'never'])\n .optional(),\n lastmod: z.union([z.string(), z.date()]).optional(),\n alternateRefs: z\n .array(\n z.object({\n href: z.string(),\n hreflang: z.string(),\n }),\n )\n .optional(),\n images: z\n .array(\n z.object({\n loc: z.string(),\n caption: z.string().optional(),\n title: z.string().optional(),\n }),\n )\n .optional(),\n news: z\n .object({\n publication: z.object({\n name: z.string(),\n language: z.string(),\n }),\n publicationDate: z.union([z.string(), z.date()]),\n title: z.string(),\n })\n .optional(),\n})\n\nconst pageBaseSchema = z.object({\n path: z.string(),\n sitemap: pageSitemapOptionsSchema.optional(),\n fromCrawl: z.boolean().optional(),\n})\n\nconst pagePrerenderOptionsSchema = z.object({\n enabled: z.boolean().optional(),\n autoSubfolderIndex: z.boolean().optional(),\n crawlLinks: z.boolean().optional(),\n retryCount: z.number().optional(),\n retryDelay: z.number().optional(),\n onSuccess: z\n .function()\n .args(\n z.object({\n page: pageBaseSchema,\n html: z.string(),\n }),\n )\n .returns(z.any())\n .optional(),\n})\n\nexport const pageSchema = pageBaseSchema.extend({\n prerender: pagePrerenderOptionsSchema.optional(),\n})\n\nexport type Page = z.infer<typeof pageSchema>\n"],"names":[],"mappings":"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"}
|
|
@@ -160,7 +160,7 @@ Add the file in: "${config.routesDirectory}/${rootPathId}.tsx"`;
|
|
|
160
160
|
node.cleanedPath = removeGroups(
|
|
161
161
|
removeUnderscores(removeLayoutSegments(node.path)) ?? ""
|
|
162
162
|
);
|
|
163
|
-
const routeCode =
|
|
163
|
+
const routeCode = fs.readFileSync(node.fullPath, "utf-8");
|
|
164
164
|
if (!node.isVirtualParentRoute && !node.isVirtual) ;
|
|
165
165
|
const cleanedPathIsEmpty = (node.cleanedPath || "").length === 0;
|
|
166
166
|
const nonPathRoute = node._fsRouteType === "pathless_layout" && node.isNonPath;
|
|
@@ -489,11 +489,7 @@ ${((_a = routeNode.children) == null ? void 0 : _a.length) ? `${routeNode.variab
|
|
|
489
489
|
function buildStartDeclarationFile({
|
|
490
490
|
serverRoutesRelativePath
|
|
491
491
|
}) {
|
|
492
|
-
|
|
493
|
-
return [
|
|
494
|
-
'/// <reference types="vite/client" />',
|
|
495
|
-
`import '${serverRoutesPath}'`
|
|
496
|
-
].join("\n") + "\n";
|
|
492
|
+
return [`import '${serverRoutesRelativePath}'`].join("\n") + "\n";
|
|
497
493
|
}
|
|
498
494
|
function removeGroups(s) {
|
|
499
495
|
return s.replace(possiblyNestedRouteGroupPatternRegex, "");
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"plugin.js","sources":["../../../src/start-server-routes-plugin/plugin.ts"],"sourcesContent":["import path, { isAbsolute, join, normalize } from 'node:path'\nimport fs from 'node:fs'\nimport fsp from 'node:fs/promises'\nimport {\n format,\n logging,\n multiSortBy,\n physicalGetRouteNodes,\n removeExt,\n removeUnderscores,\n replaceBackslash,\n resetRegex,\n rootPathId,\n routePathToVariable,\n trimPathLeft,\n virtualGetRouteNodes,\n writeIfDifferent,\n} from '@tanstack/router-generator'\nimport { rootRouteId } from '@tanstack/router-core'\nimport { fillTemplate, getTargetTemplate } from './template'\nimport type { GetRouteNodesResult, RouteNode } from '@tanstack/router-generator'\nimport type { Config } from './config'\nimport type { Plugin } from 'vite'\n\nlet lock = false\nconst checkLock = () => lock\nconst setLock = (bool: boolean) => {\n lock = bool\n}\n\nexport function TanStackStartServerRoutesVite(config: Config): Plugin {\n let ROOT: string = process.cwd()\n\n const getRoutesDirectoryPath = () => {\n return isAbsolute(config.routesDirectory)\n ? config.routesDirectory\n : join(ROOT, config.routesDirectory)\n }\n\n const generate = async () => {\n if (checkLock()) {\n return\n }\n\n setLock(true)\n\n try {\n await generator(config, ROOT)\n } catch (err) {\n console.error(err)\n console.info()\n } finally {\n setLock(false)\n }\n }\n\n const handleFile = async (file: string) => {\n const filePath = normalize(file)\n\n const routesDirectoryPath = getRoutesDirectoryPath()\n if (filePath.startsWith(routesDirectoryPath)) {\n await generate()\n }\n }\n\n return {\n name: 'tanstack-start-server-routes-plugin',\n configureServer(server) {\n server.watcher.on('all', (event, path) => {\n handleFile(path)\n })\n },\n configResolved(config) {\n ROOT = config.root\n },\n async buildStart() {\n await generate()\n // if (this.environment.name === 'server') {\n // }\n },\n sharedDuringBuild: true,\n resolveId(id) {\n if (id === 'tanstack:server-routes') {\n const generatedRouteTreePath = getGeneratedRouteTreePath(ROOT)\n return generatedRouteTreePath\n }\n return null\n },\n }\n}\n\n// Maybe import this from `@tanstack/router-core` in the future???\nlet latestTask = 0\nconst routeGroupPatternRegex = /\\(.+\\)/g\nconst possiblyNestedRouteGroupPatternRegex = /\\([^/]+\\)\\/?/g\n\nlet isFirst = false\nlet skipMessage = false\n\nfunction getGeneratedRouteTreePath(root: string) {\n return path.resolve(root, '.tanstack-start/server-routes/routeTree.gen.ts')\n}\n\nasync function generator(config: Config, root: string) {\n const generatedServerRouteTreePath = getGeneratedRouteTreePath(root)\n const ROUTE_TEMPLATE = getTargetTemplate(config.target)\n const logger = logging({ disabled: config.disableLogging })\n\n if (!isFirst) {\n // logger.log('♻️ Generating server routes...')\n isFirst = true\n } else if (skipMessage) {\n skipMessage = false\n } else {\n // logger.log('♻️ Regenerating server routes...')\n }\n\n const taskId = latestTask + 1\n latestTask = taskId\n\n const checkLatest = () => {\n if (latestTask !== taskId) {\n skipMessage = true\n return false\n }\n\n return true\n }\n\n const start = Date.now()\n\n let getRouteNodesResult: GetRouteNodesResult\n\n if (config.virtualRouteConfig) {\n getRouteNodesResult = await virtualGetRouteNodes(config, root)\n } else {\n getRouteNodesResult = await physicalGetRouteNodes(config, root)\n }\n\n const { rootRouteNode, routeNodes: beforeRouteNodes } = getRouteNodesResult\n if (rootRouteNode === undefined) {\n let errorMessage = `rootRouteNode must not be undefined. Make sure you've added your root route into the route-tree.`\n if (!config.virtualRouteConfig) {\n errorMessage += `\\nMake sure that you add a \"${rootPathId}.tsx\" file to your routes directory.\\nAdd the file in: \"${config.routesDirectory}/${rootPathId}.tsx\"`\n }\n throw new Error(errorMessage)\n }\n\n const preRouteNodes = multiSortBy(beforeRouteNodes, [\n (d) => (d.routePath === '/' ? -1 : 1),\n (d) => d.routePath?.split('/').length,\n (d) =>\n d.filePath.match(new RegExp(`[./]${config.indexToken}[.]`)) ? 1 : -1,\n (d) =>\n d.filePath.match(\n /[./](component|errorComponent|pendingComponent|loader|lazy)[.]/,\n )\n ? 1\n : -1,\n (d) =>\n d.filePath.match(new RegExp(`[./]${config.routeToken}[.]`)) ? -1 : 1,\n (d) => (d.routePath?.endsWith('/') ? -1 : 1),\n (d) => d.routePath,\n ]).filter((d) => ![`/${rootPathId}`].includes(d.routePath || ''))\n\n const routeTree: Array<RouteNode> = []\n\n // Loop over the flat list of routeNodes and\n // build up a tree based on the routeNodes' routePath\n const routeNodes: Array<RouteNode> = []\n\n // the handleRootNode function is not being collapsed into the handleNode function\n // because it requires only a subset of the logic that the handleNode function requires\n // and it's easier to read and maintain this way\n const handleRootNode = async (node?: RouteNode) => {\n if (!node) {\n // currently this is not being handled, but it could be in the future\n // for example to handle a virtual root route\n return\n }\n\n // from here on, we are only handling the root node that's present in the file system\n const routeCode = fs.readFileSync(node.fullPath, 'utf-8')\n\n if (!routeCode) {\n const _rootTemplate = ROUTE_TEMPLATE.rootRoute\n const replaced = await fillTemplate(config, _rootTemplate.template(), {\n tsrImports: _rootTemplate.imports.tsrImports(),\n tsrPath: rootPathId,\n tsrExportStart: _rootTemplate.imports.tsrExportStart(),\n tsrExportEnd: _rootTemplate.imports.tsrExportEnd(),\n })\n\n await writeIfDifferent(\n node.fullPath,\n '', // Empty string because the file doesn't exist yet\n replaced,\n {\n beforeWrite: () => {\n // logger.log(`🟡 Creating ${node.fullPath}`)\n },\n },\n )\n }\n }\n\n await handleRootNode(rootRouteNode)\n\n const handleNode = async (node: RouteNode) => {\n // Do not remove this as we need to set the lastIndex to 0 as it\n // is necessary to reset the regex's index when using the global flag\n // otherwise it might not match the next time it's used\n resetRegex(routeGroupPatternRegex)\n\n let parentRoute = hasParentRoute(routeNodes, node, node.routePath)\n\n // if the parent route is a virtual parent route, we need to find the real parent route\n if (parentRoute?.isVirtualParentRoute && parentRoute.children?.length) {\n // only if this sub-parent route returns a valid parent route, we use it, if not leave it as it\n const possibleParentRoute = hasParentRoute(\n parentRoute.children,\n node,\n node.routePath,\n )\n if (possibleParentRoute) {\n parentRoute = possibleParentRoute\n }\n }\n\n if (parentRoute) node.parent = parentRoute\n\n node.path = determineNodePath(node)\n\n const trimmedPath = trimPathLeft(node.path ?? '')\n\n const split = trimmedPath.split('/')\n const lastRouteSegment = split[split.length - 1] ?? trimmedPath\n\n node.isNonPath =\n lastRouteSegment.startsWith('_') ||\n routeGroupPatternRegex.test(lastRouteSegment)\n\n node.cleanedPath = removeGroups(\n removeUnderscores(removeLayoutSegments(node.path)) ?? '',\n )\n\n const routeCode = node.fullPath\n ? fs.readFileSync(node.fullPath, 'utf-8')\n : ''\n\n // Ensure the boilerplate for the route exists, which can be skipped for virtual parent routes and virtual routes\n if (!node.isVirtualParentRoute && !node.isVirtual) {\n // const escapedRoutePath = node.routePath?.replaceAll('$', '$$') ?? ''\n // let replaced = routeCode\n // await writeIfDifferent(node.fullPath, routeCode, replaced, {\n // beforeWrite: () => {\n // // logger.log(`🟡 Updating ${node.fullPath}`)\n // },\n // })\n }\n\n const cleanedPathIsEmpty = (node.cleanedPath || '').length === 0\n const nonPathRoute =\n node._fsRouteType === 'pathless_layout' && node.isNonPath\n\n node.isVirtualParentRequired =\n node._fsRouteType === 'pathless_layout' || nonPathRoute\n ? !cleanedPathIsEmpty\n : false\n\n if (!node.isVirtual && node.isVirtualParentRequired) {\n const parentRoutePath = removeLastSegmentFromPath(node.routePath) || '/'\n const parentVariableName = routePathToVariable(parentRoutePath)\n\n const anchorRoute = routeNodes.find(\n (d) => d.routePath === parentRoutePath,\n )\n\n if (!anchorRoute) {\n const parentNode: RouteNode = {\n ...node,\n path: removeLastSegmentFromPath(node.path) || '/',\n filePath: removeLastSegmentFromPath(node.filePath) || '/',\n fullPath: removeLastSegmentFromPath(node.fullPath) || '/',\n routePath: parentRoutePath,\n variableName: parentVariableName,\n isVirtual: true,\n _fsRouteType: 'layout', // layout since this route will wrap other routes\n isVirtualParentRoute: true,\n isVirtualParentRequired: false,\n }\n\n parentNode.children = parentNode.children ?? []\n parentNode.children.push(node)\n\n node.parent = parentNode\n\n if (node._fsRouteType === 'pathless_layout') {\n // since `node.path` is used as the `id` on the route definition, we need to update it\n node.path = determineNodePath(node)\n }\n\n await handleNode(parentNode)\n } else {\n anchorRoute.children = anchorRoute.children ?? []\n anchorRoute.children.push(node)\n\n node.parent = anchorRoute\n }\n }\n\n if (\n !routeCode\n .split('\\n')\n .some((line) => line.trim().startsWith('export const ServerRoute'))\n ) {\n return\n }\n\n if (node.parent) {\n if (!node.isVirtualParentRequired) {\n node.parent.children = node.parent.children ?? []\n node.parent.children.push(node)\n }\n } else {\n routeTree.push(node)\n }\n\n routeNodes.push(node)\n }\n\n for (const node of preRouteNodes) {\n await handleNode(node)\n }\n\n // This is run against the `routeNodes` array since it\n // has the accumulated (intended) Server Route nodes\n // Since TSR allows multiple way of defining a route,\n // we need to ensure that a user hasn't defined the\n // same route in multiple ways (i.e. `flat`, `nested`, `virtual`)\n checkRouteFullPathUniqueness(routeNodes, config)\n\n function buildRouteTreeConfig(nodes: Array<RouteNode>, depth = 1): string {\n const children = nodes.map((node) => {\n if (node._fsRouteType === '__root') {\n return\n }\n\n if (node._fsRouteType === 'pathless_layout' && !node.children?.length) {\n return\n }\n\n const route = `${node.variableName}Route`\n\n if (node.children?.length) {\n const childConfigs = buildRouteTreeConfig(node.children, depth + 1)\n\n const childrenDeclaration = `interface ${route}Children {\n ${node.children.map((child) => `${child.variableName}Route: typeof ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`\n\n const children = `const ${route}Children: ${route}Children = {\n ${node.children.map((child) => `${child.variableName}Route: ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`\n\n const routeWithChildren = `const ${route}WithChildren = ${route}._addFileChildren(${route}Children)`\n\n return [\n childConfigs,\n childrenDeclaration,\n children,\n routeWithChildren,\n ].join('\\n\\n')\n }\n\n return undefined\n })\n\n return children.filter(Boolean).join('\\n\\n')\n }\n\n const routeConfigChildrenText = buildRouteTreeConfig(routeTree)\n\n const sortedRouteNodes = multiSortBy(routeNodes, [\n (d) => (d.routePath?.includes(`/${rootPathId}`) ? -1 : 1),\n (d) => d.routePath?.split('/').length,\n (d) => (d.routePath?.endsWith(config.indexToken) ? -1 : 1),\n (d) => d,\n ])\n\n const imports = Object.entries({\n createFileRoute: sortedRouteNodes.some((d) => d.isVirtual),\n })\n .filter((d) => d[1])\n .map((d) => d[0])\n\n const virtualRouteNodes = sortedRouteNodes.filter((d) => d.isVirtual)\n\n function getImportPath(node: RouteNode) {\n return replaceBackslash(\n removeExt(\n path.relative(\n path.dirname(generatedServerRouteTreePath),\n path.resolve(config.routesDirectory, node.filePath),\n ),\n ),\n )\n }\n\n const rootRouteExists = fs.existsSync(rootRouteNode.fullPath)\n const rootRouteCode = rootRouteExists\n ? fs.readFileSync(rootRouteNode.fullPath, 'utf-8')\n : ''\n const hasServerRootRoute =\n rootRouteExists && rootRouteCode.includes('export const ServerRoute')\n\n const routeImports = [\n ...config.routeTreeFileHeader,\n `// This file was automatically generated by TanStack Router.\n// You should NOT make any changes in this file as it will be overwritten.\n// Additionally, you should also exclude this file from your linter and/or formatter to prevent it from being checked or modified.`,\n imports.length\n ? `import { ${imports.join(', ')} } from '${ROUTE_TEMPLATE.fullPkg}'\\n`\n : '',\n '// Import Routes',\n [\n `import type { FileRoutesByPath, CreateServerFileRoute } from '${ROUTE_TEMPLATE.fullPkg}'`,\n `import { createServerRoute, createServerFileRoute } from '${ROUTE_TEMPLATE.fullPkg}'`,\n hasServerRootRoute\n ? `import { ServerRoute as rootRouteImport } from './${getImportPath(rootRouteNode)}'`\n : '',\n ...sortedRouteNodes\n .filter((d) => !d.isVirtual)\n .map((node) => {\n return `import { ServerRoute as ${\n node.variableName\n }RouteImport } from './${getImportPath(node)}'`\n }),\n ].join('\\n'),\n virtualRouteNodes.length ? '// Create Virtual Routes' : '',\n virtualRouteNodes\n .map((node) => {\n return `const ${\n node.variableName\n }RouteImport = createFileRoute('${node.routePath}')()`\n })\n .join('\\n'),\n '// Create/Update Routes',\n !hasServerRootRoute\n ? `\n const rootRoute = createServerRoute()\n `\n : '',\n sortedRouteNodes\n .map((node) => {\n return [\n [\n `const ${node.variableName}Route = ${node.variableName}RouteImport.update({\n ${[\n `id: '${node.path}'`,\n !node.isNonPath ? `path: '${node.cleanedPath}'` : undefined,\n `getParentRoute: () => ${node.parent?.variableName ?? 'root'}Route`,\n ]\n .filter(Boolean)\n .join(',')}\n } as any)`,\n ].join(''),\n ].join('\\n\\n')\n })\n .join('\\n\\n'),\n '',\n\n '// Populate the FileRoutesByPath interface',\n `declare module '${ROUTE_TEMPLATE.fullPkg}' {\n interface FileRoutesByPath {\n ${routeNodes\n .map((routeNode) => {\n const filePathId = routeNode.routePath\n\n return `'${filePathId}': {\n id: '${filePathId}'\n path: '${inferPath(routeNode)}'\n fullPath: '${inferFullPath(routeNode)}'\n preLoaderRoute: typeof ${routeNode.variableName}RouteImport\n parentRoute: typeof ${\n routeNode.isVirtualParentRequired\n ? `${routeNode.parent?.variableName}Route`\n : routeNode.parent?.variableName\n ? `${routeNode.parent.variableName}RouteImport`\n : 'rootRoute'\n }\n }`\n })\n .join('\\n')}\n }\n}`,\n `// Add type-safety to the createFileRoute function across the route tree`,\n routeNodes\n .map((routeNode) => {\n return `declare module './${getImportPath(routeNode)}' {\nconst createServerFileRoute: CreateServerFileRoute<\nFileRoutesByPath['${routeNode.routePath}']['parentRoute'],\nFileRoutesByPath['${routeNode.routePath}']['id'],\nFileRoutesByPath['${routeNode.routePath}']['path'],\nFileRoutesByPath['${routeNode.routePath}']['fullPath'],\n${routeNode.children?.length ? `${routeNode.variableName}RouteChildren` : 'unknown'}\n>\n}`\n })\n .join('\\n'),\n '// Create and export the route tree',\n routeConfigChildrenText,\n `export interface FileRoutesByFullPath {\n ${[...createRouteNodesByFullPath(routeNodes).entries()].map(\n ([fullPath, routeNode]) => {\n return `'${fullPath}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n },\n )}\n}`,\n `export interface FileRoutesByTo {\n ${[...createRouteNodesByTo(routeNodes).entries()].map(([to, routeNode]) => {\n return `'${to}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n })}\n}`,\n `export interface FileRoutesById {\n '${rootRouteId}': typeof rootRoute,\n ${[...createRouteNodesById(routeNodes).entries()].map(([id, routeNode]) => {\n return `'${id}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n })}\n}`,\n `export interface FileRouteTypes {\n fileRoutesByFullPath: FileRoutesByFullPath\n fullPaths: ${routeNodes.length > 0 ? [...createRouteNodesByFullPath(routeNodes).keys()].map((fullPath) => `'${fullPath}'`).join('|') : 'never'}\n fileRoutesByTo: FileRoutesByTo\n to: ${routeNodes.length > 0 ? [...createRouteNodesByTo(routeNodes).keys()].map((to) => `'${to}'`).join('|') : 'never'}\n id: ${[`'${rootRouteId}'`, ...[...createRouteNodesById(routeNodes).keys()].map((id) => `'${id}'`)].join('|')}\n fileRoutesById: FileRoutesById\n}`,\n `export interface RootRouteChildren {\n ${routeTree.map((child) => `${child.variableName}Route: typeof ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`,\n `const rootRouteChildren: RootRouteChildren = {\n ${routeTree.map((child) => `${child.variableName}Route: ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`,\n `export const routeTree = rootRoute._addFileChildren(rootRouteChildren)._addFileTypes<FileRouteTypes>()`,\n ]\n .filter(Boolean)\n .join('\\n\\n')\n\n const createRouteManifest = () => {\n const routesManifest = {\n [rootRouteId]: {\n filePath: rootRouteNode.filePath,\n children: routeTree.map((d) => d.routePath),\n },\n ...Object.fromEntries(\n routeNodes.map((d) => {\n const filePathId = d.routePath\n\n return [\n filePathId,\n {\n filePath: d.filePath,\n parent: d.parent?.routePath ? d.parent.routePath : undefined,\n children: d.children?.map((childRoute) => childRoute.routePath),\n },\n ]\n }),\n ),\n }\n\n return JSON.stringify(\n {\n routes: routesManifest,\n },\n null,\n 2,\n )\n }\n\n const includeManifest = ['react', 'solid']\n const routeConfigFileContent = !includeManifest.includes(config.target)\n ? routeImports\n : [\n routeImports,\n '\\n',\n '/* ROUTE_MANIFEST_START',\n createRouteManifest(),\n 'ROUTE_MANIFEST_END */',\n ].join('\\n')\n\n if (!checkLatest()) return\n\n const existingRouteTreeContent = await fsp\n .readFile(path.resolve(generatedServerRouteTreePath), 'utf-8')\n .catch((err) => {\n if (err.code === 'ENOENT') {\n return ''\n }\n\n throw err\n })\n\n if (!checkLatest()) return\n\n // Ensure the directory exists\n await fsp.mkdir(path.dirname(path.resolve(generatedServerRouteTreePath)), {\n recursive: true,\n })\n\n if (!checkLatest()) return\n\n // Write the route tree file, if it has changed\n const routeTreeWriteResult = await writeIfDifferent(\n path.resolve(generatedServerRouteTreePath),\n await format(existingRouteTreeContent, config),\n await format(routeConfigFileContent, config),\n {\n beforeWrite: () => {\n // logger.log(`🟡 Updating ${generatedRouteTreePath}`)\n },\n },\n )\n\n // Write declaration file\n const startDeclarationFilePath = path.join(\n path.resolve(root, config.srcDirectory),\n 'tanstack-start.d.ts',\n )\n const serverRoutesRelativePath = removeExt(\n path.relative(\n path.dirname(startDeclarationFilePath),\n generatedServerRouteTreePath,\n ),\n )\n const startDeclarationFileContent = buildStartDeclarationFile({\n serverRoutesRelativePath,\n })\n if (!fs.existsSync(startDeclarationFilePath)) {\n await writeIfDifferent(\n startDeclarationFilePath,\n '',\n startDeclarationFileContent,\n {\n beforeWrite: () => {\n logger.log(`🟡 Creating tanstack-start.d.ts`)\n },\n },\n )\n } else {\n const existingDeclarationFileContent = await fsp\n .readFile(startDeclarationFilePath, 'utf-8')\n .catch((err) => {\n if (err.code === 'ENOENT') {\n return ''\n }\n throw err\n })\n await writeIfDifferent(\n startDeclarationFilePath,\n existingDeclarationFileContent,\n startDeclarationFileContent,\n {\n beforeWrite: () => {\n logger.log(`🟡 Updating tanstack-start.d.ts`)\n },\n },\n )\n }\n\n if (routeTreeWriteResult && !checkLatest()) {\n return\n }\n\n // logger.log(\n // `✅ Processed ${routeNodes.length === 1 ? 'server route' : 'server routes'} in ${\n // Date.now() - start\n // }ms`,\n // )\n}\n\nfunction buildStartDeclarationFile({\n serverRoutesRelativePath,\n}: {\n serverRoutesRelativePath: string\n}) {\n const serverRoutesPath = replaceBackslash(serverRoutesRelativePath)\n return (\n [\n '/// <reference types=\"vite/client\" />',\n `import '${serverRoutesPath}'`,\n ].join('\\n') + '\\n'\n )\n}\n\nfunction removeGroups(s: string) {\n return s.replace(possiblyNestedRouteGroupPatternRegex, '')\n}\n\n/**\n * The `node.path` is used as the `id` in the route definition.\n * This function checks if the given node has a parent and if so, it determines the correct path for the given node.\n * @param node - The node to determine the path for.\n * @returns The correct path for the given node.\n */\nfunction determineNodePath(node: RouteNode) {\n return (node.path = node.parent\n ? node.routePath?.replace(node.parent.routePath ?? '', '') || '/'\n : node.routePath)\n}\n\n/**\n * Removes the last segment from a given path. Segments are considered to be separated by a '/'.\n *\n * @param {string} routePath - The path from which to remove the last segment. Defaults to '/'.\n * @returns {string} The path with the last segment removed.\n * @example\n * removeLastSegmentFromPath('/workspace/_auth/foo') // '/workspace/_auth'\n */\nfunction removeLastSegmentFromPath(routePath: string = '/'): string {\n const segments = routePath.split('/')\n segments.pop() // Remove the last segment\n return segments.join('/')\n}\n\n/**\n * Removes all segments from a given path that start with an underscore ('_').\n *\n * @param {string} routePath - The path from which to remove segments. Defaults to '/'.\n * @returns {string} The path with all underscore-prefixed segments removed.\n * @example\n * removeLayoutSegments('/workspace/_auth/foo') // '/workspace/foo'\n */\nfunction removeLayoutSegments(routePath: string = '/'): string {\n const segments = routePath.split('/')\n const newSegments = segments.filter((segment) => !segment.startsWith('_'))\n return newSegments.join('/')\n}\n\nfunction hasParentRoute(\n routes: Array<RouteNode>,\n node: RouteNode,\n routePathToCheck: string | undefined,\n): RouteNode | null {\n if (!routePathToCheck || routePathToCheck === '/') {\n return null\n }\n\n const sortedNodes = multiSortBy(routes, [\n (d) => d.routePath!.length * -1,\n (d) => d.variableName,\n ]).filter((d) => d.routePath !== `/${rootPathId}`)\n\n for (const route of sortedNodes) {\n if (route.routePath === '/') continue\n\n if (\n routePathToCheck.startsWith(`${route.routePath}/`) &&\n route.routePath !== routePathToCheck\n ) {\n return route\n }\n }\n\n const segments = routePathToCheck.split('/')\n segments.pop() // Remove the last segment\n const parentRoutePath = segments.join('/')\n\n return hasParentRoute(routes, node, parentRoutePath)\n}\n\n/**\n * Gets the final variable name for a route\n */\nconst getResolvedRouteNodeVariableName = (routeNode: RouteNode): string => {\n return routeNode.children?.length\n ? `${routeNode.variableName}RouteWithChildren`\n : `${routeNode.variableName}Route`\n}\n\n/**\n * Creates a map from fullPath to routeNode\n */\nconst createRouteNodesByFullPath = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n routeNodes.map((routeNode) => [inferFullPath(routeNode), routeNode]),\n )\n}\n\n/**\n * Create a map from 'to' to a routeNode\n */\nconst createRouteNodesByTo = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n dedupeBranchesAndIndexRoutes(routeNodes).map((routeNode) => [\n inferTo(routeNode),\n routeNode,\n ]),\n )\n}\n\n/**\n * Create a map from 'id' to a routeNode\n */\nconst createRouteNodesById = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n routeNodes.map((routeNode) => {\n const id = routeNode.routePath ?? ''\n return [id, routeNode]\n }),\n )\n}\n\n/**\n * Infers the full path for use by TS\n */\nconst inferFullPath = (routeNode: RouteNode): string => {\n const fullPath = removeGroups(\n removeUnderscores(removeLayoutSegments(routeNode.routePath)) ?? '',\n )\n\n return routeNode.cleanedPath === '/' ? fullPath : fullPath.replace(/\\/$/, '')\n}\n\n/**\n * Infers the path for use by TS\n */\nconst inferPath = (routeNode: RouteNode): string => {\n return routeNode.cleanedPath === '/'\n ? routeNode.cleanedPath\n : (routeNode.cleanedPath?.replace(/\\/$/, '') ?? '')\n}\n\n/**\n * Infers to path\n */\nconst inferTo = (routeNode: RouteNode): string => {\n const fullPath = inferFullPath(routeNode)\n\n if (fullPath === '/') return fullPath\n\n return fullPath.replace(/\\/$/, '')\n}\n\n/**\n * Dedupes branches and index routes\n */\nconst dedupeBranchesAndIndexRoutes = (\n routes: Array<RouteNode>,\n): Array<RouteNode> => {\n return routes.filter((route) => {\n if (route.children?.find((child) => child.cleanedPath === '/')) return false\n return true\n })\n}\n\nfunction checkUnique<TElement>(routes: Array<TElement>, key: keyof TElement) {\n // Check no two routes have the same `key`\n // if they do, throw an error with the conflicting filePaths\n const keys = routes.map((d) => d[key])\n const uniqueKeys = new Set(keys)\n if (keys.length !== uniqueKeys.size) {\n const duplicateKeys = keys.filter((d, i) => keys.indexOf(d) !== i)\n const conflictingFiles = routes.filter((d) =>\n duplicateKeys.includes(d[key]),\n )\n return conflictingFiles\n }\n return undefined\n}\n\nfunction checkRouteFullPathUniqueness(\n _routes: Array<RouteNode>,\n config: Config,\n) {\n const routes = _routes.map((d) => {\n const inferredFullPath = inferFullPath(d)\n return { ...d, inferredFullPath }\n })\n\n const conflictingFiles = checkUnique(routes, 'inferredFullPath')\n\n if (conflictingFiles !== undefined) {\n const errorMessage = `Conflicting configuration paths were found for the following route${conflictingFiles.length > 1 ? 's' : ''}: ${conflictingFiles\n .map((p) => `\"${p.inferredFullPath}\"`)\n .join(', ')}.\nPlease ensure each Server Route has a unique full path.\nConflicting files: \\n ${conflictingFiles.map((d) => path.resolve(config.routesDirectory, d.filePath)).join('\\n ')}\\n`\n console.error(errorMessage)\n process.exit(1)\n }\n}\n"],"names":["path","config","children"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"plugin.js","sources":["../../../src/start-server-routes-plugin/plugin.ts"],"sourcesContent":["import path, { isAbsolute, join, normalize } from 'node:path'\nimport fs from 'node:fs'\nimport fsp from 'node:fs/promises'\nimport {\n format,\n logging,\n multiSortBy,\n physicalGetRouteNodes,\n removeExt,\n removeUnderscores,\n replaceBackslash,\n resetRegex,\n rootPathId,\n routePathToVariable,\n trimPathLeft,\n virtualGetRouteNodes,\n writeIfDifferent,\n} from '@tanstack/router-generator'\nimport { rootRouteId } from '@tanstack/router-core'\nimport { fillTemplate, getTargetTemplate } from './template'\nimport type { GetRouteNodesResult, RouteNode } from '@tanstack/router-generator'\nimport type { Config } from './config'\nimport type { Plugin } from 'vite'\n\nlet lock = false\nconst checkLock = () => lock\nconst setLock = (bool: boolean) => {\n lock = bool\n}\n\nexport function TanStackStartServerRoutesVite(config: Config): Plugin {\n let ROOT: string = process.cwd()\n\n const getRoutesDirectoryPath = () => {\n return isAbsolute(config.routesDirectory)\n ? config.routesDirectory\n : join(ROOT, config.routesDirectory)\n }\n\n const generate = async () => {\n if (checkLock()) {\n return\n }\n\n setLock(true)\n\n try {\n await generator(config, ROOT)\n } catch (err) {\n console.error(err)\n console.info()\n } finally {\n setLock(false)\n }\n }\n\n const handleFile = async (file: string) => {\n const filePath = normalize(file)\n\n const routesDirectoryPath = getRoutesDirectoryPath()\n if (filePath.startsWith(routesDirectoryPath)) {\n await generate()\n }\n }\n\n return {\n name: 'tanstack-start-server-routes-plugin',\n configureServer(server) {\n server.watcher.on('all', (event, path) => {\n handleFile(path)\n })\n },\n configResolved(config) {\n ROOT = config.root\n },\n async buildStart() {\n await generate()\n // if (this.environment.name === 'server') {\n // }\n },\n sharedDuringBuild: true,\n resolveId(id) {\n if (id === 'tanstack:server-routes') {\n const generatedRouteTreePath = getGeneratedRouteTreePath(ROOT)\n return generatedRouteTreePath\n }\n return null\n },\n }\n}\n\n// Maybe import this from `@tanstack/router-core` in the future???\nlet latestTask = 0\nconst routeGroupPatternRegex = /\\(.+\\)/g\nconst possiblyNestedRouteGroupPatternRegex = /\\([^/]+\\)\\/?/g\n\nlet isFirst = false\nlet skipMessage = false\n\nfunction getGeneratedRouteTreePath(root: string) {\n return path.resolve(root, '.tanstack-start/server-routes/routeTree.gen.ts')\n}\n\nasync function generator(config: Config, root: string) {\n const generatedServerRouteTreePath = getGeneratedRouteTreePath(root)\n const ROUTE_TEMPLATE = getTargetTemplate(config.target)\n const logger = logging({ disabled: config.disableLogging })\n\n if (!isFirst) {\n // logger.log('♻️ Generating server routes...')\n isFirst = true\n } else if (skipMessage) {\n skipMessage = false\n } else {\n // logger.log('♻️ Regenerating server routes...')\n }\n\n const taskId = latestTask + 1\n latestTask = taskId\n\n const checkLatest = () => {\n if (latestTask !== taskId) {\n skipMessage = true\n return false\n }\n\n return true\n }\n\n const start = Date.now()\n\n let getRouteNodesResult: GetRouteNodesResult\n\n if (config.virtualRouteConfig) {\n getRouteNodesResult = await virtualGetRouteNodes(config, root)\n } else {\n getRouteNodesResult = await physicalGetRouteNodes(config, root)\n }\n\n const { rootRouteNode, routeNodes: beforeRouteNodes } = getRouteNodesResult\n if (rootRouteNode === undefined) {\n let errorMessage = `rootRouteNode must not be undefined. Make sure you've added your root route into the route-tree.`\n if (!config.virtualRouteConfig) {\n errorMessage += `\\nMake sure that you add a \"${rootPathId}.tsx\" file to your routes directory.\\nAdd the file in: \"${config.routesDirectory}/${rootPathId}.tsx\"`\n }\n throw new Error(errorMessage)\n }\n\n const preRouteNodes = multiSortBy(beforeRouteNodes, [\n (d) => (d.routePath === '/' ? -1 : 1),\n (d) => d.routePath?.split('/').length,\n (d) =>\n d.filePath.match(new RegExp(`[./]${config.indexToken}[.]`)) ? 1 : -1,\n (d) =>\n d.filePath.match(\n /[./](component|errorComponent|pendingComponent|loader|lazy)[.]/,\n )\n ? 1\n : -1,\n (d) =>\n d.filePath.match(new RegExp(`[./]${config.routeToken}[.]`)) ? -1 : 1,\n (d) => (d.routePath?.endsWith('/') ? -1 : 1),\n (d) => d.routePath,\n ]).filter((d) => ![`/${rootPathId}`].includes(d.routePath || ''))\n\n const routeTree: Array<RouteNode> = []\n\n // Loop over the flat list of routeNodes and\n // build up a tree based on the routeNodes' routePath\n const routeNodes: Array<RouteNode> = []\n\n // the handleRootNode function is not being collapsed into the handleNode function\n // because it requires only a subset of the logic that the handleNode function requires\n // and it's easier to read and maintain this way\n const handleRootNode = async (node?: RouteNode) => {\n if (!node) {\n // currently this is not being handled, but it could be in the future\n // for example to handle a virtual root route\n return\n }\n\n // from here on, we are only handling the root node that's present in the file system\n const routeCode = fs.readFileSync(node.fullPath, 'utf-8')\n\n if (!routeCode) {\n const _rootTemplate = ROUTE_TEMPLATE.rootRoute\n const replaced = await fillTemplate(config, _rootTemplate.template(), {\n tsrImports: _rootTemplate.imports.tsrImports(),\n tsrPath: rootPathId,\n tsrExportStart: _rootTemplate.imports.tsrExportStart(),\n tsrExportEnd: _rootTemplate.imports.tsrExportEnd(),\n })\n\n await writeIfDifferent(\n node.fullPath,\n '', // Empty string because the file doesn't exist yet\n replaced,\n {\n beforeWrite: () => {\n // logger.log(`🟡 Creating ${node.fullPath}`)\n },\n },\n )\n }\n }\n\n await handleRootNode(rootRouteNode)\n\n const handleNode = async (node: RouteNode) => {\n // Do not remove this as we need to set the lastIndex to 0 as it\n // is necessary to reset the regex's index when using the global flag\n // otherwise it might not match the next time it's used\n resetRegex(routeGroupPatternRegex)\n\n let parentRoute = hasParentRoute(routeNodes, node, node.routePath)\n\n // if the parent route is a virtual parent route, we need to find the real parent route\n if (parentRoute?.isVirtualParentRoute && parentRoute.children?.length) {\n // only if this sub-parent route returns a valid parent route, we use it, if not leave it as it\n const possibleParentRoute = hasParentRoute(\n parentRoute.children,\n node,\n node.routePath,\n )\n if (possibleParentRoute) {\n parentRoute = possibleParentRoute\n }\n }\n\n if (parentRoute) node.parent = parentRoute\n\n node.path = determineNodePath(node)\n\n const trimmedPath = trimPathLeft(node.path ?? '')\n\n const split = trimmedPath.split('/')\n const lastRouteSegment = split[split.length - 1] ?? trimmedPath\n\n node.isNonPath =\n lastRouteSegment.startsWith('_') ||\n routeGroupPatternRegex.test(lastRouteSegment)\n\n node.cleanedPath = removeGroups(\n removeUnderscores(removeLayoutSegments(node.path)) ?? '',\n )\n\n const routeCode = fs.readFileSync(node.fullPath, 'utf-8')\n\n // Ensure the boilerplate for the route exists, which can be skipped for virtual parent routes and virtual routes\n if (!node.isVirtualParentRoute && !node.isVirtual) {\n // const escapedRoutePath = node.routePath?.replaceAll('$', '$$') ?? ''\n // let replaced = routeCode\n // await writeIfDifferent(node.fullPath, routeCode, replaced, {\n // beforeWrite: () => {\n // // logger.log(`🟡 Updating ${node.fullPath}`)\n // },\n // })\n }\n\n const cleanedPathIsEmpty = (node.cleanedPath || '').length === 0\n const nonPathRoute =\n node._fsRouteType === 'pathless_layout' && node.isNonPath\n\n node.isVirtualParentRequired =\n node._fsRouteType === 'pathless_layout' || nonPathRoute\n ? !cleanedPathIsEmpty\n : false\n\n if (!node.isVirtual && node.isVirtualParentRequired) {\n const parentRoutePath = removeLastSegmentFromPath(node.routePath) || '/'\n const parentVariableName = routePathToVariable(parentRoutePath)\n\n const anchorRoute = routeNodes.find(\n (d) => d.routePath === parentRoutePath,\n )\n\n if (!anchorRoute) {\n const parentNode: RouteNode = {\n ...node,\n path: removeLastSegmentFromPath(node.path) || '/',\n filePath: removeLastSegmentFromPath(node.filePath) || '/',\n fullPath: removeLastSegmentFromPath(node.fullPath) || '/',\n routePath: parentRoutePath,\n variableName: parentVariableName,\n isVirtual: true,\n _fsRouteType: 'layout', // layout since this route will wrap other routes\n isVirtualParentRoute: true,\n isVirtualParentRequired: false,\n }\n\n parentNode.children = parentNode.children ?? []\n parentNode.children.push(node)\n\n node.parent = parentNode\n\n if (node._fsRouteType === 'pathless_layout') {\n // since `node.path` is used as the `id` on the route definition, we need to update it\n node.path = determineNodePath(node)\n }\n\n await handleNode(parentNode)\n } else {\n anchorRoute.children = anchorRoute.children ?? []\n anchorRoute.children.push(node)\n\n node.parent = anchorRoute\n }\n }\n\n if (\n !routeCode\n .split('\\n')\n .some((line) => line.trim().startsWith('export const ServerRoute'))\n ) {\n return\n }\n\n if (node.parent) {\n if (!node.isVirtualParentRequired) {\n node.parent.children = node.parent.children ?? []\n node.parent.children.push(node)\n }\n } else {\n routeTree.push(node)\n }\n\n routeNodes.push(node)\n }\n\n for (const node of preRouteNodes) {\n await handleNode(node)\n }\n\n // This is run against the `routeNodes` array since it\n // has the accumulated (intended) Server Route nodes\n // Since TSR allows multiple way of defining a route,\n // we need to ensure that a user hasn't defined the\n // same route in multiple ways (i.e. `flat`, `nested`, `virtual`)\n checkRouteFullPathUniqueness(routeNodes, config)\n\n function buildRouteTreeConfig(nodes: Array<RouteNode>, depth = 1): string {\n const children = nodes.map((node) => {\n if (node._fsRouteType === '__root') {\n return\n }\n\n if (node._fsRouteType === 'pathless_layout' && !node.children?.length) {\n return\n }\n\n const route = `${node.variableName}Route`\n\n if (node.children?.length) {\n const childConfigs = buildRouteTreeConfig(node.children, depth + 1)\n\n const childrenDeclaration = `interface ${route}Children {\n ${node.children.map((child) => `${child.variableName}Route: typeof ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`\n\n const children = `const ${route}Children: ${route}Children = {\n ${node.children.map((child) => `${child.variableName}Route: ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`\n\n const routeWithChildren = `const ${route}WithChildren = ${route}._addFileChildren(${route}Children)`\n\n return [\n childConfigs,\n childrenDeclaration,\n children,\n routeWithChildren,\n ].join('\\n\\n')\n }\n\n return undefined\n })\n\n return children.filter(Boolean).join('\\n\\n')\n }\n\n const routeConfigChildrenText = buildRouteTreeConfig(routeTree)\n\n const sortedRouteNodes = multiSortBy(routeNodes, [\n (d) => (d.routePath?.includes(`/${rootPathId}`) ? -1 : 1),\n (d) => d.routePath?.split('/').length,\n (d) => (d.routePath?.endsWith(config.indexToken) ? -1 : 1),\n (d) => d,\n ])\n\n const imports = Object.entries({\n createFileRoute: sortedRouteNodes.some((d) => d.isVirtual),\n })\n .filter((d) => d[1])\n .map((d) => d[0])\n\n const virtualRouteNodes = sortedRouteNodes.filter((d) => d.isVirtual)\n\n function getImportPath(node: RouteNode) {\n return replaceBackslash(\n removeExt(\n path.relative(\n path.dirname(generatedServerRouteTreePath),\n path.resolve(config.routesDirectory, node.filePath),\n ),\n ),\n )\n }\n\n const rootRouteExists = fs.existsSync(rootRouteNode.fullPath)\n const rootRouteCode = rootRouteExists\n ? fs.readFileSync(rootRouteNode.fullPath, 'utf-8')\n : ''\n const hasServerRootRoute =\n rootRouteExists && rootRouteCode.includes('export const ServerRoute')\n\n const routeImports = [\n ...config.routeTreeFileHeader,\n `// This file was automatically generated by TanStack Router.\n// You should NOT make any changes in this file as it will be overwritten.\n// Additionally, you should also exclude this file from your linter and/or formatter to prevent it from being checked or modified.`,\n imports.length\n ? `import { ${imports.join(', ')} } from '${ROUTE_TEMPLATE.fullPkg}'\\n`\n : '',\n '// Import Routes',\n [\n `import type { FileRoutesByPath, CreateServerFileRoute } from '${ROUTE_TEMPLATE.fullPkg}'`,\n `import { createServerRoute, createServerFileRoute } from '${ROUTE_TEMPLATE.fullPkg}'`,\n hasServerRootRoute\n ? `import { ServerRoute as rootRouteImport } from './${getImportPath(rootRouteNode)}'`\n : '',\n ...sortedRouteNodes\n .filter((d) => !d.isVirtual)\n .map((node) => {\n return `import { ServerRoute as ${\n node.variableName\n }RouteImport } from './${getImportPath(node)}'`\n }),\n ].join('\\n'),\n virtualRouteNodes.length ? '// Create Virtual Routes' : '',\n virtualRouteNodes\n .map((node) => {\n return `const ${\n node.variableName\n }RouteImport = createFileRoute('${node.routePath}')()`\n })\n .join('\\n'),\n '// Create/Update Routes',\n !hasServerRootRoute\n ? `\n const rootRoute = createServerRoute()\n `\n : '',\n sortedRouteNodes\n .map((node) => {\n return [\n [\n `const ${node.variableName}Route = ${node.variableName}RouteImport.update({\n ${[\n `id: '${node.path}'`,\n !node.isNonPath ? `path: '${node.cleanedPath}'` : undefined,\n `getParentRoute: () => ${node.parent?.variableName ?? 'root'}Route`,\n ]\n .filter(Boolean)\n .join(',')}\n } as any)`,\n ].join(''),\n ].join('\\n\\n')\n })\n .join('\\n\\n'),\n '',\n\n '// Populate the FileRoutesByPath interface',\n `declare module '${ROUTE_TEMPLATE.fullPkg}' {\n interface FileRoutesByPath {\n ${routeNodes\n .map((routeNode) => {\n const filePathId = routeNode.routePath\n\n return `'${filePathId}': {\n id: '${filePathId}'\n path: '${inferPath(routeNode)}'\n fullPath: '${inferFullPath(routeNode)}'\n preLoaderRoute: typeof ${routeNode.variableName}RouteImport\n parentRoute: typeof ${\n routeNode.isVirtualParentRequired\n ? `${routeNode.parent?.variableName}Route`\n : routeNode.parent?.variableName\n ? `${routeNode.parent.variableName}RouteImport`\n : 'rootRoute'\n }\n }`\n })\n .join('\\n')}\n }\n}`,\n `// Add type-safety to the createFileRoute function across the route tree`,\n routeNodes\n .map((routeNode) => {\n return `declare module './${getImportPath(routeNode)}' {\nconst createServerFileRoute: CreateServerFileRoute<\nFileRoutesByPath['${routeNode.routePath}']['parentRoute'],\nFileRoutesByPath['${routeNode.routePath}']['id'],\nFileRoutesByPath['${routeNode.routePath}']['path'],\nFileRoutesByPath['${routeNode.routePath}']['fullPath'],\n${routeNode.children?.length ? `${routeNode.variableName}RouteChildren` : 'unknown'}\n>\n}`\n })\n .join('\\n'),\n '// Create and export the route tree',\n routeConfigChildrenText,\n `export interface FileRoutesByFullPath {\n ${[...createRouteNodesByFullPath(routeNodes).entries()].map(\n ([fullPath, routeNode]) => {\n return `'${fullPath}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n },\n )}\n}`,\n `export interface FileRoutesByTo {\n ${[...createRouteNodesByTo(routeNodes).entries()].map(([to, routeNode]) => {\n return `'${to}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n })}\n}`,\n `export interface FileRoutesById {\n '${rootRouteId}': typeof rootRoute,\n ${[...createRouteNodesById(routeNodes).entries()].map(([id, routeNode]) => {\n return `'${id}': typeof ${getResolvedRouteNodeVariableName(routeNode)}`\n })}\n}`,\n `export interface FileRouteTypes {\n fileRoutesByFullPath: FileRoutesByFullPath\n fullPaths: ${routeNodes.length > 0 ? [...createRouteNodesByFullPath(routeNodes).keys()].map((fullPath) => `'${fullPath}'`).join('|') : 'never'}\n fileRoutesByTo: FileRoutesByTo\n to: ${routeNodes.length > 0 ? [...createRouteNodesByTo(routeNodes).keys()].map((to) => `'${to}'`).join('|') : 'never'}\n id: ${[`'${rootRouteId}'`, ...[...createRouteNodesById(routeNodes).keys()].map((id) => `'${id}'`)].join('|')}\n fileRoutesById: FileRoutesById\n}`,\n `export interface RootRouteChildren {\n ${routeTree.map((child) => `${child.variableName}Route: typeof ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`,\n `const rootRouteChildren: RootRouteChildren = {\n ${routeTree.map((child) => `${child.variableName}Route: ${getResolvedRouteNodeVariableName(child)}`).join(',')}\n}`,\n `export const routeTree = rootRoute._addFileChildren(rootRouteChildren)._addFileTypes<FileRouteTypes>()`,\n ]\n .filter(Boolean)\n .join('\\n\\n')\n\n const createRouteManifest = () => {\n const routesManifest = {\n [rootRouteId]: {\n filePath: rootRouteNode.filePath,\n children: routeTree.map((d) => d.routePath),\n },\n ...Object.fromEntries(\n routeNodes.map((d) => {\n const filePathId = d.routePath\n\n return [\n filePathId,\n {\n filePath: d.filePath,\n parent: d.parent?.routePath ? d.parent.routePath : undefined,\n children: d.children?.map((childRoute) => childRoute.routePath),\n },\n ]\n }),\n ),\n }\n\n return JSON.stringify(\n {\n routes: routesManifest,\n },\n null,\n 2,\n )\n }\n\n const includeManifest = ['react', 'solid']\n const routeConfigFileContent = !includeManifest.includes(config.target)\n ? routeImports\n : [\n routeImports,\n '\\n',\n '/* ROUTE_MANIFEST_START',\n createRouteManifest(),\n 'ROUTE_MANIFEST_END */',\n ].join('\\n')\n\n if (!checkLatest()) return\n\n const existingRouteTreeContent = await fsp\n .readFile(path.resolve(generatedServerRouteTreePath), 'utf-8')\n .catch((err) => {\n if (err.code === 'ENOENT') {\n return ''\n }\n\n throw err\n })\n\n if (!checkLatest()) return\n\n // Ensure the directory exists\n await fsp.mkdir(path.dirname(path.resolve(generatedServerRouteTreePath)), {\n recursive: true,\n })\n\n if (!checkLatest()) return\n\n // Write the route tree file, if it has changed\n const routeTreeWriteResult = await writeIfDifferent(\n path.resolve(generatedServerRouteTreePath),\n await format(existingRouteTreeContent, config),\n await format(routeConfigFileContent, config),\n {\n beforeWrite: () => {\n // logger.log(`🟡 Updating ${generatedRouteTreePath}`)\n },\n },\n )\n\n // Write declaration file\n const startDeclarationFilePath = path.join(\n path.resolve(root, config.srcDirectory),\n 'tanstack-start.d.ts',\n )\n const serverRoutesRelativePath = removeExt(\n path.relative(\n path.dirname(startDeclarationFilePath),\n generatedServerRouteTreePath,\n ),\n )\n const startDeclarationFileContent = buildStartDeclarationFile({\n serverRoutesRelativePath,\n })\n if (!fs.existsSync(startDeclarationFilePath)) {\n await writeIfDifferent(\n startDeclarationFilePath,\n '',\n startDeclarationFileContent,\n {\n beforeWrite: () => {\n logger.log(`🟡 Creating tanstack-start.d.ts`)\n },\n },\n )\n } else {\n const existingDeclarationFileContent = await fsp\n .readFile(startDeclarationFilePath, 'utf-8')\n .catch((err) => {\n if (err.code === 'ENOENT') {\n return ''\n }\n throw err\n })\n await writeIfDifferent(\n startDeclarationFilePath,\n existingDeclarationFileContent,\n startDeclarationFileContent,\n {\n beforeWrite: () => {\n logger.log(`🟡 Updating tanstack-start.d.ts`)\n },\n },\n )\n }\n\n if (routeTreeWriteResult && !checkLatest()) {\n return\n }\n\n // logger.log(\n // `✅ Processed ${routeNodes.length === 1 ? 'server route' : 'server routes'} in ${\n // Date.now() - start\n // }ms`,\n // )\n}\n\nfunction buildStartDeclarationFile({\n serverRoutesRelativePath,\n}: {\n serverRoutesRelativePath: string\n}) {\n return [`import '${serverRoutesRelativePath}'`].join('\\n') + '\\n'\n}\n\nfunction removeGroups(s: string) {\n return s.replace(possiblyNestedRouteGroupPatternRegex, '')\n}\n\n/**\n * The `node.path` is used as the `id` in the route definition.\n * This function checks if the given node has a parent and if so, it determines the correct path for the given node.\n * @param node - The node to determine the path for.\n * @returns The correct path for the given node.\n */\nfunction determineNodePath(node: RouteNode) {\n return (node.path = node.parent\n ? node.routePath?.replace(node.parent.routePath ?? '', '') || '/'\n : node.routePath)\n}\n\n/**\n * Removes the last segment from a given path. Segments are considered to be separated by a '/'.\n *\n * @param {string} routePath - The path from which to remove the last segment. Defaults to '/'.\n * @returns {string} The path with the last segment removed.\n * @example\n * removeLastSegmentFromPath('/workspace/_auth/foo') // '/workspace/_auth'\n */\nfunction removeLastSegmentFromPath(routePath: string = '/'): string {\n const segments = routePath.split('/')\n segments.pop() // Remove the last segment\n return segments.join('/')\n}\n\n/**\n * Removes all segments from a given path that start with an underscore ('_').\n *\n * @param {string} routePath - The path from which to remove segments. Defaults to '/'.\n * @returns {string} The path with all underscore-prefixed segments removed.\n * @example\n * removeLayoutSegments('/workspace/_auth/foo') // '/workspace/foo'\n */\nfunction removeLayoutSegments(routePath: string = '/'): string {\n const segments = routePath.split('/')\n const newSegments = segments.filter((segment) => !segment.startsWith('_'))\n return newSegments.join('/')\n}\n\nfunction hasParentRoute(\n routes: Array<RouteNode>,\n node: RouteNode,\n routePathToCheck: string | undefined,\n): RouteNode | null {\n if (!routePathToCheck || routePathToCheck === '/') {\n return null\n }\n\n const sortedNodes = multiSortBy(routes, [\n (d) => d.routePath!.length * -1,\n (d) => d.variableName,\n ]).filter((d) => d.routePath !== `/${rootPathId}`)\n\n for (const route of sortedNodes) {\n if (route.routePath === '/') continue\n\n if (\n routePathToCheck.startsWith(`${route.routePath}/`) &&\n route.routePath !== routePathToCheck\n ) {\n return route\n }\n }\n\n const segments = routePathToCheck.split('/')\n segments.pop() // Remove the last segment\n const parentRoutePath = segments.join('/')\n\n return hasParentRoute(routes, node, parentRoutePath)\n}\n\n/**\n * Gets the final variable name for a route\n */\nconst getResolvedRouteNodeVariableName = (routeNode: RouteNode): string => {\n return routeNode.children?.length\n ? `${routeNode.variableName}RouteWithChildren`\n : `${routeNode.variableName}Route`\n}\n\n/**\n * Creates a map from fullPath to routeNode\n */\nconst createRouteNodesByFullPath = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n routeNodes.map((routeNode) => [inferFullPath(routeNode), routeNode]),\n )\n}\n\n/**\n * Create a map from 'to' to a routeNode\n */\nconst createRouteNodesByTo = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n dedupeBranchesAndIndexRoutes(routeNodes).map((routeNode) => [\n inferTo(routeNode),\n routeNode,\n ]),\n )\n}\n\n/**\n * Create a map from 'id' to a routeNode\n */\nconst createRouteNodesById = (\n routeNodes: Array<RouteNode>,\n): Map<string, RouteNode> => {\n return new Map(\n routeNodes.map((routeNode) => {\n const id = routeNode.routePath ?? ''\n return [id, routeNode]\n }),\n )\n}\n\n/**\n * Infers the full path for use by TS\n */\nconst inferFullPath = (routeNode: RouteNode): string => {\n const fullPath = removeGroups(\n removeUnderscores(removeLayoutSegments(routeNode.routePath)) ?? '',\n )\n\n return routeNode.cleanedPath === '/' ? fullPath : fullPath.replace(/\\/$/, '')\n}\n\n/**\n * Infers the path for use by TS\n */\nconst inferPath = (routeNode: RouteNode): string => {\n return routeNode.cleanedPath === '/'\n ? routeNode.cleanedPath\n : (routeNode.cleanedPath?.replace(/\\/$/, '') ?? '')\n}\n\n/**\n * Infers to path\n */\nconst inferTo = (routeNode: RouteNode): string => {\n const fullPath = inferFullPath(routeNode)\n\n if (fullPath === '/') return fullPath\n\n return fullPath.replace(/\\/$/, '')\n}\n\n/**\n * Dedupes branches and index routes\n */\nconst dedupeBranchesAndIndexRoutes = (\n routes: Array<RouteNode>,\n): Array<RouteNode> => {\n return routes.filter((route) => {\n if (route.children?.find((child) => child.cleanedPath === '/')) return false\n return true\n })\n}\n\nfunction checkUnique<TElement>(routes: Array<TElement>, key: keyof TElement) {\n // Check no two routes have the same `key`\n // if they do, throw an error with the conflicting filePaths\n const keys = routes.map((d) => d[key])\n const uniqueKeys = new Set(keys)\n if (keys.length !== uniqueKeys.size) {\n const duplicateKeys = keys.filter((d, i) => keys.indexOf(d) !== i)\n const conflictingFiles = routes.filter((d) =>\n duplicateKeys.includes(d[key]),\n )\n return conflictingFiles\n }\n return undefined\n}\n\nfunction checkRouteFullPathUniqueness(\n _routes: Array<RouteNode>,\n config: Config,\n) {\n const routes = _routes.map((d) => {\n const inferredFullPath = inferFullPath(d)\n return { ...d, inferredFullPath }\n })\n\n const conflictingFiles = checkUnique(routes, 'inferredFullPath')\n\n if (conflictingFiles !== undefined) {\n const errorMessage = `Conflicting configuration paths were found for the following route${conflictingFiles.length > 1 ? 's' : ''}: ${conflictingFiles\n .map((p) => `\"${p.inferredFullPath}\"`)\n .join(', ')}.\nPlease ensure each Server Route has a unique full path.\nConflicting files: \\n ${conflictingFiles.map((d) => path.resolve(config.routesDirectory, d.filePath)).join('\\n ')}\\n`\n console.error(errorMessage)\n process.exit(1)\n }\n}\n"],"names":["path","config","children"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tanstack/start-plugin-core",
|
|
3
|
-
"version": "1.120.4-alpha.
|
|
3
|
+
"version": "1.120.4-alpha.2",
|
|
4
4
|
"description": "Modern and scalable routing for React applications",
|
|
5
5
|
"author": "Tanner Linsley",
|
|
6
6
|
"license": "MIT",
|
|
@@ -61,17 +61,12 @@
|
|
|
61
61
|
"nitropack": "^2.11.8",
|
|
62
62
|
"pathe": "^2.0.3",
|
|
63
63
|
"ufo": "^1.5.4",
|
|
64
|
+
"vite": "^6.0.0",
|
|
64
65
|
"xmlbuilder2": "^3.1.1",
|
|
65
66
|
"zod": "^3.24.2",
|
|
66
|
-
"@tanstack/router-core": "^1.120.4-alpha.
|
|
67
|
-
"@tanstack/router-
|
|
68
|
-
"@tanstack/router-
|
|
69
|
-
},
|
|
70
|
-
"devDependencies": {
|
|
71
|
-
"vite": "^6.0.0"
|
|
72
|
-
},
|
|
73
|
-
"peerDependencies": {
|
|
74
|
-
"vite": ">=6.0.0"
|
|
67
|
+
"@tanstack/router-core": "^1.120.4-alpha.1",
|
|
68
|
+
"@tanstack/router-generator": "^1.120.4-alpha.1",
|
|
69
|
+
"@tanstack/router-utils": "^1.120.4-alpha.1"
|
|
75
70
|
},
|
|
76
71
|
"scripts": {}
|
|
77
72
|
}
|
|
@@ -80,52 +80,12 @@ export function nitroPlugin(
|
|
|
80
80
|
rollupConfig: {
|
|
81
81
|
plugins: [virtualBundlePlugin(getSsrBundle())],
|
|
82
82
|
},
|
|
83
|
-
routeRules: {
|
|
84
|
-
// TODO: We need to expose *some* kind of routeRules configuration
|
|
85
|
-
// and it needs to translate to this for now. But we should
|
|
86
|
-
// be cognizant of the probability that we will not use Nitro's
|
|
87
|
-
// routeRules configuration in the future.
|
|
88
|
-
...(options.shell?.enabled && options.shell.autoRedirect
|
|
89
|
-
? {
|
|
90
|
-
'/**': {
|
|
91
|
-
// @ts-expect-error We are using this as a marker
|
|
92
|
-
__TSS_SHELL: true,
|
|
93
|
-
redirect: {
|
|
94
|
-
to: `${options.shell.prerender.outputPath.replace(/[/]{1,}$/, '')}`,
|
|
95
|
-
statusCode: 200,
|
|
96
|
-
},
|
|
97
|
-
},
|
|
98
|
-
}
|
|
99
|
-
: {}),
|
|
100
|
-
},
|
|
101
83
|
}
|
|
102
84
|
|
|
103
85
|
const nitro = await createNitro(nitroConfig)
|
|
104
86
|
|
|
105
87
|
await buildNitroEnvironment(nitro, () => build(nitro))
|
|
106
88
|
|
|
107
|
-
if (options.shell?.enabled) {
|
|
108
|
-
options.prerender = {
|
|
109
|
-
...options.prerender,
|
|
110
|
-
enabled: true,
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
const maskUrl = new URL(
|
|
114
|
-
options.shell.maskPath,
|
|
115
|
-
'http://localhost',
|
|
116
|
-
)
|
|
117
|
-
|
|
118
|
-
maskUrl.searchParams.set('__TSS_SHELL', 'true')
|
|
119
|
-
|
|
120
|
-
options.pages.push({
|
|
121
|
-
path: maskUrl.toString().replace('http://localhost', ''),
|
|
122
|
-
prerender: options.shell.prerender,
|
|
123
|
-
sitemap: {
|
|
124
|
-
exclude: true,
|
|
125
|
-
},
|
|
126
|
-
})
|
|
127
|
-
}
|
|
128
|
-
|
|
129
89
|
if (options.prerender?.enabled) {
|
|
130
90
|
await prerender({
|
|
131
91
|
options,
|
package/src/plugin.ts
CHANGED
package/src/prerender.ts
CHANGED
|
@@ -20,7 +20,7 @@ export async function prerender({
|
|
|
20
20
|
nitro: Nitro
|
|
21
21
|
builder: ViteBuilder
|
|
22
22
|
}) {
|
|
23
|
-
|
|
23
|
+
nitro.logger.info('Prendering pages...')
|
|
24
24
|
|
|
25
25
|
// If prerender is enabled but no pages are provided, default to prerendering the root page
|
|
26
26
|
if (options.prerender?.enabled && !options.pages.length) {
|
|
@@ -44,14 +44,6 @@ export async function prerender({
|
|
|
44
44
|
|
|
45
45
|
const nodeNitro = await createNitro({
|
|
46
46
|
...nitro.options._config,
|
|
47
|
-
routeRules: {
|
|
48
|
-
// Filter out our shell redirect rule if it exists
|
|
49
|
-
...Object.fromEntries(
|
|
50
|
-
Object.entries(nitro.options._config.routeRules ?? {}).filter(
|
|
51
|
-
([_, value]) => !(value as any).__TSS_SHELL,
|
|
52
|
-
),
|
|
53
|
-
),
|
|
54
|
-
},
|
|
55
47
|
preset: 'nitro-prerender',
|
|
56
48
|
logLevel: 0,
|
|
57
49
|
output: {
|
|
@@ -98,14 +90,14 @@ export async function prerender({
|
|
|
98
90
|
// Crawl all pages
|
|
99
91
|
const pages = await prerenderPages()
|
|
100
92
|
|
|
101
|
-
|
|
93
|
+
nitro.logger.info(`Prerendered ${pages.length} pages:`)
|
|
102
94
|
pages.forEach((page) => {
|
|
103
|
-
|
|
95
|
+
nitro.logger.info(`- ${page}`)
|
|
104
96
|
})
|
|
105
97
|
|
|
106
98
|
// TODO: Write the prerendered pages to the output directory
|
|
107
99
|
} catch (error) {
|
|
108
|
-
|
|
100
|
+
nitro.logger.error(error)
|
|
109
101
|
} finally {
|
|
110
102
|
// Ensure server is always closed
|
|
111
103
|
// server.process.kill()
|
|
@@ -131,7 +123,7 @@ export async function prerender({
|
|
|
131
123
|
const seen = new Set<string>()
|
|
132
124
|
const retriesByPath = new Map<string, number>()
|
|
133
125
|
const concurrency = options.prerender?.concurrency ?? os.cpus().length
|
|
134
|
-
|
|
126
|
+
nitro.logger.info(`Concurrency: ${concurrency}`)
|
|
135
127
|
const queue = new Queue({ concurrency })
|
|
136
128
|
|
|
137
129
|
options.pages.forEach((_page) => {
|
|
@@ -173,7 +165,7 @@ export async function prerender({
|
|
|
173
165
|
|
|
174
166
|
// Add the task
|
|
175
167
|
queue.add(async () => {
|
|
176
|
-
|
|
168
|
+
nitro.logger.info(`Crawling: ${page.path}`)
|
|
177
169
|
const retries = retriesByPath.get(page.path) || 0
|
|
178
170
|
try {
|
|
179
171
|
// Fetch the route
|
|
@@ -187,29 +179,23 @@ export async function prerender({
|
|
|
187
179
|
)
|
|
188
180
|
|
|
189
181
|
if (!res.ok) {
|
|
190
|
-
throw new Error(`Failed to fetch ${page.path}: ${res.statusText}
|
|
191
|
-
cause: res,
|
|
192
|
-
})
|
|
182
|
+
throw new Error(`Failed to fetch ${page.path}: ${res.statusText}`)
|
|
193
183
|
}
|
|
194
184
|
|
|
195
|
-
const cleanPagePath = (
|
|
196
|
-
prerenderOptions.outputPath || page.path
|
|
197
|
-
).split(/[?#]/)[0]!
|
|
198
|
-
|
|
199
185
|
// Guess route type and populate fileName
|
|
200
186
|
const contentType = res.headers.get('content-type') || ''
|
|
201
187
|
const isImplicitHTML =
|
|
202
|
-
!
|
|
188
|
+
!page.path.endsWith('.html') && contentType.includes('html')
|
|
203
189
|
// &&
|
|
204
190
|
// !JsonSigRx.test(dataBuff.subarray(0, 32).toString('utf8'))
|
|
205
|
-
const routeWithIndex =
|
|
206
|
-
?
|
|
207
|
-
:
|
|
191
|
+
const routeWithIndex = page.path.endsWith('/')
|
|
192
|
+
? page.path + 'index'
|
|
193
|
+
: page.path
|
|
208
194
|
|
|
209
195
|
const htmlPath =
|
|
210
|
-
|
|
211
|
-
? joinURL(
|
|
212
|
-
:
|
|
196
|
+
page.path.endsWith('/') || prerenderOptions.autoSubfolderIndex
|
|
197
|
+
? joinURL(page.path, 'index.html')
|
|
198
|
+
: page.path + '.html'
|
|
213
199
|
|
|
214
200
|
const filename = withoutBase(
|
|
215
201
|
isImplicitHTML ? htmlPath : routeWithIndex,
|
|
@@ -241,7 +227,9 @@ export async function prerender({
|
|
|
241
227
|
}
|
|
242
228
|
} catch (error) {
|
|
243
229
|
if (retries < (prerenderOptions.retryCount ?? 0)) {
|
|
244
|
-
|
|
230
|
+
nitro.logger.warn(
|
|
231
|
+
`Encountered error, retrying: ${page.path} in 500ms`,
|
|
232
|
+
)
|
|
245
233
|
await new Promise((resolve) =>
|
|
246
234
|
setTimeout(resolve, prerenderOptions.retryDelay),
|
|
247
235
|
)
|
package/src/schema.ts
CHANGED
|
@@ -121,7 +121,6 @@ export function createTanStackStartOptionsSchema(
|
|
|
121
121
|
})
|
|
122
122
|
.and(pagePrerenderOptionsSchema.optional())
|
|
123
123
|
.optional(),
|
|
124
|
-
shell: shellSchema.optional(),
|
|
125
124
|
})
|
|
126
125
|
.optional()
|
|
127
126
|
.default({})
|
|
@@ -171,7 +170,6 @@ const pageBaseSchema = z.object({
|
|
|
171
170
|
|
|
172
171
|
const pagePrerenderOptionsSchema = z.object({
|
|
173
172
|
enabled: z.boolean().optional(),
|
|
174
|
-
outputPath: z.string().optional(),
|
|
175
173
|
autoSubfolderIndex: z.boolean().optional(),
|
|
176
174
|
crawlLinks: z.boolean().optional(),
|
|
177
175
|
retryCount: z.number().optional(),
|
|
@@ -188,22 +186,6 @@ const pagePrerenderOptionsSchema = z.object({
|
|
|
188
186
|
.optional(),
|
|
189
187
|
})
|
|
190
188
|
|
|
191
|
-
const shellSchema = z.object({
|
|
192
|
-
enabled: z.boolean().optional().default(true),
|
|
193
|
-
maskPath: z.string().optional().default('/'),
|
|
194
|
-
autoRedirect: z.boolean().optional().default(true),
|
|
195
|
-
prerender: pagePrerenderOptionsSchema
|
|
196
|
-
.optional()
|
|
197
|
-
.default({})
|
|
198
|
-
.transform((opts) => ({
|
|
199
|
-
outputPath: opts.outputPath ?? '/_shell',
|
|
200
|
-
crawlLinks: false,
|
|
201
|
-
retryCount: 0,
|
|
202
|
-
...opts,
|
|
203
|
-
enabled: true,
|
|
204
|
-
})),
|
|
205
|
-
})
|
|
206
|
-
|
|
207
189
|
export const pageSchema = pageBaseSchema.extend({
|
|
208
190
|
prerender: pagePrerenderOptionsSchema.optional(),
|
|
209
191
|
})
|
|
@@ -244,9 +244,7 @@ async function generator(config: Config, root: string) {
|
|
|
244
244
|
removeUnderscores(removeLayoutSegments(node.path)) ?? '',
|
|
245
245
|
)
|
|
246
246
|
|
|
247
|
-
const routeCode = node.fullPath
|
|
248
|
-
? fs.readFileSync(node.fullPath, 'utf-8')
|
|
249
|
-
: ''
|
|
247
|
+
const routeCode = fs.readFileSync(node.fullPath, 'utf-8')
|
|
250
248
|
|
|
251
249
|
// Ensure the boilerplate for the route exists, which can be skipped for virtual parent routes and virtual routes
|
|
252
250
|
if (!node.isVirtualParentRoute && !node.isVirtual) {
|
|
@@ -684,13 +682,7 @@ function buildStartDeclarationFile({
|
|
|
684
682
|
}: {
|
|
685
683
|
serverRoutesRelativePath: string
|
|
686
684
|
}) {
|
|
687
|
-
|
|
688
|
-
return (
|
|
689
|
-
[
|
|
690
|
-
'/// <reference types="vite/client" />',
|
|
691
|
-
`import '${serverRoutesPath}'`,
|
|
692
|
-
].join('\n') + '\n'
|
|
693
|
-
)
|
|
685
|
+
return [`import '${serverRoutesRelativePath}'`].join('\n') + '\n'
|
|
694
686
|
}
|
|
695
687
|
|
|
696
688
|
function removeGroups(s: string) {
|