@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,177 @@
1
+ /**
2
+ * Cloudflare dev-mode bindings via wrangler's getPlatformProxy().
3
+ *
4
+ * This Vite plugin starts a local workerd process (via Miniflare) that
5
+ * emulates Cloudflare bindings (KV, D1, R2, Queues, Durable Objects, etc.)
6
+ * during development. The proxy env is injected into the request ALS so
7
+ * `getCloudflareBindings()` works identically in dev and production.
8
+ *
9
+ * Design doc: design/35-cloudflare-primitives.md §"Dev Experience"
10
+ *
11
+ * Usage:
12
+ * ```ts
13
+ * // vite.config.ts
14
+ * import { timber } from '@timber-js/app'
15
+ * import { cloudflare } from '@timber-js/app/adapters/cloudflare'
16
+ * import { cloudflareDevBindings } from '@timber-js/app/adapters/cloudflare/dev'
17
+ *
18
+ * export default defineConfig({
19
+ * plugins: [
20
+ * timber({ adapter: cloudflare() }),
21
+ * cloudflareDevBindings(),
22
+ * ],
23
+ * })
24
+ * ```
25
+ *
26
+ * Requires `wrangler` as a dev dependency. Install it:
27
+ * ```bash
28
+ * pnpm add -D wrangler
29
+ * ```
30
+ */
31
+
32
+ import type { Plugin, ViteDevServer } from 'vite';
33
+ import { runWithBindings } from './cloudflare.js';
34
+
35
+ /**
36
+ * Dynamically import wrangler. Extracted as a named export so tests
37
+ * can mock it without needing wrangler installed.
38
+ * @internal
39
+ */
40
+ export async function loadWrangler(): Promise<{ getPlatformProxy: (opts: any) => Promise<any> }> {
41
+ try {
42
+ return await import('wrangler');
43
+ } catch {
44
+ throw new Error(
45
+ '[timber] Could not import wrangler. ' +
46
+ 'cloudflareDevBindings() requires wrangler as a dev dependency.\n' +
47
+ 'Install it: pnpm add -D wrangler'
48
+ );
49
+ }
50
+ }
51
+
52
+ // ─── Contract ─────────────────────────────────────────────────────────────
53
+ //
54
+ // The dev request wrapper is stored on the ViteDevServer instance using a
55
+ // well-known Symbol. This allows cross-plugin communication without import
56
+ // dependencies — the dev-server plugin checks for this Symbol per-request
57
+ // and wraps the handler call when present.
58
+ //
59
+ // The wrapper type is: <T>(fn: () => T) => T
60
+ // It runs `fn` inside the Cloudflare bindings ALS so getCloudflareBindings()
61
+ // works throughout the request lifecycle.
62
+
63
+ /**
64
+ * Symbol key used to store the dev request wrapper on ViteDevServer.
65
+ *
66
+ * @internal Exported for testing. The dev-server plugin uses
67
+ * `Symbol.for('timber:dev-request-wrapper')` directly to avoid
68
+ * importing from this module.
69
+ */
70
+ export const DEV_REQUEST_WRAPPER_KEY = Symbol.for('timber:dev-request-wrapper');
71
+
72
+ /** Options for the Cloudflare dev bindings plugin. */
73
+ export interface CloudflareDevBindingsOptions {
74
+ /**
75
+ * Path to the wrangler configuration file.
76
+ * @default Auto-detected by wrangler (wrangler.jsonc, wrangler.json, wrangler.toml)
77
+ */
78
+ configPath?: string;
79
+
80
+ /**
81
+ * Whether to persist binding data (KV, D1, R2) between dev server restarts.
82
+ *
83
+ * - `true` — persist in wrangler's default location (`.wrangler/state/`)
84
+ * - `{ path: string }` — persist in a custom directory
85
+ * - `false` — start fresh on every restart
86
+ *
87
+ * @default true
88
+ */
89
+ persist?: boolean | { path: string };
90
+
91
+ /**
92
+ * Wrangler environment to use (for multi-environment configs).
93
+ * Maps to `[env.<name>]` sections in wrangler.toml / wrangler.jsonc.
94
+ */
95
+ environment?: string;
96
+
97
+ /**
98
+ * @internal Override the wrangler loader for testing.
99
+ * Not part of the public API — may change without notice.
100
+ */
101
+ _loadWrangler?: () => Promise<{ getPlatformProxy: (opts: any) => Promise<any> }>;
102
+ }
103
+
104
+ /**
105
+ * Vite plugin that provides Cloudflare bindings in dev mode via
106
+ * wrangler's `getPlatformProxy()`.
107
+ *
108
+ * Starts a local workerd process on dev server startup and injects
109
+ * the proxy env into the per-request ALS. This makes
110
+ * `getCloudflareBindings()` return real (emulated) bindings during
111
+ * development — KV, D1, R2, Queues, Durable Objects, and all other
112
+ * bindings declared in wrangler.jsonc.
113
+ *
114
+ * The proxy is automatically disposed when the dev server closes.
115
+ *
116
+ * **Requirements:**
117
+ * - `wrangler` installed as a dev dependency
118
+ * - A `wrangler.jsonc` (or `wrangler.toml`) with binding declarations
119
+ *
120
+ * @example
121
+ * ```ts
122
+ * import { cloudflareDevBindings } from '@timber-js/app/adapters/cloudflare/dev'
123
+ *
124
+ * export default defineConfig({
125
+ * plugins: [
126
+ * timber({ adapter: cloudflare() }),
127
+ * cloudflareDevBindings(),
128
+ * ],
129
+ * })
130
+ * ```
131
+ */
132
+ export function cloudflareDevBindings(options: CloudflareDevBindingsOptions = {}): Plugin {
133
+ return {
134
+ name: 'timber-cloudflare-dev-bindings',
135
+
136
+ // Dev mode only — never active during production builds.
137
+ // See design/35-cloudflare-primitives.md §"Security Considerations":
138
+ // "Dev bindings are dev-only. apply: 'serve' ensures it never runs
139
+ // in production builds."
140
+ apply: 'serve',
141
+
142
+ async configureServer(server: ViteDevServer) {
143
+ // Dynamic import — wrangler is an optional peer dependency.
144
+ // Users must install it themselves: `pnpm add -D wrangler`
145
+ const { getPlatformProxy } = await (options._loadWrangler ?? loadWrangler)();
146
+
147
+ // Build getPlatformProxy options from our public API.
148
+ const proxyOptions: Record<string, unknown> = {
149
+ persist: options.persist ?? true,
150
+ };
151
+ if (options.configPath) {
152
+ proxyOptions.configPath = options.configPath;
153
+ }
154
+ if (options.environment) {
155
+ proxyOptions.environment = options.environment;
156
+ }
157
+
158
+ const proxy = await getPlatformProxy(proxyOptions);
159
+
160
+ // Store a request wrapper on the server instance. The dev-server
161
+ // plugin checks for this Symbol per-request and wraps the RSC
162
+ // handler call so getCloudflareBindings() works throughout the
163
+ // request lifecycle (server components, middleware, server actions).
164
+ (server as any)[DEV_REQUEST_WRAPPER_KEY] = <T>(fn: () => T): T => {
165
+ return runWithBindings(proxy.env as Record<string, unknown>, fn);
166
+ };
167
+
168
+ // Dispose the proxy when the dev server closes. This shuts down
169
+ // the local workerd process started by getPlatformProxy().
170
+ server.httpServer?.on('close', () => {
171
+ proxy.dispose().catch(() => {
172
+ // Disposal errors are non-fatal — the process is shutting down.
173
+ });
174
+ });
175
+ },
176
+ };
177
+ }
@@ -31,8 +31,27 @@ function generateHeadersFile(): string {
31
31
  // can access KV, D1, DO, R2, Queues, etc. via getCloudflareBindings().
32
32
  // No global fallback — if called outside a request, it throws.
33
33
  // See design/11-platform.md §"Platform Target" and design/25-production-deployments.md.
34
-
35
- const bindingsAls = new AsyncLocalStorage<Record<string, unknown>>();
34
+ //
35
+ // The ALS is stored on globalThis via Symbol.for so it survives Vite's
36
+ // module instance split in dev mode. The Vite host process (where
37
+ // cloudflareDevBindings → runWithBindings runs) and the RSC module runner
38
+ // (where getCloudflareBindings runs) load separate instances of this file,
39
+ // each with their own module-level variables. globalThis + Symbol.for
40
+ // ensures both share the same ALS — same pattern React uses for
41
+ // Symbol.for('react.element').
42
+
43
+ const BINDINGS_ALS_KEY = Symbol.for('timber:cf-bindings-als');
44
+
45
+ function getBindingsAls(): AsyncLocalStorage<Record<string, unknown>> {
46
+ let als = (globalThis as any)[BINDINGS_ALS_KEY] as
47
+ | AsyncLocalStorage<Record<string, unknown>>
48
+ | undefined;
49
+ if (!als) {
50
+ als = new AsyncLocalStorage<Record<string, unknown>>();
51
+ (globalThis as any)[BINDINGS_ALS_KEY] = als;
52
+ }
53
+ return als;
54
+ }
36
55
 
37
56
  /**
38
57
  * Get Cloudflare Worker bindings for the current request.
@@ -55,10 +74,8 @@ const bindingsAls = new AsyncLocalStorage<Record<string, unknown>>();
55
74
  * }
56
75
  * ```
57
76
  */
58
- export function getCloudflareBindings<
59
- T extends Record<string, unknown> = Record<string, unknown>,
60
- >(): T {
61
- const env = bindingsAls.getStore();
77
+ export function getCloudflareBindings<T = Record<string, unknown>>(): T {
78
+ const env = getBindingsAls().getStore();
62
79
  if (!env) {
63
80
  throw new Error(
64
81
  'getCloudflareBindings() called outside a Cloudflare Workers request context. ' +
@@ -74,7 +91,85 @@ export function getCloudflareBindings<
74
91
  * @internal Used by wrapWithExecutionContext.
75
92
  */
76
93
  export function runWithBindings<T>(env: Record<string, unknown>, fn: () => T): T {
77
- return bindingsAls.run(env, fn);
94
+ return getBindingsAls().run(env, fn);
95
+ }
96
+
97
+ // ─── Binding configuration types ──────────────────────────────────────────────
98
+ // Declarative binding config that maps to wrangler.jsonc binding sections.
99
+ // See design/35-cloudflare-primitives.md §"Binding Declarations in Adapter Config".
100
+
101
+ /** KV namespace binding. `id` defaults to `''` (filled in by wrangler or dashboard). */
102
+ export interface CloudflareKVBinding {
103
+ name: string;
104
+ id?: string;
105
+ }
106
+
107
+ /** D1 database binding. `database_id` is required (from Cloudflare dashboard). */
108
+ export interface CloudflareD1Binding {
109
+ name: string;
110
+ database_id: string;
111
+ }
112
+
113
+ /** R2 bucket binding. `bucket_name` must match the bucket created in Cloudflare. */
114
+ export interface CloudflareR2Binding {
115
+ name: string;
116
+ bucket_name: string;
117
+ }
118
+
119
+ /** Queue producer binding. `queue` is the queue name in Cloudflare. */
120
+ export interface CloudflareQueueProducer {
121
+ name: string;
122
+ queue: string;
123
+ }
124
+
125
+ /** Queue consumer declaration. Attached to the worker, not a binding name. */
126
+ export interface CloudflareQueueConsumer {
127
+ queue: string;
128
+ max_batch_size?: number;
129
+ max_retries?: number;
130
+ dead_letter_queue?: string;
131
+ }
132
+
133
+ /** Durable Object binding. `script_name` is for external DO references. */
134
+ export interface CloudflareDurableObjectBinding {
135
+ name: string;
136
+ class_name: string;
137
+ script_name?: string;
138
+ }
139
+
140
+ /**
141
+ * Declarative Cloudflare bindings configuration.
142
+ *
143
+ * These are converted to the appropriate wrangler.jsonc sections
144
+ * (`kv_namespaces`, `d1_databases`, `r2_buckets`, `queues`, `durable_objects`).
145
+ * The `wrangler` escape hatch overrides any generated binding sections
146
+ * if there's a conflict.
147
+ *
148
+ * @example
149
+ * ```ts
150
+ * cloudflare({
151
+ * bindings: {
152
+ * kv: [{ name: 'TIMBER_CACHE' }],
153
+ * d1: [{ name: 'MY_DB', database_id: 'xxxx' }],
154
+ * r2: [{ name: 'MY_BUCKET', bucket_name: 'my-bucket' }],
155
+ * queues: {
156
+ * producers: [{ name: 'EMAIL_QUEUE', queue: 'email-queue' }],
157
+ * consumers: [{ queue: 'email-queue', max_batch_size: 10 }],
158
+ * },
159
+ * durableObjects: [{ name: 'MY_DO', class_name: 'MyDurableObject' }],
160
+ * },
161
+ * })
162
+ * ```
163
+ */
164
+ export interface CloudflareBindings {
165
+ kv?: CloudflareKVBinding[];
166
+ d1?: CloudflareD1Binding[];
167
+ r2?: CloudflareR2Binding[];
168
+ queues?: {
169
+ producers?: CloudflareQueueProducer[];
170
+ consumers?: CloudflareQueueConsumer[];
171
+ };
172
+ durableObjects?: CloudflareDurableObjectBinding[];
78
173
  }
79
174
 
80
175
  /** Options for the Cloudflare Workers adapter. */
@@ -91,11 +186,42 @@ export interface CloudflareAdapterOptions {
91
186
  */
92
187
  compatibilityFlags?: string[];
93
188
 
189
+ /**
190
+ * Declarative Cloudflare bindings. Generates the appropriate
191
+ * `kv_namespaces`, `d1_databases`, `r2_buckets`, `queues`, and
192
+ * `durable_objects` sections in wrangler.jsonc.
193
+ *
194
+ * If both `bindings` and `wrangler` specify the same section,
195
+ * `wrangler` wins (it's the escape hatch).
196
+ */
197
+ bindings?: CloudflareBindings;
198
+
94
199
  /**
95
200
  * Custom wrangler.jsonc fields to merge.
96
- * Overrides generated values.
201
+ * Overrides generated values (including bindings-generated sections).
97
202
  */
98
203
  wrangler?: Record<string, unknown>;
204
+
205
+ /**
206
+ * Path to a module that exports additional Worker handlers (queue, scheduled, email, etc.).
207
+ * The module is imported in the generated `_worker.js` and its named exports are
208
+ * spread into the default export alongside `fetch`.
209
+ *
210
+ * The module receives `(batch/event, env, ctx)` — standard Cloudflare handler signatures.
211
+ * It does NOT have access to timber's request pipeline (no ALS, no getCloudflareBindings).
212
+ * Use `env` directly for bindings.
213
+ *
214
+ * @example
215
+ * ```ts
216
+ * // timber.config.ts
217
+ * adapter: cloudflare({ workerHandlers: './src/worker-handlers.ts' })
218
+ *
219
+ * // src/worker-handlers.ts
220
+ * export async function queue(batch, env) { ... }
221
+ * export async function scheduled(controller, env, ctx) { ... }
222
+ * ```
223
+ */
224
+ workerHandlers?: string;
99
225
  }
100
226
 
101
227
  /**
@@ -151,9 +277,35 @@ export function cloudflare(options: CloudflareAdapterOptions = {}): TimberPlatfo
151
277
  await writeFile(join(outDir, '_timber-manifest-init.js'), config.manifestInit);
152
278
  }
153
279
 
280
+ // Compile optional worker handlers (queue, scheduled, etc.)
281
+ // Uses Vite's build API to bundle the TypeScript source into ESM.
282
+ let hasWorkerHandlers = false;
283
+ if (options.workerHandlers) {
284
+ const handlersEntry = join(process.cwd(), options.workerHandlers);
285
+ const { build: viteBuild } = await import('vite');
286
+ await viteBuild({
287
+ configFile: false,
288
+ logLevel: 'info',
289
+ build: {
290
+ lib: {
291
+ entry: handlersEntry,
292
+ formats: ['es'],
293
+ fileName: () => '_worker-handlers.js',
294
+ },
295
+ outDir,
296
+ emptyOutDir: false,
297
+ minify: false,
298
+ rollupOptions: {
299
+ external: [/^node:/],
300
+ },
301
+ },
302
+ });
303
+ hasWorkerHandlers = true;
304
+ }
305
+
154
306
  // Generate the Workers entry point
155
307
  const hasManifestInit = !!config.manifestInit;
156
- const workerEntry = generateWorkerEntry(outDir, outDir, hasManifestInit);
308
+ const workerEntry = generateWorkerEntry(outDir, outDir, hasManifestInit, hasWorkerHandlers);
157
309
  await writeFile(join(outDir, '_worker.js'), workerEntry);
158
310
 
159
311
  // Generate wrangler.jsonc
@@ -184,12 +336,12 @@ export function cloudflare(options: CloudflareAdapterOptions = {}): TimberPlatfo
184
336
  export function wrapWithExecutionContext(
185
337
  adapter: TimberPlatformAdapter,
186
338
  handler: (req: Request) => Promise<Response>
187
- ): ExportedHandler<Record<string, unknown>> {
339
+ ): CfExportedHandler<Record<string, unknown>> {
188
340
  return {
189
341
  async fetch(
190
342
  request: Request,
191
343
  env: Record<string, unknown>,
192
- ctx: ExecutionContext
344
+ ctx: CfExecutionContext
193
345
  ): Promise<Response> {
194
346
  // Bind the adapter's waitUntil to the Workers execution context
195
347
  const originalWaitUntil = adapter.waitUntil;
@@ -214,7 +366,8 @@ export function wrapWithExecutionContext(
214
366
  export function generateWorkerEntry(
215
367
  buildDir: string,
216
368
  outDir: string,
217
- hasManifestInit = false
369
+ hasManifestInit = false,
370
+ hasWorkerHandlers = false
218
371
  ): string {
219
372
  // The RSC entry is the main request handler — it exports the fetch handler as default.
220
373
  // The Vite RSC plugin outputs it to rsc/index.js.
@@ -229,20 +382,171 @@ export function generateWorkerEntry(
229
382
  // ESM guarantees imports are evaluated in order.
230
383
  const manifestImport = hasManifestInit ? "import './_timber-manifest-init.js'\n" : '';
231
384
 
385
+ // Optional additional Worker handlers (queue, scheduled, email, etc.)
386
+ // Compiled by buildOutput via Vite's build API into _worker-handlers.js.
387
+ // The module's named exports are spread into the default export alongside fetch.
388
+ const handlersImport = hasWorkerHandlers
389
+ ? "import * as workerHandlers from './_worker-handlers.js'\n"
390
+ : '';
391
+ const handlersSpread = hasWorkerHandlers ? ' ...workerHandlers,\n' : '';
392
+
232
393
  return `// Generated by @timber-js/app/adapters/cloudflare
233
394
  // Do not edit — this file is regenerated on each build.
234
395
 
396
+ import { AsyncLocalStorage } from 'node:async_hooks'
235
397
  ${manifestImport}import handler from '${rscEntryRelative}'
236
-
398
+ ${handlersImport}
237
399
  // Set TIMBER_RUNTIME for instrumentation.ts conditional SDK initialization.
238
400
  // See design/25-production-deployments.md §"TIMBER_RUNTIME".
239
401
  globalThis.process ??= { env: {} }
240
402
  process.env.TIMBER_RUNTIME = 'cloudflare'
241
403
 
242
- export default { fetch: handler }
404
+ // Bind Cloudflare env to ALS so getCloudflareBindings() works at runtime.
405
+ // Uses the same Symbol.for key as getCloudflareBindings() reads from.
406
+ const ALS_KEY = Symbol.for('timber:cf-bindings-als')
407
+ let bindingsAls = globalThis[ALS_KEY]
408
+ if (!bindingsAls) {
409
+ bindingsAls = new AsyncLocalStorage()
410
+ globalThis[ALS_KEY] = bindingsAls
411
+ }
412
+
413
+ export default {
414
+ ${handlersSpread} async fetch(request, env, ctx) {
415
+ return bindingsAls.run(env, () => handler(request))
416
+ }
417
+ }
243
418
  `;
244
419
  }
245
420
 
421
+ /** Wrangler binding sections generated from `CloudflareBindings`. */
422
+ export interface WranglerBindingsConfig {
423
+ kv_namespaces?: Array<{ binding: string; id: string }>;
424
+ d1_databases?: Array<{ binding: string; database_id: string }>;
425
+ r2_buckets?: Array<{ binding: string; bucket_name: string }>;
426
+ queues?: {
427
+ producers?: Array<{ binding: string; queue: string }>;
428
+ consumers?: Array<Record<string, unknown>>;
429
+ };
430
+ durable_objects?: {
431
+ bindings: Array<Record<string, string>>;
432
+ };
433
+ }
434
+
435
+ /**
436
+ * Convert declarative `CloudflareBindings` into wrangler.jsonc binding sections.
437
+ *
438
+ * Maps:
439
+ * - `kv` → `kv_namespaces`
440
+ * - `d1` → `d1_databases`
441
+ * - `r2` → `r2_buckets`
442
+ * - `queues` → `queues` (producers + consumers)
443
+ * - `durableObjects` → `durable_objects`
444
+ *
445
+ * Empty arrays are omitted from the output.
446
+ *
447
+ * @internal Exported for testing.
448
+ */
449
+ export function generateBindingsConfig(
450
+ bindings: CloudflareBindings | undefined
451
+ ): WranglerBindingsConfig {
452
+ if (!bindings) return {};
453
+
454
+ const result: WranglerBindingsConfig = {};
455
+
456
+ // KV namespaces
457
+ if (bindings.kv && bindings.kv.length > 0) {
458
+ result.kv_namespaces = bindings.kv.map((kv) => ({
459
+ binding: kv.name,
460
+ id: kv.id ?? '',
461
+ }));
462
+ }
463
+
464
+ // D1 databases
465
+ if (bindings.d1 && bindings.d1.length > 0) {
466
+ result.d1_databases = bindings.d1.map((d1) => ({
467
+ binding: d1.name,
468
+ database_id: d1.database_id,
469
+ }));
470
+ }
471
+
472
+ // R2 buckets
473
+ if (bindings.r2 && bindings.r2.length > 0) {
474
+ result.r2_buckets = bindings.r2.map((r2) => ({
475
+ binding: r2.name,
476
+ bucket_name: r2.bucket_name,
477
+ }));
478
+ }
479
+
480
+ // Queues (producers and/or consumers)
481
+ if (bindings.queues) {
482
+ const queues: Record<string, unknown> = {};
483
+ if (bindings.queues.producers && bindings.queues.producers.length > 0) {
484
+ queues.producers = bindings.queues.producers.map((p) => ({
485
+ binding: p.name,
486
+ queue: p.queue,
487
+ }));
488
+ }
489
+ if (bindings.queues.consumers && bindings.queues.consumers.length > 0) {
490
+ queues.consumers = bindings.queues.consumers.map((c) => {
491
+ const consumer: Record<string, unknown> = { queue: c.queue };
492
+ if (c.max_batch_size !== undefined) consumer.max_batch_size = c.max_batch_size;
493
+ if (c.max_retries !== undefined) consumer.max_retries = c.max_retries;
494
+ if (c.dead_letter_queue !== undefined) consumer.dead_letter_queue = c.dead_letter_queue;
495
+ return consumer;
496
+ });
497
+ }
498
+ if (Object.keys(queues).length > 0) {
499
+ result.queues = queues;
500
+ }
501
+ }
502
+
503
+ // Durable Objects
504
+ if (bindings.durableObjects && bindings.durableObjects.length > 0) {
505
+ result.durable_objects = {
506
+ bindings: bindings.durableObjects.map((dobj) => {
507
+ const entry: Record<string, string> = {
508
+ name: dobj.name,
509
+ class_name: dobj.class_name,
510
+ };
511
+ if (dobj.script_name) entry.script_name = dobj.script_name;
512
+ return entry;
513
+ }),
514
+ };
515
+ }
516
+
517
+ return result;
518
+ }
519
+
520
+ /**
521
+ * One-level deep merge: for each key in `override`, if both the base and
522
+ * override values are plain objects (not arrays, not null), merge their
523
+ * keys with override winning on conflicts. Everything else (primitives,
524
+ * arrays, null) is replaced outright — matching the intuitive behavior
525
+ * where `kv_namespaces: [...]` replaces fully but
526
+ * `durable_objects: { migrations }` merges with generated `bindings`.
527
+ */
528
+ function shallowDeepMerge(
529
+ base: Record<string, unknown>,
530
+ override: Record<string, unknown>
531
+ ): Record<string, unknown> {
532
+ const result = { ...base };
533
+ for (const key of Object.keys(override)) {
534
+ const baseVal = result[key];
535
+ const overVal = override[key];
536
+ if (isPlainObject(baseVal) && isPlainObject(overVal)) {
537
+ result[key] = { ...baseVal, ...overVal };
538
+ } else {
539
+ result[key] = overVal;
540
+ }
541
+ }
542
+ return result;
543
+ }
544
+
545
+ /** True for `{}` literals — false for arrays, null, Date, etc. */
546
+ function isPlainObject(val: unknown): val is Record<string, unknown> {
547
+ return val !== null && typeof val === 'object' && !Array.isArray(val);
548
+ }
549
+
246
550
  /** @internal Exported for testing. */
247
551
  export function generateWranglerConfig(
248
552
  config: TimberConfig,
@@ -267,12 +571,20 @@ export function generateWranglerConfig(
267
571
  },
268
572
  };
269
573
 
270
- // Merge user overrides
574
+ // Layer 1: merge bindings-generated sections into base
575
+ const bindingsConfig = generateBindingsConfig(options.bindings);
576
+ const merged = { ...base, ...bindingsConfig };
577
+
578
+ // Layer 2: wrangler escape hatch with deep merge for nested plain objects.
579
+ // A shallow spread would replace entire sections like durable_objects and
580
+ // queues — e.g. adding migrations via wrangler would drop the generated
581
+ // durable_objects.bindings. Deep merge preserves generated keys while
582
+ // letting wrangler override on actual key-level conflicts.
271
583
  if (options.wrangler) {
272
- return { ...base, ...options.wrangler };
584
+ return shallowDeepMerge(merged, options.wrangler);
273
585
  }
274
586
 
275
- return base;
587
+ return merged;
276
588
  }
277
589
 
278
590
  // ─── Preview ─────────────────────────────────────────────────────────────────
@@ -310,16 +622,18 @@ function spawnPreviewProcess(command: string, args: string[], cwd: string): Prom
310
622
  }
311
623
 
312
624
  // ─── Cloudflare Workers type stubs ───────────────────────────────────────────
313
- // Minimal type declarations so this file compiles without @cloudflare/workers-types.
314
- // In production builds, users install @cloudflare/workers-types themselves.
315
-
316
- declare global {
317
- interface ExecutionContext {
318
- waitUntil(promise: Promise<unknown>): void;
319
- passThroughOnException(): void;
320
- }
625
+ // Local type declarations so this file compiles without @cloudflare/workers-types.
626
+ // These are NOT declared globally — that would clash with @cloudflare/workers-types
627
+ // when users install it. Instead, they're module-scoped and used only within
628
+ // this file's function signatures.
629
+
630
+ /** @internal Minimal stub — use @cloudflare/workers-types for full types. */
631
+ export interface CfExecutionContext {
632
+ waitUntil(promise: Promise<unknown>): void;
633
+ passThroughOnException(): void;
634
+ }
321
635
 
322
- interface ExportedHandler<Env = Record<string, unknown>> {
323
- fetch?(request: Request, env: Env, ctx: ExecutionContext): Promise<Response> | Response;
324
- }
636
+ /** @internal Minimal stub use @cloudflare/workers-types for full types. */
637
+ export interface CfExportedHandler<Env = Record<string, unknown>> {
638
+ fetch?(request: Request, env: Env, ctx: CfExecutionContext): Promise<Response> | Response;
325
639
  }
@@ -22,9 +22,11 @@
22
22
  export function generateCompressModule(): string {
23
23
  return `// Generated by @timber-js/app — response compression for self-hosted deployments.
24
24
  // Do not edit — this file is regenerated on each build.
25
- // Uses CompressionStream (Web API) for gzip. Brotli is left to CDNs/reverse
26
- // proxies at streaming quality levels its ratio advantage is marginal and
27
- // node:zlib buffers output internally, breaking streaming.
25
+ // Uses node:zlib createGzip() (C++ native) on Node.js, falls back to
26
+ // CompressionStream (Web API) on other runtimes. Brotli is left to CDNs/reverse
27
+ // proxies at streaming quality levels its ratio advantage is marginal.
28
+ import { Readable } from 'node:stream';
29
+ import { createGzip, constants } from 'node:zlib';
28
30
 
29
31
  const COMPRESSIBLE_TYPES = new Set([
30
32
  'text/html', 'text/css', 'text/plain', 'text/xml', 'text/javascript',
@@ -71,7 +73,25 @@ function shouldCompress(response) {
71
73
  }
72
74
 
73
75
  function compressWithGzip(body) {
74
- return body.pipeThrough(new CompressionStream('gzip'));
76
+ // Use node:zlib (C++ native) for gzip compression. The Web Streams
77
+ // CompressionStream works but every chunk crosses the JS/Promise boundary.
78
+ // node:zlib.createGzip() compresses entirely in C++ with zero per-chunk
79
+ // Promise overhead.
80
+ //
81
+ // Convert: Web ReadableStream → Node Readable → pipe through gzip →
82
+ // Node Readable → Readable.toWeb() → Web ReadableStream
83
+ try {
84
+ const nodeReadable = Readable.fromWeb(body);
85
+ // Z_SYNC_FLUSH ensures each chunk is flushed immediately so the browser
86
+ // receives the HTML shell before Suspense boundaries resolve. Without it,
87
+ // gzip buffers internally and breaks streaming.
88
+ const gzip = createGzip({ flush: constants.Z_SYNC_FLUSH });
89
+ const compressed = nodeReadable.pipe(gzip);
90
+ return Readable.toWeb(compressed);
91
+ } catch {
92
+ // Fallback: CompressionStream (CF Workers, or if node:stream unavailable)
93
+ return body.pipeThrough(new CompressionStream('gzip'));
94
+ }
75
95
  }
76
96
 
77
97
  export function compressResponse(request, response) {