@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
@@ -12,25 +12,49 @@
12
12
  * 05-streaming.md §"deferSuspenseFor and the Hold Window"
13
13
  */
14
14
 
15
- import type { ClientBootstrapConfig } from '#/server/html-injectors.js';
16
- import type { LayoutEntry } from '#/server/deny-renderer.js';
17
- import { renderDenyPage } from '#/server/deny-renderer.js';
18
- import type { RouteMatch } from '#/server/pipeline.js';
19
- import { SsrStreamError } from '#/server/primitives.js';
20
- import type { LayoutComponentEntry } from '#/server/route-element-builder.js';
21
- import type { ManifestSegmentNode } from '#/server/route-matcher.js';
22
- import type { NavContext } from '#/server/ssr-entry.js';
15
+ import type { ClientBootstrapConfig } from '../html-injectors.js';
16
+ import { flightInitScript } from '../flight-scripts.js';
17
+ import type { LayoutEntry } from '../deny-renderer.js';
18
+ import { renderDenyPage } from '../deny-renderer.js';
19
+ import type { RouteMatch } from '../pipeline.js';
20
+ import { SsrStreamError } from '../primitives.js';
21
+ import type { LayoutComponentEntry } from '../route-element-builder.js';
22
+ import type { ManifestSegmentNode } from '../route-matcher.js';
23
+ import type { NavContext } from '../ssr-entry.js';
23
24
 
24
25
  import {
25
26
  buildRedirectResponse,
26
27
  buildSegmentInfo,
27
28
  createDebugChannelSink,
28
29
  isAbortError,
29
- parseCookiesFromHeader,
30
30
  } from './helpers.js';
31
- import { renderErrorPage } from './error-renderer.js';
31
+ import { getCookiesForSsr } from '../request-context.js';
32
+ import { getDenyStatus } from '../deny-page-resolver.js';
33
+ import { renderErrorPage, type GlobalErrorFile } from './error-renderer.js';
32
34
  import { callSsr } from './ssr-bridge.js';
33
35
  import type { RenderSignals } from './rsc-stream.js';
36
+ import { recordTiming } from '../server-timing.js';
37
+ import { teeWithErrorPropagation } from '../stream-utils.js';
38
+
39
+ /**
40
+ * Maximum number of queueMicrotask yields during signal promotion.
41
+ * The known rejection chain depth for a withSpan-wrapped component is 3+
42
+ * microtasks. We use 5 to tolerate additional async wrappers or React
43
+ * Flight internals adding extra microtask hops. Do not increase beyond 8 —
44
+ * a deeper chain indicates a structural problem.
45
+ */
46
+ const SIGNAL_PROMOTION_MAX_YIELDS = 5;
47
+
48
+ /**
49
+ * Test-only observable: records how many microtask yields were consumed
50
+ * during the last signal promotion check. Import this in tests to assert
51
+ * the actual rejection chain depth and detect regressions if wrappers
52
+ * add more async hops.
53
+ *
54
+ * Only populated when the object is imported and exists — production
55
+ * bundles that don't import it pay zero cost (the typeof check is free).
56
+ */
57
+ export const __test_only_signalPromotionDepth: { value: number } = { value: 0 };
34
58
 
35
59
  interface SsrRenderOptions {
36
60
  req: Request;
@@ -44,6 +68,8 @@ interface SsrRenderOptions {
44
68
  clientJsDisabled: boolean;
45
69
  headHtml: string;
46
70
  deferSuspenseFor: number;
71
+ /** Tier 2 global-error.tsx file, if present in app/. */
72
+ globalError?: GlobalErrorFile;
47
73
  }
48
74
 
49
75
  /**
@@ -77,11 +103,26 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
77
103
  clientJsDisabled,
78
104
  headHtml,
79
105
  deferSuspenseFor,
106
+ globalError,
80
107
  } = opts;
81
108
 
82
109
  // Tee the RSC stream — one copy goes to SSR for HTML rendering,
83
110
  // the other is inlined in the HTML for client-side hydration.
84
- const [ssrStream, inlineStream] = rscStream.tee();
111
+ // Uses teeWithErrorPropagation to ensure errors on either branch
112
+ // cancel the other (preventing hung responses) and to monitor
113
+ // buffer divergence between SSR and inline consumers.
114
+ //
115
+ // When client JS is disabled, skip the tee — the inline stream would
116
+ // never be consumed, causing monotonic buffer divergence that eventually
117
+ // triggers the 5MB guard and cancels the SSR stream.
118
+ let ssrStream: ReadableStream<Uint8Array>;
119
+ let inlineStream: ReadableStream<Uint8Array> | undefined;
120
+ if (clientJsDisabled) {
121
+ ssrStream = rscStream;
122
+ inlineStream = undefined;
123
+ } else {
124
+ [ssrStream, inlineStream] = teeWithErrorPropagation(rscStream);
125
+ }
85
126
 
86
127
  // Embed segment metadata in HTML for initial hydration.
87
128
  // The client reads this to populate its segment cache before the
@@ -91,26 +132,33 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
91
132
  ? ''
92
133
  : `<script>self.__timber_segments=${JSON.stringify(buildSegmentInfo(segments, layoutComponents))}</script>`;
93
134
 
94
- // Embed route params in HTML so useParams() works on initial hydration.
95
- // Without this, useParams() returns {} until the first client navigation.
135
+ // Embed route params in HTML so useSegmentParams() works on initial hydration.
136
+ // Without this, useSegmentParams() returns {} until the first client navigation.
96
137
  const paramsScript =
97
- clientJsDisabled || Object.keys(match.params).length === 0
138
+ clientJsDisabled || Object.keys(match.segmentParams).length === 0
98
139
  ? ''
99
- : `<script>self.__timber_params=${JSON.stringify(match.params)}</script>`;
140
+ : `<script>self.__timber_params=${JSON.stringify(match.segmentParams)}</script>`;
100
141
 
101
142
  const navContext: NavContext = {
102
143
  pathname: new URL(req.url).pathname,
103
- params: match.params,
144
+ params: match.segmentParams,
104
145
  searchParams: Object.fromEntries(new URL(req.url).searchParams),
105
146
  statusCode: 200,
106
147
  responseHeaders,
107
- headHtml: headHtml + clientBootstrap.preloadLinks + segmentScript + paramsScript,
148
+ headHtml:
149
+ headHtml +
150
+ clientBootstrap.preloadLinks +
151
+ segmentScript +
152
+ paramsScript +
153
+ // Initialize __timber_f in <head> so it exists before any streaming
154
+ // chunk scripts arrive in <body>. See flight-scripts.ts, LOCAL-415.
155
+ (clientJsDisabled ? '' : flightInitScript()),
108
156
  bootstrapScriptContent: clientBootstrap.bootstrapScriptContent,
109
157
  // Skip RSC inline stream when client JS is disabled — no client to hydrate.
110
158
  rscStream: clientJsDisabled ? undefined : inlineStream,
111
159
  deferSuspenseFor: deferSuspenseFor > 0 ? deferSuspenseFor : undefined,
112
160
  signal: req.signal,
113
- cookies: parseCookiesFromHeader(req.headers.get('cookie') ?? ''),
161
+ cookies: getCookiesForSsr(),
114
162
  };
115
163
 
116
164
  // Helper: check if render-phase signals were captured and return the
@@ -118,14 +166,19 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
118
166
  // promotion from Suspense) and failed SSR (signal outside Suspense).
119
167
  //
120
168
  // When `skipHandledDeny` is true (SSR success path), skip DenySignal
121
- // promotion if the denial was already handled by a TimberErrorBoundary
122
- // (e.g., slot error boundary). The boundary sets navContext._denyHandledByBoundary
123
- // during SSR rendering. See LOCAL-298.
169
+ // re-render if the denial was already handled by a TimberErrorBoundary.
170
+ // The boundary renders the deny page as its fallback AND sets the correct
171
+ // HTTP status code on navContext — no redundant renderDenyPage() needed.
172
+ // See TIM-664, LOCAL-298.
124
173
  function checkCapturedSignals(skipHandledDeny = false): Response | Promise<Response> | null {
125
174
  if (signals.redirectSignal) {
126
175
  return buildRedirectResponse(req, signals.redirectSignal, responseHeaders);
127
176
  }
128
177
  if (signals.denySignal && !(skipHandledDeny && navContext._denyHandledByBoundary)) {
178
+ // Deny was not handled by an in-tree error boundary.
179
+ // Fall back to renderDenyPage (separate React.cache scope).
180
+ // For single-render behavior, add a 'use client' status-code file
181
+ // (e.g., 4xx.tsx) — the error boundary handles it in-tree.
129
182
  return renderDenyPage(
130
183
  signals.denySignal,
131
184
  segments,
@@ -147,7 +200,8 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
147
200
  req,
148
201
  match,
149
202
  responseHeaders,
150
- clientBootstrap
203
+ clientBootstrap,
204
+ globalError
151
205
  );
152
206
  }
153
207
  return null;
@@ -156,22 +210,71 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
156
210
  try {
157
211
  const ssrResponse = await callSsr(ssrStream, navContext);
158
212
 
159
- // Signal promotion: yield one microtask so async component rejections
160
- // propagate to the RSC onError callback, then check if any signals
161
- // were captured during rendering inside Suspense boundaries.
162
- // The Response hasn't been sent yet — it's an unconsumed stream.
213
+ // Record SSR sub-phase timings for Server-Timing header (detailed mode).
214
+ // These are populated by handleSsr() in the SSR environment and passed
215
+ // back via navContext._ssrTimings across the RSC→SSR boundary.
216
+ if (navContext._ssrTimings) {
217
+ const t = navContext._ssrTimings;
218
+ recordTiming({ name: 'ssr-decode', dur: t.decodeMs, desc: 'RSC Flight decode' });
219
+ recordTiming({ name: 'ssr-shell', dur: t.shellMs, desc: 'Fizz onShellReady' });
220
+ recordTiming({ name: 'ssr-pipeline', dur: t.pipelineMs, desc: 'stream transforms' });
221
+ recordTiming({
222
+ name: 'ssr-total',
223
+ dur: t.totalMs,
224
+ desc: t.nodeStreams ? 'SSR (Node streams)' : 'SSR (Web Streams)',
225
+ });
226
+ }
227
+
228
+ // Signal promotion: check if any signals were captured during rendering
229
+ // inside Suspense boundaries. If no signals are present yet, yield
230
+ // multiple microtasks so async component rejections propagate through
231
+ // the full rejection chain to the RSC onError callback before we
232
+ // commit the response.
163
233
  //
164
- // Uses queueMicrotask instead of setTimeout(0) to avoid yielding to
165
- // the full event loop (timers phase). Microtask resolution happens
166
- // within the same tick, eliminating per-request idle time under load.
234
+ // The known rejection chain depth for a withSpan-wrapped page:
235
+ // 1. PageComponent throws DenySignal/RedirectSignal
236
+ // 2. withSpan catches and re-throws (microtask 1)
237
+ // 3. TracedPage promise rejects (microtask 2)
238
+ // 4. React Flight rejection handler → onError (microtask 3+)
239
+ //
240
+ // A single queueMicrotask only advances the queue by one step, which
241
+ // is insufficient for chains deeper than 1. We retry up to
242
+ // SIGNAL_PROMOTION_MAX_YIELDS (5) times, exiting early as soon as a
243
+ // signal is detected. This tolerates future wrappers adding depth
244
+ // without requiring code changes.
245
+ //
246
+ // When signals are already captured (onSignal already fired), skip the
247
+ // yield entirely — react immediately. Uses queueMicrotask instead of
248
+ // setTimeout(0) to avoid yielding to the full event loop (timers phase).
167
249
  // See design/05-streaming.md §"deferSuspenseFor and the Hold Window"
168
- await new Promise<void>((r) => queueMicrotask(r));
250
+ if (!signals.redirectSignal && !signals.denySignal && !signals.renderError) {
251
+ let yieldsConsumed = 0;
252
+ for (let i = 0; i < SIGNAL_PROMOTION_MAX_YIELDS; i++) {
253
+ await new Promise<void>((r) => queueMicrotask(r));
254
+ yieldsConsumed++;
255
+ if (signals.redirectSignal || signals.denySignal || signals.renderError) break;
256
+ }
257
+ __test_only_signalPromotionDepth.value = yieldsConsumed;
258
+ }
169
259
 
170
260
  const promoted = checkCapturedSignals(/* skipHandledDeny */ true);
171
261
  if (promoted) {
172
262
  ssrResponse.body?.cancel();
173
263
  return promoted;
174
264
  }
265
+
266
+ // Check if a DenySignal was caught in-tree by AccessGate or
267
+ // PageDenyBoundary. The deny page was rendered as a normal element
268
+ // (no error in the Flight stream), so SSR succeeded with status 200.
269
+ // Override the status code from the ALS-stored deny status. See TIM-666.
270
+ const denyStatus = getDenyStatus();
271
+ if (denyStatus) {
272
+ return new Response(ssrResponse.body, {
273
+ status: denyStatus,
274
+ headers: ssrResponse.headers,
275
+ });
276
+ }
277
+
175
278
  return ssrResponse;
176
279
  } catch (ssrError) {
177
280
  // Connection abort — the client disconnected (page refresh, navigation
@@ -182,19 +285,34 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
182
285
 
183
286
  // SsrStreamError: SSR's renderToReadableStream failed because the RSC
184
287
  // stream contained an uncontained error (e.g., slot without error boundary).
185
- // Render the deny/error page WITHOUT layout wrapping to avoid re-executing
186
- // server components (which call headers()/cookies() and fail in SSR's
187
- // separate ALS scope). See LOCAL-293.
188
- if (ssrError instanceof SsrStreamError) {
288
+ //
289
+ // Error pages now use SSR-only rendering (Fizz directly, no RSC Flight),
290
+ // which imports layouts in the SSR environment where ALS is active. This
291
+ // means we CAN include layouts — they'll call headers()/cookies() correctly.
292
+ //
293
+ // Deny pages still go through RSC → SSR (plain props), so they pass empty
294
+ // layouts to avoid re-executing server components in the RSC environment's
295
+ // stale ALS scope. See LOCAL-293.
296
+ // Name-based check: SsrStreamError is thrown in the SSR environment
297
+ // but caught in the RSC environment. Since RSC and SSR are separate
298
+ // Vite module graphs, `instanceof` fails (different class identities).
299
+ // Use error.name instead. See design/18-build-system.md §"Virtual Module
300
+ // Resolution Quirks".
301
+ const isSsrStreamError =
302
+ ssrError instanceof SsrStreamError ||
303
+ (ssrError instanceof Error && ssrError.name === 'SsrStreamError');
304
+ if (isSsrStreamError) {
189
305
  if (signals.redirectSignal) {
190
306
  return buildRedirectResponse(req, signals.redirectSignal, responseHeaders);
191
307
  }
192
308
  if (signals.denySignal) {
193
- // Render deny page without layouts pass empty layout list
309
+ // SSR failed (SsrStreamError) the deny was in the shell and
310
+ // the error boundary didn't catch it during Fizz rendering.
311
+ // Fall back to renderDenyPage (separate render). See TIM-664.
194
312
  return renderDenyPage(
195
313
  signals.denySignal,
196
314
  segments,
197
- [] as LayoutEntry[],
315
+ layoutComponents as LayoutEntry[],
198
316
  req,
199
317
  match,
200
318
  responseHeaders,
@@ -204,19 +322,34 @@ export async function renderSsrResponse(opts: SsrRenderOptions): Promise<Respons
204
322
  );
205
323
  }
206
324
  if (signals.renderError) {
325
+ // RenderError: TSX error pages use SSR-only path (layouts OK),
326
+ // MDX error pages use RSC path (plain props, no Error object).
207
327
  return renderErrorPage(
208
328
  signals.renderError.error,
209
329
  signals.renderError.status,
210
330
  segments,
211
- [] as LayoutEntry[],
331
+ layoutComponents as LayoutEntry[],
212
332
  req,
213
333
  match,
214
334
  responseHeaders,
215
- clientBootstrap
335
+ clientBootstrap,
336
+ globalError
216
337
  );
217
338
  }
218
- // No captured signal — return bare 500
219
- return new Response(null, { status: 500, headers: responseHeaders });
339
+ // No captured signal — unhandled error in the RSC stream.
340
+ // Render the error page using the original error (cause).
341
+ const cause = (ssrError as { cause?: unknown }).cause ?? ssrError;
342
+ return renderErrorPage(
343
+ cause,
344
+ 500,
345
+ segments,
346
+ layoutComponents as LayoutEntry[],
347
+ req,
348
+ match,
349
+ responseHeaders,
350
+ clientBootstrap,
351
+ globalError
352
+ );
220
353
  }
221
354
 
222
355
  // SSR shell rendering failed — the error was outside Suspense.
@@ -0,0 +1,60 @@
1
+ /**
2
+ * loadModule — enriched error context for route manifest .load() failures.
3
+ *
4
+ * Wraps the lazy `load()` functions from the route manifest with a
5
+ * try/catch that re-throws with the file path and original cause.
6
+ *
7
+ * Callers that need fallthrough behavior (error renderers) can use
8
+ * `.catch(() => null)` or try/catch — the decision stays at the call site.
9
+ *
10
+ * See design/spike-TIM-551-dynamic-import-audit.md §"Proposed Wrapping Strategy"
11
+ */
12
+
13
+ /** A manifest file reference with a lazy import function and file path. */
14
+ export interface ManifestLoader {
15
+ load: () => Promise<unknown>;
16
+ filePath: string;
17
+ }
18
+
19
+ /**
20
+ * Custom error class for module load failures.
21
+ *
22
+ * Preserves the original error as `cause` while providing a
23
+ * human-readable message with the file path.
24
+ */
25
+ export class ModuleLoadError extends Error {
26
+ /** The file path that failed to load. */
27
+ readonly filePath: string;
28
+
29
+ constructor(filePath: string, cause: unknown) {
30
+ const originalMessage = cause instanceof Error ? cause.message : String(cause);
31
+ super(`[timber] Failed to load module ${filePath}\n ${originalMessage}`, { cause });
32
+ this.name = 'ModuleLoadError';
33
+ this.filePath = filePath;
34
+ }
35
+ }
36
+
37
+ /**
38
+ * Load a route manifest module with enriched error context.
39
+ *
40
+ * On success: returns the module object (same as `loader.load()`).
41
+ * On failure: throws `ModuleLoadError` with file path and original cause.
42
+ *
43
+ * For error rendering paths that need fallthrough instead of throwing,
44
+ * callers should catch at the call site:
45
+ *
46
+ * ```ts
47
+ * // Throwing (default) — route-element-builder, api-handler, etc.
48
+ * const mod = await loadModule(segment.page);
49
+ *
50
+ * // Fallthrough — error-renderer, error-boundary-wrapper
51
+ * const mod = await loadModule(segment.error).catch(() => null);
52
+ * ```
53
+ */
54
+ export async function loadModule<T = Record<string, unknown>>(loader: ManifestLoader): Promise<T> {
55
+ try {
56
+ return (await loader.load()) as T;
57
+ } catch (error) {
58
+ throw new ModuleLoadError(loader.filePath, error);
59
+ }
60
+ }