@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
@@ -0,0 +1,65 @@
1
+ /**
2
+ * Client chunk grouping strategy for @vitejs/plugin-rsc.
3
+ *
4
+ * Groups client reference modules by layout boundary to balance route-scoped
5
+ * code splitting with HTTP request count. A constant group name would collapse
6
+ * all routes into one chunk (every page downloads every client component).
7
+ * Per-serverChunk grouping creates many sub-500B files. Layout-boundary
8
+ * grouping is the middle ground.
9
+ *
10
+ * See design/27-chunking-strategy.md, TIM-440, TIM-499.
11
+ */
12
+
13
+ /**
14
+ * Derive a chunk group name for a client reference module.
15
+ *
16
+ * Groups by the first non-group route segment under appDir so that all
17
+ * client components belonging to the same layout boundary land in one
18
+ * chunk. For example:
19
+ * - `facade:app/dashboard/settings/page.tsx` → `"client-dashboard"`
20
+ * - `facade:app/(group-a)/group-page-a/page.tsx` → `"client-group-page-a"`
21
+ * - `facade:app/layout.tsx` (root layout) → `"client-shared"`
22
+ * - `shared:...` (shared across chunks) → `"client-shared"`
23
+ *
24
+ * This balances route-scoped code splitting with HTTP request count:
25
+ * fewer chunks than per-serverChunk, but still avoids downloading every
26
+ * client component on every page.
27
+ */
28
+ export function clientChunkGroup(
29
+ meta: { id: string; normalizedId: string; serverChunk: string },
30
+ appDir: string
31
+ ): string {
32
+ const { normalizedId, serverChunk } = meta;
33
+
34
+ // Shared chunks (not associated with a single route entry) get one group.
35
+ if (serverChunk.startsWith('shared:')) return 'client-shared';
36
+
37
+ // Derive the layout boundary from the file's location relative to the
38
+ // app directory. normalizedId is root-relative (e.g. "app/dashboard/shell.tsx"
39
+ // or "src/app/dashboard/shell.tsx"). We find the "app/" prefix and walk
40
+ // segments after it to find the first non-group directory.
41
+ const relPath = normalizedId.replace(/\\/g, '/');
42
+
43
+ // Find the app directory boundary in the path. The last segment of appDir
44
+ // is the app folder name (usually "app").
45
+ const appDirName = appDir.replace(/\\/g, '/').split('/').pop() || 'app';
46
+ const appIdx = relPath.indexOf(appDirName + '/');
47
+ if (appIdx === -1) return 'client-shared';
48
+
49
+ const withinApp = relPath.slice(appIdx + appDirName.length + 1);
50
+ const segments = withinApp.split('/');
51
+
52
+ // Find first directory segment that isn't a route group like (group-a)
53
+ for (const seg of segments) {
54
+ // Skip the filename itself
55
+ if (seg.includes('.')) break;
56
+ // Skip route groups — parenthesized segments like (group-a)
57
+ if (seg.startsWith('(') && seg.endsWith(')')) continue;
58
+ // Found a real route segment
59
+ return `client-${seg}`;
60
+ }
61
+
62
+ // Root-level files (layout.tsx, page.tsx directly in app/) go into the
63
+ // shared group since the root layout is loaded on every page.
64
+ return 'client-shared';
65
+ }
@@ -11,7 +11,7 @@
11
11
  import type { Plugin } from 'vite';
12
12
  import { existsSync } from 'node:fs';
13
13
  import { join } from 'node:path';
14
- import type { PluginContext } from '#/index.js';
14
+ import type { PluginContext } from '../index.js';
15
15
 
16
16
  const CONFIG_FILE_NAMES = [
17
17
  'content-collections.ts',
@@ -0,0 +1,288 @@
1
+ /**
2
+ * timber-dev-browser-logs — Forwards browser console output to the server terminal.
3
+ *
4
+ * Injects a small inline script in dev mode that intercepts browser
5
+ * `console.error`, `console.warn`, and `console.info`, then forwards
6
+ * messages to the server via Vite's HMR WebSocket.
7
+ *
8
+ * The server-side listener formats and prints the messages with color-coded
9
+ * prefixes: [browser:error], [browser:warn], [browser:info].
10
+ *
11
+ * Dev-only: this plugin only runs during `vite dev`.
12
+ * No runtime overhead in production.
13
+ *
14
+ * See TIM-513 for design context.
15
+ * See design/18-build-system.md §"Dev Server" for sub-plugin architecture.
16
+ */
17
+
18
+ import type { Plugin, ViteDevServer } from 'vite';
19
+ import type { PluginContext } from '../index.js';
20
+
21
+ // ─── Types ───────────────────────────────────────────────────────────────
22
+
23
+ /** Log levels forwarded from the browser. */
24
+ export type BrowserLogLevel = 'error' | 'warn' | 'info';
25
+
26
+ /** Configuration value for devBrowserLogs in timber.config.ts. */
27
+ export type DevBrowserLogsConfig = BrowserLogLevel | 'none' | undefined;
28
+
29
+ /**
30
+ * Payload sent from browser to server via Vite's HMR WebSocket.
31
+ * Kept small — truncated before sending.
32
+ */
33
+ export interface BrowserLogPayload {
34
+ level: BrowserLogLevel;
35
+ /** Serialized message string. */
36
+ message: string;
37
+ /** Error stack trace, if the first argument was an Error. */
38
+ stack: string | null;
39
+ /** Source URL and line number, if available. */
40
+ source: string | null;
41
+ /** Timestamp in ms (Date.now()) from the browser. */
42
+ timestamp: number;
43
+ }
44
+
45
+ // ─── Constants ───────────────────────────────────────────────────────────
46
+
47
+ /** Max message size in bytes before truncation. */
48
+ const MAX_MESSAGE_BYTES = 2048;
49
+
50
+ /** HMR event name for browser→server log forwarding. */
51
+ const HMR_EVENT = 'timber:browser-log';
52
+
53
+ /** ANSI color codes for terminal output. */
54
+ const COLORS = {
55
+ red: '\x1b[31m',
56
+ yellow: '\x1b[33m',
57
+ blue: '\x1b[34m',
58
+ dim: '\x1b[2m',
59
+ reset: '\x1b[0m',
60
+ } as const;
61
+
62
+ /** Level severity ordering for threshold comparison. */
63
+ const LEVEL_SEVERITY: Record<BrowserLogLevel | 'none', number> = {
64
+ none: 0,
65
+ info: 1,
66
+ warn: 2,
67
+ error: 3,
68
+ };
69
+
70
+ // ─── Formatting ──────────────────────────────────────────────────────────
71
+
72
+ /**
73
+ * Format a browser log payload for server terminal output.
74
+ *
75
+ * Produces color-coded output like:
76
+ * [browser:error] Uncaught TypeError: x is not a function
77
+ * at App (app.tsx:10:5)
78
+ * [browser:warn] Deprecation warning (app/page.tsx:42:12)
79
+ */
80
+ export function formatBrowserLog(payload: BrowserLogPayload): string {
81
+ const { level, message, stack, source } = payload;
82
+
83
+ // Color-coded prefix
84
+ const color = level === 'error' ? COLORS.red : level === 'warn' ? COLORS.yellow : COLORS.blue;
85
+ const prefix = `${color}[browser:${level}]${COLORS.reset}`;
86
+
87
+ // Source suffix
88
+ const sourceSuffix = source ? ` ${COLORS.dim}(${source})${COLORS.reset}` : '';
89
+
90
+ let output = `${prefix} ${message}${sourceSuffix}`;
91
+
92
+ // Append stack trace indented
93
+ if (stack) {
94
+ const indented = stack
95
+ .split('\n')
96
+ .map((line) => ` ${COLORS.dim}${line}${COLORS.reset}`)
97
+ .join('\n');
98
+ output += `\n${indented}`;
99
+ }
100
+
101
+ return output;
102
+ }
103
+
104
+ // ─── Level Filtering ─────────────────────────────────────────────────────
105
+
106
+ /**
107
+ * Check if a log at `level` should be forwarded given the configured threshold.
108
+ *
109
+ * The threshold acts as a minimum severity:
110
+ * - 'error' → only errors
111
+ * - 'warn' → errors + warnings
112
+ * - 'info' → errors + warnings + info
113
+ * - 'none' → nothing
114
+ */
115
+ export function shouldForwardLevel(
116
+ level: BrowserLogLevel,
117
+ threshold: BrowserLogLevel | 'none'
118
+ ): boolean {
119
+ if (threshold === 'none') return false;
120
+ return LEVEL_SEVERITY[level] >= LEVEL_SEVERITY[threshold];
121
+ }
122
+
123
+ // ─── Truncation ──────────────────────────────────────────────────────────
124
+
125
+ /**
126
+ * Truncate a message to `maxBytes` to avoid flooding the terminal.
127
+ * Appends a suffix indicating how many bytes were dropped.
128
+ */
129
+ export function truncateMessage(message: string, maxBytes: number): string {
130
+ if (message.length <= maxBytes) return message;
131
+
132
+ const truncated = message.slice(0, maxBytes);
133
+ const droppedBytes = message.length - maxBytes;
134
+ return `${truncated}… [truncated ${droppedBytes} bytes]`;
135
+ }
136
+
137
+ // ─── Client Injection Script ─────────────────────────────────────────────
138
+
139
+ /**
140
+ * Generate the inline script injected into the browser in dev mode.
141
+ *
142
+ * This script:
143
+ * 1. Saves references to the original console methods
144
+ * 2. Wraps `console.error`, `console.warn`, `console.info`
145
+ * 3. Calls the original first (browser devtools still work)
146
+ * 4. Serializes and forwards via `import.meta.hot.send()`
147
+ * 5. Truncates messages to MAX_MESSAGE_BYTES
148
+ *
149
+ * The script is minimal and self-contained — no imports, no dependencies.
150
+ */
151
+ export function generateClientScript(threshold: BrowserLogLevel | 'none'): string {
152
+ if (threshold === 'none') return '';
153
+
154
+ // Only intercept levels that meet the threshold
155
+ const levels: BrowserLogLevel[] = ['error', 'warn', 'info'].filter((l) =>
156
+ shouldForwardLevel(l as BrowserLogLevel, threshold)
157
+ ) as BrowserLogLevel[];
158
+
159
+ if (levels.length === 0) return '';
160
+
161
+ return `
162
+ (function() {
163
+ if (!import.meta.hot) return;
164
+ var MAX_BYTES = ${MAX_MESSAGE_BYTES};
165
+ var levels = ${JSON.stringify(levels)};
166
+ var originals = {};
167
+
168
+ function serialize(arg) {
169
+ if (arg === null) return 'null';
170
+ if (arg === undefined) return 'undefined';
171
+ if (arg instanceof Error) return arg.stack || arg.message || String(arg);
172
+ if (typeof arg === 'object') {
173
+ try { return JSON.stringify(arg); } catch(e) { return String(arg); }
174
+ }
175
+ return String(arg);
176
+ }
177
+
178
+ function truncate(s) {
179
+ if (s.length <= MAX_BYTES) return s;
180
+ return s.slice(0, MAX_BYTES) + '... [truncated ' + (s.length - MAX_BYTES) + ' bytes]';
181
+ }
182
+
183
+ levels.forEach(function(level) {
184
+ originals[level] = console[level];
185
+ console[level] = function() {
186
+ originals[level].apply(console, arguments);
187
+ try {
188
+ var args = Array.prototype.slice.call(arguments);
189
+ var firstArg = args[0];
190
+ var stack = null;
191
+ var source = null;
192
+
193
+ if (firstArg instanceof Error) {
194
+ stack = firstArg.stack || null;
195
+ source = null;
196
+ }
197
+
198
+ var message = args.map(serialize).join(' ');
199
+ message = truncate(message);
200
+
201
+ import.meta.hot.send('${HMR_EVENT}', {
202
+ level: level,
203
+ message: message,
204
+ stack: stack,
205
+ source: source,
206
+ timestamp: Date.now()
207
+ });
208
+ } catch(e) {
209
+ // Never let log forwarding break the page
210
+ }
211
+ };
212
+ });
213
+ })();
214
+ `.trim();
215
+ }
216
+
217
+ // ─── Plugin ──────────────────────────────────────────────────────────────
218
+
219
+ /**
220
+ * Create the timber-dev-browser-logs Vite plugin.
221
+ *
222
+ * - `configureServer`: Listens for HMR messages and prints them to the terminal
223
+ * - `transformIndexHtml`: Injects the client-side interception script
224
+ *
225
+ * Only active during `vite dev` (apply: 'serve').
226
+ */
227
+ export function timberDevBrowserLogs(ctx: PluginContext): Plugin {
228
+ return {
229
+ name: 'timber-dev-browser-logs',
230
+ apply: 'serve',
231
+
232
+ configureServer(server: ViteDevServer) {
233
+ const threshold = ctx.config.devBrowserLogs ?? 'warn';
234
+ if (threshold === 'none') {
235
+ // Clear any stale script from a prior server instance (e.g. HMR reconfigure)
236
+ delete (globalThis as Record<string, unknown>).__timber_dev_browser_log_script;
237
+ return;
238
+ }
239
+
240
+ // Register the client injection script on globalThis so the RSC entry
241
+ // can include it in headHtml for all Timber route responses.
242
+ // transformIndexHtml only runs for Vite's index.html fallback, not
243
+ // for responses served by createTimberMiddleware (TIM-575).
244
+ const script = generateClientScript(threshold);
245
+ if (script) {
246
+ (globalThis as Record<string, unknown>).__timber_dev_browser_log_script =
247
+ `<script type="module">${script}</script>`;
248
+ }
249
+
250
+ // Listen for browser log messages via HMR WebSocket
251
+ server.hot.on(HMR_EVENT, (payload: BrowserLogPayload) => {
252
+ try {
253
+ // Validate level
254
+ if (!shouldForwardLevel(payload.level, threshold)) return;
255
+
256
+ // Truncate server-side too (defense in depth)
257
+ payload.message = truncateMessage(payload.message, MAX_MESSAGE_BYTES);
258
+
259
+ const formatted = formatBrowserLog(payload);
260
+
261
+ // Use the correct console method for the log level
262
+ if (payload.level === 'error') {
263
+ process.stderr.write(formatted + '\n');
264
+ } else {
265
+ process.stdout.write(formatted + '\n');
266
+ }
267
+ } catch {
268
+ // Never let log forwarding crash the server
269
+ }
270
+ });
271
+ },
272
+
273
+ transformIndexHtml() {
274
+ const threshold = ctx.config.devBrowserLogs ?? 'warn';
275
+ const script = generateClientScript(threshold);
276
+ if (!script) return [];
277
+
278
+ return [
279
+ {
280
+ tag: 'script',
281
+ attrs: { type: 'module' },
282
+ children: script,
283
+ injectTo: 'head' as const,
284
+ },
285
+ ];
286
+ },
287
+ };
288
+ }
@@ -180,6 +180,63 @@ export function formatTerminalError(error: Error, phase: ErrorPhase, projectRoot
180
180
  return lines.join('\n');
181
181
  }
182
182
 
183
+ // ─── RSC Debug Context ──────────────────────────────────────────────────────
184
+
185
+ /**
186
+ * Component info extracted from the RSC debug channel.
187
+ * Contains only names, environments, and stack frames — never source code.
188
+ */
189
+ export interface RscDebugComponentInfo {
190
+ name: string;
191
+ env: string | null;
192
+ stack: unknown[] | null;
193
+ }
194
+
195
+ /**
196
+ * Format RSC debug component info into a readable string for the overlay.
197
+ *
198
+ * Renders the server component tree that was active when an error occurred,
199
+ * including component names and source locations from stack frames. This
200
+ * gives developers visibility into which server components were rendering
201
+ * without exposing source code.
202
+ *
203
+ * Returns an empty string if no components are provided.
204
+ */
205
+ export function formatRscDebugContext(components: RscDebugComponentInfo[]): string {
206
+ if (!components || components.length === 0) return '';
207
+
208
+ // Deduplicate by name — the debug channel may emit the same component
209
+ // multiple times (e.g., when re-rendered or when multiple instances exist).
210
+ const seen = new Set<string>();
211
+ const unique: RscDebugComponentInfo[] = [];
212
+ for (const c of components) {
213
+ if (!seen.has(c.name)) {
214
+ seen.add(c.name);
215
+ unique.push(c);
216
+ }
217
+ }
218
+
219
+ const lines: string[] = ['Server Component Tree:'];
220
+ for (let i = 0; i < unique.length; i++) {
221
+ const c = unique[i]!;
222
+ const indent = ' '.repeat(i + 1);
223
+ const envLabel = c.env ? ` [${c.env}]` : '';
224
+
225
+ // Extract file location from stack frames if available
226
+ let locStr = '';
227
+ if (c.stack && c.stack.length > 0) {
228
+ const frame = c.stack[0] as [string, string, number, number] | undefined;
229
+ if (Array.isArray(frame) && frame.length >= 3) {
230
+ locStr = ` (${frame[1]}:${frame[2]})`;
231
+ }
232
+ }
233
+
234
+ lines.push(`${indent}${c.name}${envLabel}${locStr}`);
235
+ }
236
+
237
+ return lines.join('\n');
238
+ }
239
+
183
240
  // ─── Overlay Integration ────────────────────────────────────────────────────
184
241
 
185
242
  /**
@@ -188,13 +245,19 @@ export function formatTerminalError(error: Error, phase: ErrorPhase, projectRoot
188
245
  * Uses `server.ssrFixStacktrace()` to map stack traces back to source,
189
246
  * then sends the error via `server.hot.send()` for the browser overlay.
190
247
  *
248
+ * When `rscDebugComponents` is provided (dev mode only), the server
249
+ * component tree context is appended to the error message. This helps
250
+ * developers identify which server component caused the error without
251
+ * exposing source code.
252
+ *
191
253
  * The dev server remains running — errors are handled, not fatal.
192
254
  */
193
255
  export function sendErrorToOverlay(
194
256
  server: ViteDevServer,
195
257
  error: Error,
196
258
  phase: ErrorPhase,
197
- projectRoot: string
259
+ projectRoot: string,
260
+ rscDebugComponents?: RscDebugComponentInfo[]
198
261
  ): void {
199
262
  // Fix stack trace to use source-mapped positions
200
263
  server.ssrFixStacktrace(error);
@@ -212,6 +275,12 @@ export function sendErrorToOverlay(
212
275
  message = `${error.message}\n\nComponent Stack:\n${componentStack.trim()}`;
213
276
  }
214
277
 
278
+ // Append RSC debug context if available (dev mode only)
279
+ const debugContext = formatRscDebugContext(rscDebugComponents ?? []);
280
+ if (debugContext) {
281
+ message = `${message}\n\n${debugContext}`;
282
+ }
283
+
215
284
  // Send to browser via Vite's error overlay protocol
216
285
  try {
217
286
  server.hot.send({
@@ -13,7 +13,7 @@
13
13
  */
14
14
 
15
15
  import type { Plugin, ViteDevServer } from 'vite';
16
- import type { PluginContext } from '#/index.js';
16
+ import type { PluginContext } from '../index.js';
17
17
 
18
18
  // ─── Types ───────────────────────────────────────────────────────────────
19
19
 
@@ -15,10 +15,10 @@
15
15
  import type { Plugin, ViteDevServer, DevEnvironment } from 'vite';
16
16
  import type { IncomingMessage, ServerResponse } from 'node:http';
17
17
  import { join } from 'node:path';
18
- import type { PluginContext } from '#/index.js';
19
- import { setViteServer } from '#/server/dev-warnings.js';
18
+ import type { PluginContext } from '../index.js';
19
+ import { setViteServer } from '../server/dev-warnings.js';
20
20
  import { sendErrorToOverlay, classifyErrorPhase, parseFirstAppFrame } from './dev-error-overlay.js';
21
- import { compressResponse } from '#/server/compress.js';
21
+ import { compressResponse } from '../server/compress.js';
22
22
 
23
23
  // ─── Constants ────────────────────────────────────────────────────────────
24
24
 
@@ -161,12 +161,26 @@ function createTimberMiddleware(server: ViteDevServer, projectRoot: string) {
161
161
 
162
162
  // Wire pipeline errors into the browser error overlay.
163
163
  // setDevPipelineErrorHandler is only defined in dev (rsc-entry.ts exports it).
164
+ // The handler receives optional RSC debug component data (component names,
165
+ // environments, stack frames) from the Flight debug channel for render errors.
164
166
  const setHandler = rscModule.setDevPipelineErrorHandler as
165
- | ((fn: (error: Error, phase: string) => void) => void)
167
+ | ((
168
+ fn: (
169
+ error: Error,
170
+ phase: string,
171
+ debugComponents?: Array<{ name: string; env: string | null; stack: unknown[] | null }>
172
+ ) => void
173
+ ) => void)
166
174
  | undefined;
167
175
  if (typeof setHandler === 'function') {
168
- setHandler((error) => {
169
- sendErrorToOverlay(server, error, classifyErrorPhase(error, projectRoot), projectRoot);
176
+ setHandler((error, _phase, debugComponents) => {
177
+ sendErrorToOverlay(
178
+ server,
179
+ error,
180
+ classifyErrorPhase(error, projectRoot),
181
+ projectRoot,
182
+ debugComponents
183
+ );
170
184
  });
171
185
  }
172
186
  } catch (error) {
@@ -191,8 +205,20 @@ function createTimberMiddleware(server: ViteDevServer, projectRoot: string) {
191
205
  // Convert Node IncomingMessage → Web Request
192
206
  const webRequest = toWebRequest(req);
193
207
 
194
- // Run the full pipeline
195
- const webResponse = await handler(webRequest);
208
+ // Check for a platform-specific request wrapper (e.g., Cloudflare
209
+ // dev bindings). The cloudflareDevBindings() plugin stores a wrapper
210
+ // function on the server instance using a well-known Symbol. The
211
+ // wrapper runs the handler inside an ALS context so platform APIs
212
+ // like getCloudflareBindings() work during dev.
213
+ // See design/35-cloudflare-primitives.md §"Dev Experience".
214
+ const wrapper = (server as any)[Symbol.for('timber:dev-request-wrapper')] as
215
+ | (<T>(fn: () => T) => T)
216
+ | undefined;
217
+
218
+ // Run the full pipeline (wrapped if a platform dev plugin is active)
219
+ const webResponse = wrapper
220
+ ? await wrapper(() => handler(webRequest))
221
+ : await handler(webRequest);
196
222
 
197
223
  // Compress the response if the client supports it.
198
224
  // In dev mode, compression is always enabled for parity with production.
@@ -317,6 +343,17 @@ async function sendWebResponse(nodeRes: ServerResponse, webResponse: Response):
317
343
  nodeRes.flushHeaders();
318
344
 
319
345
  const reader = webResponse.body.getReader();
346
+
347
+ // Cancel the reader when the client disconnects. This causes any pending
348
+ // reader.read() to reject, breaking the pump loop. Critical for SSE and
349
+ // other infinite streams — without this, disconnected clients leak readers.
350
+ let clientDisconnected = false;
351
+ const onClose = () => {
352
+ clientDisconnected = true;
353
+ reader.cancel('Client disconnected').catch(() => {});
354
+ };
355
+ nodeRes.on('close', onClose);
356
+
320
357
  try {
321
358
  while (true) {
322
359
  const { done, value } = await reader.read();
@@ -325,9 +362,18 @@ async function sendWebResponse(nodeRes: ServerResponse, webResponse: Response):
325
362
  // don't need back-pressure here — just keep pushing chunks.
326
363
  nodeRes.write(value);
327
364
  }
365
+ } catch (err) {
366
+ // reader.cancel() from the close handler causes read() to reject.
367
+ // This is expected on client disconnect — not an error.
368
+ if (!clientDisconnected) {
369
+ throw err;
370
+ }
328
371
  } finally {
372
+ nodeRes.off('close', onClose);
329
373
  reader.releaseLock();
330
- nodeRes.end();
374
+ if (!nodeRes.writableEnded) {
375
+ nodeRes.end();
376
+ }
331
377
  }
332
378
  }
333
379