@timber-js/app 0.2.0-alpha.7 → 0.2.0-alpha.71

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 (500) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-BJARkOcu.js} +1 -1
  3. package/dist/_chunks/als-registry-BJARkOcu.js.map +1 -0
  4. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  5. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  6. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  7. package/dist/_chunks/define-CGuYoRHU.js +199 -0
  8. package/dist/_chunks/define-CGuYoRHU.js.map +1 -0
  9. package/dist/_chunks/define-Dz1bqwaS.js +106 -0
  10. package/dist/_chunks/define-Dz1bqwaS.js.map +1 -0
  11. package/dist/_chunks/define-cookie-B5mewxwM.js +93 -0
  12. package/dist/_chunks/define-cookie-B5mewxwM.js.map +1 -0
  13. package/dist/_chunks/error-boundary-D9hzsveV.js +216 -0
  14. package/dist/_chunks/error-boundary-D9hzsveV.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → format-Rn922VH2.js} +3 -20
  16. package/dist/_chunks/format-Rn922VH2.js.map +1 -0
  17. package/dist/_chunks/{tracing-Cwn7697K.js → handler-store-BVePM7hp.js} +68 -3
  18. package/dist/_chunks/handler-store-BVePM7hp.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-CEdHHviP.js} +171 -97
  20. package/dist/_chunks/interception-CEdHHviP.js.map +1 -0
  21. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  22. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  23. package/dist/_chunks/{request-context-DIkVh_jG.js → request-context-CywiO4jV.js} +181 -69
  24. package/dist/_chunks/request-context-CywiO4jV.js.map +1 -0
  25. package/dist/_chunks/schema-bridge-C4SwjCQD.js +86 -0
  26. package/dist/_chunks/schema-bridge-C4SwjCQD.js.map +1 -0
  27. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  28. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  29. package/dist/_chunks/segment-context-hzuJ048X.js +72 -0
  30. package/dist/_chunks/segment-context-hzuJ048X.js.map +1 -0
  31. package/dist/_chunks/stale-reload-BLUC_Pl_.js +64 -0
  32. package/dist/_chunks/stale-reload-BLUC_Pl_.js.map +1 -0
  33. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-DAhgj8Gx.js} +1 -1
  34. package/dist/_chunks/use-query-states-DAhgj8Gx.js.map +1 -0
  35. package/dist/_chunks/wrappers-LZbghvn0.js +63 -0
  36. package/dist/_chunks/wrappers-LZbghvn0.js.map +1 -0
  37. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  38. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  39. package/dist/adapters/cloudflare-dev.js +73 -0
  40. package/dist/adapters/cloudflare-dev.js.map +1 -0
  41. package/dist/adapters/cloudflare.d.ts +148 -12
  42. package/dist/adapters/cloudflare.d.ts.map +1 -1
  43. package/dist/adapters/cloudflare.js +135 -11
  44. package/dist/adapters/cloudflare.js.map +1 -1
  45. package/dist/adapters/compress-module.d.ts.map +1 -1
  46. package/dist/adapters/nitro.d.ts +17 -1
  47. package/dist/adapters/nitro.d.ts.map +1 -1
  48. package/dist/adapters/nitro.js +56 -13
  49. package/dist/adapters/nitro.js.map +1 -1
  50. package/dist/cache/cache-api.d.ts +24 -0
  51. package/dist/cache/cache-api.d.ts.map +1 -0
  52. package/dist/cache/fast-hash.d.ts +22 -0
  53. package/dist/cache/fast-hash.d.ts.map +1 -0
  54. package/dist/cache/handler-store.d.ts +31 -0
  55. package/dist/cache/handler-store.d.ts.map +1 -0
  56. package/dist/cache/index.d.ts +7 -5
  57. package/dist/cache/index.d.ts.map +1 -1
  58. package/dist/cache/index.js +111 -73
  59. package/dist/cache/index.js.map +1 -1
  60. package/dist/cache/singleflight.d.ts +18 -1
  61. package/dist/cache/singleflight.d.ts.map +1 -1
  62. package/dist/cache/timber-cache.d.ts +1 -1
  63. package/dist/cache/timber-cache.d.ts.map +1 -1
  64. package/dist/client/error-boundary.d.ts +12 -5
  65. package/dist/client/error-boundary.d.ts.map +1 -1
  66. package/dist/client/error-boundary.js +1 -125
  67. package/dist/client/error-reconstituter.d.ts +54 -0
  68. package/dist/client/error-reconstituter.d.ts.map +1 -0
  69. package/dist/client/form.d.ts +2 -2
  70. package/dist/client/form.d.ts.map +1 -1
  71. package/dist/client/history.d.ts +19 -4
  72. package/dist/client/history.d.ts.map +1 -1
  73. package/dist/client/index.d.ts +6 -5
  74. package/dist/client/index.d.ts.map +1 -1
  75. package/dist/client/index.js +537 -166
  76. package/dist/client/index.js.map +1 -1
  77. package/dist/client/link-pending-store.d.ts +78 -0
  78. package/dist/client/link-pending-store.d.ts.map +1 -0
  79. package/dist/client/link.d.ts +90 -32
  80. package/dist/client/link.d.ts.map +1 -1
  81. package/dist/client/nav-link-store.d.ts +36 -0
  82. package/dist/client/nav-link-store.d.ts.map +1 -0
  83. package/dist/client/navigation-api-types.d.ts +90 -0
  84. package/dist/client/navigation-api-types.d.ts.map +1 -0
  85. package/dist/client/navigation-api.d.ts +115 -0
  86. package/dist/client/navigation-api.d.ts.map +1 -0
  87. package/dist/client/navigation-context.d.ts +13 -2
  88. package/dist/client/navigation-context.d.ts.map +1 -1
  89. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  90. package/dist/client/navigation-root.d.ts.map +1 -0
  91. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  92. package/dist/client/router.d.ts +70 -4
  93. package/dist/client/router.d.ts.map +1 -1
  94. package/dist/client/rsc-fetch.d.ts +38 -3
  95. package/dist/client/rsc-fetch.d.ts.map +1 -1
  96. package/dist/client/segment-cache.d.ts +1 -1
  97. package/dist/client/segment-cache.d.ts.map +1 -1
  98. package/dist/client/segment-context.d.ts +1 -1
  99. package/dist/client/segment-context.d.ts.map +1 -1
  100. package/dist/client/segment-merger.d.ts.map +1 -1
  101. package/dist/client/segment-outlet.d.ts +63 -0
  102. package/dist/client/segment-outlet.d.ts.map +1 -0
  103. package/dist/client/ssr-data.d.ts +13 -4
  104. package/dist/client/ssr-data.d.ts.map +1 -1
  105. package/dist/client/stale-reload.d.ts +15 -0
  106. package/dist/client/stale-reload.d.ts.map +1 -1
  107. package/dist/client/top-loader.d.ts +3 -3
  108. package/dist/client/top-loader.d.ts.map +1 -1
  109. package/dist/client/use-params.d.ts +6 -4
  110. package/dist/client/use-params.d.ts.map +1 -1
  111. package/dist/client/use-query-states.d.ts +1 -1
  112. package/dist/client/use-query-states.d.ts.map +1 -1
  113. package/dist/codec.d.ts +23 -0
  114. package/dist/codec.d.ts.map +1 -0
  115. package/dist/codec.js +2 -0
  116. package/dist/cookies/define-cookie.d.ts +35 -14
  117. package/dist/cookies/define-cookie.d.ts.map +1 -1
  118. package/dist/cookies/index.d.ts +2 -0
  119. package/dist/cookies/index.d.ts.map +1 -1
  120. package/dist/cookies/index.js +3 -84
  121. package/dist/fonts/css.d.ts +1 -0
  122. package/dist/fonts/css.d.ts.map +1 -1
  123. package/dist/index.d.ts +154 -38
  124. package/dist/index.d.ts.map +1 -1
  125. package/dist/index.js +12092 -11916
  126. package/dist/index.js.map +1 -1
  127. package/dist/plugins/adapter-build.d.ts +1 -1
  128. package/dist/plugins/adapter-build.d.ts.map +1 -1
  129. package/dist/plugins/build-manifest.d.ts +2 -2
  130. package/dist/plugins/build-manifest.d.ts.map +1 -1
  131. package/dist/plugins/build-report.d.ts +3 -3
  132. package/dist/plugins/build-report.d.ts.map +1 -1
  133. package/dist/plugins/client-chunks.d.ts +32 -0
  134. package/dist/plugins/client-chunks.d.ts.map +1 -0
  135. package/dist/plugins/content.d.ts +1 -1
  136. package/dist/plugins/content.d.ts.map +1 -1
  137. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  138. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  139. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  140. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  141. package/dist/plugins/dev-logs.d.ts +1 -1
  142. package/dist/plugins/dev-logs.d.ts.map +1 -1
  143. package/dist/plugins/dev-server.d.ts +1 -1
  144. package/dist/plugins/dev-server.d.ts.map +1 -1
  145. package/dist/plugins/entries.d.ts +1 -1
  146. package/dist/plugins/entries.d.ts.map +1 -1
  147. package/dist/plugins/fonts.d.ts +19 -5
  148. package/dist/plugins/fonts.d.ts.map +1 -1
  149. package/dist/plugins/mdx.d.ts +1 -1
  150. package/dist/plugins/mdx.d.ts.map +1 -1
  151. package/dist/plugins/routing.d.ts +1 -1
  152. package/dist/plugins/routing.d.ts.map +1 -1
  153. package/dist/plugins/server-bundle.d.ts.map +1 -1
  154. package/dist/plugins/shims.d.ts +6 -5
  155. package/dist/plugins/shims.d.ts.map +1 -1
  156. package/dist/plugins/static-build.d.ts +1 -1
  157. package/dist/plugins/static-build.d.ts.map +1 -1
  158. package/dist/routing/codegen.d.ts +2 -2
  159. package/dist/routing/codegen.d.ts.map +1 -1
  160. package/dist/routing/index.d.ts +2 -0
  161. package/dist/routing/index.d.ts.map +1 -1
  162. package/dist/routing/index.js +3 -2
  163. package/dist/routing/scanner.d.ts.map +1 -1
  164. package/dist/routing/segment-classify.d.ts +46 -0
  165. package/dist/routing/segment-classify.d.ts.map +1 -0
  166. package/dist/routing/status-file-lint.d.ts +2 -1
  167. package/dist/routing/status-file-lint.d.ts.map +1 -1
  168. package/dist/routing/types.d.ts +16 -4
  169. package/dist/routing/types.d.ts.map +1 -1
  170. package/dist/rsc-runtime/rsc.d.ts +1 -1
  171. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  172. package/dist/rsc-runtime/ssr.d.ts +12 -0
  173. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  174. package/dist/schema-bridge.d.ts +76 -0
  175. package/dist/schema-bridge.d.ts.map +1 -0
  176. package/dist/search-params/define.d.ts +139 -0
  177. package/dist/search-params/define.d.ts.map +1 -0
  178. package/dist/search-params/index.d.ts +4 -6
  179. package/dist/search-params/index.d.ts.map +1 -1
  180. package/dist/search-params/index.js +4 -474
  181. package/dist/search-params/registry.d.ts +1 -1
  182. package/dist/search-params/wrappers.d.ts +53 -0
  183. package/dist/search-params/wrappers.d.ts.map +1 -0
  184. package/dist/segment-params/define.d.ts +78 -0
  185. package/dist/segment-params/define.d.ts.map +1 -0
  186. package/dist/segment-params/index.d.ts +7 -0
  187. package/dist/segment-params/index.d.ts.map +1 -0
  188. package/dist/segment-params/index.js +4 -0
  189. package/dist/server/access-gate.d.ts +4 -0
  190. package/dist/server/access-gate.d.ts.map +1 -1
  191. package/dist/server/action-client.d.ts +12 -1
  192. package/dist/server/action-client.d.ts.map +1 -1
  193. package/dist/server/action-encryption.d.ts +76 -0
  194. package/dist/server/action-encryption.d.ts.map +1 -0
  195. package/dist/server/action-handler.d.ts.map +1 -1
  196. package/dist/server/actions.d.ts +3 -6
  197. package/dist/server/actions.d.ts.map +1 -1
  198. package/dist/server/als-registry.d.ts +32 -4
  199. package/dist/server/als-registry.d.ts.map +1 -1
  200. package/dist/server/build-manifest.d.ts +2 -2
  201. package/dist/server/build-manifest.d.ts.map +1 -1
  202. package/dist/server/debug.d.ts +1 -1
  203. package/dist/server/default-logger.d.ts +22 -0
  204. package/dist/server/default-logger.d.ts.map +1 -0
  205. package/dist/server/deny-page-resolver.d.ts +52 -0
  206. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  207. package/dist/server/deny-renderer.d.ts.map +1 -1
  208. package/dist/server/dev-warnings.d.ts +0 -14
  209. package/dist/server/dev-warnings.d.ts.map +1 -1
  210. package/dist/server/early-hints.d.ts +13 -5
  211. package/dist/server/early-hints.d.ts.map +1 -1
  212. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  213. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  214. package/dist/server/fallback-error.d.ts +4 -3
  215. package/dist/server/fallback-error.d.ts.map +1 -1
  216. package/dist/server/flight-injection-state.d.ts +66 -0
  217. package/dist/server/flight-injection-state.d.ts.map +1 -0
  218. package/dist/server/flight-scripts.d.ts +42 -0
  219. package/dist/server/flight-scripts.d.ts.map +1 -0
  220. package/dist/server/flush.d.ts.map +1 -1
  221. package/dist/server/form-data.d.ts +29 -0
  222. package/dist/server/form-data.d.ts.map +1 -1
  223. package/dist/server/html-injectors.d.ts +51 -11
  224. package/dist/server/html-injectors.d.ts.map +1 -1
  225. package/dist/server/index.d.ts +5 -3
  226. package/dist/server/index.d.ts.map +1 -1
  227. package/dist/server/index.js +2176 -1663
  228. package/dist/server/index.js.map +1 -1
  229. package/dist/server/logger.d.ts +25 -7
  230. package/dist/server/logger.d.ts.map +1 -1
  231. package/dist/server/middleware-runner.d.ts +19 -4
  232. package/dist/server/middleware-runner.d.ts.map +1 -1
  233. package/dist/server/node-stream-transforms.d.ts +113 -0
  234. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  235. package/dist/server/page-deny-boundary.d.ts +31 -0
  236. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  237. package/dist/server/pipeline-interception.d.ts +1 -1
  238. package/dist/server/pipeline-interception.d.ts.map +1 -1
  239. package/dist/server/pipeline-metadata.d.ts +6 -0
  240. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  241. package/dist/server/pipeline.d.ts +32 -10
  242. package/dist/server/pipeline.d.ts.map +1 -1
  243. package/dist/server/primitives.d.ts +30 -3
  244. package/dist/server/primitives.d.ts.map +1 -1
  245. package/dist/server/render-timeout.d.ts +51 -0
  246. package/dist/server/render-timeout.d.ts.map +1 -0
  247. package/dist/server/request-context.d.ts +76 -37
  248. package/dist/server/request-context.d.ts.map +1 -1
  249. package/dist/server/route-element-builder.d.ts +27 -1
  250. package/dist/server/route-element-builder.d.ts.map +1 -1
  251. package/dist/server/route-handler.d.ts.map +1 -1
  252. package/dist/server/route-matcher.d.ts +9 -2
  253. package/dist/server/route-matcher.d.ts.map +1 -1
  254. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  255. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  256. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  257. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  258. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  259. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  260. package/dist/server/rsc-entry/index.d.ts +8 -3
  261. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  262. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  263. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  264. package/dist/server/rsc-entry/rsc-stream.d.ts +10 -1
  265. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  266. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  267. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  268. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  269. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  270. package/dist/server/safe-load.d.ts +46 -0
  271. package/dist/server/safe-load.d.ts.map +1 -0
  272. package/dist/server/sitemap-generator.d.ts +129 -0
  273. package/dist/server/sitemap-generator.d.ts.map +1 -0
  274. package/dist/server/sitemap-handler.d.ts +22 -0
  275. package/dist/server/sitemap-handler.d.ts.map +1 -0
  276. package/dist/server/slot-resolver.d.ts +1 -1
  277. package/dist/server/slot-resolver.d.ts.map +1 -1
  278. package/dist/server/ssr-entry.d.ts +22 -0
  279. package/dist/server/ssr-entry.d.ts.map +1 -1
  280. package/dist/server/ssr-render.d.ts +39 -21
  281. package/dist/server/ssr-render.d.ts.map +1 -1
  282. package/dist/server/ssr-wrappers.d.ts +50 -0
  283. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  284. package/dist/server/status-code-resolver.d.ts +1 -1
  285. package/dist/server/status-code-resolver.d.ts.map +1 -1
  286. package/dist/server/stream-utils.d.ts +36 -0
  287. package/dist/server/stream-utils.d.ts.map +1 -0
  288. package/dist/server/tracing.d.ts +10 -0
  289. package/dist/server/tracing.d.ts.map +1 -1
  290. package/dist/server/tree-builder.d.ts +22 -19
  291. package/dist/server/tree-builder.d.ts.map +1 -1
  292. package/dist/server/types.d.ts +1 -4
  293. package/dist/server/types.d.ts.map +1 -1
  294. package/dist/server/version-skew.d.ts +61 -0
  295. package/dist/server/version-skew.d.ts.map +1 -0
  296. package/dist/server/waituntil-bridge.d.ts.map +1 -1
  297. package/dist/shared/merge-search-params.d.ts +22 -0
  298. package/dist/shared/merge-search-params.d.ts.map +1 -0
  299. package/dist/shims/font-google.d.ts +1 -1
  300. package/dist/shims/font-google.d.ts.map +1 -1
  301. package/dist/shims/font-google.js +42 -0
  302. package/dist/shims/font-google.js.map +1 -0
  303. package/dist/shims/font-local.d.ts +26 -0
  304. package/dist/shims/font-local.d.ts.map +1 -0
  305. package/dist/shims/font-local.js +20 -0
  306. package/dist/shims/font-local.js.map +1 -0
  307. package/dist/shims/navigation-client.d.ts +1 -1
  308. package/dist/shims/navigation-client.d.ts.map +1 -1
  309. package/dist/shims/navigation.d.ts +1 -1
  310. package/dist/shims/navigation.d.ts.map +1 -1
  311. package/dist/utils/directive-parser.d.ts +5 -2
  312. package/dist/utils/directive-parser.d.ts.map +1 -1
  313. package/dist/utils/state-machine.d.ts +80 -0
  314. package/dist/utils/state-machine.d.ts.map +1 -0
  315. package/package.json +37 -17
  316. package/src/adapters/cloudflare-dev.ts +177 -0
  317. package/src/adapters/cloudflare.ts +342 -28
  318. package/src/adapters/compress-module.ts +24 -4
  319. package/src/adapters/nitro.ts +58 -9
  320. package/src/adapters/wrangler.d.ts +7 -0
  321. package/src/cache/cache-api.ts +38 -0
  322. package/src/cache/fast-hash.ts +34 -0
  323. package/src/cache/handler-store.ts +68 -0
  324. package/src/cache/index.ts +9 -5
  325. package/src/cache/singleflight.ts +62 -4
  326. package/src/cache/timber-cache.ts +40 -29
  327. package/src/cli.ts +0 -0
  328. package/src/client/browser-entry.ts +314 -142
  329. package/src/client/error-boundary.tsx +48 -16
  330. package/src/client/error-reconstituter.tsx +65 -0
  331. package/src/client/form.tsx +2 -2
  332. package/src/client/history.ts +26 -4
  333. package/src/client/index.ts +13 -4
  334. package/src/client/link-pending-store.ts +136 -0
  335. package/src/client/link.tsx +346 -105
  336. package/src/client/nav-link-store.ts +47 -0
  337. package/src/client/navigation-api-types.ts +112 -0
  338. package/src/client/navigation-api.ts +332 -0
  339. package/src/client/navigation-context.ts +27 -6
  340. package/src/client/navigation-root.tsx +346 -0
  341. package/src/client/nuqs-adapter.tsx +16 -3
  342. package/src/client/router.ts +302 -77
  343. package/src/client/rsc-fetch.ts +93 -5
  344. package/src/client/segment-cache.ts +1 -1
  345. package/src/client/segment-context.ts +6 -1
  346. package/src/client/segment-merger.ts +2 -8
  347. package/src/client/segment-outlet.tsx +86 -0
  348. package/src/client/ssr-data.ts +13 -5
  349. package/src/client/stale-reload.ts +73 -6
  350. package/src/client/top-loader.tsx +22 -13
  351. package/src/client/use-navigation-pending.ts +1 -1
  352. package/src/client/use-params.ts +7 -5
  353. package/src/client/use-query-states.ts +2 -2
  354. package/src/codec.ts +34 -0
  355. package/src/cookies/define-cookie.ts +72 -21
  356. package/src/cookies/index.ts +7 -0
  357. package/src/fonts/css.ts +2 -1
  358. package/src/index.ts +328 -92
  359. package/src/plugins/adapter-build.ts +8 -2
  360. package/src/plugins/build-manifest.ts +13 -2
  361. package/src/plugins/build-report.ts +3 -3
  362. package/src/plugins/client-chunks.ts +65 -0
  363. package/src/plugins/content.ts +1 -1
  364. package/src/plugins/dev-browser-logs.ts +288 -0
  365. package/src/plugins/dev-error-overlay.ts +70 -1
  366. package/src/plugins/dev-logs.ts +1 -1
  367. package/src/plugins/dev-server.ts +55 -9
  368. package/src/plugins/entries.ts +70 -9
  369. package/src/plugins/fonts.ts +167 -61
  370. package/src/plugins/mdx.ts +1 -1
  371. package/src/plugins/routing.ts +57 -17
  372. package/src/plugins/server-action-exports.ts +1 -1
  373. package/src/plugins/server-bundle.ts +32 -1
  374. package/src/plugins/shims.ts +76 -33
  375. package/src/plugins/static-build.ts +10 -6
  376. package/src/routing/codegen.ts +165 -105
  377. package/src/routing/index.ts +2 -0
  378. package/src/routing/scanner.ts +93 -23
  379. package/src/routing/segment-classify.ts +89 -0
  380. package/src/routing/status-file-lint.ts +3 -2
  381. package/src/routing/types.ts +17 -4
  382. package/src/rsc-runtime/rsc.ts +2 -0
  383. package/src/rsc-runtime/ssr.ts +50 -0
  384. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  385. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  386. package/src/search-params/define.ts +482 -0
  387. package/src/search-params/index.ts +13 -19
  388. package/src/search-params/registry.ts +1 -1
  389. package/src/search-params/wrappers.ts +85 -0
  390. package/src/segment-params/define.ts +279 -0
  391. package/src/segment-params/index.ts +28 -0
  392. package/src/server/access-gate.tsx +70 -29
  393. package/src/server/action-client.ts +28 -3
  394. package/src/server/action-encryption.ts +144 -0
  395. package/src/server/action-handler.ts +20 -3
  396. package/src/server/actions.ts +10 -9
  397. package/src/server/als-registry.ts +32 -4
  398. package/src/server/build-manifest.ts +10 -4
  399. package/src/server/compress.ts +25 -7
  400. package/src/server/debug.ts +1 -1
  401. package/src/server/default-logger.ts +99 -0
  402. package/src/server/deny-page-resolver.ts +154 -0
  403. package/src/server/deny-renderer.ts +24 -38
  404. package/src/server/dev-warnings.ts +2 -28
  405. package/src/server/early-hints.ts +36 -15
  406. package/src/server/error-boundary-wrapper.ts +74 -22
  407. package/src/server/fallback-error.ts +31 -15
  408. package/src/server/flight-injection-state.ts +113 -0
  409. package/src/server/flight-scripts.ts +62 -0
  410. package/src/server/flush.ts +2 -1
  411. package/src/server/form-data.ts +76 -0
  412. package/src/server/html-injectors.ts +277 -117
  413. package/src/server/index.ts +9 -5
  414. package/src/server/logger.ts +44 -36
  415. package/src/server/middleware-runner.ts +31 -4
  416. package/src/server/node-stream-transforms.ts +509 -0
  417. package/src/server/page-deny-boundary.tsx +56 -0
  418. package/src/server/pipeline-interception.ts +17 -16
  419. package/src/server/pipeline-metadata.ts +13 -0
  420. package/src/server/pipeline.ts +195 -51
  421. package/src/server/primitives.ts +47 -5
  422. package/src/server/render-timeout.ts +108 -0
  423. package/src/server/request-context.ts +240 -117
  424. package/src/server/route-element-builder.ts +284 -197
  425. package/src/server/route-handler.ts +24 -4
  426. package/src/server/route-matcher.ts +24 -20
  427. package/src/server/rsc-entry/api-handler.ts +15 -16
  428. package/src/server/rsc-entry/error-renderer.ts +300 -89
  429. package/src/server/rsc-entry/helpers.ts +134 -5
  430. package/src/server/rsc-entry/index.ts +202 -113
  431. package/src/server/rsc-entry/rsc-payload.ts +100 -21
  432. package/src/server/rsc-entry/rsc-stream.ts +74 -18
  433. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  434. package/src/server/rsc-entry/ssr-renderer.ts +173 -40
  435. package/src/server/safe-load.ts +60 -0
  436. package/src/server/sitemap-generator.ts +338 -0
  437. package/src/server/sitemap-handler.ts +126 -0
  438. package/src/server/slot-resolver.ts +243 -228
  439. package/src/server/ssr-entry.ts +211 -32
  440. package/src/server/ssr-render.ts +289 -67
  441. package/src/server/ssr-wrappers.tsx +139 -0
  442. package/src/server/status-code-resolver.ts +1 -1
  443. package/src/server/stream-utils.ts +213 -0
  444. package/src/server/tracing.ts +37 -3
  445. package/src/server/tree-builder.ts +92 -58
  446. package/src/server/types.ts +3 -6
  447. package/src/server/version-skew.ts +104 -0
  448. package/src/server/waituntil-bridge.ts +4 -1
  449. package/src/shared/merge-search-params.ts +55 -0
  450. package/src/shims/font-google.ts +1 -1
  451. package/src/shims/font-local.ts +34 -0
  452. package/src/shims/navigation-client.ts +1 -1
  453. package/src/shims/navigation.ts +2 -1
  454. package/src/utils/directive-parser.ts +5 -2
  455. package/src/utils/state-machine.ts +111 -0
  456. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  457. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  458. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  459. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  460. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  461. package/dist/_chunks/ssr-data-MjmprTmO.js +0 -88
  462. package/dist/_chunks/ssr-data-MjmprTmO.js.map +0 -1
  463. package/dist/_chunks/tracing-Cwn7697K.js.map +0 -1
  464. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  465. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  466. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  467. package/dist/cache/register-cached-function.d.ts +0 -17
  468. package/dist/cache/register-cached-function.d.ts.map +0 -1
  469. package/dist/client/error-boundary.js.map +0 -1
  470. package/dist/client/link-status-provider.d.ts +0 -11
  471. package/dist/client/link-status-provider.d.ts.map +0 -1
  472. package/dist/client/transition-root.d.ts.map +0 -1
  473. package/dist/cookies/index.js.map +0 -1
  474. package/dist/plugins/cache-transform.d.ts +0 -36
  475. package/dist/plugins/cache-transform.d.ts.map +0 -1
  476. package/dist/plugins/dynamic-transform.d.ts +0 -72
  477. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  478. package/dist/search-params/analyze.d.ts +0 -54
  479. package/dist/search-params/analyze.d.ts.map +0 -1
  480. package/dist/search-params/builtin-codecs.d.ts +0 -105
  481. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  482. package/dist/search-params/codecs.d.ts +0 -53
  483. package/dist/search-params/codecs.d.ts.map +0 -1
  484. package/dist/search-params/create.d.ts +0 -106
  485. package/dist/search-params/create.d.ts.map +0 -1
  486. package/dist/search-params/index.js.map +0 -1
  487. package/dist/server/prerender.d.ts +0 -77
  488. package/dist/server/prerender.d.ts.map +0 -1
  489. package/dist/server/response-cache.d.ts +0 -53
  490. package/dist/server/response-cache.d.ts.map +0 -1
  491. package/src/cache/register-cached-function.ts +0 -99
  492. package/src/client/link-status-provider.tsx +0 -30
  493. package/src/client/transition-root.tsx +0 -160
  494. package/src/plugins/cache-transform.ts +0 -199
  495. package/src/plugins/dynamic-transform.ts +0 -161
  496. package/src/search-params/analyze.ts +0 -192
  497. package/src/search-params/builtin-codecs.ts +0 -228
  498. package/src/search-params/create.ts +0 -321
  499. package/src/server/prerender.ts +0 -139
  500. package/src/server/response-cache.ts +0 -277
@@ -18,7 +18,7 @@ import {
18
18
  decodeReply,
19
19
  decodeAction,
20
20
  renderToReadableStream,
21
- } from '#/rsc-runtime/rsc.js';
21
+ } from '../rsc-runtime/rsc.js';
22
22
 
23
23
  import { validateCsrf, type CsrfConfig } from './csrf.js';
24
24
  import { executeAction, type RevalidateRenderer } from './actions.js';
@@ -31,6 +31,8 @@ import { handleActionError } from './action-client.js';
31
31
  import { enforceBodyLimits, enforceFieldLimit, type BodyLimitsConfig } from './body-limits.js';
32
32
  import { parseFormData } from './form-data.js';
33
33
  import type { FormFlashData } from './form-flash.js';
34
+ import { checkVersionSkew, applyReloadHeaders } from './version-skew.js';
35
+ import { logActionError } from './logger.js';
34
36
 
35
37
  // ─── Types ────────────────────────────────────────────────────────────────
36
38
 
@@ -90,6 +92,21 @@ export async function handleActionRequest(
90
92
  req: Request,
91
93
  config: ActionDispatchConfig
92
94
  ): Promise<Response | FormRerender | null> {
95
+ // Version skew detection — reject actions from stale clients (TIM-446).
96
+ // On mismatch, return a structured RSC error response that the client
97
+ // handles by showing a brief "App updated" message and reloading.
98
+ const skewCheck = checkVersionSkew(req);
99
+ if (!skewCheck.ok) {
100
+ const reloadHeaders = new Headers({
101
+ 'Content-Type': RSC_CONTENT_TYPE,
102
+ });
103
+ applyReloadHeaders(reloadHeaders);
104
+ // Return the reload signal as an RSC stream so createFromFetch can
105
+ // decode it. The client checks X-Timber-Reload before processing.
106
+ const rscStream = renderToReadableStream({ _versionSkew: true });
107
+ return new Response(rscStream, { status: 200, headers: reloadHeaders });
108
+ }
109
+
93
110
  // CSRF validation — reject cross-origin mutation requests.
94
111
  const csrfResult = validateCsrf(req, config.csrf);
95
112
  if (!csrfResult.ok) {
@@ -177,7 +194,7 @@ async function handleRscAction(
177
194
  });
178
195
  } catch (error) {
179
196
  // Log full error server-side for debugging
180
- console.error('[timber] server action error:', error);
197
+ logActionError({ method: req.method, path: new URL(req.url).pathname, error });
181
198
 
182
199
  // Return structured error response — ActionError gets its code/data,
183
200
  // unexpected errors get sanitized { code: 'INTERNAL_ERROR' }
@@ -293,7 +310,7 @@ async function handleFormAction(
293
310
  renderer: config.revalidateRenderer,
294
311
  });
295
312
  } catch (error) {
296
- console.error('[timber] server action error:', error);
313
+ logActionError({ method: req.method, path: new URL(req.url).pathname, error });
297
314
 
298
315
  // Return the error as flash data for re-render.
299
316
  // handleActionError produces { serverError } for ActionErrors
@@ -3,7 +3,7 @@
3
3
  *
4
4
  * - revalidatePath(path) re-renders the route at that path and returns the RSC
5
5
  * flight payload for inline reconciliation.
6
- * - revalidateTag(tag) invalidates cached shells and 'use cache' entries by tag.
6
+ * - revalidateTag(tag) invalidates timber.cache entries by tag.
7
7
  *
8
8
  * Both are callable from anywhere on the server — actions, API routes, handlers.
9
9
  *
@@ -14,7 +14,7 @@
14
14
  * See design/08-forms-and-actions.md
15
15
  */
16
16
 
17
- import type { CacheHandler } from '#/cache/index';
17
+ import { getCacheHandler } from '../cache/handler-store';
18
18
  import { RedirectSignal } from './primitives';
19
19
  import { withSpan } from './tracing';
20
20
  import { revalidationAls, type RevalidationState } from './als-registry.js';
@@ -37,8 +37,6 @@ export type { RevalidationState } from './als-registry.js';
37
37
 
38
38
  /** Options for creating the action handler. */
39
39
  export interface ActionHandlerConfig {
40
- /** Cache handler for tag invalidation. */
41
- cacheHandler?: CacheHandler;
42
40
  /** Renderer for producing RSC payloads during revalidation. */
43
41
  renderer?: RevalidateRenderer;
44
42
  }
@@ -112,8 +110,8 @@ export function revalidatePath(path: string): void {
112
110
  }
113
111
 
114
112
  /**
115
- * Invalidate all pre-rendered shells and 'use cache' entries tagged with `tag`.
116
- * Does not return a payload — the next request for an invalidated route re-renders fresh.
113
+ * Invalidate all timber.cache entries tagged with `tag`.
114
+ * Does not return a payload — the next request for an invalidated entry re-executes.
117
115
  *
118
116
  * @param tag - The cache tag to invalidate (e.g. 'products', 'user:123').
119
117
  */
@@ -172,9 +170,12 @@ export async function executeAction(
172
170
  }
173
171
  });
174
172
 
175
- // Process tag invalidation
176
- if (state.tags.length > 0 && config.cacheHandler) {
177
- await Promise.all(state.tags.map((tag) => config.cacheHandler!.invalidate({ tag })));
173
+ // Process tag invalidation via the module-level cache handler singleton.
174
+ // setCacheHandler() is called at boot from rsc-entry when timber.config.ts
175
+ // provides a cacheHandler; otherwise falls back to in-memory LRU (TIM-599).
176
+ if (state.tags.length > 0) {
177
+ const handler = getCacheHandler();
178
+ await Promise.all(state.tags.map((tag) => handler.invalidate({ tag })));
178
179
  }
179
180
 
180
181
  // Process path revalidation — build element tree (not yet serialized)
@@ -21,6 +21,7 @@
21
21
  */
22
22
 
23
23
  import { AsyncLocalStorage } from 'node:async_hooks';
24
+ import type { DebugComponentEntry } from './rsc-entry/helpers.js';
24
25
 
25
26
  // ─── Request Context ──────────────────────────────────────────────────────
26
27
  // Used by: request-context.ts (headers(), cookies(), searchParams())
@@ -39,17 +40,44 @@ export interface RequestContextStore {
39
40
  /** Original (pre-overlay) frozen headers, kept for overlay merging. */
40
41
  originalHeaders: Headers;
41
42
  /**
42
- * Promise resolving to the route's typed search params (when search-params.ts
43
- * exists) or to the raw URLSearchParams. Stored as a Promise so the framework
44
- * can later support partial pre-rendering where param resolution is deferred.
43
+ * Promise resolving to the raw URLSearchParams for the current request.
44
+ * To get typed parsed params, import a search params definition and
45
+ * call `.parse(searchParams())`.
45
46
  */
46
- searchParamsPromise: Promise<URLSearchParams | Record<string, unknown>>;
47
+ searchParamsPromise: Promise<URLSearchParams>;
48
+ /**
49
+ * Raw search string from the request URL (e.g. "?foo=bar&baz=1").
50
+ * Available synchronously for use in `redirect()` with `preserveSearchParams`.
51
+ */
52
+ searchString: string;
53
+ /**
54
+ * Promise resolving to the coerced segment params for the current request.
55
+ * Set by the pipeline after route matching and param coercion, before
56
+ * middleware and rendering. Pages and layouts read params via
57
+ * `rawSegmentParams()` instead of receiving them as a prop.
58
+ *
59
+ * See design/07-routing.md §"params.ts — Convention File for Typed Params"
60
+ */
61
+ segmentParamsPromise?: Promise<Record<string, string | string[]>>;
47
62
  /** Outgoing Set-Cookie entries (name → serialized value + options). Last write wins. */
48
63
  cookieJar: Map<string, CookieEntry>;
49
64
  /** Whether the response has flushed (headers committed). */
50
65
  flushed: boolean;
51
66
  /** Whether the current context allows cookie mutation. */
52
67
  mutableContext: boolean;
68
+ /**
69
+ * Set by AccessGate or PageDenyBoundary when a DenySignal is caught
70
+ * server-side (inside the React tree, before React Flight sees it).
71
+ * The pipeline reads this after render to set the HTTP status code.
72
+ * See TIM-666.
73
+ */
74
+ denyStatus?: number;
75
+ /**
76
+ * Dev-only: getter for the current request's RSC debug components.
77
+ * Set by renderRoute() so onPipelineError can include component tree
78
+ * context for render-phase errors without module-level shared state.
79
+ */
80
+ debugComponentsGetter?: () => DebugComponentEntry[];
53
81
  }
54
82
 
55
83
  /** A single outgoing cookie entry in the cookie jar. */
@@ -85,6 +85,12 @@ export function collectRouteCss(segments: SegmentWithFiles[], manifest: BuildMan
85
85
  * via injectHead() before </head>.
86
86
  */
87
87
  export function buildCssLinkTags(cssUrls: string[]): string {
88
+ // Emit <link rel="stylesheet"> tags as a fallback for platforms where
89
+ // React's Float system (via @vitejs/plugin-rsc preinit with
90
+ // data-precedence) doesn't handle CSS injection. In practice, Float
91
+ // deduplicates and these may be dropped. No preload hints — Float
92
+ // already starts the fetch via preinit(), and redundant preloads
93
+ // cause "ignored due to unknown as/type" browser warnings.
88
94
  return cssUrls.map((url) => `<link rel="stylesheet" href="${url}">`).join('');
89
95
  }
90
96
 
@@ -95,10 +101,10 @@ export function buildCssLinkTags(cssUrls: string[]): string {
95
101
  * into 103 Early Hints responses. This avoids platform-specific 103
96
102
  * sending code.
97
103
  *
98
- * Example output: `</assets/root.css>; rel=preload; as=style, </assets/page.css>; rel=preload; as=style`
104
+ * Example output: `</assets/root.css>; as=style; rel=preload, </assets/page.css>; as=style; rel=preload`
99
105
  */
100
106
  export function buildLinkHeaders(cssUrls: string[]): string {
101
- return cssUrls.map((url) => `<${url}>; rel=preload; as=style`).join(', ');
107
+ return cssUrls.map((url) => `<${url}>; as=style; rel=preload`).join(', ');
102
108
  }
103
109
 
104
110
  // ─── Font utilities ──────────────────────────────────────────────────────
@@ -153,10 +159,10 @@ export function buildFontPreloadTags(fonts: ManifestFontEntry[]): string {
153
159
  *
154
160
  * Cloudflare CDN converts Link headers with rel=preload into 103 Early Hints.
155
161
  *
156
- * Example: `</fonts/inter.woff2>; rel=preload; as=font; crossorigin`
162
+ * Example: `</fonts/inter.woff2>; as=font; rel=preload; crossorigin`
157
163
  */
158
164
  export function buildFontLinkHeaders(fonts: ManifestFontEntry[]): string {
159
- return fonts.map((f) => `<${f.href}>; rel=preload; as=font; crossorigin`).join(', ');
165
+ return fonts.map((f) => `<${f.href}>; as=font; rel=preload; crossorigin`).join(', ');
160
166
  }
161
167
 
162
168
  // ─── JS chunk utilities ──────────────────────────────────────────────────
@@ -160,15 +160,33 @@ export function compressResponse(request: Request, response: Response): Response
160
160
  });
161
161
  }
162
162
 
163
- // ─── Gzip (CompressionStream API) ────────────────────────────────────────
163
+ // ─── Gzip (node:zlib with Z_SYNC_FLUSH) ──────────────────────────────────
164
+ //
165
+ // Uses node:zlib's createGzip with Z_SYNC_FLUSH so each chunk is flushed
166
+ // to the output immediately. The Web Platform CompressionStream API buffers
167
+ // internally and does NOT flush per-chunk — this kills streaming because
168
+ // the browser doesn't receive the HTML shell until the gzip stream closes
169
+ // (i.e. after all Suspense boundaries resolve).
170
+ //
171
+ // Z_SYNC_FLUSH adds ~2–5% size overhead vs Z_NO_FLUSH but preserves
172
+ // correct streaming behavior: the shell renders instantly, Suspense
173
+ // fallbacks are visible immediately, and streamed content appears
174
+ // progressively.
175
+
176
+ import { createGzip, constants } from 'node:zlib';
177
+ import { Readable } from 'node:stream';
164
178
 
165
179
  /**
166
- * Compress a ReadableStream with gzip using the Web Platform CompressionStream API.
167
- * Available in Node 18+, Bun, and Deno — no npm dependency needed.
180
+ * Compress a ReadableStream with gzip, flushing each chunk immediately.
181
+ *
182
+ * Uses node:zlib's createGzip with Z_SYNC_FLUSH to ensure each HTML chunk
183
+ * (shell, Suspense resolution, RSC payload) is delivered to the browser
184
+ * as soon as it's available — preserving streaming semantics.
168
185
  */
169
186
  function compressWithGzip(body: ReadableStream<Uint8Array>): ReadableStream<Uint8Array> {
170
- const compressionStream = new CompressionStream('gzip');
171
- // Cast needed: CompressionStream's WritableStream<BufferSource> type is wider
172
- // than ReadableStream's Uint8Array, but Uint8Array is a valid BufferSource.
173
- return body.pipeThrough(compressionStream as unknown as TransformStream<Uint8Array, Uint8Array>);
187
+ const gzip = createGzip({ flush: constants.Z_SYNC_FLUSH });
188
+ const nodeInput = Readable.fromWeb(body as import('stream/web').ReadableStream);
189
+ nodeInput.pipe(gzip);
190
+
191
+ return Readable.toWeb(gzip) as ReadableStream<Uint8Array>;
174
192
  }
@@ -48,7 +48,7 @@
48
48
  *
49
49
  * This is the ONLY function that should gate client-visible dev behavior:
50
50
  * - Dev error pages with stack traces
51
- * - Detailed Server-Timing response headers
51
+ * - Server-Timing mode default (`'detailed'` in dev, `'total'` in prod)
52
52
  * - Error messages in action `INTERNAL_ERROR` payloads
53
53
  * - Pipeline error handler wiring (Vite overlay)
54
54
  *
@@ -0,0 +1,99 @@
1
+ /**
2
+ * DefaultLogger — human-readable stderr logging when no custom logger is configured.
3
+ *
4
+ * Ships as the fallback so production deployments always have error visibility,
5
+ * even without an `instrumentation.ts` logger export. Output is one line per
6
+ * event, designed for `fly logs`, `kubectl logs`, Cloudflare dashboard tails, etc.
7
+ *
8
+ * Format:
9
+ * [timber] ERROR message key=value key=value trace_id=4bf92f35
10
+ * [timber] WARN message key=value key=value trace_id=4bf92f35
11
+ * [timber] INFO message method=GET path=/dashboard status=200 durationMs=43 trace_id=4bf92f35
12
+ *
13
+ * Behavior:
14
+ * - Suppressed entirely in dev mode (dev logging handles all output)
15
+ * - `debug` suppressed unless TIMBER_DEBUG is set
16
+ * - Replaced entirely when a custom logger is set via `setLogger()`
17
+ *
18
+ * See design/17-logging.md §"DefaultLogger"
19
+ */
20
+
21
+ import { isDevMode, isDebug } from './debug.js';
22
+ import { formatSsrError } from './error-formatter.js';
23
+ import type { TimberLogger } from './logger.js';
24
+
25
+ /**
26
+ * Format data fields as `key=value` pairs for human-readable output.
27
+ * - `error` key is serialized via formatSsrError for stack trace cleanup
28
+ * - `trace_id` is truncated to 8 chars for readability (full ID in OTEL)
29
+ * - Other values are stringified inline
30
+ */
31
+ function formatDataFields(data?: Record<string, unknown>): string {
32
+ if (!data) return '';
33
+
34
+ const parts: string[] = [];
35
+ let traceId: string | undefined;
36
+
37
+ for (const [key, value] of Object.entries(data)) {
38
+ if (key === 'trace_id') {
39
+ // Defer trace_id to the end
40
+ traceId = typeof value === 'string' ? value : String(value);
41
+ continue;
42
+ }
43
+ if (key === 'error') {
44
+ // Serialize errors with formatSsrError for clean output
45
+ parts.push(`error=${formatSsrError(value)}`);
46
+ continue;
47
+ }
48
+ if (value === undefined || value === null) continue;
49
+ parts.push(`${key}=${value}`);
50
+ }
51
+
52
+ // trace_id always last, truncated to 8 chars for readability
53
+ if (traceId) {
54
+ parts.push(`trace_id=${traceId.slice(0, 8)}`);
55
+ }
56
+
57
+ return parts.length > 0 ? ' ' + parts.join(' ') : '';
58
+ }
59
+
60
+ /** Pad level string to fixed width for alignment. */
61
+ function padLevel(level: string): string {
62
+ return level.padEnd(5);
63
+ }
64
+
65
+ export function createDefaultLogger(): TimberLogger {
66
+ return {
67
+ error(msg: string, data?: Record<string, unknown>): void {
68
+ // Errors are ALWAYS logged, including dev mode. Suppressing errors
69
+ // in dev causes silent 500s with no stack trace, making route.ts
70
+ // and render errors impossible to debug. See TIM-555.
71
+ const fields = formatDataFields(data);
72
+ process.stderr.write(`[timber] ${padLevel('ERROR')} ${msg}${fields}\n`);
73
+ },
74
+
75
+ warn(msg: string, data?: Record<string, unknown>): void {
76
+ // Warnings are always logged — same rationale as errors.
77
+ const fields = formatDataFields(data);
78
+ process.stderr.write(`[timber] ${padLevel('WARN')} ${msg}${fields}\n`);
79
+ },
80
+
81
+ info(msg: string, data?: Record<string, unknown>): void {
82
+ // info is suppressed by default — per-request lines are too noisy
83
+ // without a custom logger. Enable with TIMBER_DEBUG.
84
+ if (isDevMode()) return;
85
+ if (!isDebug()) return;
86
+ const fields = formatDataFields(data);
87
+ process.stderr.write(`[timber] ${padLevel('INFO')} ${msg}${fields}\n`);
88
+ },
89
+
90
+ debug(msg: string, data?: Record<string, unknown>): void {
91
+ // debug is suppressed in dev (dev logger handles it) and in
92
+ // production unless TIMBER_DEBUG is explicitly set.
93
+ if (isDevMode()) return;
94
+ if (!isDebug()) return;
95
+ const fields = formatDataFields(data);
96
+ process.stderr.write(`[timber] ${padLevel('DEBUG')} ${msg}${fields}\n`);
97
+ },
98
+ };
99
+ }
@@ -0,0 +1,154 @@
1
+ /**
2
+ * Deny Page Resolver — resolves status-code file components for in-tree deny handling.
3
+ *
4
+ * When AccessGate or PageDenyBoundary catches a DenySignal, they need to
5
+ * render the matching deny page (403.tsx, 4xx.tsx, error.tsx) as a normal
6
+ * element in the React tree. This module resolves the deny page chain from
7
+ * the segment chain — a list of fallback components ordered by specificity.
8
+ *
9
+ * The chain is built during buildRouteElement and passed as a prop to
10
+ * AccessGate and PageDenyBoundary. At catch time, the first matching
11
+ * component is rendered.
12
+ *
13
+ * See design/10-error-handling.md §"Status-Code Files", TIM-666.
14
+ */
15
+
16
+ import { createElement } from 'react';
17
+
18
+ import type { ManifestSegmentNode } from './route-matcher.js';
19
+ import { loadModule } from './safe-load.js';
20
+ import { requestContextAls } from './als-registry.js';
21
+
22
+ // ─── Types ────────────────────────────────────────────────────────────────
23
+
24
+ /** A single entry in the deny page fallback chain. */
25
+ export interface DenyPageEntry {
26
+ /** Status code filter: specific (403), category (400 = any 4xx), or null (catch-all). */
27
+ status: number | null;
28
+ /** The component to render (server or client — both work). */
29
+ component: (...args: unknown[]) => unknown;
30
+ }
31
+
32
+ // ─── Resolver ─────────────────────────────────────────────────────────────
33
+
34
+ /**
35
+ * Build the deny page fallback chain from the segment chain.
36
+ *
37
+ * Walks segments from `startIndex` outward (toward root) and collects
38
+ * status-code file components in fallback order:
39
+ * 1. Specific status files (403.tsx, 404.tsx) — exact match
40
+ * 2. Category catch-alls (4xx.tsx) — matches any 4xx
41
+ * 3. error.tsx — catches everything
42
+ *
43
+ * Each segment is checked in this order. The chain is ordered so the
44
+ * FIRST match wins at catch time.
45
+ */
46
+ export async function buildDenyPageChain(
47
+ segments: ManifestSegmentNode[],
48
+ startIndex: number
49
+ ): Promise<DenyPageEntry[]> {
50
+ const chain: DenyPageEntry[] = [];
51
+
52
+ // Pass 1: Status files (specific + category) across ALL segments.
53
+ // These have higher priority than error.tsx — a root 4xx.tsx should
54
+ // match before a leaf error.tsx. Walking inner → outer ensures the
55
+ // nearest match wins within each priority tier.
56
+ for (let i = startIndex; i >= 0; i--) {
57
+ const segment = segments[i];
58
+ if (!segment.statusFiles) continue;
59
+
60
+ // Specific status files (403.tsx, 404.tsx, etc.)
61
+ for (const [key, file] of Object.entries(segment.statusFiles)) {
62
+ if (key !== '4xx' && key !== '5xx') {
63
+ const status = parseInt(key, 10);
64
+ if (!isNaN(status)) {
65
+ const mod = await loadModule(file).catch(() => null);
66
+ if (mod?.default) {
67
+ chain.push({ status, component: mod.default as (...args: unknown[]) => unknown });
68
+ }
69
+ }
70
+ }
71
+ }
72
+
73
+ // Category catch-alls (4xx.tsx, 5xx.tsx)
74
+ for (const [key, file] of Object.entries(segment.statusFiles)) {
75
+ if (key === '4xx' || key === '5xx') {
76
+ const mod = await loadModule(file).catch(() => null);
77
+ if (mod?.default) {
78
+ const categoryStatus = key === '4xx' ? 400 : 500;
79
+ chain.push({
80
+ status: categoryStatus,
81
+ component: mod.default as (...args: unknown[]) => unknown,
82
+ });
83
+ }
84
+ }
85
+ }
86
+ }
87
+
88
+ // Pass 2: error.tsx files — lowest priority catch-all.
89
+ // Only added AFTER all status files so they never shadow a specific
90
+ // or category status file from an ancestor segment.
91
+ for (let i = startIndex; i >= 0; i--) {
92
+ const segment = segments[i];
93
+ if (segment.error) {
94
+ const mod = await loadModule(segment.error).catch(() => null);
95
+ if (mod?.default) {
96
+ chain.push({ status: null, component: mod.default as (...args: unknown[]) => unknown });
97
+ }
98
+ }
99
+ }
100
+
101
+ return chain;
102
+ }
103
+
104
+ // ─── Matcher ──────────────────────────────────────────────────────────────
105
+
106
+ /**
107
+ * Find the first deny page in the chain that matches the given status code.
108
+ * Returns a React element for the matching component, or null if no match.
109
+ */
110
+ export function renderMatchingDenyPage(
111
+ chain: DenyPageEntry[],
112
+ status: number,
113
+ data: unknown
114
+ ): React.ReactElement | null {
115
+ const h = createElement as (...args: unknown[]) => React.ReactElement;
116
+
117
+ for (const entry of chain) {
118
+ if (entry.status === status) {
119
+ return h(entry.component, { status, dangerouslyPassData: data });
120
+ }
121
+ if (entry.status === 400 && status >= 400 && status <= 499) {
122
+ return h(entry.component, { status, dangerouslyPassData: data });
123
+ }
124
+ if (entry.status === 500 && status >= 500 && status <= 599) {
125
+ return h(entry.component, { status, dangerouslyPassData: data });
126
+ }
127
+ if (entry.status === null) {
128
+ return h(entry.component, { status, dangerouslyPassData: data });
129
+ }
130
+ }
131
+ return null;
132
+ }
133
+
134
+ // ─── ALS Helper ───────────────────────────────────────────────────────────
135
+
136
+ /**
137
+ * Set the deny status in the request context ALS.
138
+ * Called from AccessGate / PageDenyBoundary when a DenySignal is caught.
139
+ * The pipeline reads this after render to set the HTTP status code.
140
+ */
141
+ export function setDenyStatus(status: number): void {
142
+ const store = requestContextAls.getStore();
143
+ if (store) {
144
+ store.denyStatus = status;
145
+ }
146
+ }
147
+
148
+ /**
149
+ * Read the deny status from the request context ALS.
150
+ * Returns undefined if no deny was caught during render.
151
+ */
152
+ export function getDenyStatus(): number | undefined {
153
+ return requestContextAls.getStore()?.denyStatus;
154
+ }
@@ -16,18 +16,21 @@
16
16
  */
17
17
 
18
18
  import { createElement } from 'react';
19
- import { renderToReadableStream } from '#/rsc-runtime/rsc.js';
19
+ import { renderToReadableStream } from '../rsc-runtime/rsc.js';
20
20
 
21
21
  import { DenySignal } from './primitives.js';
22
22
  import { logRenderError } from './logger.js';
23
+ import { loadModule } from './safe-load.js';
23
24
  import { isDebug } from './debug.js';
24
25
  import { resolveMetadata, renderMetadataToElements } from './metadata.js';
25
26
  import { resolveManifestStatusFile } from './manifest-status-resolver.js';
26
27
  import type { ManifestSegmentNode } from './route-matcher.js';
27
28
  import type { RouteMatch } from './pipeline.js';
28
29
  import type { NavContext } from './ssr-entry.js';
30
+ import { flightInitScript } from './flight-scripts.js';
29
31
  import type { ClientBootstrapConfig } from './html-injectors.js';
30
32
  import type { Metadata } from './types.js';
33
+ import { teeWithErrorPropagation } from './stream-utils.js';
31
34
 
32
35
  /** RSC content type for client navigation payload requests. */
33
36
  const RSC_CONTENT_TYPE = 'text/x-component';
@@ -108,7 +111,7 @@ export async function renderDenyPage(
108
111
  }
109
112
 
110
113
  // Load the status-code page component
111
- const mod = (await resolution.file.load()) as Record<string, unknown>;
114
+ const mod = await loadModule(resolution.file);
112
115
  if (!mod.default) {
113
116
  return new Response(null, { status: deny.status, headers: responseHeaders });
114
117
  }
@@ -116,9 +119,6 @@ export async function renderDenyPage(
116
119
  const ErrorPageComponent = mod.default as (...args: unknown[]) => unknown;
117
120
  const h = createElement as (...args: unknown[]) => React.ReactElement;
118
121
 
119
- // Check shell opt-out: export const shell = false
120
- const shellEnabled = mod.shell !== false;
121
-
122
122
  // 4xx status-code pages receive { status, dangerouslyPassData }
123
123
  // per design/10-error-handling.md §"Status-Code File Props"
124
124
  let element = h(ErrorPageComponent, {
@@ -126,23 +126,14 @@ export async function renderDenyPage(
126
126
  dangerouslyPassData: deny.data,
127
127
  });
128
128
 
129
- // Wrap in layouts unless shell is explicitly disabled
130
- if (shellEnabled) {
131
- const resolvedSegments = new Set(segments.slice(0, resolution.segmentIndex + 1));
132
- const layoutsToWrap = layoutComponents.filter((lc) => resolvedSegments.has(lc.segment));
133
- for (let i = layoutsToWrap.length - 1; i >= 0; i--) {
134
- const { component } = layoutsToWrap[i];
135
- element = h(component, null, element);
136
- }
137
- } else if (isDebug()) {
138
- // Dev-mode: warn if shell=false might conflict with Suspense
139
- // The actual Suspense boundary check happens at render time in the pipeline.
140
- // This is a preemptive log for developer awareness.
141
- console.warn(
142
- `[timber] Status-code file ${resolution.file.filePath} exports shell = false. ` +
143
- 'If deny() fires inside a Suspense boundary, layouts are already committed and ' +
144
- 'cannot be unwrapped. The shell opt-out will be ignored in that case.'
145
- );
129
+ // Always wrap in layout shell.
130
+ // NOTE: Shell opt-out (export const shell = false) is a future feature
131
+ // pending a proper design doc. See design backlog.
132
+ const resolvedSegments = new Set(segments.slice(0, resolution.segmentIndex + 1));
133
+ const layoutsToWrap = layoutComponents.filter((lc) => resolvedSegments.has(lc.segment));
134
+ for (let i = layoutsToWrap.length - 1; i >= 0; i--) {
135
+ const { component } = layoutsToWrap[i];
136
+ element = h(component, null, element);
146
137
  }
147
138
 
148
139
  // Build head HTML from error page metadata (if any)
@@ -170,15 +161,15 @@ export async function renderDenyPage(
170
161
  debugChannel: createDebugChannelSink(),
171
162
  });
172
163
 
173
- const [ssrStream, inlineStream] = rscStream.tee();
164
+ const [ssrStream, inlineStream] = teeWithErrorPropagation(rscStream);
174
165
 
175
166
  const navContext: NavContext = {
176
167
  pathname: new URL(req.url).pathname,
177
- params: match.params,
168
+ params: match.segmentParams,
178
169
  searchParams: Object.fromEntries(new URL(req.url).searchParams),
179
170
  statusCode: deny.status,
180
171
  responseHeaders,
181
- headHtml,
172
+ headHtml: headHtml + flightInitScript(),
182
173
  bootstrapScriptContent: clientBootstrap.bootstrapScriptContent,
183
174
  rscStream: inlineStream,
184
175
  };
@@ -206,7 +197,7 @@ export async function renderDenyPageAsRsc(
206
197
  return new Response(null, { status: deny.status, headers: responseHeaders });
207
198
  }
208
199
 
209
- const mod = (await resolution.file.load()) as Record<string, unknown>;
200
+ const mod = await loadModule(resolution.file);
210
201
  if (!mod.default) {
211
202
  responseHeaders.set('content-type', `${RSC_CONTENT_TYPE}; charset=utf-8`);
212
203
  return new Response(null, { status: deny.status, headers: responseHeaders });
@@ -215,22 +206,17 @@ export async function renderDenyPageAsRsc(
215
206
  const ErrorPageComponent = mod.default as (...args: unknown[]) => unknown;
216
207
  const h = createElement as (...args: unknown[]) => React.ReactElement;
217
208
 
218
- // Check shell opt-out
219
- const shellEnabled = mod.shell !== false;
220
-
221
209
  let element = h(ErrorPageComponent, {
222
210
  status: deny.status,
223
211
  dangerouslyPassData: deny.data,
224
212
  });
225
213
 
226
- // Wrap in layouts unless shell is explicitly disabled
227
- if (shellEnabled) {
228
- const resolvedSegments = new Set(segments.slice(0, resolution.segmentIndex + 1));
229
- const layoutsToWrap = layoutComponents.filter((lc) => resolvedSegments.has(lc.segment));
230
- for (let i = layoutsToWrap.length - 1; i >= 0; i--) {
231
- const { component } = layoutsToWrap[i];
232
- element = h(component, null, element);
233
- }
214
+ // Always wrap in layout shell.
215
+ const resolvedSegments = new Set(segments.slice(0, resolution.segmentIndex + 1));
216
+ const layoutsToWrap = layoutComponents.filter((lc) => resolvedSegments.has(lc.segment));
217
+ for (let i = layoutsToWrap.length - 1; i >= 0; i--) {
218
+ const { component } = layoutsToWrap[i];
219
+ element = h(component, null, element);
234
220
  }
235
221
 
236
222
  const rscStream = renderToReadableStream(element, {
@@ -272,7 +258,7 @@ async function renderDenyPageJson(
272
258
  // JSON status files are loaded as modules that export the JSON content.
273
259
  // The manifest's load() imports the .json file, which Vite handles as a
274
260
  // default export of the parsed JSON object.
275
- const mod = (await resolution.file.load()) as Record<string, unknown>;
261
+ const mod = await loadModule(resolution.file);
276
262
  const jsonContent = mod.default ?? mod;
277
263
 
278
264
  responseHeaders.set('content-type', 'application/json; charset=utf-8');