toiljs 0.0.15 → 0.0.16

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.
Files changed (217) hide show
  1. package/.babelrc +13 -13
  2. package/.gitattributes +2 -2
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +38 -38
  4. package/.github/ISSUE_TEMPLATE/bug_report.yml +90 -90
  5. package/.github/ISSUE_TEMPLATE/config.yml +8 -8
  6. package/.github/ISSUE_TEMPLATE/feature_request.md +20 -20
  7. package/.github/PULL_REQUEST_TEMPLATE.md +43 -43
  8. package/.github/changelog-config.json +45 -45
  9. package/.github/dependabot.yml +27 -27
  10. package/.github/workflows/ci.yml +191 -191
  11. package/.prettierrc.json +11 -11
  12. package/.vscode/settings.json +9 -9
  13. package/CHANGELOG.md +5 -5
  14. package/LICENSE +187 -187
  15. package/README.md +339 -315
  16. package/as-pect.asconfig.json +34 -34
  17. package/as-pect.config.js +65 -65
  18. package/assets/logo.svg +36 -36
  19. package/build/backend/.tsbuildinfo +1 -1
  20. package/build/cli/.tsbuildinfo +1 -1
  21. package/build/cli/index.js +0 -0
  22. package/build/client/.tsbuildinfo +1 -1
  23. package/build/client/dev/devtools.d.ts +6 -0
  24. package/build/client/dev/devtools.js +442 -0
  25. package/build/client/dev/error-overlay.d.ts +9 -0
  26. package/build/client/dev/error-overlay.js +19 -4
  27. package/build/client/navigation/prefetch.d.ts +1 -0
  28. package/build/client/navigation/prefetch.js +35 -0
  29. package/build/client/routing/Router.js +1 -1
  30. package/build/client/routing/hooks.js +6 -2
  31. package/build/client/routing/loader.d.ts +23 -0
  32. package/build/client/routing/loader.js +53 -7
  33. package/build/client/routing/mount.js +4 -3
  34. package/build/compiler/.tsbuildinfo +1 -1
  35. package/build/compiler/config.d.ts +16 -0
  36. package/build/compiler/config.js +7 -0
  37. package/build/compiler/docs.js +16 -16
  38. package/build/compiler/index.d.ts +2 -2
  39. package/build/compiler/index.js +1 -1
  40. package/build/compiler/plugin.js +156 -0
  41. package/build/compiler/prerender.d.ts +1 -0
  42. package/build/compiler/prerender.js +1 -1
  43. package/build/compiler/seo.d.ts +1 -1
  44. package/build/compiler/seo.js +5 -4
  45. package/build/compiler/ssg.js +32 -1
  46. package/build/io/.tsbuildinfo +1 -1
  47. package/build/logger/.tsbuildinfo +1 -1
  48. package/build/shared/.tsbuildinfo +1 -1
  49. package/eslint.config.js +48 -48
  50. package/examples/basic/client/404.tsx +11 -11
  51. package/examples/basic/client/components/.gitkeep +1 -1
  52. package/examples/basic/client/global-error.tsx +13 -13
  53. package/examples/basic/client/layout.tsx +25 -25
  54. package/examples/basic/client/public/images/.gitkeep +1 -1
  55. package/examples/basic/client/public/images/logo.svg +36 -36
  56. package/examples/basic/client/public/robots.txt +2 -2
  57. package/examples/basic/client/routes/docs/[...slug].tsx +12 -12
  58. package/examples/basic/client/routes/features/error/error.tsx +16 -16
  59. package/examples/basic/client/routes/features/template/b.tsx +14 -14
  60. package/examples/basic/client/routes/files/[[...slug]].tsx +21 -21
  61. package/examples/basic/client/routes/gallery/layout.tsx +13 -13
  62. package/examples/basic/client/routes/io.tsx +24 -24
  63. package/examples/basic/client/routes/loader-demo/loading.tsx +13 -13
  64. package/examples/basic/client/routes/search.tsx +61 -61
  65. package/examples/basic/client/toil.tsx +5 -5
  66. package/package.json +155 -148
  67. package/presets/eslint.js +88 -88
  68. package/presets/no-uint8array-tostring.js +200 -200
  69. package/presets/prettier.json +18 -18
  70. package/presets/tsconfig.json +37 -37
  71. package/src/backend/index.ts +160 -160
  72. package/src/cli/proc.ts +50 -50
  73. package/src/cli/updates.ts +69 -69
  74. package/src/cli/validate.ts +31 -31
  75. package/src/client/channel/channel.ts +146 -146
  76. package/src/client/components/Form.tsx +65 -65
  77. package/src/client/components/Script.tsx +113 -113
  78. package/src/client/components/Slot.tsx +21 -21
  79. package/src/client/dev/devtools.tsx +973 -0
  80. package/src/client/dev/error-overlay.tsx +30 -4
  81. package/src/client/head/head.ts +167 -167
  82. package/src/client/head/metadata.ts +112 -112
  83. package/src/client/index.ts +89 -89
  84. package/src/client/navigation/NavLink.tsx +86 -86
  85. package/src/client/navigation/navigation.ts +235 -235
  86. package/src/client/navigation/prefetch.ts +169 -130
  87. package/src/client/navigation/scroll.ts +53 -53
  88. package/src/client/routing/Router.tsx +8 -2
  89. package/src/client/routing/action.ts +122 -122
  90. package/src/client/routing/error-boundary.tsx +43 -43
  91. package/src/client/routing/hooks.ts +21 -6
  92. package/src/client/routing/loader.ts +325 -235
  93. package/src/client/routing/match.ts +47 -47
  94. package/src/client/routing/mount.tsx +54 -52
  95. package/src/client/routing/params-context.ts +10 -10
  96. package/src/client/routing/slot-context.ts +7 -7
  97. package/src/client/search/search.ts +189 -189
  98. package/src/client/search/use-page-search.ts +73 -73
  99. package/src/client/types.ts +73 -73
  100. package/src/compiler/config.ts +219 -182
  101. package/src/compiler/docs.ts +228 -228
  102. package/src/compiler/generate.ts +394 -394
  103. package/src/compiler/index.ts +64 -57
  104. package/src/compiler/pages.ts +70 -70
  105. package/src/compiler/plugin.ts +170 -2
  106. package/src/compiler/prerender.ts +156 -156
  107. package/src/compiler/seo.ts +397 -390
  108. package/src/compiler/ssg.ts +162 -126
  109. package/src/io/BinaryReader.ts +340 -340
  110. package/src/io/BinaryWriter.ts +385 -385
  111. package/src/io/FastMap.ts +127 -127
  112. package/src/io/index.ts +11 -11
  113. package/src/io/lengths.ts +14 -14
  114. package/src/io/types.ts +18 -18
  115. package/src/logger/index.ts +22 -22
  116. package/src/server/index.ts +10 -10
  117. package/src/server/main.ts +13 -13
  118. package/src/server/tsconfig.json +4 -4
  119. package/src/shared/index.ts +10 -10
  120. package/std/client/index.d.ts +15 -15
  121. package/std/client/package.json +3 -3
  122. package/test/assembly/example.spec.ts +7 -7
  123. package/test/channel.test.ts +21 -21
  124. package/test/dom/Link.test.tsx +47 -47
  125. package/test/dom/NavLink.test.tsx +37 -37
  126. package/test/dom/error-overlay.test.tsx +44 -44
  127. package/test/dom/loader.test.tsx +121 -121
  128. package/test/dom/navigation.test.ts +59 -59
  129. package/test/dom/revalidate.test.tsx +38 -38
  130. package/test/dom/route-head.test.tsx +78 -78
  131. package/test/dom/router-loading.test.tsx +44 -44
  132. package/test/dom/scroll.test.ts +56 -56
  133. package/test/dom/use-metadata.test.tsx +58 -58
  134. package/test/io.test.ts +93 -93
  135. package/test/navlink.test.ts +28 -28
  136. package/test/placeholder.test.ts +9 -9
  137. package/test/routes.test.ts +76 -76
  138. package/test/seo.test.ts +175 -164
  139. package/test/slot-layouts.test.ts +69 -69
  140. package/test/ssg.test.ts +36 -36
  141. package/test/update.test.ts +44 -44
  142. package/test/validate.test.ts +42 -42
  143. package/toil-routes.d.ts +7 -0
  144. package/toilconfig.json +30 -30
  145. package/tsconfig.backend.json +13 -13
  146. package/tsconfig.base.json +35 -35
  147. package/tsconfig.cli.json +13 -13
  148. package/tsconfig.client.json +14 -14
  149. package/tsconfig.compiler.json +13 -13
  150. package/tsconfig.io.json +12 -12
  151. package/tsconfig.json +22 -22
  152. package/tsconfig.logger.json +12 -12
  153. package/tsconfig.server.json +10 -10
  154. package/tsconfig.shared.json +12 -12
  155. package/vitest.config.ts +26 -26
  156. package/.idea/codeStyles/Project.xml +0 -54
  157. package/.idea/codeStyles/codeStyleConfig.xml +0 -5
  158. package/.idea/inspectionProfiles/Project_Default.xml +0 -6
  159. package/.idea/modules.xml +0 -8
  160. package/.idea/prettier.xml +0 -7
  161. package/.idea/toiljs.iml +0 -8
  162. package/.idea/vcs.xml +0 -6
  163. package/.toil/entry.tsx +0 -9
  164. package/.toil/index.html +0 -12
  165. package/.toil/routes.ts +0 -9
  166. package/build/cli/configure.d.ts +0 -16
  167. package/build/cli/configure.js +0 -272
  168. package/build/cli/create.d.ts +0 -16
  169. package/build/cli/create.js +0 -420
  170. package/build/cli/diagnostics.d.ts +0 -55
  171. package/build/cli/diagnostics.js +0 -333
  172. package/build/cli/doctor.d.ts +0 -6
  173. package/build/cli/doctor.js +0 -249
  174. package/build/cli/features.d.ts +0 -25
  175. package/build/cli/features.js +0 -107
  176. package/build/cli/index.d.ts +0 -2
  177. package/build/cli/proc.d.ts +0 -6
  178. package/build/cli/proc.js +0 -31
  179. package/build/cli/ui.d.ts +0 -9
  180. package/build/cli/ui.js +0 -75
  181. package/build/cli/update.d.ts +0 -7
  182. package/build/cli/update.js +0 -117
  183. package/build/cli/updates.d.ts +0 -10
  184. package/build/cli/updates.js +0 -45
  185. package/build/cli/validate.d.ts +0 -4
  186. package/build/cli/validate.js +0 -19
  187. package/build/client/Link.d.ts +0 -8
  188. package/build/client/Link.js +0 -44
  189. package/build/client/NavLink.d.ts +0 -14
  190. package/build/client/NavLink.js +0 -37
  191. package/build/client/Router.d.ts +0 -7
  192. package/build/client/Router.js +0 -55
  193. package/build/client/channel.d.ts +0 -23
  194. package/build/client/channel.js +0 -94
  195. package/build/client/error-boundary.d.ts +0 -16
  196. package/build/client/error-boundary.js +0 -19
  197. package/build/client/head.d.ts +0 -26
  198. package/build/client/head.js +0 -87
  199. package/build/client/hooks.d.ts +0 -17
  200. package/build/client/hooks.js +0 -48
  201. package/build/client/lazy.d.ts +0 -16
  202. package/build/client/lazy.js +0 -53
  203. package/build/client/match.d.ts +0 -2
  204. package/build/client/match.js +0 -32
  205. package/build/client/mount.d.ts +0 -2
  206. package/build/client/mount.js +0 -13
  207. package/build/client/navigation.d.ts +0 -13
  208. package/build/client/navigation.js +0 -97
  209. package/build/client/params-context.d.ts +0 -2
  210. package/build/client/params-context.js +0 -2
  211. package/build/client/prefetch.d.ts +0 -11
  212. package/build/client/prefetch.js +0 -100
  213. package/build/client/runtime.d.ts +0 -31
  214. package/build/client/runtime.js +0 -112
  215. package/build/client/scroll.d.ts +0 -8
  216. package/build/client/scroll.js +0 -36
  217. package/toil-env.d.ts +0 -16
@@ -1,126 +1,162 @@
1
- /**
2
- * Build-time SSG for dynamic routes. After the client bundle is written, this loads each dynamic
3
- * route that exports `generateStaticParams`, enumerates its concrete URLs, runs the route's
4
- * `generateMetadata` per URL, and bakes a `<url>/index.html` (so JS-less crawlers get per-page tags)
5
- * plus a `sitemap.xml` entry. Opt-in: a route without `generateStaticParams` is untouched, and the
6
- * whole pass is skipped when no such route exists or `seo` is unconfigured. Build-only.
7
- *
8
- * Runs from `build()` (not the prerender Vite plugin) so it can reuse `createViteConfig` without the
9
- * `vite.ts` <-> `prerender.ts` import cycle; it spins up a short-lived SSR server to load route source.
10
- */
11
- import fs from 'node:fs';
12
- import path from 'node:path';
13
-
14
- import { createServer } from 'vite';
15
-
16
- import { type ResolvedToilConfig } from './config.js';
17
- import { scanRoutes } from './routes.js';
18
- import { injectSeoHtml, routeSeo, sitemapXml } from './seo.js';
19
- import { createViteConfig } from './vite.js';
20
-
21
- type StaticParams = Record<string, string | string[]>;
22
-
23
- interface RouteModule {
24
- generateStaticParams?: () => StaticParams[] | Promise<StaticParams[]>;
25
- generateMetadata?: (args: {
26
- params: StaticParams;
27
- searchParams: URLSearchParams;
28
- data: unknown;
29
- }) => unknown;
30
- loader?: (args: { params: StaticParams; searchParams: URLSearchParams }) => unknown;
31
- metadata?: Record<string, unknown>;
32
- }
33
-
34
- /** Substitutes `:param` / `*catch-all` segments in a route pattern with concrete param values. */
35
- export function fillPattern(pattern: string, params: StaticParams): string {
36
- return pattern.replace(/[:*]([A-Za-z0-9_]+)/g, (_m, name: string) => {
37
- const value = params[name] as string | string[] | undefined;
38
- if (Array.isArray(value)) return value.join('/');
39
- return value ?? '';
40
- });
41
- }
42
-
43
- /** Coerces an unknown module export to a typed Metadata-ish record, or null. */
44
- function asMetadata(value: unknown): Record<string, unknown> | null {
45
- return typeof value === 'object' && value !== null ? (value as Record<string, unknown>) : null;
46
- }
47
-
48
- /**
49
- * Pre-renders every dynamic route that opts in via `generateStaticParams`. Bakes per-URL HTML into
50
- * `outDir` and rewrites `sitemap.xml` with the generated URLs. Returns the list of generated URLs.
51
- */
52
- export async function prerenderStaticParams(cfg: ResolvedToilConfig): Promise<string[]> {
53
- if (!cfg.seo) return [];
54
- const outDir = path.resolve(cfg.root, cfg.outDir);
55
- // Prefer the clean shell stashed by the prerender plugin (no per-route SEO baked in); fall back
56
- // to the built index.html.
57
- const stashed = path.join(cfg.toilDir, 'shell.html');
58
- const shellPath = fs.existsSync(stashed) ? stashed : path.join(outDir, 'index.html');
59
- if (!fs.existsSync(shellPath)) return [];
60
-
61
- const allRoutes = scanRoutes(cfg.routesAbsDir);
62
- const dynamic = allRoutes.filter(
63
- (r) => r.slot === undefined && !r.intercept && /[:*]/.test(r.pattern),
64
- );
65
- if (dynamic.length === 0) return [];
66
-
67
- const shell = fs.readFileSync(shellPath, 'utf8');
68
- const server = await createServer({
69
- ...(await createViteConfig(cfg)),
70
- server: { middlewareMode: true, hmr: false },
71
- appType: 'custom',
72
- logLevel: 'silent',
73
- });
74
-
75
- const generated: string[] = [];
76
- const warn = (msg: string): void => {
77
- process.stderr.write(` toil: SSG ${msg}\n`);
78
- };
79
- try {
80
- for (const route of dynamic) {
81
- let mod: RouteModule;
82
- try {
83
- mod = (await server.ssrLoadModule(route.file)) as RouteModule;
84
- } catch (err) {
85
- warn(`skipped ${route.pattern} (${err instanceof Error ? err.message : String(err)})`);
86
- continue;
87
- }
88
- if (typeof mod.generateStaticParams !== 'function') continue;
89
- const paramSets = await mod.generateStaticParams();
90
- for (const params of paramSets) {
91
- const url = fillPattern(route.pattern, params);
92
- let metadata: Record<string, unknown> | null = null;
93
- try {
94
- if (typeof mod.generateMetadata === 'function') {
95
- const searchParams = new URLSearchParams();
96
- const data =
97
- typeof mod.loader === 'function'
98
- ? await mod.loader({ params, searchParams })
99
- : undefined;
100
- metadata = asMetadata(await mod.generateMetadata({ params, searchParams, data }));
101
- } else if (mod.metadata) {
102
- metadata = asMetadata(mod.metadata);
103
- }
104
- } catch (err) {
105
- warn(`metadata failed for ${url} (${err instanceof Error ? err.message : String(err)})`);
106
- }
107
- const html = injectSeoHtml(shell, routeSeo(cfg.seo, metadata, url));
108
- const target = path.join(outDir, url.replace(/^\//, ''), 'index.html');
109
- fs.mkdirSync(path.dirname(target), { recursive: true });
110
- fs.writeFileSync(target, html);
111
- generated.push(url);
112
- }
113
- }
114
- } finally {
115
- await server.close();
116
- }
117
-
118
- if (generated.length > 0) {
119
- const sitemap = sitemapXml(cfg.seo, allRoutes, generated);
120
- if (sitemap) fs.writeFileSync(path.join(outDir, 'sitemap.xml'), sitemap);
121
- process.stdout.write(
122
- ` ✓ prerendered ${String(generated.length)} dynamic route${generated.length === 1 ? '' : 's'}\n`,
123
- );
124
- }
125
- return generated;
126
- }
1
+ /**
2
+ * Build-time SSG for dynamic routes. After the client bundle is written, this loads each dynamic
3
+ * route that exports `generateStaticParams`, enumerates its concrete URLs, runs the route's
4
+ * `generateMetadata` per URL, and bakes a `<url>/index.html` (so JS-less crawlers get per-page tags)
5
+ * plus a `sitemap.xml` entry. Opt-in: a route without `generateStaticParams` is untouched, and the
6
+ * whole pass is skipped when no such route exists or `seo` is unconfigured. Build-only.
7
+ *
8
+ * Runs from `build()` (not the prerender Vite plugin) so it can reuse `createViteConfig` without the
9
+ * `vite.ts` <-> `prerender.ts` import cycle; it spins up a short-lived SSR server to load route source.
10
+ */
11
+ import fs from 'node:fs';
12
+ import path from 'node:path';
13
+
14
+ import { createServer } from 'vite';
15
+
16
+ import { type ResolvedToilConfig } from './config.js';
17
+ import { extractStaticMetadata, loadTypeScript } from './prerender.js';
18
+ import { scanRoutes } from './routes.js';
19
+ import { injectSeoHtml, joinUrl, llmsTxt, routeSeo, sitemapXml, type LlmsPage } from './seo.js';
20
+ import { createViteConfig } from './vite.js';
21
+
22
+ /** Reads a string field off a metadata record, or undefined. */
23
+ function metaString(meta: Record<string, unknown> | null, key: string): string | undefined {
24
+ const value = meta?.[key];
25
+ return typeof value === 'string' ? value : undefined;
26
+ }
27
+
28
+ type StaticParams = Record<string, string | string[]>;
29
+
30
+ interface RouteModule {
31
+ generateStaticParams?: () => StaticParams[] | Promise<StaticParams[]>;
32
+ generateMetadata?: (args: {
33
+ params: StaticParams;
34
+ searchParams: URLSearchParams;
35
+ data: unknown;
36
+ }) => unknown;
37
+ loader?: (args: { params: StaticParams; searchParams: URLSearchParams }) => unknown;
38
+ metadata?: Record<string, unknown>;
39
+ }
40
+
41
+ /** Substitutes `:param` / `*catch-all` segments in a route pattern with concrete param values. */
42
+ export function fillPattern(pattern: string, params: StaticParams): string {
43
+ return pattern.replace(/[:*]([A-Za-z0-9_]+)/g, (_m, name: string) => {
44
+ const value = params[name] as string | string[] | undefined;
45
+ if (Array.isArray(value)) return value.join('/');
46
+ return value ?? '';
47
+ });
48
+ }
49
+
50
+ /** Coerces an unknown module export to a typed Metadata-ish record, or null. */
51
+ function asMetadata(value: unknown): Record<string, unknown> | null {
52
+ return typeof value === 'object' && value !== null ? (value as Record<string, unknown>) : null;
53
+ }
54
+
55
+ /**
56
+ * Pre-renders every dynamic route that opts in via `generateStaticParams`. Bakes per-URL HTML into
57
+ * `outDir` and rewrites `sitemap.xml` with the generated URLs. Returns the list of generated URLs.
58
+ */
59
+ export async function prerenderStaticParams(cfg: ResolvedToilConfig): Promise<string[]> {
60
+ if (!cfg.seo) return [];
61
+ const outDir = path.resolve(cfg.root, cfg.outDir);
62
+ // Prefer the clean shell stashed by the prerender plugin (no per-route SEO baked in); fall back
63
+ // to the built index.html.
64
+ const stashed = path.join(cfg.toilDir, 'shell.html');
65
+ const shellPath = fs.existsSync(stashed) ? stashed : path.join(outDir, 'index.html');
66
+ if (!fs.existsSync(shellPath)) return [];
67
+
68
+ const allRoutes = scanRoutes(cfg.routesAbsDir);
69
+ const dynamic = allRoutes.filter(
70
+ (r) => r.slot === undefined && !r.intercept && /[:*]/.test(r.pattern),
71
+ );
72
+ if (dynamic.length === 0) return [];
73
+
74
+ const shell = fs.readFileSync(shellPath, 'utf8');
75
+ const server = await createServer({
76
+ ...(await createViteConfig(cfg)),
77
+ server: { middlewareMode: true, hmr: false },
78
+ appType: 'custom',
79
+ logLevel: 'silent',
80
+ });
81
+
82
+ const baseUrl = cfg.seo.url;
83
+ const generated: string[] = [];
84
+ const dynamicPages: LlmsPage[] = [];
85
+ const warn = (msg: string): void => {
86
+ process.stderr.write(` toil: SSG ${msg}\n`);
87
+ };
88
+ try {
89
+ for (const route of dynamic) {
90
+ let mod: RouteModule;
91
+ try {
92
+ mod = (await server.ssrLoadModule(route.file)) as RouteModule;
93
+ } catch (err) {
94
+ warn(`skipped ${route.pattern} (${err instanceof Error ? err.message : String(err)})`);
95
+ continue;
96
+ }
97
+ if (typeof mod.generateStaticParams !== 'function') continue;
98
+ const paramSets = await mod.generateStaticParams();
99
+ for (const params of paramSets) {
100
+ const url = fillPattern(route.pattern, params);
101
+ let metadata: Record<string, unknown> | null = null;
102
+ try {
103
+ if (typeof mod.generateMetadata === 'function') {
104
+ const searchParams = new URLSearchParams();
105
+ const data =
106
+ typeof mod.loader === 'function'
107
+ ? await mod.loader({ params, searchParams })
108
+ : undefined;
109
+ metadata = asMetadata(await mod.generateMetadata({ params, searchParams, data }));
110
+ } else if (mod.metadata) {
111
+ metadata = asMetadata(mod.metadata);
112
+ }
113
+ } catch (err) {
114
+ warn(`metadata failed for ${url} (${err instanceof Error ? err.message : String(err)})`);
115
+ }
116
+ const html = injectSeoHtml(shell, routeSeo(cfg.seo, metadata, url));
117
+ const target = path.join(outDir, url.replace(/^\//, ''), 'index.html');
118
+ fs.mkdirSync(path.dirname(target), { recursive: true });
119
+ fs.writeFileSync(target, html);
120
+ generated.push(url);
121
+ dynamicPages.push({
122
+ title: metaString(metadata, 'title') ?? url,
123
+ url: baseUrl !== undefined ? joinUrl(baseUrl, url) : url,
124
+ description: metaString(metadata, 'description'),
125
+ });
126
+ }
127
+ }
128
+ } finally {
129
+ await server.close();
130
+ }
131
+
132
+ if (generated.length > 0) {
133
+ const sitemap = sitemapXml(cfg.seo, allRoutes, generated);
134
+ if (sitemap) fs.writeFileSync(path.join(outDir, 'sitemap.xml'), sitemap);
135
+ process.stdout.write(
136
+ ` ✓ prerendered ${String(generated.length)} dynamic route${generated.length === 1 ? '' : 's'}\n`,
137
+ );
138
+ }
139
+
140
+ // Rewrite llms.txt with the full page index: every static route's resolved title/description plus
141
+ // the enumerated dynamic pages, so AI crawlers get the whole site, not just the static paths.
142
+ if (baseUrl !== undefined) {
143
+ const ts = await loadTypeScript(cfg.root);
144
+ const staticPages: LlmsPage[] = allRoutes
145
+ .filter((r) => r.slot === undefined && !r.intercept && !/[:*]/.test(r.pattern))
146
+ .map((r): LlmsPage => {
147
+ const meta = ts ? extractStaticMetadata(ts, r.file) : null;
148
+ return {
149
+ title: metaString(meta, 'title') ?? (r.pattern === '/' ? 'Home' : r.pattern),
150
+ url: joinUrl(baseUrl, r.pattern),
151
+ description: metaString(meta, 'description'),
152
+ };
153
+ });
154
+ const pages = [...staticPages, ...dynamicPages];
155
+ if (pages.length > 0) {
156
+ const llms = llmsTxt(cfg.seo, allRoutes, pages);
157
+ if (llms) fs.writeFileSync(path.join(outDir, 'llms.txt'), llms);
158
+ }
159
+ }
160
+
161
+ return generated;
162
+ }