@timber-js/app 0.2.0-alpha.9 → 0.2.0-alpha.91

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 (619) hide show
  1. package/dist/_chunks/actions-DLnUaR65.js +421 -0
  2. package/dist/_chunks/actions-DLnUaR65.js.map +1 -0
  3. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
  4. package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
  5. package/dist/_chunks/chunk-BYIpzuS7.js +39 -0
  6. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  7. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  8. package/dist/_chunks/define-C77ScO0m.js +106 -0
  9. package/dist/_chunks/define-C77ScO0m.js.map +1 -0
  10. package/dist/_chunks/define-Itxvcd7F.js +199 -0
  11. package/dist/_chunks/define-Itxvcd7F.js.map +1 -0
  12. package/dist/_chunks/define-cookie-BowvzoP0.js +94 -0
  13. package/dist/_chunks/define-cookie-BowvzoP0.js.map +1 -0
  14. package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
  15. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
  16. package/dist/_chunks/format-CYBGxKtc.js +14 -0
  17. package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
  18. package/dist/_chunks/{interception-BOoWmLUA.js → interception-ErnB33JX.js} +301 -133
  19. package/dist/_chunks/interception-ErnB33JX.js.map +1 -0
  20. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
  21. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
  22. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  23. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  24. package/dist/_chunks/request-context-CK5tZqIP.js +478 -0
  25. package/dist/_chunks/request-context-CK5tZqIP.js.map +1 -0
  26. package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
  27. package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
  28. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  29. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  30. package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
  31. package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
  32. package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
  33. package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
  34. package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
  35. package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
  36. package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
  37. package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
  38. package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
  39. package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
  40. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-BiV5GJgm.js} +7 -4
  41. package/dist/_chunks/use-query-states-BiV5GJgm.js.map +1 -0
  42. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  43. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  44. package/dist/adapters/cloudflare-dev.js +73 -0
  45. package/dist/adapters/cloudflare-dev.js.map +1 -0
  46. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  47. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  48. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  49. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  50. package/dist/adapters/cloudflare.d.ts +148 -12
  51. package/dist/adapters/cloudflare.d.ts.map +1 -1
  52. package/dist/adapters/cloudflare.js +135 -11
  53. package/dist/adapters/cloudflare.js.map +1 -1
  54. package/dist/adapters/compress-module.d.ts.map +1 -1
  55. package/dist/adapters/nitro.d.ts +17 -1
  56. package/dist/adapters/nitro.d.ts.map +1 -1
  57. package/dist/adapters/nitro.js +56 -13
  58. package/dist/adapters/nitro.js.map +1 -1
  59. package/dist/cache/cache-api.d.ts +24 -0
  60. package/dist/cache/cache-api.d.ts.map +1 -0
  61. package/dist/cache/handler-store.d.ts +31 -0
  62. package/dist/cache/handler-store.d.ts.map +1 -0
  63. package/dist/cache/index.d.ts +23 -7
  64. package/dist/cache/index.d.ts.map +1 -1
  65. package/dist/cache/index.js +142 -80
  66. package/dist/cache/index.js.map +1 -1
  67. package/dist/cache/singleflight.d.ts +18 -1
  68. package/dist/cache/singleflight.d.ts.map +1 -1
  69. package/dist/cache/sizeof.d.ts +22 -0
  70. package/dist/cache/sizeof.d.ts.map +1 -0
  71. package/dist/cache/timber-cache.d.ts +1 -1
  72. package/dist/cache/timber-cache.d.ts.map +1 -1
  73. package/dist/cli.d.ts +6 -1
  74. package/dist/cli.d.ts.map +1 -1
  75. package/dist/cli.js +8 -3
  76. package/dist/cli.js.map +1 -1
  77. package/dist/client/browser-dev.d.ts +27 -1
  78. package/dist/client/browser-dev.d.ts.map +1 -1
  79. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  80. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  81. package/dist/client/browser-entry/hmr.d.ts +21 -0
  82. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  83. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  84. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  85. package/dist/client/browser-entry/index.d.ts +30 -0
  86. package/dist/client/browser-entry/index.d.ts.map +1 -0
  87. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  88. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  89. package/dist/client/browser-entry/router-init.d.ts +23 -0
  90. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  91. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  92. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  93. package/dist/client/browser-entry/scroll.d.ts +19 -0
  94. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  95. package/dist/client/error-boundary.d.ts +12 -5
  96. package/dist/client/error-boundary.d.ts.map +1 -1
  97. package/dist/client/error-boundary.js +10 -4
  98. package/dist/client/error-boundary.js.map +1 -1
  99. package/dist/client/error-reconstituter.d.ts +54 -0
  100. package/dist/client/error-reconstituter.d.ts.map +1 -0
  101. package/dist/client/form.d.ts +6 -3
  102. package/dist/client/form.d.ts.map +1 -1
  103. package/dist/client/history.d.ts +19 -4
  104. package/dist/client/history.d.ts.map +1 -1
  105. package/dist/client/index.d.ts +9 -21
  106. package/dist/client/index.d.ts.map +1 -1
  107. package/dist/client/index.js +229 -1018
  108. package/dist/client/index.js.map +1 -1
  109. package/dist/client/internal.d.ts +18 -0
  110. package/dist/client/internal.d.ts.map +1 -0
  111. package/dist/client/internal.js +890 -0
  112. package/dist/client/internal.js.map +1 -0
  113. package/dist/client/link-pending-store.d.ts +63 -0
  114. package/dist/client/link-pending-store.d.ts.map +1 -0
  115. package/dist/client/link.d.ts +62 -55
  116. package/dist/client/link.d.ts.map +1 -1
  117. package/dist/client/nav-link-store.d.ts +36 -0
  118. package/dist/client/nav-link-store.d.ts.map +1 -0
  119. package/dist/client/navigation-api-types.d.ts +90 -0
  120. package/dist/client/navigation-api-types.d.ts.map +1 -0
  121. package/dist/client/navigation-api.d.ts +115 -0
  122. package/dist/client/navigation-api.d.ts.map +1 -0
  123. package/dist/client/navigation-context.d.ts +13 -2
  124. package/dist/client/navigation-context.d.ts.map +1 -1
  125. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  126. package/dist/client/navigation-root.d.ts.map +1 -0
  127. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  128. package/dist/client/router-ref.d.ts +1 -1
  129. package/dist/client/router.d.ts +70 -4
  130. package/dist/client/router.d.ts.map +1 -1
  131. package/dist/client/rsc-fetch.d.ts +38 -3
  132. package/dist/client/rsc-fetch.d.ts.map +1 -1
  133. package/dist/client/segment-cache.d.ts +1 -1
  134. package/dist/client/segment-cache.d.ts.map +1 -1
  135. package/dist/client/segment-outlet.d.ts +63 -0
  136. package/dist/client/segment-outlet.d.ts.map +1 -0
  137. package/dist/client/ssr-data.d.ts +13 -4
  138. package/dist/client/ssr-data.d.ts.map +1 -1
  139. package/dist/client/stale-reload.d.ts +15 -0
  140. package/dist/client/stale-reload.d.ts.map +1 -1
  141. package/dist/client/top-loader.d.ts +5 -5
  142. package/dist/client/top-loader.d.ts.map +1 -1
  143. package/dist/client/use-link-status.d.ts +5 -5
  144. package/dist/client/use-link-status.d.ts.map +1 -1
  145. package/dist/client/use-params.d.ts +6 -4
  146. package/dist/client/use-params.d.ts.map +1 -1
  147. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  148. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  149. package/dist/client/use-query-states.d.ts +1 -1
  150. package/dist/client/use-query-states.d.ts.map +1 -1
  151. package/dist/client/use-router.d.ts +1 -1
  152. package/dist/codec.d.ts +33 -0
  153. package/dist/codec.d.ts.map +1 -0
  154. package/dist/codec.js +2 -0
  155. package/dist/config-types.d.ts +266 -0
  156. package/dist/config-types.d.ts.map +1 -0
  157. package/dist/config-validation.d.ts +51 -0
  158. package/dist/config-validation.d.ts.map +1 -0
  159. package/dist/content/index.d.ts +1 -10
  160. package/dist/content/index.d.ts.map +1 -1
  161. package/dist/content/index.js +0 -2
  162. package/dist/cookies/define-cookie.d.ts +35 -14
  163. package/dist/cookies/define-cookie.d.ts.map +1 -1
  164. package/dist/cookies/index.js +1 -83
  165. package/dist/fonts/bundle.d.ts +48 -0
  166. package/dist/fonts/bundle.d.ts.map +1 -0
  167. package/dist/fonts/css.d.ts +1 -0
  168. package/dist/fonts/css.d.ts.map +1 -1
  169. package/dist/fonts/dev-middleware.d.ts +22 -0
  170. package/dist/fonts/dev-middleware.d.ts.map +1 -0
  171. package/dist/fonts/pipeline.d.ts +138 -0
  172. package/dist/fonts/pipeline.d.ts.map +1 -0
  173. package/dist/fonts/transform.d.ts +72 -0
  174. package/dist/fonts/transform.d.ts.map +1 -0
  175. package/dist/fonts/types.d.ts +45 -1
  176. package/dist/fonts/types.d.ts.map +1 -1
  177. package/dist/fonts/virtual-modules.d.ts +59 -0
  178. package/dist/fonts/virtual-modules.d.ts.map +1 -0
  179. package/dist/index.d.ts +45 -190
  180. package/dist/index.d.ts.map +1 -1
  181. package/dist/index.js +4294 -2453
  182. package/dist/index.js.map +1 -1
  183. package/dist/plugin-context.d.ts +107 -0
  184. package/dist/plugin-context.d.ts.map +1 -0
  185. package/dist/plugins/adapter-build.d.ts +1 -1
  186. package/dist/plugins/adapter-build.d.ts.map +1 -1
  187. package/dist/plugins/build-manifest.d.ts +2 -2
  188. package/dist/plugins/build-manifest.d.ts.map +1 -1
  189. package/dist/plugins/build-report.d.ts +3 -3
  190. package/dist/plugins/build-report.d.ts.map +1 -1
  191. package/dist/plugins/client-chunks.d.ts +32 -0
  192. package/dist/plugins/client-chunks.d.ts.map +1 -0
  193. package/dist/plugins/content.d.ts +1 -1
  194. package/dist/plugins/content.d.ts.map +1 -1
  195. package/dist/plugins/dev-404-page.d.ts +56 -0
  196. package/dist/plugins/dev-404-page.d.ts.map +1 -0
  197. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  198. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  199. package/dist/plugins/dev-error-overlay.d.ts +49 -9
  200. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  201. package/dist/plugins/dev-error-page.d.ts +58 -0
  202. package/dist/plugins/dev-error-page.d.ts.map +1 -0
  203. package/dist/plugins/dev-logs.d.ts +1 -1
  204. package/dist/plugins/dev-logs.d.ts.map +1 -1
  205. package/dist/plugins/dev-server.d.ts +1 -1
  206. package/dist/plugins/dev-server.d.ts.map +1 -1
  207. package/dist/plugins/dev-terminal-error.d.ts +28 -0
  208. package/dist/plugins/dev-terminal-error.d.ts.map +1 -0
  209. package/dist/plugins/entries.d.ts +1 -1
  210. package/dist/plugins/entries.d.ts.map +1 -1
  211. package/dist/plugins/fonts.d.ts +17 -73
  212. package/dist/plugins/fonts.d.ts.map +1 -1
  213. package/dist/plugins/mdx.d.ts +1 -1
  214. package/dist/plugins/mdx.d.ts.map +1 -1
  215. package/dist/plugins/routing.d.ts +1 -1
  216. package/dist/plugins/routing.d.ts.map +1 -1
  217. package/dist/plugins/server-bundle.d.ts.map +1 -1
  218. package/dist/plugins/shims.d.ts +6 -5
  219. package/dist/plugins/shims.d.ts.map +1 -1
  220. package/dist/plugins/static-build.d.ts +4 -4
  221. package/dist/plugins/static-build.d.ts.map +1 -1
  222. package/dist/routing/codegen-shared.d.ts +38 -0
  223. package/dist/routing/codegen-shared.d.ts.map +1 -0
  224. package/dist/routing/codegen-types.d.ts +36 -0
  225. package/dist/routing/codegen-types.d.ts.map +1 -0
  226. package/dist/routing/codegen.d.ts +2 -2
  227. package/dist/routing/codegen.d.ts.map +1 -1
  228. package/dist/routing/convention-lint.d.ts +41 -0
  229. package/dist/routing/convention-lint.d.ts.map +1 -0
  230. package/dist/routing/index.d.ts +2 -0
  231. package/dist/routing/index.d.ts.map +1 -1
  232. package/dist/routing/index.js +3 -2
  233. package/dist/routing/link-codegen.d.ts +90 -0
  234. package/dist/routing/link-codegen.d.ts.map +1 -0
  235. package/dist/routing/scanner.d.ts.map +1 -1
  236. package/dist/routing/segment-classify.d.ts +46 -0
  237. package/dist/routing/segment-classify.d.ts.map +1 -0
  238. package/dist/routing/status-file-lint.d.ts +2 -1
  239. package/dist/routing/status-file-lint.d.ts.map +1 -1
  240. package/dist/routing/types.d.ts +16 -4
  241. package/dist/routing/types.d.ts.map +1 -1
  242. package/dist/rsc-runtime/rsc.d.ts +1 -1
  243. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  244. package/dist/rsc-runtime/ssr.d.ts +12 -0
  245. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  246. package/dist/schema-bridge.d.ts +76 -0
  247. package/dist/schema-bridge.d.ts.map +1 -0
  248. package/dist/search-params/define.d.ts +139 -0
  249. package/dist/search-params/define.d.ts.map +1 -0
  250. package/dist/search-params/index.d.ts +4 -7
  251. package/dist/search-params/index.d.ts.map +1 -1
  252. package/dist/search-params/index.js +32 -441
  253. package/dist/search-params/index.js.map +1 -1
  254. package/dist/search-params/registry.d.ts +2 -2
  255. package/dist/search-params/registry.d.ts.map +1 -1
  256. package/dist/search-params/wrappers.d.ts +53 -0
  257. package/dist/search-params/wrappers.d.ts.map +1 -0
  258. package/dist/segment-params/define.d.ts +78 -0
  259. package/dist/segment-params/define.d.ts.map +1 -0
  260. package/dist/segment-params/index.d.ts +3 -0
  261. package/dist/segment-params/index.d.ts.map +1 -0
  262. package/dist/segment-params/index.js +2 -0
  263. package/dist/server/access-gate.d.ts +4 -0
  264. package/dist/server/access-gate.d.ts.map +1 -1
  265. package/dist/server/action-client.d.ts +41 -6
  266. package/dist/server/action-client.d.ts.map +1 -1
  267. package/dist/server/action-encryption.d.ts +76 -0
  268. package/dist/server/action-encryption.d.ts.map +1 -0
  269. package/dist/server/action-handler.d.ts +7 -0
  270. package/dist/server/action-handler.d.ts.map +1 -1
  271. package/dist/server/actions.d.ts +3 -6
  272. package/dist/server/actions.d.ts.map +1 -1
  273. package/dist/server/als-registry.d.ts +32 -4
  274. package/dist/server/als-registry.d.ts.map +1 -1
  275. package/dist/server/build-manifest.d.ts +2 -2
  276. package/dist/server/build-manifest.d.ts.map +1 -1
  277. package/dist/server/debug.d.ts +1 -1
  278. package/dist/server/default-logger.d.ts +22 -0
  279. package/dist/server/default-logger.d.ts.map +1 -0
  280. package/dist/server/deny-page-resolver.d.ts +52 -0
  281. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  282. package/dist/server/deny-renderer.d.ts.map +1 -1
  283. package/dist/server/dev-holding-server.d.ts +52 -0
  284. package/dist/server/dev-holding-server.d.ts.map +1 -0
  285. package/dist/server/dev-source-map.d.ts +22 -0
  286. package/dist/server/dev-source-map.d.ts.map +1 -0
  287. package/dist/server/dev-warnings.d.ts +1 -21
  288. package/dist/server/dev-warnings.d.ts.map +1 -1
  289. package/dist/server/early-hints.d.ts +13 -5
  290. package/dist/server/early-hints.d.ts.map +1 -1
  291. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  292. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  293. package/dist/server/fallback-error.d.ts +12 -7
  294. package/dist/server/fallback-error.d.ts.map +1 -1
  295. package/dist/server/flight-injection-state.d.ts +66 -0
  296. package/dist/server/flight-injection-state.d.ts.map +1 -0
  297. package/dist/server/flight-scripts.d.ts +42 -0
  298. package/dist/server/flight-scripts.d.ts.map +1 -0
  299. package/dist/server/flush.d.ts.map +1 -1
  300. package/dist/server/form-data.d.ts +29 -0
  301. package/dist/server/form-data.d.ts.map +1 -1
  302. package/dist/server/html-injectors.d.ts +51 -11
  303. package/dist/server/html-injectors.d.ts.map +1 -1
  304. package/dist/server/index.d.ts +5 -43
  305. package/dist/server/index.d.ts.map +1 -1
  306. package/dist/server/index.js +195 -2800
  307. package/dist/server/index.js.map +1 -1
  308. package/dist/server/internal.d.ts +46 -0
  309. package/dist/server/internal.d.ts.map +1 -0
  310. package/dist/server/internal.js +2900 -0
  311. package/dist/server/internal.js.map +1 -0
  312. package/dist/server/logger.d.ts +25 -7
  313. package/dist/server/logger.d.ts.map +1 -1
  314. package/dist/server/middleware-runner.d.ts +19 -4
  315. package/dist/server/middleware-runner.d.ts.map +1 -1
  316. package/dist/server/node-stream-transforms.d.ts +113 -0
  317. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  318. package/dist/server/page-deny-boundary.d.ts +31 -0
  319. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  320. package/dist/server/pipeline-interception.d.ts +1 -1
  321. package/dist/server/pipeline-interception.d.ts.map +1 -1
  322. package/dist/server/pipeline-metadata.d.ts +6 -0
  323. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  324. package/dist/server/pipeline.d.ts +52 -10
  325. package/dist/server/pipeline.d.ts.map +1 -1
  326. package/dist/server/primitives.d.ts +69 -18
  327. package/dist/server/primitives.d.ts.map +1 -1
  328. package/dist/server/render-timeout.d.ts +51 -0
  329. package/dist/server/render-timeout.d.ts.map +1 -0
  330. package/dist/server/request-context.d.ts +112 -43
  331. package/dist/server/request-context.d.ts.map +1 -1
  332. package/dist/server/route-element-builder.d.ts +27 -1
  333. package/dist/server/route-element-builder.d.ts.map +1 -1
  334. package/dist/server/route-handler.d.ts.map +1 -1
  335. package/dist/server/route-matcher.d.ts +16 -2
  336. package/dist/server/route-matcher.d.ts.map +1 -1
  337. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  338. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  339. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  340. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  341. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  342. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  343. package/dist/server/rsc-entry/index.d.ts +20 -3
  344. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  345. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  346. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  347. package/dist/server/rsc-entry/rsc-stream.d.ts +14 -1
  348. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  349. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  350. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  351. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  352. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  353. package/dist/server/safe-load.d.ts +46 -0
  354. package/dist/server/safe-load.d.ts.map +1 -0
  355. package/dist/server/sensitive-fields.d.ts +74 -0
  356. package/dist/server/sensitive-fields.d.ts.map +1 -0
  357. package/dist/server/sitemap-generator.d.ts +129 -0
  358. package/dist/server/sitemap-generator.d.ts.map +1 -0
  359. package/dist/server/sitemap-handler.d.ts +22 -0
  360. package/dist/server/sitemap-handler.d.ts.map +1 -0
  361. package/dist/server/slot-resolver.d.ts +1 -1
  362. package/dist/server/slot-resolver.d.ts.map +1 -1
  363. package/dist/server/ssr-entry.d.ts +23 -0
  364. package/dist/server/ssr-entry.d.ts.map +1 -1
  365. package/dist/server/ssr-render.d.ts +39 -21
  366. package/dist/server/ssr-render.d.ts.map +1 -1
  367. package/dist/server/ssr-wrappers.d.ts +50 -0
  368. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  369. package/dist/server/status-code-resolver.d.ts +1 -1
  370. package/dist/server/status-code-resolver.d.ts.map +1 -1
  371. package/dist/server/stream-utils.d.ts +36 -0
  372. package/dist/server/stream-utils.d.ts.map +1 -0
  373. package/dist/server/tracing.d.ts +4 -4
  374. package/dist/server/tracing.d.ts.map +1 -1
  375. package/dist/server/tree-builder.d.ts +22 -19
  376. package/dist/server/tree-builder.d.ts.map +1 -1
  377. package/dist/server/types.d.ts +1 -4
  378. package/dist/server/types.d.ts.map +1 -1
  379. package/dist/server/version-skew.d.ts +61 -0
  380. package/dist/server/version-skew.d.ts.map +1 -0
  381. package/dist/shared/merge-search-params.d.ts +22 -0
  382. package/dist/shared/merge-search-params.d.ts.map +1 -0
  383. package/dist/shims/font-google.d.ts +1 -1
  384. package/dist/shims/font-google.d.ts.map +1 -1
  385. package/dist/shims/font-google.js +42 -0
  386. package/dist/shims/font-google.js.map +1 -0
  387. package/dist/shims/font-local.d.ts +26 -0
  388. package/dist/shims/font-local.d.ts.map +1 -0
  389. package/dist/shims/font-local.js +20 -0
  390. package/dist/shims/font-local.js.map +1 -0
  391. package/dist/shims/headers.d.ts +2 -1
  392. package/dist/shims/headers.d.ts.map +1 -1
  393. package/dist/shims/navigation-client.d.ts +1 -1
  394. package/dist/shims/navigation-client.d.ts.map +1 -1
  395. package/dist/shims/navigation.d.ts +3 -2
  396. package/dist/shims/navigation.d.ts.map +1 -1
  397. package/dist/utils/directive-parser.d.ts +5 -2
  398. package/dist/utils/directive-parser.d.ts.map +1 -1
  399. package/dist/utils/state-machine.d.ts +80 -0
  400. package/dist/utils/state-machine.d.ts.map +1 -0
  401. package/package.json +51 -16
  402. package/src/adapters/cloudflare-dev.ts +177 -0
  403. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  404. package/src/adapters/cloudflare.ts +342 -28
  405. package/src/adapters/compress-module.ts +24 -4
  406. package/src/adapters/nitro.ts +52 -8
  407. package/src/adapters/wrangler.d.ts +7 -0
  408. package/src/cache/cache-api.ts +38 -0
  409. package/src/cache/handler-store.ts +68 -0
  410. package/src/cache/index.ts +81 -18
  411. package/src/cache/singleflight.ts +62 -4
  412. package/src/cache/sizeof.ts +31 -0
  413. package/src/cache/timber-cache.ts +24 -20
  414. package/src/cli.ts +16 -6
  415. package/src/client/browser-dev.ts +128 -1
  416. package/src/client/browser-entry/action-dispatch.ts +116 -0
  417. package/src/client/browser-entry/hmr.ts +81 -0
  418. package/src/client/browser-entry/hydrate.ts +145 -0
  419. package/src/client/browser-entry/index.ts +143 -0
  420. package/src/client/browser-entry/post-hydration.ts +119 -0
  421. package/src/client/browser-entry/router-init.ts +193 -0
  422. package/src/client/browser-entry/rsc-stream.ts +157 -0
  423. package/src/client/browser-entry/scroll.ts +27 -0
  424. package/src/client/error-boundary.tsx +48 -16
  425. package/src/client/error-reconstituter.tsx +65 -0
  426. package/src/client/form.tsx +14 -7
  427. package/src/client/history.ts +26 -4
  428. package/src/client/index.ts +65 -38
  429. package/src/client/internal.ts +57 -0
  430. package/src/client/link-pending-store.ts +111 -0
  431. package/src/client/link.tsx +342 -113
  432. package/src/client/nav-link-store.ts +47 -0
  433. package/src/client/navigation-api-types.ts +112 -0
  434. package/src/client/navigation-api.ts +332 -0
  435. package/src/client/navigation-context.ts +31 -6
  436. package/src/client/navigation-root.tsx +342 -0
  437. package/src/client/nuqs-adapter.tsx +16 -3
  438. package/src/client/router-ref.ts +1 -1
  439. package/src/client/router.ts +299 -72
  440. package/src/client/rsc-fetch.ts +97 -8
  441. package/src/client/segment-cache.ts +1 -1
  442. package/src/client/segment-outlet.tsx +86 -0
  443. package/src/client/ssr-data.ts +13 -5
  444. package/src/client/stale-reload.ts +72 -3
  445. package/src/client/top-loader.tsx +18 -6
  446. package/src/client/use-link-status.ts +7 -7
  447. package/src/client/use-params.ts +7 -5
  448. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  449. package/src/client/use-query-states.ts +9 -3
  450. package/src/client/use-router.ts +1 -1
  451. package/src/codec.ts +49 -0
  452. package/src/config-types.ts +264 -0
  453. package/src/config-validation.ts +303 -0
  454. package/src/content/index.ts +5 -13
  455. package/src/cookies/define-cookie.ts +78 -25
  456. package/src/cookies/index.ts +8 -0
  457. package/src/fonts/bundle.ts +142 -0
  458. package/src/fonts/css.ts +2 -1
  459. package/src/fonts/dev-middleware.ts +74 -0
  460. package/src/fonts/pipeline.ts +275 -0
  461. package/src/fonts/transform.ts +353 -0
  462. package/src/fonts/types.ts +50 -1
  463. package/src/fonts/virtual-modules.ts +159 -0
  464. package/src/index.ts +314 -355
  465. package/src/plugin-context.ts +240 -0
  466. package/src/plugins/adapter-build.ts +9 -3
  467. package/src/plugins/build-manifest.ts +13 -2
  468. package/src/plugins/build-report.ts +3 -3
  469. package/src/plugins/client-chunks.ts +65 -0
  470. package/src/plugins/content.ts +1 -1
  471. package/src/plugins/dev-404-page.ts +418 -0
  472. package/src/plugins/dev-browser-logs.ts +288 -0
  473. package/src/plugins/dev-error-overlay.ts +286 -42
  474. package/src/plugins/dev-error-page.ts +536 -0
  475. package/src/plugins/dev-logs.ts +1 -1
  476. package/src/plugins/dev-server.ts +146 -19
  477. package/src/plugins/dev-terminal-error.ts +217 -0
  478. package/src/plugins/entries.ts +111 -10
  479. package/src/plugins/fonts.ts +133 -638
  480. package/src/plugins/mdx.ts +1 -1
  481. package/src/plugins/routing.ts +213 -31
  482. package/src/plugins/server-action-exports.ts +1 -1
  483. package/src/plugins/server-bundle.ts +32 -1
  484. package/src/plugins/shims.ts +136 -35
  485. package/src/plugins/static-build.ts +17 -11
  486. package/src/routing/codegen-shared.ts +74 -0
  487. package/src/routing/codegen-types.ts +37 -0
  488. package/src/routing/codegen.ts +112 -173
  489. package/src/routing/convention-lint.ts +356 -0
  490. package/src/routing/index.ts +2 -0
  491. package/src/routing/link-codegen.ts +262 -0
  492. package/src/routing/scanner.ts +93 -23
  493. package/src/routing/segment-classify.ts +89 -0
  494. package/src/routing/status-file-lint.ts +3 -2
  495. package/src/routing/types.ts +17 -4
  496. package/src/rsc-runtime/rsc.ts +2 -0
  497. package/src/rsc-runtime/ssr.ts +50 -0
  498. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  499. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  500. package/src/search-params/define.ts +482 -0
  501. package/src/search-params/index.ts +14 -20
  502. package/src/search-params/registry.ts +2 -2
  503. package/src/search-params/wrappers.ts +85 -0
  504. package/src/segment-params/define.ts +279 -0
  505. package/src/segment-params/index.ts +9 -0
  506. package/src/server/access-gate.tsx +70 -29
  507. package/src/server/action-client.ts +88 -15
  508. package/src/server/action-encryption.ts +144 -0
  509. package/src/server/action-handler.ts +53 -6
  510. package/src/server/actions.ts +10 -9
  511. package/src/server/als-registry.ts +34 -6
  512. package/src/server/build-manifest.ts +10 -4
  513. package/src/server/compress.ts +25 -7
  514. package/src/server/debug.ts +1 -1
  515. package/src/server/default-logger.ts +99 -0
  516. package/src/server/deny-page-resolver.ts +154 -0
  517. package/src/server/deny-renderer.ts +24 -38
  518. package/src/server/dev-holding-server.ts +185 -0
  519. package/src/server/dev-source-map.ts +31 -0
  520. package/src/server/dev-warnings.ts +4 -49
  521. package/src/server/early-hints.ts +36 -15
  522. package/src/server/error-boundary-wrapper.ts +74 -22
  523. package/src/server/fallback-error.ts +74 -102
  524. package/src/server/flight-injection-state.ts +113 -0
  525. package/src/server/flight-scripts.ts +62 -0
  526. package/src/server/flush.ts +2 -1
  527. package/src/server/form-data.ts +76 -0
  528. package/src/server/html-injectors.ts +280 -120
  529. package/src/server/index.ts +25 -177
  530. package/src/server/internal.ts +169 -0
  531. package/src/server/logger.ts +44 -36
  532. package/src/server/middleware-runner.ts +31 -4
  533. package/src/server/node-stream-transforms.ts +509 -0
  534. package/src/server/page-deny-boundary.tsx +56 -0
  535. package/src/server/pipeline-interception.ts +17 -16
  536. package/src/server/pipeline-metadata.ts +13 -0
  537. package/src/server/pipeline.ts +261 -66
  538. package/src/server/primitives.ts +111 -28
  539. package/src/server/render-timeout.ts +108 -0
  540. package/src/server/request-context.ts +293 -132
  541. package/src/server/route-element-builder.ts +283 -191
  542. package/src/server/route-handler.ts +24 -4
  543. package/src/server/route-matcher.ts +31 -20
  544. package/src/server/rsc-entry/api-handler.ts +15 -16
  545. package/src/server/rsc-entry/error-renderer.ts +305 -89
  546. package/src/server/rsc-entry/helpers.ts +134 -5
  547. package/src/server/rsc-entry/index.ts +304 -111
  548. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  549. package/src/server/rsc-entry/rsc-stream.ts +81 -13
  550. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  551. package/src/server/rsc-entry/ssr-renderer.ts +171 -38
  552. package/src/server/safe-load.ts +60 -0
  553. package/src/server/sensitive-fields.ts +230 -0
  554. package/src/server/sitemap-generator.ts +338 -0
  555. package/src/server/sitemap-handler.ts +126 -0
  556. package/src/server/slot-resolver.ts +244 -229
  557. package/src/server/ssr-entry.ts +215 -32
  558. package/src/server/ssr-render.ts +289 -67
  559. package/src/server/ssr-wrappers.tsx +139 -0
  560. package/src/server/status-code-resolver.ts +1 -1
  561. package/src/server/stream-utils.ts +213 -0
  562. package/src/server/tracing.ts +20 -9
  563. package/src/server/tree-builder.ts +92 -58
  564. package/src/server/types.ts +3 -6
  565. package/src/server/version-skew.ts +104 -0
  566. package/src/shared/merge-search-params.ts +55 -0
  567. package/src/shims/font-google.ts +1 -1
  568. package/src/shims/font-local.ts +34 -0
  569. package/src/shims/headers.ts +5 -1
  570. package/src/shims/navigation-client.ts +1 -1
  571. package/src/shims/navigation.ts +7 -2
  572. package/src/utils/directive-parser.ts +5 -2
  573. package/src/utils/state-machine.ts +111 -0
  574. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  575. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  576. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  577. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  578. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  579. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  580. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  581. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  582. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  583. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  584. package/dist/cache/register-cached-function.d.ts +0 -17
  585. package/dist/cache/register-cached-function.d.ts.map +0 -1
  586. package/dist/client/browser-entry.d.ts +0 -21
  587. package/dist/client/browser-entry.d.ts.map +0 -1
  588. package/dist/client/link-status-provider.d.ts +0 -11
  589. package/dist/client/link-status-provider.d.ts.map +0 -1
  590. package/dist/client/transition-root.d.ts.map +0 -1
  591. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  592. package/dist/cookies/index.js.map +0 -1
  593. package/dist/plugins/cache-transform.d.ts +0 -36
  594. package/dist/plugins/cache-transform.d.ts.map +0 -1
  595. package/dist/plugins/dynamic-transform.d.ts +0 -72
  596. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  597. package/dist/search-params/analyze.d.ts +0 -54
  598. package/dist/search-params/analyze.d.ts.map +0 -1
  599. package/dist/search-params/builtin-codecs.d.ts +0 -105
  600. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  601. package/dist/search-params/codecs.d.ts +0 -53
  602. package/dist/search-params/codecs.d.ts.map +0 -1
  603. package/dist/search-params/create.d.ts +0 -106
  604. package/dist/search-params/create.d.ts.map +0 -1
  605. package/dist/server/prerender.d.ts +0 -77
  606. package/dist/server/prerender.d.ts.map +0 -1
  607. package/dist/server/response-cache.d.ts +0 -54
  608. package/dist/server/response-cache.d.ts.map +0 -1
  609. package/src/cache/register-cached-function.ts +0 -103
  610. package/src/client/browser-entry.ts +0 -678
  611. package/src/client/link-status-provider.tsx +0 -30
  612. package/src/client/transition-root.tsx +0 -166
  613. package/src/plugins/cache-transform.ts +0 -199
  614. package/src/plugins/dynamic-transform.ts +0 -161
  615. package/src/search-params/analyze.ts +0 -192
  616. package/src/search-params/builtin-codecs.ts +0 -228
  617. package/src/search-params/create.ts +0 -321
  618. package/src/server/prerender.ts +0 -139
  619. package/src/server/response-cache.ts +0 -410
@@ -0,0 +1,356 @@
1
+ /**
2
+ * Convention linter — validates common misconfigurations in the route tree.
3
+ *
4
+ * Runs at scan time (build and dev startup). Each check produces a warning
5
+ * with the file path, what's wrong, and what to do about it.
6
+ *
7
+ * These are warnings, not errors — they don't block the build. The goal is
8
+ * to catch issues that would otherwise produce cryptic runtime behavior
9
+ * (silent 404s, empty pages, confusing React errors).
10
+ *
11
+ * Design doc: 07-routing.md, 10-error-handling.md
12
+ */
13
+
14
+ import { readFileSync, existsSync } from 'node:fs';
15
+ import type { RouteTree, SegmentNode } from './types.js';
16
+
17
+ // ─── Types ──────────────────────────────────────────────────────────────────
18
+
19
+ export interface ConventionWarning {
20
+ /** Warning ID for deduplication and filtering. */
21
+ id: string;
22
+ /** Human-readable single-line summary. */
23
+ summary: string;
24
+ /** Multi-line details with file path and fix suggestion. */
25
+ details: string;
26
+ /** Severity: 'warn' for potential issues, 'error' for definite misconfigurations. */
27
+ level: 'warn' | 'error';
28
+ }
29
+
30
+ // ─── Lint Rules ─────────────────────────────────────────────────────────────
31
+
32
+ /**
33
+ * Run all convention lint checks on a route tree.
34
+ *
35
+ * Returns an array of warnings. Empty array means everything looks good.
36
+ */
37
+ export function lintConventions(tree: RouteTree, appDir: string): ConventionWarning[] {
38
+ const warnings: ConventionWarning[] = [];
39
+
40
+ // Check 1: app/ directory exists (caller should check before scanning,
41
+ // but we validate the root has at least one routable file)
42
+ checkEmptyApp(tree.root, appDir, warnings);
43
+
44
+ // Check 2: route.ts without recognized HTTP method exports
45
+ checkRouteExports(tree.root, warnings);
46
+
47
+ // Check 3: Segments with layout but no page/route anywhere in subtree
48
+ // (not a misconfiguration per se — layouts without pages are valid for
49
+ // route groups. Skip this check.)
50
+
51
+ // Check 4: Root segment has no layout.tsx (no HTML shell)
52
+ checkRootLayout(tree.root, warnings);
53
+
54
+ // Check 5: page.tsx / layout.tsx without default export
55
+ checkDefaultExports(tree.root, warnings);
56
+
57
+ return warnings;
58
+ }
59
+
60
+ // ─── Check: Empty App ───────────────────────────────────────────────────────
61
+
62
+ /**
63
+ * Warn when the app/ directory has no routable files at all.
64
+ *
65
+ * This catches the "I created app/ but didn't add any pages" case where
66
+ * every request silently 404s with no guidance.
67
+ */
68
+ function checkEmptyApp(root: SegmentNode, appDir: string, warnings: ConventionWarning[]): void {
69
+ if (hasAnyRoutable(root)) return;
70
+
71
+ warnings.push({
72
+ id: 'EMPTY_APP',
73
+ summary: 'No pages or route handlers found in app/',
74
+ details:
75
+ ` Directory: ${appDir}\n\n` +
76
+ ' Your app/ directory has no page.tsx or route.ts files.\n' +
77
+ ' Every request will return 404.\n\n' +
78
+ ' To fix: Create app/page.tsx with a default export:\n\n' +
79
+ ' export default function Home() {\n return <h1>Hello</h1>;\n }\n',
80
+ level: 'warn',
81
+ });
82
+ }
83
+
84
+ /**
85
+ * Check if a segment tree has any routable files (page or route) anywhere.
86
+ */
87
+ function hasAnyRoutable(node: SegmentNode): boolean {
88
+ if (node.page || node.route) return true;
89
+ for (const child of node.children) {
90
+ if (hasAnyRoutable(child)) return true;
91
+ }
92
+ for (const [, slot] of node.slots) {
93
+ if (hasAnyRoutable(slot)) return true;
94
+ }
95
+ return false;
96
+ }
97
+
98
+ /**
99
+ * Check if a segment tree has any page files (not just route handlers).
100
+ * Used by checkRootLayout to avoid false positives for API-only apps (TIM-794).
101
+ * API-only apps (just route.ts handlers) don't need a root layout.
102
+ */
103
+ function hasAnyPage(node: SegmentNode): boolean {
104
+ if (node.page) return true;
105
+ for (const child of node.children) {
106
+ if (hasAnyPage(child)) return true;
107
+ }
108
+ for (const [, slot] of node.slots) {
109
+ if (hasAnyPage(slot)) return true;
110
+ }
111
+ return false;
112
+ }
113
+
114
+ // ─── Check: Route Exports ───────────────────────────────────────────────────
115
+
116
+ /** HTTP methods that route.ts can export. */
117
+ const HTTP_METHODS = ['GET', 'POST', 'PUT', 'PATCH', 'DELETE', 'HEAD', 'OPTIONS'];
118
+
119
+ /**
120
+ * Pattern to detect named exports that look like HTTP method handlers.
121
+ * Matches: export function GET, export const GET, export async function POST, etc.
122
+ * Also matches: export { GET } or re-exports like export { GET } from './handler'.
123
+ *
124
+ * The re-export branch uses word boundaries (\b) around method names to avoid
125
+ * matching substrings (e.g. TARGET should not match GET). See TIM-795.
126
+ */
127
+ const EXPORT_PATTERN = new RegExp(
128
+ `export\\s+(?:async\\s+)?(?:function|const|let|var)\\s+(${HTTP_METHODS.join('|')})\\b` +
129
+ `|export\\s*\\{[^}]*\\b(${HTTP_METHODS.join('|')})\\b[^}]*\\}`
130
+ );
131
+
132
+ /**
133
+ * Warn when a route.ts file doesn't appear to export any recognized HTTP methods.
134
+ *
135
+ * Uses static analysis (regex on source text) — not module loading.
136
+ * This is intentionally conservative: it may miss complex re-exports but
137
+ * catches the common case of an empty route.ts or one with wrong export names.
138
+ */
139
+ function checkRouteExports(node: SegmentNode, warnings: ConventionWarning[]): void {
140
+ if (node.route) {
141
+ const filePath = node.route.filePath;
142
+ try {
143
+ const source = readFileSync(filePath, 'utf-8');
144
+ if (!EXPORT_PATTERN.test(source)) {
145
+ // Check if there's a default export (common mistake)
146
+ const hasDefaultExport = /export\s+default\b/.test(source);
147
+ const hint = hasDefaultExport
148
+ ? ' It looks like you have a default export. route.ts uses named exports\n' +
149
+ ' for HTTP methods (GET, POST, etc.), not a default export.\n'
150
+ : '';
151
+
152
+ warnings.push({
153
+ id: 'ROUTE_NO_METHODS',
154
+ summary: `route.ts has no HTTP method exports: ${filePath}`,
155
+ details:
156
+ ` File: ${filePath}\n\n` +
157
+ ' route.ts files must export named HTTP method handlers.\n' +
158
+ ' Without them, the route matches but returns 405 Method Not Allowed.\n\n' +
159
+ hint +
160
+ ' Example:\n\n' +
161
+ ' export function GET(ctx: RouteContext) {\n' +
162
+ " return Response.json({ hello: 'world' });\n" +
163
+ ' }\n',
164
+ level: 'warn',
165
+ });
166
+ }
167
+ } catch {
168
+ // Can't read the file — skip silently
169
+ }
170
+ }
171
+
172
+ // Recurse
173
+ for (const child of node.children) {
174
+ checkRouteExports(child, warnings);
175
+ }
176
+ for (const [, slot] of node.slots) {
177
+ checkRouteExports(slot, warnings);
178
+ }
179
+ }
180
+
181
+ // ─── Check: Root Layout ─────────────────────────────────────────────────────
182
+
183
+ /**
184
+ * Warn when the root segment has no layout.tsx.
185
+ *
186
+ * Without a root layout, there's no HTML shell (<html>, <body>).
187
+ * The page will render but may have hydration issues or no proper document structure.
188
+ */
189
+ function checkRootLayout(root: SegmentNode, warnings: ConventionWarning[]): void {
190
+ if (root.layout) return;
191
+
192
+ // Only warn if there are pages — API-only apps (just route.ts handlers)
193
+ // don't need a root layout (TIM-794).
194
+ if (!hasAnyPage(root)) return;
195
+
196
+ warnings.push({
197
+ id: 'NO_ROOT_LAYOUT',
198
+ summary: 'No root layout.tsx found',
199
+ details:
200
+ ' Your app has pages but no root layout.\n' +
201
+ ' Without app/layout.tsx, pages have no <html> or <body> wrapper.\n\n' +
202
+ ' To fix: Create app/layout.tsx:\n\n' +
203
+ ' export default function RootLayout({ children }: { children: React.ReactNode }) {\n' +
204
+ ' return (\n' +
205
+ ' <html lang="en">\n' +
206
+ ' <body>{children}</body>\n' +
207
+ ' </html>\n' +
208
+ ' );\n' +
209
+ ' }\n',
210
+ level: 'warn',
211
+ });
212
+ }
213
+
214
+ // ─── Check: Default Exports ───────────────────────────────────────────
215
+
216
+ /**
217
+ * Pattern to detect a default export in source code.
218
+ * Matches: export default function, export default class, export default
219
+ * Also matches: export { X as default }, export { default } from './Impl'
220
+ *
221
+ * The third alternative catches bare `default` re-exports (TIM-790).
222
+ * The negative lookahead (?!\s+as\b) prevents matching `export { default as X }`
223
+ * which is a named export, not a default export (TIM-806).
224
+ */
225
+ const DEFAULT_EXPORT_PATTERN =
226
+ /export\s+default\b|export\s*\{[^}]*\bas\s+default\b|export\s*\{[^}]*\bdefault\b(?!\s+as\b)[^}]*\}/;
227
+
228
+ /**
229
+ * Warn when page.tsx or layout.tsx files don't have a default export.
230
+ *
231
+ * Without a default export:
232
+ * - page.tsx: the page renders nothing (empty content)
233
+ * - layout.tsx: the layout module loads but has no component to render
234
+ *
235
+ * Uses static analysis (regex on source text) — intentionally conservative.
236
+ */
237
+ function checkDefaultExports(node: SegmentNode, warnings: ConventionWarning[]): void {
238
+ // Check page files (tsx/ts/jsx/js only — not mdx, which is default-exported by the mdx compiler)
239
+ if (node.page && isScriptExtension(node.page.extension)) {
240
+ checkFileDefaultExport(node.page.filePath, 'page', warnings);
241
+ }
242
+
243
+ // Check layout files
244
+ if (node.layout && isScriptExtension(node.layout.extension)) {
245
+ checkFileDefaultExport(node.layout.filePath, 'layout', warnings);
246
+ }
247
+
248
+ // Recurse
249
+ for (const child of node.children) {
250
+ checkDefaultExports(child, warnings);
251
+ }
252
+ for (const [, slot] of node.slots) {
253
+ checkDefaultExports(slot, warnings);
254
+ }
255
+ }
256
+
257
+ function isScriptExtension(ext: string): boolean {
258
+ return ext === 'tsx' || ext === 'ts' || ext === 'jsx' || ext === 'js';
259
+ }
260
+
261
+ function checkFileDefaultExport(
262
+ filePath: string,
263
+ fileType: string,
264
+ warnings: ConventionWarning[]
265
+ ): void {
266
+ try {
267
+ const source = readFileSync(filePath, 'utf-8');
268
+ if (!DEFAULT_EXPORT_PATTERN.test(source)) {
269
+ warnings.push({
270
+ id: `NO_DEFAULT_EXPORT:${filePath}`,
271
+ summary: `${fileType}.tsx has no default export: ${filePath}`,
272
+ details:
273
+ ` File: ${filePath}\n\n` +
274
+ ` ${fileType}.tsx files must export a default React component.\n` +
275
+ ` Without a default export, the ${fileType === 'page' ? 'page renders nothing' : 'layout has no component to wrap children'}.\n\n` +
276
+ ` To fix: Add a default export:\n\n` +
277
+ ` export default function My${fileType === 'page' ? 'Page' : 'Layout'}(${fileType === 'layout' ? '{ children }' : ''}) {\n` +
278
+ ` return ${fileType === 'layout' ? '<div>{children}</div>' : '<h1>Hello</h1>'};\n` +
279
+ ` }\n`,
280
+ level: 'warn',
281
+ });
282
+ }
283
+ } catch {
284
+ // Can't read file — skip silently
285
+ }
286
+ }
287
+
288
+ // ─── Check: App Directory Exists ────────────────────────────────────────────
289
+
290
+ /**
291
+ * Check if the app/ directory exists. Called before scanning.
292
+ * Returns a warning if missing, or null if the directory exists.
293
+ */
294
+ export function checkAppDirExists(appDir: string): ConventionWarning | null {
295
+ if (existsSync(appDir)) return null;
296
+
297
+ return {
298
+ id: 'NO_APP_DIR',
299
+ summary: 'No app/ directory found',
300
+ details:
301
+ ` Expected: ${appDir}\n\n` +
302
+ ' timber.js requires an app/ directory for file-system routing.\n' +
303
+ ' Every request will return 404 until you create it.\n\n' +
304
+ ' To fix: Create the app/ directory with a page:\n\n' +
305
+ ' mkdir -p app\n' +
306
+ ' # Create app/layout.tsx and app/page.tsx\n',
307
+ level: 'error',
308
+ };
309
+ }
310
+
311
+ // ─── Formatting ─────────────────────────────────────────────────────────────
312
+
313
+ const YELLOW = '\x1b[33m';
314
+ const RED = '\x1b[31m';
315
+ const BOLD = '\x1b[1m';
316
+ const DIM = '\x1b[2m';
317
+ const RESET = '\x1b[0m';
318
+
319
+ /**
320
+ * Format warnings for terminal output.
321
+ *
322
+ * Groups by severity, uses colors, and includes fix suggestions.
323
+ */
324
+ export function formatConventionWarnings(warnings: ConventionWarning[]): string {
325
+ if (warnings.length === 0) return '';
326
+
327
+ const errors = warnings.filter((w) => w.level === 'error');
328
+ const warns = warnings.filter((w) => w.level === 'warn');
329
+
330
+ const lines: string[] = [];
331
+
332
+ if (errors.length > 0) {
333
+ lines.push(
334
+ `${RED}${BOLD}[timber]${RESET} ${RED}${errors.length} configuration error${errors.length !== 1 ? 's' : ''}:${RESET}`
335
+ );
336
+ for (const e of errors) {
337
+ lines.push('');
338
+ lines.push(` ${RED}✗${RESET} ${e.summary}`);
339
+ lines.push(`${DIM}${e.details}${RESET}`);
340
+ }
341
+ }
342
+
343
+ if (warns.length > 0) {
344
+ if (errors.length > 0) lines.push('');
345
+ lines.push(
346
+ `${YELLOW}${BOLD}[timber]${RESET} ${YELLOW}${warns.length} configuration warning${warns.length !== 1 ? 's' : ''}:${RESET}`
347
+ );
348
+ for (const w of warns) {
349
+ lines.push('');
350
+ lines.push(` ${YELLOW}⚠${RESET} ${w.summary}`);
351
+ lines.push(`${DIM}${w.details}${RESET}`);
352
+ }
353
+ }
354
+
355
+ return lines.join('\n');
356
+ }
@@ -12,3 +12,5 @@ export type {
12
12
  export { DEFAULT_PAGE_EXTENSIONS, INTERCEPTION_MARKERS } from './types.js';
13
13
  export { collectInterceptionRewrites } from './interception.js';
14
14
  export type { InterceptionRewrite } from './interception.js';
15
+ export { classifyUrlSegment } from './segment-classify.js';
16
+ export type { UrlSegment } from './segment-classify.js';
@@ -0,0 +1,262 @@
1
+ /**
2
+ * Typed `<Link>` codegen — interface augmentation generation.
3
+ *
4
+ * Extracted from `codegen.ts` to keep that file under the project's
5
+ * 500-line cap. This module owns everything that emits the
6
+ * `interface LinkFunction { ... }` augmentation blocks in the generated
7
+ * `.timber/timber-routes.d.ts`.
8
+ *
9
+ * Two augmentation blocks are emitted:
10
+ *
11
+ * 1. **Per-route discriminated union** (`formatTypedLinkOverloads`) —
12
+ * one call signature whose props is a union keyed on `href`. TS
13
+ * narrows by literal href and reports prop errors against the
14
+ * matched variant. See TIM-835 and `design/09-typescript.md`.
15
+ *
16
+ * 2. **Catch-all overloads** (`formatLinkCatchAllOverloads`) — external
17
+ * href literals (`http://`, `mailto:`, etc.) and a computed-string
18
+ * `<H extends string>` signature for runtime-computed paths.
19
+ *
20
+ * Block ordering is critical for error UX: per-route is emitted FIRST
21
+ * so that, after TS's "later overload set ordered first" merge rule,
22
+ * the discriminated union ends up LAST in resolution order — the
23
+ * overload TS reports against on failure.
24
+ */
25
+
26
+ import type { ParamEntry, RouteEntry } from './codegen-types.js';
27
+ import { buildCodecChainType, formatSearchParamsType } from './codegen-shared.js';
28
+
29
+ /** Shared Link base-props type literal used in every emitted call signature. */
30
+ export const LINK_BASE_PROPS_TYPE =
31
+ "Omit<import('react').AnchorHTMLAttributes<HTMLAnchorElement>, 'href'> & { prefetch?: boolean; scroll?: boolean; preserveSearchParams?: true | string[]; onNavigate?: import('./client/link.js').OnNavigateHandler; children?: import('react').ReactNode }";
32
+
33
+ /**
34
+ * Build a TypeScript template literal pattern for a dynamic route.
35
+ * e.g. '/products/[id]' → '/products/${string}'
36
+ * '/blog/[...slug]' → '/blog/${string}'
37
+ * '/docs/[[...path]]' → '/docs/${string}' (also matches /docs)
38
+ * '/[org]/[repo]' → '/${string}/${string}'
39
+ */
40
+ export function buildResolvedPattern(route: RouteEntry): string | null {
41
+ const parts = route.urlPath.split('/');
42
+ const templateParts = parts.map((part) => {
43
+ if (part.startsWith('[[...') && part.endsWith(']]')) {
44
+ // Optional catch-all — matches any trailing path
45
+ return '${string}';
46
+ }
47
+ if (part.startsWith('[...') && part.endsWith(']')) {
48
+ // Catch-all — matches one or more segments
49
+ return '${string}';
50
+ }
51
+ if (part.startsWith('[') && part.endsWith(']')) {
52
+ // Dynamic segment
53
+ return '${string}';
54
+ }
55
+ return part;
56
+ });
57
+ return templateParts.join('/');
58
+ }
59
+
60
+ /**
61
+ * Format the segmentParams type for Link overloads.
62
+ *
63
+ * Link params accept `string | number` for single dynamic segments
64
+ * (convenience — values are stringified at runtime). Catch-all and
65
+ * optional catch-all remain `string[]` / `string[] | undefined`.
66
+ *
67
+ * When the segment's params chain (TIM-834) declares a typed codec, the
68
+ * inferred type from the codec wins via a nested conditional.
69
+ */
70
+ export function formatLinkParamsType(params: ParamEntry[], importBase?: string): string {
71
+ if (params.length === 0) {
72
+ return '{}';
73
+ }
74
+
75
+ const fields = params.map((p) => {
76
+ const fallback = p.type === 'string' ? 'string | number' : p.type;
77
+ const codecType = buildCodecChainType(p, importBase, fallback);
78
+ return `${p.name}: ${codecType}`;
79
+ });
80
+ return `{ ${fields.join('; ')} }`;
81
+ }
82
+
83
+ /**
84
+ * Catch-all call signatures for `<Link>` — external hrefs and computed
85
+ * `string` variables. Emitted from codegen in a SEPARATE
86
+ * `declare module` block (declared AFTER the per-route block) so the TS
87
+ * "later overload set ordered first" rule places these catch-all
88
+ * signatures ahead of per-route in resolution order, leaving per-route
89
+ * as the final overload whose error message is reported on failure.
90
+ *
91
+ * The conditional `string extends H ? ... : never` protection preserves
92
+ * TIM-624's guarantee that unknown internal path literals don't match
93
+ * the catch-all — typos like `<Link href="/typo" />` still error.
94
+ */
95
+ export function formatLinkCatchAllOverloads(): string[] {
96
+ const lines: string[] = [];
97
+ const baseProps = LINK_BASE_PROPS_TYPE;
98
+
99
+ // TIM-830: the catch-all signatures accept EITHER the legacy
100
+ // `{ definition, values }` wrapper OR a flat `Record<string, unknown>`
101
+ // values object. External/computed hrefs can't be looked up in the
102
+ // runtime registry, so the wrapped form is still the reliable path;
103
+ // the flat form is kept permissive for callers migrating from typed-
104
+ // route hrefs to computed strings. `resolveHref` discriminates at
105
+ // runtime via the presence of a `definition` key.
106
+ const catchAllSearchParams =
107
+ '{ definition: SearchParamsDefinition<Record<string, unknown>>; values: Record<string, unknown> } | Record<string, unknown>';
108
+
109
+ // ExternalHref inlined here rather than referenced as an exported
110
+ // alias so the generated .d.ts stands alone without source imports.
111
+ const externalHref =
112
+ '`http://${string}` | `https://${string}` | `mailto:${string}` | `tel:${string}` | `ftp://${string}` | `//${string}` | `#${string}` | `?${string}`';
113
+
114
+ lines.push(' // Typed Link overloads — catch-all (block 2 / emitted second)');
115
+ lines.push(' interface LinkFunction {');
116
+
117
+ // (1) External/literal-protocol hrefs.
118
+ lines.push(` (props: ${baseProps} & {`);
119
+ lines.push(` href: ${externalHref}`);
120
+ lines.push(` segmentParams?: never`);
121
+ lines.push(` searchParams?: ${catchAllSearchParams}`);
122
+ lines.push(` }): import('react').JSX.Element`);
123
+
124
+ // (2) Computed/variable href — non-literal `string` only.
125
+ // `string extends H` is true only when H is the wide `string` type,
126
+ // not a specific literal. Literal internal paths (typos) that don't
127
+ // match any per-route signature collapse to `never` here, but since
128
+ // this block is NOT the "last overload" at resolution time, TS
129
+ // instead reports the error from the per-route block — which now
130
+ // says `Type '"/typo"' is not assignable to type '"/products/[id]"'`
131
+ // or similar per-route-specific guidance.
132
+ lines.push(` <H extends string>(`);
133
+ lines.push(` props: string extends H`);
134
+ lines.push(` ? ${baseProps} & {`);
135
+ lines.push(` href: H`);
136
+ lines.push(` segmentParams?: Record<string, string | number | string[]>`);
137
+ lines.push(` searchParams?: ${catchAllSearchParams}`);
138
+ lines.push(` }`);
139
+ lines.push(` : never`);
140
+ lines.push(` ): import('react').JSX.Element`);
141
+
142
+ lines.push(' }');
143
+ return lines;
144
+ }
145
+
146
+ /**
147
+ * Generate typed per-route Link call signatures via LinkFunction
148
+ * interface merging.
149
+ *
150
+ * TIM-835: This emits a SINGLE call signature whose props is a
151
+ * discriminated union keyed on `href`. TypeScript narrows the union by
152
+ * the literal `href` at the call site, then checks the rest of the
153
+ * props against the matched variant. When `segmentParams` or
154
+ * `searchParams` is wrong, TS reports the error against the matched
155
+ * variant — naming the user's actual `href` and the offending field.
156
+ *
157
+ * Before TIM-835, this function emitted N separate per-route overloads
158
+ * (one per route, sometimes two for dynamic routes). When ALL overloads
159
+ * failed, TS would pick an arbitrary failed overload (heuristically the
160
+ * "last tried") to render the diagnostic, often pointing at a route
161
+ * completely unrelated to the one the user wrote. The discriminated
162
+ * union sidesteps overload-resolution heuristics entirely.
163
+ *
164
+ * Open property shapes (`Record<string, unknown>` instead of `never`)
165
+ * for `segmentParams` on routes that don't declare them keep generic
166
+ * `LinkProps`-spreading wrappers compiling. (Aligned with TIM-833.)
167
+ *
168
+ * TIM-832: this function still emits the per-route block FIRST and the
169
+ * catch-all block follows, so per-route remains the LAST overload set in
170
+ * resolution order — the one TS reports against on failure. With a
171
+ * discriminated union there is only one call signature in this block,
172
+ * so the resolved-template-vs-pattern ordering reduces to placing the
173
+ * pattern variant before the resolved-template variant inside the union.
174
+ */
175
+ export function formatTypedLinkOverloads(routes: RouteEntry[], importBase?: string): string[] {
176
+ const lines: string[] = [];
177
+ const baseProps = LINK_BASE_PROPS_TYPE;
178
+
179
+ // Build the union variants. Each route contributes one variant for the
180
+ // pattern href (e.g. '/products/[id]') and, for dynamic routes, an
181
+ // additional variant for the resolved-template href (e.g.
182
+ // `/products/${string}`).
183
+ //
184
+ // The PATTERN variant must be listed BEFORE the resolved-template
185
+ // variant for the same route so that when the user writes the literal
186
+ // pattern href (`<Link href="/products/[id]" />`), TS narrows to the
187
+ // pattern variant (which carries the typed `segmentParams` shape)
188
+ // rather than the looser resolved-template variant. Without this
189
+ // ordering, TS would silently match the resolved-template variant
190
+ // first — swallowing typed-segmentParams type errors.
191
+ const variants: string[] = [];
192
+ for (const route of routes) {
193
+ const hasDynamicParams = route.params.length > 0;
194
+ // For routes with no dynamic params, accept an absent OR open-shape
195
+ // segmentParams. `Record<string, unknown>` keeps generic spreads
196
+ // compiling and gives a readable error if a non-object is passed.
197
+ const paramsProp = hasDynamicParams
198
+ ? `segmentParams: ${formatLinkParamsType(route.params, importBase)}`
199
+ : 'segmentParams?: Record<string, unknown>';
200
+
201
+ const searchParamsType = route.hasSearchParams
202
+ ? formatSearchParamsType(route, importBase)
203
+ : null;
204
+ // TIM-830: pattern href uses the FLAT `Partial<T>` shape — the
205
+ // runtime registry is keyed by the un-interpolated pattern.
206
+ //
207
+ // TIM-835: when the route has NO searchParams definition, use
208
+ // `Record<string, never>` (forbids any keys) instead of
209
+ // `Record<string, unknown>` so passing unexpected query params is a
210
+ // type error. Generic-spread compatibility is preserved by the
211
+ // matching open shape on `segmentParams` (which is the prop users
212
+ // typically forward through wrapper components).
213
+ const patternSearchParamsProp = searchParamsType
214
+ ? `searchParams?: Partial<${searchParamsType}>`
215
+ : 'searchParams?: Record<string, never>';
216
+
217
+ // Pattern variant FIRST (more specific).
218
+ variants.push(
219
+ `${baseProps} & { href: '${route.urlPath}'; ${paramsProp}; ${patternSearchParamsProp} }`
220
+ );
221
+
222
+ // Resolved-template variant SECOND (looser, matches interpolated hrefs).
223
+ if (hasDynamicParams) {
224
+ const templatePattern = buildResolvedPattern(route);
225
+ if (templatePattern) {
226
+ // TIM-830: resolved-template href keeps the WRAPPED
227
+ // `{ definition, values }` shape — the registry can't be looked
228
+ // up by an already-interpolated href.
229
+ const resolvedSearchParamsProp = searchParamsType
230
+ ? `searchParams?: { definition: SearchParamsDefinition<${searchParamsType}>; values: Partial<${searchParamsType}> }`
231
+ : 'searchParams?: Record<string, never>';
232
+ // TIM-835: `Record<string, never>` for segmentParams forbids ANY
233
+ // provided keys on the resolved-template variant. Without this,
234
+ // the resolved-template would shadow the pattern variant for
235
+ // literal pattern hrefs (`<Link href="/products/[id]" segmentParams={...} />`)
236
+ // because both variants accept the literal `'/products/[id]'`
237
+ // and the looser variant would silently swallow segmentParams
238
+ // type errors.
239
+ variants.push(
240
+ `${baseProps} & { href: \`${templatePattern}\`; segmentParams?: Record<string, never>; ${resolvedSearchParamsProp} }`
241
+ );
242
+ }
243
+ }
244
+ }
245
+
246
+ lines.push(' interface LinkFunction {');
247
+ if (variants.length === 0) {
248
+ // No page routes — emit nothing. The catch-all block in the second
249
+ // augmentation still provides external/computed-string signatures.
250
+ lines.push(' }');
251
+ return lines;
252
+ }
253
+ lines.push(' (');
254
+ lines.push(' props:');
255
+ for (const variant of variants) {
256
+ lines.push(` | (${variant})`);
257
+ }
258
+ lines.push(` ): import('react').JSX.Element`);
259
+ lines.push(' }');
260
+
261
+ return lines;
262
+ }