@timber-js/app 0.2.0-alpha.8 → 0.2.0-alpha.81

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 (571) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/actions-Dg-ANYHb.js +421 -0
  3. package/dist/_chunks/actions-Dg-ANYHb.js.map +1 -0
  4. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
  5. package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
  6. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  7. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  8. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  9. package/dist/_chunks/define-C77ScO0m.js +106 -0
  10. package/dist/_chunks/define-C77ScO0m.js.map +1 -0
  11. package/dist/_chunks/define-CZqDwhSu.js +199 -0
  12. package/dist/_chunks/define-CZqDwhSu.js.map +1 -0
  13. package/dist/_chunks/define-cookie-C2IkoFGN.js +94 -0
  14. package/dist/_chunks/define-cookie-C2IkoFGN.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
  16. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
  17. package/dist/_chunks/format-CYBGxKtc.js +14 -0
  18. package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-Dpn_UfAD.js} +171 -97
  20. package/dist/_chunks/interception-Dpn_UfAD.js.map +1 -0
  21. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
  22. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
  23. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  24. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  25. package/dist/_chunks/request-context-qMsWgy9C.js +478 -0
  26. package/dist/_chunks/request-context-qMsWgy9C.js.map +1 -0
  27. package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
  28. package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
  29. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  30. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  31. package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
  32. package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
  33. package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
  34. package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
  35. package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
  36. package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
  37. package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
  38. package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
  39. package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
  40. package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
  41. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-Lo_s_pw2.js} +4 -4
  42. package/dist/_chunks/use-query-states-Lo_s_pw2.js.map +1 -0
  43. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  44. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  45. package/dist/adapters/cloudflare-dev.js +73 -0
  46. package/dist/adapters/cloudflare-dev.js.map +1 -0
  47. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  48. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  49. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  50. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  51. package/dist/adapters/cloudflare.d.ts +148 -12
  52. package/dist/adapters/cloudflare.d.ts.map +1 -1
  53. package/dist/adapters/cloudflare.js +135 -11
  54. package/dist/adapters/cloudflare.js.map +1 -1
  55. package/dist/adapters/compress-module.d.ts.map +1 -1
  56. package/dist/adapters/nitro.d.ts +17 -1
  57. package/dist/adapters/nitro.d.ts.map +1 -1
  58. package/dist/adapters/nitro.js +56 -13
  59. package/dist/adapters/nitro.js.map +1 -1
  60. package/dist/cache/cache-api.d.ts +24 -0
  61. package/dist/cache/cache-api.d.ts.map +1 -0
  62. package/dist/cache/handler-store.d.ts +31 -0
  63. package/dist/cache/handler-store.d.ts.map +1 -0
  64. package/dist/cache/index.d.ts +23 -7
  65. package/dist/cache/index.d.ts.map +1 -1
  66. package/dist/cache/index.js +142 -80
  67. package/dist/cache/index.js.map +1 -1
  68. package/dist/cache/singleflight.d.ts +18 -1
  69. package/dist/cache/singleflight.d.ts.map +1 -1
  70. package/dist/cache/sizeof.d.ts +22 -0
  71. package/dist/cache/sizeof.d.ts.map +1 -0
  72. package/dist/cache/timber-cache.d.ts +1 -1
  73. package/dist/cache/timber-cache.d.ts.map +1 -1
  74. package/dist/cli.d.ts +6 -1
  75. package/dist/cli.d.ts.map +1 -1
  76. package/dist/cli.js +8 -3
  77. package/dist/cli.js.map +1 -1
  78. package/dist/client/browser-dev.d.ts +27 -1
  79. package/dist/client/browser-dev.d.ts.map +1 -1
  80. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  81. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  82. package/dist/client/browser-entry/hmr.d.ts +21 -0
  83. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  84. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  85. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  86. package/dist/client/browser-entry/index.d.ts +30 -0
  87. package/dist/client/browser-entry/index.d.ts.map +1 -0
  88. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  89. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  90. package/dist/client/browser-entry/router-init.d.ts +23 -0
  91. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  92. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  93. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  94. package/dist/client/browser-entry/scroll.d.ts +19 -0
  95. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  96. package/dist/client/error-boundary.d.ts +12 -5
  97. package/dist/client/error-boundary.d.ts.map +1 -1
  98. package/dist/client/error-boundary.js +10 -4
  99. package/dist/client/error-boundary.js.map +1 -1
  100. package/dist/client/error-reconstituter.d.ts +54 -0
  101. package/dist/client/error-reconstituter.d.ts.map +1 -0
  102. package/dist/client/form.d.ts +3 -3
  103. package/dist/client/form.d.ts.map +1 -1
  104. package/dist/client/history.d.ts +19 -4
  105. package/dist/client/history.d.ts.map +1 -1
  106. package/dist/client/index.d.ts +7 -21
  107. package/dist/client/index.d.ts.map +1 -1
  108. package/dist/client/index.js +210 -1017
  109. package/dist/client/index.js.map +1 -1
  110. package/dist/client/internal.d.ts +18 -0
  111. package/dist/client/internal.d.ts.map +1 -0
  112. package/dist/client/internal.js +890 -0
  113. package/dist/client/internal.js.map +1 -0
  114. package/dist/client/link-pending-store.d.ts +63 -0
  115. package/dist/client/link-pending-store.d.ts.map +1 -0
  116. package/dist/client/link.d.ts +90 -32
  117. package/dist/client/link.d.ts.map +1 -1
  118. package/dist/client/nav-link-store.d.ts +36 -0
  119. package/dist/client/nav-link-store.d.ts.map +1 -0
  120. package/dist/client/navigation-api-types.d.ts +90 -0
  121. package/dist/client/navigation-api-types.d.ts.map +1 -0
  122. package/dist/client/navigation-api.d.ts +115 -0
  123. package/dist/client/navigation-api.d.ts.map +1 -0
  124. package/dist/client/navigation-context.d.ts +13 -2
  125. package/dist/client/navigation-context.d.ts.map +1 -1
  126. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  127. package/dist/client/navigation-root.d.ts.map +1 -0
  128. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  129. package/dist/client/router-ref.d.ts +1 -1
  130. package/dist/client/router.d.ts +70 -4
  131. package/dist/client/router.d.ts.map +1 -1
  132. package/dist/client/rsc-fetch.d.ts +38 -3
  133. package/dist/client/rsc-fetch.d.ts.map +1 -1
  134. package/dist/client/segment-cache.d.ts +1 -1
  135. package/dist/client/segment-cache.d.ts.map +1 -1
  136. package/dist/client/segment-outlet.d.ts +63 -0
  137. package/dist/client/segment-outlet.d.ts.map +1 -0
  138. package/dist/client/ssr-data.d.ts +13 -4
  139. package/dist/client/ssr-data.d.ts.map +1 -1
  140. package/dist/client/stale-reload.d.ts +15 -0
  141. package/dist/client/stale-reload.d.ts.map +1 -1
  142. package/dist/client/top-loader.d.ts +5 -5
  143. package/dist/client/top-loader.d.ts.map +1 -1
  144. package/dist/client/use-link-status.d.ts +5 -5
  145. package/dist/client/use-link-status.d.ts.map +1 -1
  146. package/dist/client/use-params.d.ts +6 -4
  147. package/dist/client/use-params.d.ts.map +1 -1
  148. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  149. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  150. package/dist/client/use-query-states.d.ts +1 -1
  151. package/dist/client/use-query-states.d.ts.map +1 -1
  152. package/dist/client/use-router.d.ts +1 -1
  153. package/dist/codec.d.ts +33 -0
  154. package/dist/codec.d.ts.map +1 -0
  155. package/dist/codec.js +2 -0
  156. package/dist/config-types.d.ts +227 -0
  157. package/dist/config-types.d.ts.map +1 -0
  158. package/dist/content/index.d.ts +1 -10
  159. package/dist/content/index.d.ts.map +1 -1
  160. package/dist/content/index.js +0 -2
  161. package/dist/cookies/define-cookie.d.ts +35 -14
  162. package/dist/cookies/define-cookie.d.ts.map +1 -1
  163. package/dist/cookies/index.js +1 -83
  164. package/dist/fonts/css.d.ts +1 -0
  165. package/dist/fonts/css.d.ts.map +1 -1
  166. package/dist/index.d.ts +45 -192
  167. package/dist/index.d.ts.map +1 -1
  168. package/dist/index.js +12357 -11925
  169. package/dist/index.js.map +1 -1
  170. package/dist/plugin-context.d.ts +107 -0
  171. package/dist/plugin-context.d.ts.map +1 -0
  172. package/dist/plugins/adapter-build.d.ts +1 -1
  173. package/dist/plugins/adapter-build.d.ts.map +1 -1
  174. package/dist/plugins/build-manifest.d.ts +2 -2
  175. package/dist/plugins/build-manifest.d.ts.map +1 -1
  176. package/dist/plugins/build-report.d.ts +3 -3
  177. package/dist/plugins/build-report.d.ts.map +1 -1
  178. package/dist/plugins/client-chunks.d.ts +32 -0
  179. package/dist/plugins/client-chunks.d.ts.map +1 -0
  180. package/dist/plugins/content.d.ts +1 -1
  181. package/dist/plugins/content.d.ts.map +1 -1
  182. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  183. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  184. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  185. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  186. package/dist/plugins/dev-logs.d.ts +1 -1
  187. package/dist/plugins/dev-logs.d.ts.map +1 -1
  188. package/dist/plugins/dev-server.d.ts +1 -1
  189. package/dist/plugins/dev-server.d.ts.map +1 -1
  190. package/dist/plugins/entries.d.ts +1 -1
  191. package/dist/plugins/entries.d.ts.map +1 -1
  192. package/dist/plugins/fonts.d.ts +19 -5
  193. package/dist/plugins/fonts.d.ts.map +1 -1
  194. package/dist/plugins/mdx.d.ts +1 -1
  195. package/dist/plugins/mdx.d.ts.map +1 -1
  196. package/dist/plugins/routing.d.ts +1 -1
  197. package/dist/plugins/routing.d.ts.map +1 -1
  198. package/dist/plugins/server-bundle.d.ts.map +1 -1
  199. package/dist/plugins/shims.d.ts +6 -5
  200. package/dist/plugins/shims.d.ts.map +1 -1
  201. package/dist/plugins/static-build.d.ts +4 -4
  202. package/dist/plugins/static-build.d.ts.map +1 -1
  203. package/dist/routing/codegen.d.ts +2 -2
  204. package/dist/routing/codegen.d.ts.map +1 -1
  205. package/dist/routing/index.d.ts +2 -0
  206. package/dist/routing/index.d.ts.map +1 -1
  207. package/dist/routing/index.js +3 -2
  208. package/dist/routing/scanner.d.ts.map +1 -1
  209. package/dist/routing/segment-classify.d.ts +46 -0
  210. package/dist/routing/segment-classify.d.ts.map +1 -0
  211. package/dist/routing/status-file-lint.d.ts +2 -1
  212. package/dist/routing/status-file-lint.d.ts.map +1 -1
  213. package/dist/routing/types.d.ts +16 -4
  214. package/dist/routing/types.d.ts.map +1 -1
  215. package/dist/rsc-runtime/rsc.d.ts +1 -1
  216. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  217. package/dist/rsc-runtime/ssr.d.ts +12 -0
  218. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  219. package/dist/schema-bridge.d.ts +76 -0
  220. package/dist/schema-bridge.d.ts.map +1 -0
  221. package/dist/search-params/define.d.ts +139 -0
  222. package/dist/search-params/define.d.ts.map +1 -0
  223. package/dist/search-params/index.d.ts +4 -7
  224. package/dist/search-params/index.d.ts.map +1 -1
  225. package/dist/search-params/index.js +32 -441
  226. package/dist/search-params/index.js.map +1 -1
  227. package/dist/search-params/registry.d.ts +2 -2
  228. package/dist/search-params/registry.d.ts.map +1 -1
  229. package/dist/search-params/wrappers.d.ts +53 -0
  230. package/dist/search-params/wrappers.d.ts.map +1 -0
  231. package/dist/segment-params/define.d.ts +78 -0
  232. package/dist/segment-params/define.d.ts.map +1 -0
  233. package/dist/segment-params/index.d.ts +3 -0
  234. package/dist/segment-params/index.d.ts.map +1 -0
  235. package/dist/segment-params/index.js +2 -0
  236. package/dist/server/access-gate.d.ts +4 -0
  237. package/dist/server/access-gate.d.ts.map +1 -1
  238. package/dist/server/action-client.d.ts +25 -6
  239. package/dist/server/action-client.d.ts.map +1 -1
  240. package/dist/server/action-encryption.d.ts +76 -0
  241. package/dist/server/action-encryption.d.ts.map +1 -0
  242. package/dist/server/action-handler.d.ts.map +1 -1
  243. package/dist/server/actions.d.ts +3 -6
  244. package/dist/server/actions.d.ts.map +1 -1
  245. package/dist/server/als-registry.d.ts +32 -4
  246. package/dist/server/als-registry.d.ts.map +1 -1
  247. package/dist/server/build-manifest.d.ts +2 -2
  248. package/dist/server/build-manifest.d.ts.map +1 -1
  249. package/dist/server/debug.d.ts +1 -1
  250. package/dist/server/default-logger.d.ts +22 -0
  251. package/dist/server/default-logger.d.ts.map +1 -0
  252. package/dist/server/deny-page-resolver.d.ts +52 -0
  253. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  254. package/dist/server/deny-renderer.d.ts.map +1 -1
  255. package/dist/server/dev-holding-server.d.ts +52 -0
  256. package/dist/server/dev-holding-server.d.ts.map +1 -0
  257. package/dist/server/dev-warnings.d.ts +1 -21
  258. package/dist/server/dev-warnings.d.ts.map +1 -1
  259. package/dist/server/early-hints.d.ts +13 -5
  260. package/dist/server/early-hints.d.ts.map +1 -1
  261. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  262. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  263. package/dist/server/fallback-error.d.ts +4 -3
  264. package/dist/server/fallback-error.d.ts.map +1 -1
  265. package/dist/server/flight-injection-state.d.ts +66 -0
  266. package/dist/server/flight-injection-state.d.ts.map +1 -0
  267. package/dist/server/flight-scripts.d.ts +42 -0
  268. package/dist/server/flight-scripts.d.ts.map +1 -0
  269. package/dist/server/flush.d.ts.map +1 -1
  270. package/dist/server/form-data.d.ts +29 -0
  271. package/dist/server/form-data.d.ts.map +1 -1
  272. package/dist/server/html-injectors.d.ts +51 -11
  273. package/dist/server/html-injectors.d.ts.map +1 -1
  274. package/dist/server/index.d.ts +5 -43
  275. package/dist/server/index.d.ts.map +1 -1
  276. package/dist/server/index.js +37 -2798
  277. package/dist/server/index.js.map +1 -1
  278. package/dist/server/internal.d.ts +46 -0
  279. package/dist/server/internal.d.ts.map +1 -0
  280. package/dist/server/internal.js +2883 -0
  281. package/dist/server/internal.js.map +1 -0
  282. package/dist/server/logger.d.ts +25 -7
  283. package/dist/server/logger.d.ts.map +1 -1
  284. package/dist/server/middleware-runner.d.ts +19 -4
  285. package/dist/server/middleware-runner.d.ts.map +1 -1
  286. package/dist/server/node-stream-transforms.d.ts +113 -0
  287. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  288. package/dist/server/page-deny-boundary.d.ts +31 -0
  289. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  290. package/dist/server/pipeline-interception.d.ts +1 -1
  291. package/dist/server/pipeline-interception.d.ts.map +1 -1
  292. package/dist/server/pipeline-metadata.d.ts +6 -0
  293. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  294. package/dist/server/pipeline.d.ts +42 -10
  295. package/dist/server/pipeline.d.ts.map +1 -1
  296. package/dist/server/primitives.d.ts +69 -18
  297. package/dist/server/primitives.d.ts.map +1 -1
  298. package/dist/server/render-timeout.d.ts +51 -0
  299. package/dist/server/render-timeout.d.ts.map +1 -0
  300. package/dist/server/request-context.d.ts +112 -43
  301. package/dist/server/request-context.d.ts.map +1 -1
  302. package/dist/server/route-element-builder.d.ts +27 -1
  303. package/dist/server/route-element-builder.d.ts.map +1 -1
  304. package/dist/server/route-handler.d.ts.map +1 -1
  305. package/dist/server/route-matcher.d.ts +9 -2
  306. package/dist/server/route-matcher.d.ts.map +1 -1
  307. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  308. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  309. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  310. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  311. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  312. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  313. package/dist/server/rsc-entry/index.d.ts +8 -3
  314. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  315. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  316. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  317. package/dist/server/rsc-entry/rsc-stream.d.ts +4 -1
  318. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  319. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  320. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  321. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  322. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  323. package/dist/server/safe-load.d.ts +46 -0
  324. package/dist/server/safe-load.d.ts.map +1 -0
  325. package/dist/server/sitemap-generator.d.ts +129 -0
  326. package/dist/server/sitemap-generator.d.ts.map +1 -0
  327. package/dist/server/sitemap-handler.d.ts +22 -0
  328. package/dist/server/sitemap-handler.d.ts.map +1 -0
  329. package/dist/server/slot-resolver.d.ts +1 -1
  330. package/dist/server/slot-resolver.d.ts.map +1 -1
  331. package/dist/server/ssr-entry.d.ts +22 -0
  332. package/dist/server/ssr-entry.d.ts.map +1 -1
  333. package/dist/server/ssr-render.d.ts +39 -21
  334. package/dist/server/ssr-render.d.ts.map +1 -1
  335. package/dist/server/ssr-wrappers.d.ts +50 -0
  336. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  337. package/dist/server/status-code-resolver.d.ts +1 -1
  338. package/dist/server/status-code-resolver.d.ts.map +1 -1
  339. package/dist/server/stream-utils.d.ts +36 -0
  340. package/dist/server/stream-utils.d.ts.map +1 -0
  341. package/dist/server/tracing.d.ts +4 -4
  342. package/dist/server/tracing.d.ts.map +1 -1
  343. package/dist/server/tree-builder.d.ts +22 -19
  344. package/dist/server/tree-builder.d.ts.map +1 -1
  345. package/dist/server/types.d.ts +1 -4
  346. package/dist/server/types.d.ts.map +1 -1
  347. package/dist/server/version-skew.d.ts +61 -0
  348. package/dist/server/version-skew.d.ts.map +1 -0
  349. package/dist/shared/merge-search-params.d.ts +22 -0
  350. package/dist/shared/merge-search-params.d.ts.map +1 -0
  351. package/dist/shims/font-google.d.ts +1 -1
  352. package/dist/shims/font-google.d.ts.map +1 -1
  353. package/dist/shims/font-google.js +42 -0
  354. package/dist/shims/font-google.js.map +1 -0
  355. package/dist/shims/font-local.d.ts +26 -0
  356. package/dist/shims/font-local.d.ts.map +1 -0
  357. package/dist/shims/font-local.js +20 -0
  358. package/dist/shims/font-local.js.map +1 -0
  359. package/dist/shims/headers.d.ts +2 -1
  360. package/dist/shims/headers.d.ts.map +1 -1
  361. package/dist/shims/navigation-client.d.ts +1 -1
  362. package/dist/shims/navigation-client.d.ts.map +1 -1
  363. package/dist/shims/navigation.d.ts +3 -2
  364. package/dist/shims/navigation.d.ts.map +1 -1
  365. package/dist/utils/directive-parser.d.ts +5 -2
  366. package/dist/utils/directive-parser.d.ts.map +1 -1
  367. package/dist/utils/state-machine.d.ts +80 -0
  368. package/dist/utils/state-machine.d.ts.map +1 -0
  369. package/package.json +56 -22
  370. package/src/adapters/cloudflare-dev.ts +177 -0
  371. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  372. package/src/adapters/cloudflare.ts +342 -28
  373. package/src/adapters/compress-module.ts +24 -4
  374. package/src/adapters/nitro.ts +52 -8
  375. package/src/adapters/wrangler.d.ts +7 -0
  376. package/src/cache/cache-api.ts +38 -0
  377. package/src/cache/handler-store.ts +68 -0
  378. package/src/cache/index.ts +81 -18
  379. package/src/cache/singleflight.ts +62 -4
  380. package/src/cache/sizeof.ts +31 -0
  381. package/src/cache/timber-cache.ts +24 -20
  382. package/src/cli.ts +16 -6
  383. package/src/client/browser-dev.ts +128 -1
  384. package/src/client/browser-entry/action-dispatch.ts +116 -0
  385. package/src/client/browser-entry/hmr.ts +81 -0
  386. package/src/client/browser-entry/hydrate.ts +145 -0
  387. package/src/client/browser-entry/index.ts +138 -0
  388. package/src/client/browser-entry/post-hydration.ts +119 -0
  389. package/src/client/browser-entry/router-init.ts +193 -0
  390. package/src/client/browser-entry/rsc-stream.ts +157 -0
  391. package/src/client/browser-entry/scroll.ts +27 -0
  392. package/src/client/error-boundary.tsx +48 -16
  393. package/src/client/error-reconstituter.tsx +65 -0
  394. package/src/client/form.tsx +9 -7
  395. package/src/client/history.ts +26 -4
  396. package/src/client/index.ts +19 -38
  397. package/src/client/internal.ts +57 -0
  398. package/src/client/link-pending-store.ts +111 -0
  399. package/src/client/link.tsx +329 -97
  400. package/src/client/nav-link-store.ts +47 -0
  401. package/src/client/navigation-api-types.ts +112 -0
  402. package/src/client/navigation-api.ts +332 -0
  403. package/src/client/navigation-context.ts +31 -6
  404. package/src/client/navigation-root.tsx +342 -0
  405. package/src/client/nuqs-adapter.tsx +16 -3
  406. package/src/client/router-ref.ts +1 -1
  407. package/src/client/router.ts +299 -72
  408. package/src/client/rsc-fetch.ts +97 -8
  409. package/src/client/segment-cache.ts +1 -1
  410. package/src/client/segment-outlet.tsx +86 -0
  411. package/src/client/ssr-data.ts +13 -5
  412. package/src/client/stale-reload.ts +72 -3
  413. package/src/client/top-loader.tsx +16 -8
  414. package/src/client/use-link-status.ts +7 -7
  415. package/src/client/use-params.ts +7 -5
  416. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  417. package/src/client/use-query-states.ts +3 -3
  418. package/src/client/use-router.ts +1 -1
  419. package/src/codec.ts +49 -0
  420. package/src/config-types.ts +225 -0
  421. package/src/content/index.ts +5 -13
  422. package/src/cookies/define-cookie.ts +78 -25
  423. package/src/cookies/index.ts +8 -0
  424. package/src/fonts/css.ts +2 -1
  425. package/src/index.ts +295 -354
  426. package/src/plugin-context.ts +240 -0
  427. package/src/plugins/adapter-build.ts +9 -3
  428. package/src/plugins/build-manifest.ts +13 -2
  429. package/src/plugins/build-report.ts +3 -3
  430. package/src/plugins/client-chunks.ts +65 -0
  431. package/src/plugins/content.ts +1 -1
  432. package/src/plugins/dev-browser-logs.ts +288 -0
  433. package/src/plugins/dev-error-overlay.ts +70 -1
  434. package/src/plugins/dev-logs.ts +1 -1
  435. package/src/plugins/dev-server.ts +70 -9
  436. package/src/plugins/entries.ts +71 -10
  437. package/src/plugins/fonts.ts +168 -61
  438. package/src/plugins/mdx.ts +1 -1
  439. package/src/plugins/routing.ts +57 -17
  440. package/src/plugins/server-action-exports.ts +1 -1
  441. package/src/plugins/server-bundle.ts +32 -1
  442. package/src/plugins/shims.ts +135 -35
  443. package/src/plugins/static-build.ts +17 -11
  444. package/src/routing/codegen.ts +165 -105
  445. package/src/routing/index.ts +2 -0
  446. package/src/routing/scanner.ts +93 -23
  447. package/src/routing/segment-classify.ts +89 -0
  448. package/src/routing/status-file-lint.ts +3 -2
  449. package/src/routing/types.ts +17 -4
  450. package/src/rsc-runtime/rsc.ts +2 -0
  451. package/src/rsc-runtime/ssr.ts +50 -0
  452. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  453. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  454. package/src/search-params/define.ts +482 -0
  455. package/src/search-params/index.ts +14 -20
  456. package/src/search-params/registry.ts +2 -2
  457. package/src/search-params/wrappers.ts +85 -0
  458. package/src/segment-params/define.ts +279 -0
  459. package/src/segment-params/index.ts +9 -0
  460. package/src/server/access-gate.tsx +70 -29
  461. package/src/server/action-client.ts +46 -11
  462. package/src/server/action-encryption.ts +144 -0
  463. package/src/server/action-handler.ts +21 -4
  464. package/src/server/actions.ts +10 -9
  465. package/src/server/als-registry.ts +34 -6
  466. package/src/server/build-manifest.ts +10 -4
  467. package/src/server/compress.ts +25 -7
  468. package/src/server/debug.ts +1 -1
  469. package/src/server/default-logger.ts +99 -0
  470. package/src/server/deny-page-resolver.ts +154 -0
  471. package/src/server/deny-renderer.ts +24 -38
  472. package/src/server/dev-holding-server.ts +185 -0
  473. package/src/server/dev-warnings.ts +4 -49
  474. package/src/server/early-hints.ts +36 -15
  475. package/src/server/error-boundary-wrapper.ts +74 -22
  476. package/src/server/fallback-error.ts +31 -15
  477. package/src/server/flight-injection-state.ts +113 -0
  478. package/src/server/flight-scripts.ts +62 -0
  479. package/src/server/flush.ts +2 -1
  480. package/src/server/form-data.ts +76 -0
  481. package/src/server/html-injectors.ts +280 -120
  482. package/src/server/index.ts +25 -177
  483. package/src/server/internal.ts +169 -0
  484. package/src/server/logger.ts +44 -36
  485. package/src/server/middleware-runner.ts +31 -4
  486. package/src/server/node-stream-transforms.ts +509 -0
  487. package/src/server/page-deny-boundary.tsx +56 -0
  488. package/src/server/pipeline-interception.ts +17 -16
  489. package/src/server/pipeline-metadata.ts +13 -0
  490. package/src/server/pipeline.ts +227 -62
  491. package/src/server/primitives.ts +111 -28
  492. package/src/server/render-timeout.ts +108 -0
  493. package/src/server/request-context.ts +293 -132
  494. package/src/server/route-element-builder.ts +283 -191
  495. package/src/server/route-handler.ts +24 -4
  496. package/src/server/route-matcher.ts +24 -20
  497. package/src/server/rsc-entry/api-handler.ts +15 -16
  498. package/src/server/rsc-entry/error-renderer.ts +300 -89
  499. package/src/server/rsc-entry/helpers.ts +134 -5
  500. package/src/server/rsc-entry/index.ts +200 -112
  501. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  502. package/src/server/rsc-entry/rsc-stream.ts +65 -13
  503. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  504. package/src/server/rsc-entry/ssr-renderer.ts +168 -38
  505. package/src/server/safe-load.ts +60 -0
  506. package/src/server/sitemap-generator.ts +338 -0
  507. package/src/server/sitemap-handler.ts +126 -0
  508. package/src/server/slot-resolver.ts +244 -229
  509. package/src/server/ssr-entry.ts +211 -32
  510. package/src/server/ssr-render.ts +289 -67
  511. package/src/server/ssr-wrappers.tsx +139 -0
  512. package/src/server/status-code-resolver.ts +1 -1
  513. package/src/server/stream-utils.ts +213 -0
  514. package/src/server/tracing.ts +20 -9
  515. package/src/server/tree-builder.ts +92 -58
  516. package/src/server/types.ts +3 -6
  517. package/src/server/version-skew.ts +104 -0
  518. package/src/shared/merge-search-params.ts +55 -0
  519. package/src/shims/font-google.ts +1 -1
  520. package/src/shims/font-local.ts +34 -0
  521. package/src/shims/headers.ts +5 -1
  522. package/src/shims/navigation-client.ts +1 -1
  523. package/src/shims/navigation.ts +7 -2
  524. package/src/utils/directive-parser.ts +5 -2
  525. package/src/utils/state-machine.ts +111 -0
  526. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  527. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  528. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  529. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  530. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  531. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  532. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  533. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  534. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  535. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  536. package/dist/cache/register-cached-function.d.ts +0 -17
  537. package/dist/cache/register-cached-function.d.ts.map +0 -1
  538. package/dist/client/browser-entry.d.ts +0 -21
  539. package/dist/client/browser-entry.d.ts.map +0 -1
  540. package/dist/client/link-status-provider.d.ts +0 -11
  541. package/dist/client/link-status-provider.d.ts.map +0 -1
  542. package/dist/client/transition-root.d.ts.map +0 -1
  543. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  544. package/dist/cookies/index.js.map +0 -1
  545. package/dist/plugins/cache-transform.d.ts +0 -36
  546. package/dist/plugins/cache-transform.d.ts.map +0 -1
  547. package/dist/plugins/dynamic-transform.d.ts +0 -72
  548. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  549. package/dist/search-params/analyze.d.ts +0 -54
  550. package/dist/search-params/analyze.d.ts.map +0 -1
  551. package/dist/search-params/builtin-codecs.d.ts +0 -105
  552. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  553. package/dist/search-params/codecs.d.ts +0 -53
  554. package/dist/search-params/codecs.d.ts.map +0 -1
  555. package/dist/search-params/create.d.ts +0 -106
  556. package/dist/search-params/create.d.ts.map +0 -1
  557. package/dist/server/prerender.d.ts +0 -77
  558. package/dist/server/prerender.d.ts.map +0 -1
  559. package/dist/server/response-cache.d.ts +0 -54
  560. package/dist/server/response-cache.d.ts.map +0 -1
  561. package/src/cache/register-cached-function.ts +0 -103
  562. package/src/client/browser-entry.ts +0 -678
  563. package/src/client/link-status-provider.tsx +0 -30
  564. package/src/client/transition-root.tsx +0 -166
  565. package/src/plugins/cache-transform.ts +0 -199
  566. package/src/plugins/dynamic-transform.ts +0 -161
  567. package/src/search-params/analyze.ts +0 -192
  568. package/src/search-params/builtin-codecs.ts +0 -228
  569. package/src/search-params/create.ts +0 -321
  570. package/src/server/prerender.ts +0 -139
  571. package/src/server/response-cache.ts +0 -410
@@ -0,0 +1,279 @@
1
+ /**
2
+ * defineSegmentParams — factory for typed route param coercion.
3
+ *
4
+ * Creates a ParamsDefinition that coerces raw string params from the
5
+ * URL into typed values. Used by exporting from params.ts (segment-level)
6
+ * convention file.
7
+ *
8
+ * Reuses the shared Codec<T> protocol with Standard Schema auto-detection,
9
+ * same pattern as defineSearchParams. Runtime constraints are stricter:
10
+ * - serialize must return string (not null — path segments can't be omitted)
11
+ * - parse throwing → 404 (invalid param value)
12
+ *
13
+ * Design doc: design/07a-route-params-triage.md
14
+ */
15
+
16
+ import type { Codec } from '../codec.js';
17
+ import {
18
+ type StandardSchemaV1,
19
+ validateSync,
20
+ isStandardSchema,
21
+ isCodec,
22
+ } from '../schema-bridge.js';
23
+
24
+ // ---------------------------------------------------------------------------
25
+ // Server-only ALS reference for .get()
26
+ // ---------------------------------------------------------------------------
27
+
28
+ // Same pattern as search-params: eagerly registered at server startup
29
+ // to avoid dynamic imports that lose ALS context. See TIM-523.
30
+ let _getSegmentParamsFn: (() => Promise<Record<string, string | string[]>>) | undefined;
31
+
32
+ /**
33
+ * Register the getSegmentParams function. Called once at module load time
34
+ * from request-context.ts to avoid dynamic import at call time.
35
+ * @internal
36
+ */
37
+ export function _setGetSegmentParamsFn(fn: () => Promise<Record<string, string | string[]>>): void {
38
+ _getSegmentParamsFn = fn;
39
+ }
40
+
41
+ function getSegmentParamsFromAls(): Promise<Record<string, string | string[]>> {
42
+ if (!_getSegmentParamsFn) {
43
+ throw new Error(
44
+ '[timber] segmentParams.get() is only available on the server. ' +
45
+ 'Use useSegmentParams() on the client.'
46
+ );
47
+ }
48
+ return _getSegmentParamsFn();
49
+ }
50
+
51
+ // ---------------------------------------------------------------------------
52
+ // Types
53
+ // ---------------------------------------------------------------------------
54
+
55
+ /** Infer the output type from a Codec or StandardSchemaV1. */
56
+ export type InferParamField<V> =
57
+ V extends Codec<infer T> ? T : V extends StandardSchemaV1<infer T> ? T : never;
58
+
59
+ /** Acceptable field value for defineParams: a Codec or a Standard Schema. */
60
+ export type ParamField<T = unknown> = Codec<T> | StandardSchemaV1<T>;
61
+
62
+ export type { StandardSchemaV1 };
63
+
64
+ /**
65
+ * A typed route params definition.
66
+ *
67
+ * Returned by defineParams(). Provides parse (string → typed) and
68
+ * serialize (typed → string) for each declared param.
69
+ */
70
+ export interface ParamsDefinition<T extends Record<string, unknown>> {
71
+ /** Parse raw string params into typed values. Throws on invalid values. */
72
+ parse(raw: Record<string, string | string[]>): T;
73
+
74
+ /** Serialize typed values back to strings for URL construction. */
75
+ serialize(values: T): Record<string, string>;
76
+
77
+ /**
78
+ * Get typed segment params from the current request context (ALS).
79
+ *
80
+ * Server-only. Reads getSegmentParams() from ALS and coerces through
81
+ * this definition's codecs, returning fully typed params.
82
+ *
83
+ * ```ts
84
+ * // app/products/[id]/params.ts
85
+ * export const segmentParams = defineSegmentParams({ id: z.coerce.number() })
86
+ *
87
+ * // app/products/[id]/page.tsx
88
+ * import { segmentParams } from './params'
89
+ * export default async function Page() {
90
+ * const { id } = await segmentParams.get() // id: number
91
+ * }
92
+ * ```
93
+ */
94
+ get(): Promise<T>;
95
+
96
+ /** Read-only codec map. */
97
+ codecs: { [K in keyof T]: Codec<T[K]> };
98
+ }
99
+
100
+ // ---------------------------------------------------------------------------
101
+ // Internal helpers
102
+ // ---------------------------------------------------------------------------
103
+
104
+ /**
105
+ * Wrap a Standard Schema into a Codec for route params.
106
+ *
107
+ * Unlike fromSchema for search params:
108
+ * - Parse throws on failure (no fallback to default)
109
+ * - Serialize returns string (not null)
110
+ */
111
+ function fromParamSchema<T>(fieldName: string, schema: StandardSchemaV1<T>): Codec<T> {
112
+ return {
113
+ parse(value: string | string[] | undefined): T {
114
+ // Route params are always strings (single segment) or string[] (catch-all)
115
+ const input = Array.isArray(value) ? value : value;
116
+
117
+ const result = validateSync(schema, input);
118
+ if (!result.issues) {
119
+ return result.value;
120
+ }
121
+
122
+ // For route params, parse failure means the param is invalid → throw
123
+ const messages = result.issues.map((i) => i.message).join(', ');
124
+ throw new Error(`[timber] Param '${fieldName}' coercion failed: ${messages}`);
125
+ },
126
+
127
+ serialize(value: T): string | null {
128
+ if (value === null || value === undefined) {
129
+ return null;
130
+ }
131
+ // Catch-all segments produce arrays — join with '/' for path reconstruction
132
+ if (Array.isArray(value)) {
133
+ return value.join('/');
134
+ }
135
+ return String(value);
136
+ },
137
+ };
138
+ }
139
+
140
+ /**
141
+ * Resolve a field value to a Codec. Auto-detects Standard Schema objects.
142
+ */
143
+ function resolveField(fieldName: string, value: ParamField): Codec<unknown> {
144
+ if (isCodec(value)) {
145
+ return value;
146
+ }
147
+
148
+ if (isStandardSchema(value)) {
149
+ return fromParamSchema(fieldName, value);
150
+ }
151
+
152
+ throw new Error(
153
+ `[timber] defineSegmentParams: field '${fieldName}' is not a valid codec or Standard Schema. ` +
154
+ `Expected an object with { parse, serialize } methods, or a Standard Schema object ` +
155
+ `(Zod, Valibot, ArkType).`
156
+ );
157
+ }
158
+
159
+ /**
160
+ * Validate that no codec's serialize returns null.
161
+ * Route params are structural — they must produce a valid path segment.
162
+ */
163
+ function validateSerialize(codecMap: Record<string, Codec<unknown>>): void {
164
+ for (const [key, codec] of Object.entries(codecMap)) {
165
+ // Test serialize with a sample parsed value to check for null
166
+ // We can't exhaustively test, but we can check that serialize(parse("test"))
167
+ // doesn't return null for a basic input.
168
+ try {
169
+ const testValue = codec.parse('test');
170
+ const serialized = codec.serialize(testValue);
171
+ if (serialized === null) {
172
+ throw new Error(
173
+ `[timber] defineSegmentParams: field '${key}' codec.serialize() returned null.\n` +
174
+ ` Route params are path segments — they cannot be omitted.\n` +
175
+ ` Ensure serialize() always returns a string.`
176
+ );
177
+ }
178
+ } catch (e) {
179
+ // parse('test') may throw for strict codecs (e.g., number-only).
180
+ // That's fine — it means the codec validates. We only care about
181
+ // serialize returning null, which we can't test without a valid value.
182
+ if (e instanceof Error && e.message.includes('returned null')) {
183
+ throw e;
184
+ }
185
+ }
186
+ }
187
+ }
188
+
189
+ // ---------------------------------------------------------------------------
190
+ // Factory
191
+ // ---------------------------------------------------------------------------
192
+
193
+ /**
194
+ * Create a ParamsDefinition from a map of codecs and/or Standard Schema objects.
195
+ *
196
+ * ```ts
197
+ * // app/products/[id]/layout.tsx
198
+ * import { defineSegmentParams } from '@timber-js/app/segment-params'
199
+ * import { z } from 'zod/v4'
200
+ *
201
+ * export const segmentParams = defineSegmentParams({
202
+ * id: z.coerce.number().int().positive(),
203
+ * })
204
+ * ```
205
+ */
206
+ export function defineSegmentParams<C extends Record<string, ParamField>>(
207
+ codecs: C
208
+ ): ParamsDefinition<{ [K in keyof C]: InferParamField<C[K]> }> {
209
+ type T = { [K in keyof C]: InferParamField<C[K]> };
210
+
211
+ const resolvedCodecs: Record<string, Codec<unknown>> = {};
212
+
213
+ for (const [key, value] of Object.entries(codecs)) {
214
+ resolvedCodecs[key] = resolveField(key, value as ParamField);
215
+ }
216
+
217
+ // Validate that serialize doesn't return null
218
+ validateSerialize(resolvedCodecs);
219
+
220
+ // ---- parse ----
221
+ function parse(raw: Record<string, string | string[]>): T {
222
+ const result: Record<string, unknown> = {};
223
+
224
+ for (const [key, codec] of Object.entries(resolvedCodecs)) {
225
+ const rawValue = raw[key];
226
+ // Route params are always present (the route matched)
227
+ result[key] = codec.parse(rawValue);
228
+ }
229
+
230
+ return result as T;
231
+ }
232
+
233
+ // ---- serialize ----
234
+ function serialize(values: T): Record<string, string> {
235
+ const result: Record<string, string> = {};
236
+
237
+ for (const [key, codec] of Object.entries(resolvedCodecs)) {
238
+ const serialized = codec.serialize(values[key as keyof T] as unknown);
239
+ if (serialized === null) {
240
+ throw new Error(
241
+ `[timber] params.serialize: field '${key}' serialized to null. ` +
242
+ `Route params must produce a valid path segment.`
243
+ );
244
+ }
245
+ result[key] = serialized;
246
+ }
247
+
248
+ return result;
249
+ }
250
+
251
+ // ---- get ----
252
+ // ALS-backed: reads segment params from the current request context.
253
+ // Server-only — throws on client.
254
+ //
255
+ // The pipeline already coerces params via coerceSegmentParams() which
256
+ // calls parse() and stores typed values in ALS via setSegmentParams().
257
+ // We return those directly instead of re-parsing, because codecs may
258
+ // not be idempotent (e.g., a codec that only accepts raw strings would
259
+ // throw if given an already-parsed value). See TIM-574.
260
+ async function get(): Promise<T> {
261
+ if (typeof window !== 'undefined') {
262
+ throw new Error(
263
+ '[timber] segmentParams.get() is server-only. ' + 'Use useSegmentParams() on the client.'
264
+ );
265
+ }
266
+ const params = await getSegmentParamsFromAls();
267
+ // params are already coerced by the pipeline — return as-is.
268
+ return params as unknown as T;
269
+ }
270
+
271
+ const definition: ParamsDefinition<T> = {
272
+ parse,
273
+ serialize,
274
+ get,
275
+ codecs: resolvedCodecs as { [K in keyof T]: Codec<T[K]> },
276
+ };
277
+
278
+ return definition;
279
+ }
@@ -0,0 +1,9 @@
1
+ // @timber-js/app/segment-params — Typed route param coercion
2
+ //
3
+ // This is the import path for defineSegmentParams and related types.
4
+ // For search params, import from @timber-js/app/search-params instead.
5
+ //
6
+ // See design/07-routing.md §"params.ts Convention File"
7
+
8
+ export type { ParamsDefinition, InferParamField, ParamField } from './define.js';
9
+ export { defineSegmentParams } from './define.js';
@@ -17,6 +17,8 @@ import { DenySignal, RedirectSignal } from './primitives.js';
17
17
  import type { AccessGateProps, SlotAccessGateProps, ReactElement } from './tree-builder.js';
18
18
  import { withSpan, setSpanAttribute } from './tracing.js';
19
19
  import { isDebug } from './debug.js';
20
+ import type { DenyPageEntry } from './deny-page-resolver.js';
21
+ import { renderMatchingDenyPage, setDenyStatus } from './deny-page-resolver.js';
20
22
 
21
23
  // ─── AccessGate ─────────────────────────────────────────────────────────────
22
24
 
@@ -35,24 +37,20 @@ import { isDebug } from './debug.js';
35
37
  * gets the same data by calling the same cached functions (React.cache dedup).
36
38
  */
37
39
  export function AccessGate(props: AccessGateProps): ReactElement | Promise<ReactElement> {
38
- const { accessFn, params, searchParams, segmentName, verdict, children } = props;
40
+ const { accessFn, segmentName, verdict, denyPages, children } = props;
39
41
 
40
42
  // Fast path: replay pre-computed verdict from the pre-render pass.
41
- // This is synchronous — Suspense boundaries cannot interfere with the
42
- // status code because the signal throws before any async work.
43
43
  if (verdict !== undefined) {
44
44
  if (verdict === 'pass') {
45
45
  return children;
46
46
  }
47
- // Throw the stored DenySignal or RedirectSignal synchronously.
48
- // React catches this as a render-phase throw — the flush controller
49
- // produces the correct HTTP status code.
50
47
  throw verdict;
51
48
  }
52
49
 
53
- // Fallback: call accessFn directly (used by tree-builder.ts which
54
- // doesn't run a pre-render pass, and for backward compat).
55
- return accessGateFallback(accessFn, params, searchParams, segmentName, children);
50
+ // Primary path: call accessFn directly during render.
51
+ // If denyPages is provided, catch DenySignal and render the deny page
52
+ // in-tree no throw reaches React Flight, no second render pass.
53
+ return accessGateFallback(accessFn, segmentName, denyPages, children);
56
54
  }
57
55
 
58
56
  /**
@@ -61,28 +59,41 @@ export function AccessGate(props: AccessGateProps): ReactElement | Promise<React
61
59
  */
62
60
  async function accessGateFallback(
63
61
  accessFn: AccessGateProps['accessFn'],
64
- params: AccessGateProps['params'],
65
- searchParams: AccessGateProps['searchParams'],
66
62
  segmentName: AccessGateProps['segmentName'],
63
+ denyPages: DenyPageEntry[] | undefined,
67
64
  children: ReactElement
68
65
  ): Promise<ReactElement> {
69
- await withSpan('timber.access', { 'timber.segment': segmentName ?? 'unknown' }, async () => {
70
- try {
71
- await accessFn({ params, searchParams });
72
- await setSpanAttribute('timber.result', 'pass');
73
- } catch (error: unknown) {
74
- if (error instanceof DenySignal) {
75
- await setSpanAttribute('timber.result', 'deny');
76
- await setSpanAttribute('timber.deny_status', error.status);
77
- if (error.sourceFile) {
78
- await setSpanAttribute('timber.deny_file', error.sourceFile);
66
+ try {
67
+ await withSpan('timber.access', { 'timber.segment': segmentName ?? 'unknown' }, async () => {
68
+ try {
69
+ await accessFn();
70
+ await setSpanAttribute('timber.result', 'pass');
71
+ } catch (error: unknown) {
72
+ if (error instanceof DenySignal) {
73
+ await setSpanAttribute('timber.result', 'deny');
74
+ await setSpanAttribute('timber.deny_status', error.status);
75
+ if (error.sourceFile) {
76
+ await setSpanAttribute('timber.deny_file', error.sourceFile);
77
+ }
78
+ } else if (error instanceof RedirectSignal) {
79
+ await setSpanAttribute('timber.result', 'redirect');
79
80
  }
80
- } else if (error instanceof RedirectSignal) {
81
- await setSpanAttribute('timber.result', 'redirect');
81
+ throw error;
82
+ }
83
+ });
84
+ } catch (error: unknown) {
85
+ // Catch DenySignal and render the deny page in-tree.
86
+ // No throw reaches React Flight — clean stream, single render pass.
87
+ // RedirectSignal and other errors propagate normally.
88
+ if (error instanceof DenySignal && denyPages) {
89
+ const denyElement = renderMatchingDenyPage(denyPages, error.status, error.data);
90
+ if (denyElement) {
91
+ setDenyStatus(error.status);
92
+ return denyElement;
82
93
  }
83
- throw error;
84
94
  }
85
- });
95
+ throw error;
96
+ }
86
97
 
87
98
  return children;
88
99
  }
@@ -96,18 +107,27 @@ async function accessGateFallback(
96
107
  * The HTTP status code is unaffected — slot denial is a UI concern, not
97
108
  * a protocol concern. The parent layout and sibling slots still render.
98
109
  *
110
+ * DeniedComponent is passed instead of a pre-built element so that
111
+ * DenySignal.data can be forwarded as the dangerouslyPassData prop
112
+ * and the slot name can be passed as the slot prop. See TIM-488.
113
+ *
99
114
  * redirect() in slot access.ts is a dev-mode error — redirecting from a
100
115
  * slot doesn't make architectural sense.
101
116
  */
102
117
  export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactElement> {
103
- const { accessFn, params, searchParams, deniedFallback, defaultFallback, children } = props;
118
+ const { accessFn, DeniedComponent, slotName, createElement, defaultFallback, children } = props;
104
119
 
105
120
  try {
106
- await accessFn({ params, searchParams });
121
+ await accessFn();
107
122
  } catch (error: unknown) {
108
123
  // DenySignal → graceful degradation (denied.tsx → default.tsx → null)
124
+ // Build the denied element dynamically so DenySignal.data is forwarded.
109
125
  if (error instanceof DenySignal) {
110
- return deniedFallback ?? defaultFallback ?? null;
126
+ return (
127
+ buildDeniedFallback(DeniedComponent, slotName, error.data, createElement) ??
128
+ defaultFallback ??
129
+ null
130
+ );
111
131
  }
112
132
 
113
133
  // RedirectSignal in slot access → dev-mode error.
@@ -123,7 +143,11 @@ export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactE
123
143
  );
124
144
  }
125
145
  // In production, treat as a deny — render fallback rather than crash.
126
- return deniedFallback ?? defaultFallback ?? null;
146
+ return (
147
+ buildDeniedFallback(DeniedComponent, slotName, undefined, createElement) ??
148
+ defaultFallback ??
149
+ null
150
+ );
127
151
  }
128
152
 
129
153
  // Unhandled error — re-throw so error boundaries can catch it.
@@ -141,3 +165,20 @@ export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactE
141
165
  // Access passed — render slot content.
142
166
  return children;
143
167
  }
168
+
169
+ /**
170
+ * Build the denied fallback element dynamically with DenySignal data.
171
+ * Returns null if no DeniedComponent is available.
172
+ */
173
+ function buildDeniedFallback(
174
+ DeniedComponent: SlotAccessGateProps['DeniedComponent'],
175
+ slotName: string,
176
+ data: unknown,
177
+ createElement: SlotAccessGateProps['createElement']
178
+ ): ReactElement | null {
179
+ if (!DeniedComponent) return null;
180
+ return createElement(DeniedComponent, {
181
+ slot: slotName,
182
+ dangerouslyPassData: data,
183
+ });
184
+ }
@@ -138,22 +138,45 @@ export interface ActionBuilder<TCtx> {
138
138
  /** Declare the input schema. Validation errors are returned typed. */
139
139
  schema<TInput>(schema: ActionSchema<TInput>): ActionBuilderWithSchema<TCtx, TInput>;
140
140
  /** Define the action body without input validation. */
141
- action<TData>(fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>): ActionFn<TData>;
141
+ action<TData>(
142
+ fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>
143
+ ): ActionFn<undefined, TData>;
142
144
  }
143
145
 
144
146
  /** Builder after .schema() has been called. */
145
147
  export interface ActionBuilderWithSchema<TCtx, TInput> {
146
148
  /** Define the action body with validated input. */
147
- action<TData>(fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>): ActionFn<TData>;
149
+ action<TData>(fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>): ActionFn<TInput, TData>;
148
150
  }
149
151
 
150
152
  /**
151
- * The final action function. Callable two ways:
153
+ * The final action function. Callable three ways:
152
154
  * - Direct: action(input) → Promise<ActionResult<TData>>
153
155
  * - React useActionState: action(prevState, formData) → Promise<ActionResult<TData>>
156
+ * - React <form action={fn}>: action(formData) → void (return value ignored by React)
157
+ *
158
+ * The third overload exists purely for type compatibility with React's
159
+ * `<form action>` prop, which expects `(formData: FormData) => void`.
160
+ * At runtime the function still returns Promise<ActionResult>, but React
161
+ * discards it. This lets validated actions be passed directly to forms
162
+ * without casts.
163
+ */
164
+ /**
165
+ * Map schema output keys to `string | undefined` for form-facing APIs.
166
+ * HTML form values are always strings, and fields can be absent.
167
+ * Gives autocomplete for field names without lying about value types.
154
168
  */
155
- export type ActionFn<TData> = {
156
- (input?: unknown): Promise<ActionResult<TData>>;
169
+ export type InputHint<T> =
170
+ T extends Record<string, unknown> ? { [K in keyof T]: string | undefined } : T;
171
+
172
+ export type ActionFn<TInput = unknown, TData = unknown> = {
173
+ /** <form action={fn}> compatibility — React discards the return value. */
174
+ (formData: FormData): void;
175
+ /** Direct call: action(input) — optional when TInput is undefined (no-schema actions). */
176
+ (
177
+ ...args: undefined extends TInput ? [input?: TInput] : [input: TInput]
178
+ ): Promise<ActionResult<TData>>;
179
+ /** React useActionState: action(prevState, formData) */
157
180
  (prevState: ActionResult<TData> | null, formData: FormData): Promise<ActionResult<TData>>;
158
181
  };
159
182
 
@@ -183,8 +206,9 @@ async function runActionMiddleware<TCtx>(
183
206
 
184
207
  // Re-export parseFormData for use throughout the framework
185
208
  import { parseFormData } from './form-data.js';
186
- import { formatSize } from '#/utils/format.js';
209
+ import { formatSize } from '../utils/format.js';
187
210
  import { isDebug, isDevMode } from './debug.js';
211
+ import { RedirectSignal, DenySignal } from './primitives.js';
188
212
 
189
213
  /**
190
214
  * Extract validation errors from a schema error.
@@ -286,7 +310,7 @@ export function createActionClient<TCtx = Record<string, never>>(
286
310
  function buildAction<TInput, TData>(
287
311
  schema: ActionSchema<TInput> | undefined,
288
312
  fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>
289
- ): ActionFn<TData> {
313
+ ): ActionFn<TInput, TData> {
290
314
  async function actionHandler(...args: unknown[]): Promise<ActionResult<TData>> {
291
315
  try {
292
316
  // Run middleware
@@ -366,22 +390,33 @@ export function createActionClient<TCtx = Record<string, never>>(
366
390
  const data = await fn({ ctx, input });
367
391
  return { data };
368
392
  } catch (error) {
393
+ // Re-throw redirect/deny signals — these are control flow, not errors.
394
+ // They must propagate to executeAction() which converts them to proper
395
+ // HTTP responses (302 redirect, 4xx deny). Catching them here would
396
+ // wrap them as INTERNAL_ERROR and break redirect()/redirectExternal()/deny().
397
+ if (error instanceof RedirectSignal || error instanceof DenySignal) {
398
+ throw error;
399
+ }
369
400
  return handleActionError(error);
370
401
  }
371
402
  }
372
403
 
373
- return actionHandler as ActionFn<TData>;
404
+ return actionHandler as ActionFn<TInput, TData>;
374
405
  }
375
406
 
376
407
  return {
377
408
  schema<TInput>(schema: ActionSchema<TInput>) {
378
409
  return {
379
- action<TData>(fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>): ActionFn<TData> {
410
+ action<TData>(
411
+ fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>
412
+ ): ActionFn<TInput, TData> {
380
413
  return buildAction(schema, fn);
381
414
  },
382
415
  };
383
416
  },
384
- action<TData>(fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>): ActionFn<TData> {
417
+ action<TData>(
418
+ fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>
419
+ ): ActionFn<undefined, TData> {
385
420
  return buildAction(undefined, fn as (ctx: ActionContext<TCtx, unknown>) => Promise<TData>);
386
421
  },
387
422
  };
@@ -410,7 +445,7 @@ export function createActionClient<TCtx = Record<string, never>>(
410
445
  export function validated<TInput, TData>(
411
446
  schema: ActionSchema<TInput>,
412
447
  handler: (input: TInput) => Promise<TData>
413
- ): ActionFn<TData> {
448
+ ): ActionFn<TInput, TData> {
414
449
  return createActionClient()
415
450
  .schema(schema)
416
451
  .action(async ({ input }) => handler(input));