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

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 (573) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/actions-Dg-ANYHb.js +421 -0
  3. package/dist/_chunks/actions-Dg-ANYHb.js.map +1 -0
  4. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
  5. package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
  6. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  7. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  8. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  9. package/dist/_chunks/define-C77ScO0m.js +106 -0
  10. package/dist/_chunks/define-C77ScO0m.js.map +1 -0
  11. package/dist/_chunks/define-CZqDwhSu.js +199 -0
  12. package/dist/_chunks/define-CZqDwhSu.js.map +1 -0
  13. package/dist/_chunks/define-cookie-C2IkoFGN.js +94 -0
  14. package/dist/_chunks/define-cookie-C2IkoFGN.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
  16. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
  17. package/dist/_chunks/format-CYBGxKtc.js +14 -0
  18. package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-Dpn_UfAD.js} +171 -97
  20. package/dist/_chunks/interception-Dpn_UfAD.js.map +1 -0
  21. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
  22. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
  23. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  24. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  25. package/dist/_chunks/request-context-qMsWgy9C.js +478 -0
  26. package/dist/_chunks/request-context-qMsWgy9C.js.map +1 -0
  27. package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
  28. package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
  29. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  30. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  31. package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
  32. package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
  33. package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
  34. package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
  35. package/dist/_chunks/stale-reload-C2plcNtG.js +64 -0
  36. package/dist/_chunks/stale-reload-C2plcNtG.js.map +1 -0
  37. package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
  38. package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
  39. package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
  40. package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
  41. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-Lo_s_pw2.js} +4 -4
  42. package/dist/_chunks/use-query-states-Lo_s_pw2.js.map +1 -0
  43. package/dist/_chunks/wrappers-_DTmImGt.js +63 -0
  44. package/dist/_chunks/wrappers-_DTmImGt.js.map +1 -0
  45. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  46. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  47. package/dist/adapters/cloudflare-dev.js +73 -0
  48. package/dist/adapters/cloudflare-dev.js.map +1 -0
  49. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  50. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  51. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  52. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  53. package/dist/adapters/cloudflare.d.ts +148 -12
  54. package/dist/adapters/cloudflare.d.ts.map +1 -1
  55. package/dist/adapters/cloudflare.js +135 -11
  56. package/dist/adapters/cloudflare.js.map +1 -1
  57. package/dist/adapters/compress-module.d.ts.map +1 -1
  58. package/dist/adapters/nitro.d.ts +17 -1
  59. package/dist/adapters/nitro.d.ts.map +1 -1
  60. package/dist/adapters/nitro.js +56 -13
  61. package/dist/adapters/nitro.js.map +1 -1
  62. package/dist/cache/cache-api.d.ts +24 -0
  63. package/dist/cache/cache-api.d.ts.map +1 -0
  64. package/dist/cache/handler-store.d.ts +31 -0
  65. package/dist/cache/handler-store.d.ts.map +1 -0
  66. package/dist/cache/index.d.ts +23 -7
  67. package/dist/cache/index.d.ts.map +1 -1
  68. package/dist/cache/index.js +142 -80
  69. package/dist/cache/index.js.map +1 -1
  70. package/dist/cache/singleflight.d.ts +18 -1
  71. package/dist/cache/singleflight.d.ts.map +1 -1
  72. package/dist/cache/sizeof.d.ts +22 -0
  73. package/dist/cache/sizeof.d.ts.map +1 -0
  74. package/dist/cache/timber-cache.d.ts +1 -1
  75. package/dist/cache/timber-cache.d.ts.map +1 -1
  76. package/dist/cli.d.ts +6 -1
  77. package/dist/cli.d.ts.map +1 -1
  78. package/dist/cli.js +6 -1
  79. package/dist/cli.js.map +1 -1
  80. package/dist/client/browser-dev.d.ts +27 -1
  81. package/dist/client/browser-dev.d.ts.map +1 -1
  82. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  83. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  84. package/dist/client/browser-entry/hmr.d.ts +21 -0
  85. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  86. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  87. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  88. package/dist/client/browser-entry/index.d.ts +30 -0
  89. package/dist/client/browser-entry/index.d.ts.map +1 -0
  90. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  91. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  92. package/dist/client/browser-entry/router-init.d.ts +23 -0
  93. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  94. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  95. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  96. package/dist/client/browser-entry/scroll.d.ts +19 -0
  97. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  98. package/dist/client/error-boundary.d.ts +12 -5
  99. package/dist/client/error-boundary.d.ts.map +1 -1
  100. package/dist/client/error-boundary.js +10 -4
  101. package/dist/client/error-boundary.js.map +1 -1
  102. package/dist/client/error-reconstituter.d.ts +54 -0
  103. package/dist/client/error-reconstituter.d.ts.map +1 -0
  104. package/dist/client/form.d.ts +2 -2
  105. package/dist/client/form.d.ts.map +1 -1
  106. package/dist/client/history.d.ts +19 -4
  107. package/dist/client/history.d.ts.map +1 -1
  108. package/dist/client/index.d.ts +7 -21
  109. package/dist/client/index.d.ts.map +1 -1
  110. package/dist/client/index.js +210 -1017
  111. package/dist/client/index.js.map +1 -1
  112. package/dist/client/internal.d.ts +18 -0
  113. package/dist/client/internal.d.ts.map +1 -0
  114. package/dist/client/internal.js +890 -0
  115. package/dist/client/internal.js.map +1 -0
  116. package/dist/client/link-pending-store.d.ts +63 -0
  117. package/dist/client/link-pending-store.d.ts.map +1 -0
  118. package/dist/client/link.d.ts +90 -32
  119. package/dist/client/link.d.ts.map +1 -1
  120. package/dist/client/nav-link-store.d.ts +36 -0
  121. package/dist/client/nav-link-store.d.ts.map +1 -0
  122. package/dist/client/navigation-api-types.d.ts +90 -0
  123. package/dist/client/navigation-api-types.d.ts.map +1 -0
  124. package/dist/client/navigation-api.d.ts +115 -0
  125. package/dist/client/navigation-api.d.ts.map +1 -0
  126. package/dist/client/navigation-context.d.ts +13 -2
  127. package/dist/client/navigation-context.d.ts.map +1 -1
  128. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  129. package/dist/client/navigation-root.d.ts.map +1 -0
  130. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  131. package/dist/client/router-ref.d.ts +1 -1
  132. package/dist/client/router.d.ts +70 -4
  133. package/dist/client/router.d.ts.map +1 -1
  134. package/dist/client/rsc-fetch.d.ts +38 -3
  135. package/dist/client/rsc-fetch.d.ts.map +1 -1
  136. package/dist/client/segment-cache.d.ts +1 -1
  137. package/dist/client/segment-cache.d.ts.map +1 -1
  138. package/dist/client/segment-outlet.d.ts +63 -0
  139. package/dist/client/segment-outlet.d.ts.map +1 -0
  140. package/dist/client/ssr-data.d.ts +13 -4
  141. package/dist/client/ssr-data.d.ts.map +1 -1
  142. package/dist/client/stale-reload.d.ts +15 -0
  143. package/dist/client/stale-reload.d.ts.map +1 -1
  144. package/dist/client/top-loader.d.ts +5 -5
  145. package/dist/client/top-loader.d.ts.map +1 -1
  146. package/dist/client/use-link-status.d.ts +5 -5
  147. package/dist/client/use-link-status.d.ts.map +1 -1
  148. package/dist/client/use-params.d.ts +6 -4
  149. package/dist/client/use-params.d.ts.map +1 -1
  150. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  151. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  152. package/dist/client/use-query-states.d.ts +1 -1
  153. package/dist/client/use-query-states.d.ts.map +1 -1
  154. package/dist/client/use-router.d.ts +1 -1
  155. package/dist/codec.d.ts +33 -0
  156. package/dist/codec.d.ts.map +1 -0
  157. package/dist/codec.js +2 -0
  158. package/dist/config-types.d.ts +210 -0
  159. package/dist/config-types.d.ts.map +1 -0
  160. package/dist/content/index.d.ts +1 -10
  161. package/dist/content/index.d.ts.map +1 -1
  162. package/dist/content/index.js +0 -2
  163. package/dist/cookies/define-cookie.d.ts +35 -14
  164. package/dist/cookies/define-cookie.d.ts.map +1 -1
  165. package/dist/cookies/index.js +1 -83
  166. package/dist/fonts/css.d.ts +1 -0
  167. package/dist/fonts/css.d.ts.map +1 -1
  168. package/dist/index.d.ts +45 -192
  169. package/dist/index.d.ts.map +1 -1
  170. package/dist/index.js +12296 -11901
  171. package/dist/index.js.map +1 -1
  172. package/dist/plugin-context.d.ts +84 -0
  173. package/dist/plugin-context.d.ts.map +1 -0
  174. package/dist/plugins/adapter-build.d.ts +1 -1
  175. package/dist/plugins/adapter-build.d.ts.map +1 -1
  176. package/dist/plugins/build-manifest.d.ts +2 -2
  177. package/dist/plugins/build-manifest.d.ts.map +1 -1
  178. package/dist/plugins/build-report.d.ts +3 -3
  179. package/dist/plugins/build-report.d.ts.map +1 -1
  180. package/dist/plugins/client-chunks.d.ts +32 -0
  181. package/dist/plugins/client-chunks.d.ts.map +1 -0
  182. package/dist/plugins/content.d.ts +1 -1
  183. package/dist/plugins/content.d.ts.map +1 -1
  184. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  185. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  186. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  187. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  188. package/dist/plugins/dev-logs.d.ts +1 -1
  189. package/dist/plugins/dev-logs.d.ts.map +1 -1
  190. package/dist/plugins/dev-server.d.ts +1 -1
  191. package/dist/plugins/dev-server.d.ts.map +1 -1
  192. package/dist/plugins/entries.d.ts +1 -1
  193. package/dist/plugins/entries.d.ts.map +1 -1
  194. package/dist/plugins/fonts.d.ts +19 -5
  195. package/dist/plugins/fonts.d.ts.map +1 -1
  196. package/dist/plugins/mdx.d.ts +1 -1
  197. package/dist/plugins/mdx.d.ts.map +1 -1
  198. package/dist/plugins/routing.d.ts +1 -1
  199. package/dist/plugins/routing.d.ts.map +1 -1
  200. package/dist/plugins/server-bundle.d.ts.map +1 -1
  201. package/dist/plugins/shims.d.ts +6 -5
  202. package/dist/plugins/shims.d.ts.map +1 -1
  203. package/dist/plugins/static-build.d.ts +4 -4
  204. package/dist/plugins/static-build.d.ts.map +1 -1
  205. package/dist/routing/codegen.d.ts +2 -2
  206. package/dist/routing/codegen.d.ts.map +1 -1
  207. package/dist/routing/index.d.ts +2 -0
  208. package/dist/routing/index.d.ts.map +1 -1
  209. package/dist/routing/index.js +3 -2
  210. package/dist/routing/scanner.d.ts.map +1 -1
  211. package/dist/routing/segment-classify.d.ts +46 -0
  212. package/dist/routing/segment-classify.d.ts.map +1 -0
  213. package/dist/routing/status-file-lint.d.ts +2 -1
  214. package/dist/routing/status-file-lint.d.ts.map +1 -1
  215. package/dist/routing/types.d.ts +16 -4
  216. package/dist/routing/types.d.ts.map +1 -1
  217. package/dist/rsc-runtime/rsc.d.ts +1 -1
  218. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  219. package/dist/rsc-runtime/ssr.d.ts +12 -0
  220. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  221. package/dist/schema-bridge.d.ts +76 -0
  222. package/dist/schema-bridge.d.ts.map +1 -0
  223. package/dist/search-params/define.d.ts +139 -0
  224. package/dist/search-params/define.d.ts.map +1 -0
  225. package/dist/search-params/index.d.ts +4 -7
  226. package/dist/search-params/index.d.ts.map +1 -1
  227. package/dist/search-params/index.js +4 -474
  228. package/dist/search-params/registry.d.ts +2 -2
  229. package/dist/search-params/registry.d.ts.map +1 -1
  230. package/dist/search-params/wrappers.d.ts +53 -0
  231. package/dist/search-params/wrappers.d.ts.map +1 -0
  232. package/dist/segment-params/define.d.ts +78 -0
  233. package/dist/segment-params/define.d.ts.map +1 -0
  234. package/dist/segment-params/index.d.ts +6 -0
  235. package/dist/segment-params/index.d.ts.map +1 -0
  236. package/dist/segment-params/index.js +4 -0
  237. package/dist/server/access-gate.d.ts +4 -0
  238. package/dist/server/access-gate.d.ts.map +1 -1
  239. package/dist/server/action-client.d.ts +12 -1
  240. package/dist/server/action-client.d.ts.map +1 -1
  241. package/dist/server/action-encryption.d.ts +76 -0
  242. package/dist/server/action-encryption.d.ts.map +1 -0
  243. package/dist/server/action-handler.d.ts.map +1 -1
  244. package/dist/server/actions.d.ts +3 -6
  245. package/dist/server/actions.d.ts.map +1 -1
  246. package/dist/server/als-registry.d.ts +32 -4
  247. package/dist/server/als-registry.d.ts.map +1 -1
  248. package/dist/server/build-manifest.d.ts +2 -2
  249. package/dist/server/build-manifest.d.ts.map +1 -1
  250. package/dist/server/debug.d.ts +1 -1
  251. package/dist/server/default-logger.d.ts +22 -0
  252. package/dist/server/default-logger.d.ts.map +1 -0
  253. package/dist/server/deny-page-resolver.d.ts +52 -0
  254. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  255. package/dist/server/deny-renderer.d.ts.map +1 -1
  256. package/dist/server/dev-holding-server.d.ts +52 -0
  257. package/dist/server/dev-holding-server.d.ts.map +1 -0
  258. package/dist/server/dev-warnings.d.ts +1 -21
  259. package/dist/server/dev-warnings.d.ts.map +1 -1
  260. package/dist/server/early-hints.d.ts +13 -5
  261. package/dist/server/early-hints.d.ts.map +1 -1
  262. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  263. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  264. package/dist/server/fallback-error.d.ts +4 -3
  265. package/dist/server/fallback-error.d.ts.map +1 -1
  266. package/dist/server/flight-injection-state.d.ts +66 -0
  267. package/dist/server/flight-injection-state.d.ts.map +1 -0
  268. package/dist/server/flight-scripts.d.ts +42 -0
  269. package/dist/server/flight-scripts.d.ts.map +1 -0
  270. package/dist/server/flush.d.ts.map +1 -1
  271. package/dist/server/form-data.d.ts +29 -0
  272. package/dist/server/form-data.d.ts.map +1 -1
  273. package/dist/server/html-injectors.d.ts +51 -11
  274. package/dist/server/html-injectors.d.ts.map +1 -1
  275. package/dist/server/index.d.ts +6 -43
  276. package/dist/server/index.d.ts.map +1 -1
  277. package/dist/server/index.js +38 -2798
  278. package/dist/server/index.js.map +1 -1
  279. package/dist/server/internal.d.ts +46 -0
  280. package/dist/server/internal.d.ts.map +1 -0
  281. package/dist/server/internal.js +2883 -0
  282. package/dist/server/internal.js.map +1 -0
  283. package/dist/server/logger.d.ts +25 -7
  284. package/dist/server/logger.d.ts.map +1 -1
  285. package/dist/server/middleware-runner.d.ts +19 -4
  286. package/dist/server/middleware-runner.d.ts.map +1 -1
  287. package/dist/server/node-stream-transforms.d.ts +113 -0
  288. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  289. package/dist/server/page-deny-boundary.d.ts +31 -0
  290. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  291. package/dist/server/pipeline-interception.d.ts +1 -1
  292. package/dist/server/pipeline-interception.d.ts.map +1 -1
  293. package/dist/server/pipeline-metadata.d.ts +6 -0
  294. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  295. package/dist/server/pipeline.d.ts +42 -10
  296. package/dist/server/pipeline.d.ts.map +1 -1
  297. package/dist/server/primitives.d.ts +69 -18
  298. package/dist/server/primitives.d.ts.map +1 -1
  299. package/dist/server/render-timeout.d.ts +51 -0
  300. package/dist/server/render-timeout.d.ts.map +1 -0
  301. package/dist/server/request-context.d.ts +112 -43
  302. package/dist/server/request-context.d.ts.map +1 -1
  303. package/dist/server/route-element-builder.d.ts +27 -1
  304. package/dist/server/route-element-builder.d.ts.map +1 -1
  305. package/dist/server/route-handler.d.ts.map +1 -1
  306. package/dist/server/route-matcher.d.ts +9 -2
  307. package/dist/server/route-matcher.d.ts.map +1 -1
  308. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  309. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  310. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  311. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  312. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  313. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  314. package/dist/server/rsc-entry/index.d.ts +8 -3
  315. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  316. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  317. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  318. package/dist/server/rsc-entry/rsc-stream.d.ts +4 -1
  319. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  320. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  321. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  322. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  323. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  324. package/dist/server/safe-load.d.ts +46 -0
  325. package/dist/server/safe-load.d.ts.map +1 -0
  326. package/dist/server/sitemap-generator.d.ts +129 -0
  327. package/dist/server/sitemap-generator.d.ts.map +1 -0
  328. package/dist/server/sitemap-handler.d.ts +22 -0
  329. package/dist/server/sitemap-handler.d.ts.map +1 -0
  330. package/dist/server/slot-resolver.d.ts +1 -1
  331. package/dist/server/slot-resolver.d.ts.map +1 -1
  332. package/dist/server/ssr-entry.d.ts +22 -0
  333. package/dist/server/ssr-entry.d.ts.map +1 -1
  334. package/dist/server/ssr-render.d.ts +39 -21
  335. package/dist/server/ssr-render.d.ts.map +1 -1
  336. package/dist/server/ssr-wrappers.d.ts +50 -0
  337. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  338. package/dist/server/status-code-resolver.d.ts +1 -1
  339. package/dist/server/status-code-resolver.d.ts.map +1 -1
  340. package/dist/server/stream-utils.d.ts +36 -0
  341. package/dist/server/stream-utils.d.ts.map +1 -0
  342. package/dist/server/tracing.d.ts +4 -4
  343. package/dist/server/tracing.d.ts.map +1 -1
  344. package/dist/server/tree-builder.d.ts +22 -19
  345. package/dist/server/tree-builder.d.ts.map +1 -1
  346. package/dist/server/types.d.ts +1 -4
  347. package/dist/server/types.d.ts.map +1 -1
  348. package/dist/server/version-skew.d.ts +61 -0
  349. package/dist/server/version-skew.d.ts.map +1 -0
  350. package/dist/shared/merge-search-params.d.ts +22 -0
  351. package/dist/shared/merge-search-params.d.ts.map +1 -0
  352. package/dist/shims/font-google.d.ts +1 -1
  353. package/dist/shims/font-google.d.ts.map +1 -1
  354. package/dist/shims/font-google.js +42 -0
  355. package/dist/shims/font-google.js.map +1 -0
  356. package/dist/shims/font-local.d.ts +26 -0
  357. package/dist/shims/font-local.d.ts.map +1 -0
  358. package/dist/shims/font-local.js +20 -0
  359. package/dist/shims/font-local.js.map +1 -0
  360. package/dist/shims/headers.d.ts +2 -1
  361. package/dist/shims/headers.d.ts.map +1 -1
  362. package/dist/shims/navigation-client.d.ts +1 -1
  363. package/dist/shims/navigation-client.d.ts.map +1 -1
  364. package/dist/shims/navigation.d.ts +3 -2
  365. package/dist/shims/navigation.d.ts.map +1 -1
  366. package/dist/utils/directive-parser.d.ts +5 -2
  367. package/dist/utils/directive-parser.d.ts.map +1 -1
  368. package/dist/utils/state-machine.d.ts +80 -0
  369. package/dist/utils/state-machine.d.ts.map +1 -0
  370. package/package.json +53 -23
  371. package/src/adapters/cloudflare-dev.ts +177 -0
  372. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  373. package/src/adapters/cloudflare.ts +342 -28
  374. package/src/adapters/compress-module.ts +24 -4
  375. package/src/adapters/nitro.ts +52 -8
  376. package/src/adapters/wrangler.d.ts +7 -0
  377. package/src/cache/cache-api.ts +38 -0
  378. package/src/cache/handler-store.ts +68 -0
  379. package/src/cache/index.ts +81 -18
  380. package/src/cache/singleflight.ts +62 -4
  381. package/src/cache/sizeof.ts +31 -0
  382. package/src/cache/timber-cache.ts +24 -20
  383. package/src/cli.ts +6 -1
  384. package/src/client/browser-dev.ts +128 -1
  385. package/src/client/browser-entry/action-dispatch.ts +116 -0
  386. package/src/client/browser-entry/hmr.ts +81 -0
  387. package/src/client/browser-entry/hydrate.ts +145 -0
  388. package/src/client/browser-entry/index.ts +138 -0
  389. package/src/client/browser-entry/post-hydration.ts +119 -0
  390. package/src/client/browser-entry/router-init.ts +193 -0
  391. package/src/client/browser-entry/rsc-stream.ts +157 -0
  392. package/src/client/browser-entry/scroll.ts +27 -0
  393. package/src/client/error-boundary.tsx +48 -16
  394. package/src/client/error-reconstituter.tsx +65 -0
  395. package/src/client/form.tsx +2 -2
  396. package/src/client/history.ts +26 -4
  397. package/src/client/index.ts +19 -38
  398. package/src/client/internal.ts +57 -0
  399. package/src/client/link-pending-store.ts +111 -0
  400. package/src/client/link.tsx +329 -97
  401. package/src/client/nav-link-store.ts +47 -0
  402. package/src/client/navigation-api-types.ts +112 -0
  403. package/src/client/navigation-api.ts +332 -0
  404. package/src/client/navigation-context.ts +31 -6
  405. package/src/client/navigation-root.tsx +342 -0
  406. package/src/client/nuqs-adapter.tsx +16 -3
  407. package/src/client/router-ref.ts +1 -1
  408. package/src/client/router.ts +299 -72
  409. package/src/client/rsc-fetch.ts +97 -8
  410. package/src/client/segment-cache.ts +1 -1
  411. package/src/client/segment-outlet.tsx +86 -0
  412. package/src/client/ssr-data.ts +13 -5
  413. package/src/client/stale-reload.ts +72 -3
  414. package/src/client/top-loader.tsx +16 -8
  415. package/src/client/use-link-status.ts +7 -7
  416. package/src/client/use-params.ts +7 -5
  417. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  418. package/src/client/use-query-states.ts +3 -3
  419. package/src/client/use-router.ts +1 -1
  420. package/src/codec.ts +49 -0
  421. package/src/config-types.ts +208 -0
  422. package/src/content/index.ts +5 -13
  423. package/src/cookies/define-cookie.ts +78 -25
  424. package/src/cookies/index.ts +8 -0
  425. package/src/fonts/css.ts +2 -1
  426. package/src/index.ts +258 -354
  427. package/src/plugin-context.ts +200 -0
  428. package/src/plugins/adapter-build.ts +8 -2
  429. package/src/plugins/build-manifest.ts +13 -2
  430. package/src/plugins/build-report.ts +3 -3
  431. package/src/plugins/client-chunks.ts +65 -0
  432. package/src/plugins/content.ts +1 -1
  433. package/src/plugins/dev-browser-logs.ts +288 -0
  434. package/src/plugins/dev-error-overlay.ts +70 -1
  435. package/src/plugins/dev-logs.ts +1 -1
  436. package/src/plugins/dev-server.ts +70 -9
  437. package/src/plugins/entries.ts +71 -10
  438. package/src/plugins/fonts.ts +168 -61
  439. package/src/plugins/mdx.ts +1 -1
  440. package/src/plugins/routing.ts +57 -17
  441. package/src/plugins/server-action-exports.ts +1 -1
  442. package/src/plugins/server-bundle.ts +32 -1
  443. package/src/plugins/shims.ts +135 -35
  444. package/src/plugins/static-build.ts +17 -11
  445. package/src/routing/codegen.ts +165 -105
  446. package/src/routing/index.ts +2 -0
  447. package/src/routing/scanner.ts +93 -23
  448. package/src/routing/segment-classify.ts +89 -0
  449. package/src/routing/status-file-lint.ts +3 -2
  450. package/src/routing/types.ts +17 -4
  451. package/src/rsc-runtime/rsc.ts +2 -0
  452. package/src/rsc-runtime/ssr.ts +50 -0
  453. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  454. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  455. package/src/search-params/define.ts +482 -0
  456. package/src/search-params/index.ts +14 -20
  457. package/src/search-params/registry.ts +2 -2
  458. package/src/search-params/wrappers.ts +85 -0
  459. package/src/segment-params/define.ts +279 -0
  460. package/src/segment-params/index.ts +29 -0
  461. package/src/server/access-gate.tsx +70 -29
  462. package/src/server/action-client.ts +21 -2
  463. package/src/server/action-encryption.ts +144 -0
  464. package/src/server/action-handler.ts +21 -4
  465. package/src/server/actions.ts +10 -9
  466. package/src/server/als-registry.ts +34 -6
  467. package/src/server/build-manifest.ts +10 -4
  468. package/src/server/compress.ts +25 -7
  469. package/src/server/debug.ts +1 -1
  470. package/src/server/default-logger.ts +99 -0
  471. package/src/server/deny-page-resolver.ts +154 -0
  472. package/src/server/deny-renderer.ts +24 -38
  473. package/src/server/dev-holding-server.ts +185 -0
  474. package/src/server/dev-warnings.ts +4 -49
  475. package/src/server/early-hints.ts +36 -15
  476. package/src/server/error-boundary-wrapper.ts +74 -22
  477. package/src/server/fallback-error.ts +31 -15
  478. package/src/server/flight-injection-state.ts +113 -0
  479. package/src/server/flight-scripts.ts +62 -0
  480. package/src/server/flush.ts +2 -1
  481. package/src/server/form-data.ts +76 -0
  482. package/src/server/html-injectors.ts +280 -120
  483. package/src/server/index.ts +26 -177
  484. package/src/server/internal.ts +169 -0
  485. package/src/server/logger.ts +44 -36
  486. package/src/server/middleware-runner.ts +31 -4
  487. package/src/server/node-stream-transforms.ts +509 -0
  488. package/src/server/page-deny-boundary.tsx +56 -0
  489. package/src/server/pipeline-interception.ts +17 -16
  490. package/src/server/pipeline-metadata.ts +13 -0
  491. package/src/server/pipeline.ts +227 -62
  492. package/src/server/primitives.ts +111 -28
  493. package/src/server/render-timeout.ts +108 -0
  494. package/src/server/request-context.ts +293 -132
  495. package/src/server/route-element-builder.ts +283 -191
  496. package/src/server/route-handler.ts +24 -4
  497. package/src/server/route-matcher.ts +24 -20
  498. package/src/server/rsc-entry/api-handler.ts +15 -16
  499. package/src/server/rsc-entry/error-renderer.ts +300 -89
  500. package/src/server/rsc-entry/helpers.ts +134 -5
  501. package/src/server/rsc-entry/index.ts +200 -112
  502. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  503. package/src/server/rsc-entry/rsc-stream.ts +65 -13
  504. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  505. package/src/server/rsc-entry/ssr-renderer.ts +168 -38
  506. package/src/server/safe-load.ts +60 -0
  507. package/src/server/sitemap-generator.ts +338 -0
  508. package/src/server/sitemap-handler.ts +126 -0
  509. package/src/server/slot-resolver.ts +244 -229
  510. package/src/server/ssr-entry.ts +211 -32
  511. package/src/server/ssr-render.ts +289 -67
  512. package/src/server/ssr-wrappers.tsx +139 -0
  513. package/src/server/status-code-resolver.ts +1 -1
  514. package/src/server/stream-utils.ts +213 -0
  515. package/src/server/tracing.ts +20 -9
  516. package/src/server/tree-builder.ts +92 -58
  517. package/src/server/types.ts +3 -6
  518. package/src/server/version-skew.ts +104 -0
  519. package/src/shared/merge-search-params.ts +55 -0
  520. package/src/shims/font-google.ts +1 -1
  521. package/src/shims/font-local.ts +34 -0
  522. package/src/shims/headers.ts +5 -1
  523. package/src/shims/navigation-client.ts +1 -1
  524. package/src/shims/navigation.ts +7 -2
  525. package/src/utils/directive-parser.ts +5 -2
  526. package/src/utils/state-machine.ts +111 -0
  527. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  528. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  529. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  530. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  531. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  532. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  533. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  534. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  535. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  536. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  537. package/dist/cache/register-cached-function.d.ts +0 -17
  538. package/dist/cache/register-cached-function.d.ts.map +0 -1
  539. package/dist/client/browser-entry.d.ts +0 -21
  540. package/dist/client/browser-entry.d.ts.map +0 -1
  541. package/dist/client/link-status-provider.d.ts +0 -11
  542. package/dist/client/link-status-provider.d.ts.map +0 -1
  543. package/dist/client/transition-root.d.ts.map +0 -1
  544. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  545. package/dist/cookies/index.js.map +0 -1
  546. package/dist/plugins/cache-transform.d.ts +0 -36
  547. package/dist/plugins/cache-transform.d.ts.map +0 -1
  548. package/dist/plugins/dynamic-transform.d.ts +0 -72
  549. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  550. package/dist/search-params/analyze.d.ts +0 -54
  551. package/dist/search-params/analyze.d.ts.map +0 -1
  552. package/dist/search-params/builtin-codecs.d.ts +0 -105
  553. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  554. package/dist/search-params/codecs.d.ts +0 -53
  555. package/dist/search-params/codecs.d.ts.map +0 -1
  556. package/dist/search-params/create.d.ts +0 -106
  557. package/dist/search-params/create.d.ts.map +0 -1
  558. package/dist/search-params/index.js.map +0 -1
  559. package/dist/server/prerender.d.ts +0 -77
  560. package/dist/server/prerender.d.ts.map +0 -1
  561. package/dist/server/response-cache.d.ts +0 -54
  562. package/dist/server/response-cache.d.ts.map +0 -1
  563. package/src/cache/register-cached-function.ts +0 -103
  564. package/src/client/browser-entry.ts +0 -678
  565. package/src/client/link-status-provider.tsx +0 -30
  566. package/src/client/transition-root.tsx +0 -166
  567. package/src/plugins/cache-transform.ts +0 -199
  568. package/src/plugins/dynamic-transform.ts +0 -161
  569. package/src/search-params/analyze.ts +0 -192
  570. package/src/search-params/builtin-codecs.ts +0 -228
  571. package/src/search-params/create.ts +0 -321
  572. package/src/server/prerender.ts +0 -139
  573. package/src/server/response-cache.ts +0 -410
@@ -0,0 +1,68 @@
1
+ /**
2
+ * Module-level cache handler singleton.
3
+ *
4
+ * Lazily initialized to MemoryCacheHandler on first access. The framework
5
+ * replaces this at boot from timber.config.ts via setCacheHandler().
6
+ *
7
+ * This module avoids importing from ./index to prevent circular dependencies.
8
+ */
9
+
10
+ // Inline the interface to avoid circular import with index.ts
11
+ interface CacheHandlerLike {
12
+ get(key: string): Promise<{ value: unknown; stale: boolean } | null>;
13
+ set(key: string, value: unknown, opts: { ttl: number; tags: string[] }): Promise<void>;
14
+ invalidate(opts: { key?: string; tag?: string }): Promise<void>;
15
+ }
16
+
17
+ let handler: CacheHandlerLike | null = null;
18
+
19
+ /** Replace the active cache handler. Called by the framework at boot. */
20
+ export function setCacheHandler(h: CacheHandlerLike): void {
21
+ handler = h;
22
+ }
23
+
24
+ /**
25
+ * Get the active cache handler. Creates a default MemoryCacheHandler on
26
+ * first access if none has been set via setCacheHandler().
27
+ */
28
+ export function getCacheHandler(): CacheHandlerLike {
29
+ if (!handler) {
30
+ // Inline a minimal LRU cache to avoid circular dep with index.ts.
31
+ // In production, the framework always calls setCacheHandler() at boot.
32
+ handler = createDefaultHandler();
33
+ }
34
+ return handler;
35
+ }
36
+
37
+ function createDefaultHandler(): CacheHandlerLike {
38
+ const store = new Map<string, { value: unknown; expiresAt: number; tags: string[] }>();
39
+ const maxEntries = 1000;
40
+
41
+ return {
42
+ async get(key) {
43
+ const entry = store.get(key);
44
+ if (!entry) return null;
45
+ store.delete(key);
46
+ store.set(key, entry);
47
+ const stale = Date.now() > entry.expiresAt;
48
+ return { value: entry.value, stale };
49
+ },
50
+ async set(key, value, opts) {
51
+ if (store.has(key)) store.delete(key);
52
+ while (store.size >= maxEntries) {
53
+ const oldest = store.keys().next().value;
54
+ if (oldest !== undefined) store.delete(oldest);
55
+ else break;
56
+ }
57
+ store.set(key, { value, expiresAt: Date.now() + opts.ttl * 1000, tags: opts.tags });
58
+ },
59
+ async invalidate(opts) {
60
+ if (opts.key) store.delete(opts.key);
61
+ if (opts.tag) {
62
+ for (const [key, entry] of store) {
63
+ if (entry.tags.includes(opts.tag)) store.delete(key);
64
+ }
65
+ }
66
+ },
67
+ };
68
+ }
@@ -1,5 +1,7 @@
1
1
  // @timber-js/app/cache — Caching primitives
2
2
 
3
+ import { estimateByteSize } from './sizeof.js';
4
+
3
5
  export interface CacheHandler {
4
6
  get(key: string): Promise<{ value: unknown; stale: boolean } | null>;
5
7
  set(key: string, value: unknown, opts: { ttl: number; tags: string[] }): Promise<void>;
@@ -12,19 +14,40 @@ export interface CacheOptions<Fn extends (...args: any[]) => any> {
12
14
  key?: (...args: Parameters<Fn>) => string;
13
15
  staleWhileRevalidate?: boolean;
14
16
  tags?: string[] | ((...args: Parameters<Fn>) => string[]);
17
+ /** Timeout (ms) for singleflight-coalesced calls. Prevents hung fn() from
18
+ * permanently blocking all future callers for the same cache key. See TIM-518. */
19
+ timeoutMs?: number;
15
20
  }
16
21
 
17
22
  export interface MemoryCacheHandlerOptions {
18
23
  /** Maximum number of entries. Oldest accessed entries are evicted first. Default: 1000. */
24
+ maxEntries?: number;
25
+ /**
26
+ * @deprecated Use `maxEntries` instead. Will be removed in a future release.
27
+ * Alias for `maxEntries` — maximum number of entries (not bytes).
28
+ */
19
29
  maxSize?: number;
30
+ /** Maximum total byte budget for all cached values. Oldest entries are evicted when exceeded. Default: no limit. */
31
+ maxBytes?: number;
32
+ /** Maximum byte size for a single cache entry. Entries exceeding this are silently dropped. Default: no limit. */
33
+ maxEntryBytes?: number;
20
34
  }
21
35
 
22
36
  export class MemoryCacheHandler implements CacheHandler {
23
- private store = new Map<string, { value: unknown; expiresAt: number; tags: string[] }>();
24
- private maxSize: number;
37
+ private store = new Map<
38
+ string,
39
+ { value: unknown; expiresAt: number; tags: string[]; byteSize: number }
40
+ >();
41
+ private maxEntries: number;
42
+ private maxBytes: number | undefined;
43
+ private maxEntryBytes: number | undefined;
44
+ private currentBytes = 0;
25
45
 
26
46
  constructor(opts?: MemoryCacheHandlerOptions) {
27
- this.maxSize = opts?.maxSize ?? 1000;
47
+ // maxEntries takes precedence over deprecated maxSize
48
+ this.maxEntries = opts?.maxEntries ?? opts?.maxSize ?? 1000;
49
+ this.maxBytes = opts?.maxBytes;
50
+ this.maxEntryBytes = opts?.maxEntryBytes;
28
51
  }
29
52
 
30
53
  async get(key: string) {
@@ -40,18 +63,33 @@ export class MemoryCacheHandler implements CacheHandler {
40
63
  }
41
64
 
42
65
  async set(key: string, value: unknown, opts: { ttl: number; tags: string[] }) {
43
- // If key already exists, delete first to refresh insertion order
66
+ const byteSize = estimateByteSize(value);
67
+
68
+ // Reject entries exceeding per-entry byte limit
69
+ if (this.maxEntryBytes !== undefined && byteSize > this.maxEntryBytes) {
70
+ return;
71
+ }
72
+
73
+ // If key already exists, delete first to refresh insertion order and reclaim bytes
44
74
  if (this.store.has(key)) {
75
+ const existing = this.store.get(key)!;
76
+ this.currentBytes -= existing.byteSize;
45
77
  this.store.delete(key);
46
78
  }
47
79
 
48
- // Evict oldest entries (front of Map) if at capacity
49
- while (this.store.size >= this.maxSize) {
50
- const oldest = this.store.keys().next().value;
51
- if (oldest !== undefined) {
52
- this.store.delete(oldest);
53
- } else {
54
- break;
80
+ // Evict oldest entries (front of Map) if at entry count capacity
81
+ while (this.store.size >= this.maxEntries) {
82
+ this.evictOldest();
83
+ }
84
+
85
+ // Evict oldest entries if byte budget would be exceeded
86
+ if (this.maxBytes !== undefined) {
87
+ while (this.currentBytes + byteSize > this.maxBytes && this.store.size > 0) {
88
+ this.evictOldest();
89
+ }
90
+ // If the single entry exceeds the total byte budget, don't store it
91
+ if (this.currentBytes + byteSize > this.maxBytes) {
92
+ return;
55
93
  }
56
94
  }
57
95
 
@@ -59,16 +97,23 @@ export class MemoryCacheHandler implements CacheHandler {
59
97
  value,
60
98
  expiresAt: Date.now() + opts.ttl * 1000,
61
99
  tags: opts.tags,
100
+ byteSize,
62
101
  });
102
+ this.currentBytes += byteSize;
63
103
  }
64
104
 
65
105
  async invalidate(opts: { key?: string; tag?: string }) {
66
106
  if (opts.key) {
67
- this.store.delete(opts.key);
107
+ const entry = this.store.get(opts.key);
108
+ if (entry) {
109
+ this.currentBytes -= entry.byteSize;
110
+ this.store.delete(opts.key);
111
+ }
68
112
  }
69
113
  if (opts.tag) {
70
114
  for (const [key, entry] of this.store) {
71
115
  if (entry.tags.includes(opts.tag)) {
116
+ this.currentBytes -= entry.byteSize;
72
117
  this.store.delete(key);
73
118
  }
74
119
  }
@@ -79,13 +124,31 @@ export class MemoryCacheHandler implements CacheHandler {
79
124
  get size(): number {
80
125
  return this.store.size;
81
126
  }
127
+
128
+ /** Estimated total byte size of all cached values. */
129
+ get bytes(): number {
130
+ return this.currentBytes;
131
+ }
132
+
133
+ /** Evict the oldest entry (front of Map). */
134
+ private evictOldest(): void {
135
+ const oldest = this.store.keys().next().value;
136
+ if (oldest !== undefined) {
137
+ const entry = this.store.get(oldest)!;
138
+ this.currentBytes -= entry.byteSize;
139
+ this.store.delete(oldest);
140
+ }
141
+ }
82
142
  }
83
143
 
84
144
  export { RedisCacheHandler } from './redis-handler';
85
145
  export type { RedisClient } from './redis-handler';
86
- export { createCache } from './timber-cache';
87
- export { registerCachedFunction } from './register-cached-function';
88
- export type { RegisterCachedFunctionOptions } from './register-cached-function';
89
- export { stableStringify } from './stable-stringify';
90
- export { createSingleflight } from './singleflight';
91
- export type { Singleflight } from './singleflight';
146
+ export { cache } from './cache-api';
147
+ export { setCacheHandler, getCacheHandler } from './handler-store';
148
+ // NOTE: registerCachedFunction (runtime for 'use cache' directive) removed.
149
+ // Future feature pending design doc. See design/06-caching.md.
150
+ export { estimateByteSize } from './sizeof';
151
+ // stableStringify, createSingleflight, and SingleflightTimeoutError are
152
+ // internal utilities used by the cache implementation. They are not
153
+ // re-exported here — import directly from the source files if needed
154
+ // within the package. See TIM-720.
@@ -3,24 +3,82 @@
3
3
  * execution. All callers receive the same result (or error).
4
4
  *
5
5
  * Per-process, in-memory. Each process coalesces independently.
6
+ *
7
+ * An optional `timeoutMs` prevents hung `fn()` calls from permanently
8
+ * blocking all future callers for that key. When set, `fn()` is raced
9
+ * against a timeout — if the timeout fires first, the promise rejects
10
+ * with `SingleflightTimeoutError`, `finally` cleans up the key, and
11
+ * subsequent callers can retry. See TIM-518.
6
12
  */
13
+
14
+ export interface SingleflightOptions {
15
+ /** Maximum time (ms) a coalesced call may run before being rejected. */
16
+ timeoutMs?: number;
17
+ }
18
+
7
19
  export interface Singleflight {
8
20
  do<T>(key: string, fn: () => Promise<T>): Promise<T>;
9
21
  }
10
22
 
11
- export function createSingleflight(): Singleflight {
23
+ /**
24
+ * Error thrown when a singleflight call exceeds `timeoutMs`.
25
+ * Exported so callers can distinguish timeout from other errors.
26
+ */
27
+ export class SingleflightTimeoutError extends Error {
28
+ constructor(key: string, timeoutMs: number) {
29
+ super(`Singleflight timeout: key "${key}" exceeded ${timeoutMs}ms`);
30
+ this.name = 'SingleflightTimeoutError';
31
+ }
32
+ }
33
+
34
+ export function createSingleflight(opts?: SingleflightOptions): Singleflight {
12
35
  const inflight = new Map<string, Promise<unknown>>();
36
+ const timeoutMs = opts?.timeoutMs;
13
37
 
14
38
  return {
15
39
  do<T>(key: string, fn: () => Promise<T>): Promise<T> {
16
40
  const existing = inflight.get(key);
17
41
  if (existing) return existing as Promise<T>;
18
42
 
19
- const promise = fn().finally(() => {
43
+ let promise: Promise<T>;
44
+
45
+ if (timeoutMs != null && timeoutMs > 0) {
46
+ // Race fn() against a timeout to prevent hung calls from
47
+ // permanently blocking the key. See TIM-518.
48
+ promise = new Promise<T>((resolve, reject) => {
49
+ const timer = setTimeout(
50
+ () => reject(new SingleflightTimeoutError(key, timeoutMs)),
51
+ timeoutMs
52
+ );
53
+ // Wrap in try/catch so a synchronous throw from fn()
54
+ // (e.g. argument validation) still clears the timer.
55
+ // Without this, the timer leaks until expiry.
56
+ try {
57
+ fn().then(
58
+ (value) => {
59
+ clearTimeout(timer);
60
+ resolve(value);
61
+ },
62
+ (err) => {
63
+ clearTimeout(timer);
64
+ reject(err);
65
+ }
66
+ );
67
+ } catch (err) {
68
+ clearTimeout(timer);
69
+ reject(err);
70
+ }
71
+ });
72
+ } else {
73
+ promise = fn();
74
+ }
75
+
76
+ const tracked = promise.finally(() => {
20
77
  inflight.delete(key);
21
78
  });
22
- inflight.set(key, promise);
23
- return promise;
79
+
80
+ inflight.set(key, tracked);
81
+ return tracked as Promise<T>;
24
82
  },
25
83
  };
26
84
  }
@@ -0,0 +1,31 @@
1
+ /**
2
+ * Lightweight byte-size estimation for cache entries.
3
+ *
4
+ * Estimates the in-memory byte cost of a JavaScript value using
5
+ * JSON.stringify().length * 2 (UTF-16 char width). This is a rough
6
+ * approximation — it doesn't account for V8 object overhead, Map
7
+ * metadata, or non-serializable values — but it's fast and good
8
+ * enough for cache budget enforcement.
9
+ *
10
+ * Values that fail JSON serialization (circular references, BigInt,
11
+ * etc.) return 0, allowing the entry to be cached without counting
12
+ * toward the byte budget. This is a conservative choice: it's better
13
+ * to cache and undercount than to reject the entry.
14
+ */
15
+
16
+ /**
17
+ * Estimate the byte size of a value.
18
+ *
19
+ * Uses `JSON.stringify(value).length * 2` to approximate UTF-16
20
+ * in-memory size. Returns 0 if the value is not serializable.
21
+ */
22
+ export function estimateByteSize(value: unknown): number {
23
+ try {
24
+ const json = JSON.stringify(value);
25
+ if (json === undefined) return 0;
26
+ // Each JS string character is 2 bytes (UTF-16)
27
+ return json.length * 2;
28
+ } catch {
29
+ return 0;
30
+ }
31
+ }
@@ -1,10 +1,10 @@
1
1
  import type { CacheHandler, CacheOptions } from './index';
2
2
  import { stableStringify } from './stable-stringify';
3
3
  import { createSingleflight } from './singleflight';
4
- import { addSpanEventSync } from '#/server/tracing.js';
4
+ import { addSpanEventSync } from '../server/tracing.js';
5
5
  import { fnv1aHash } from './fast-hash.js';
6
6
 
7
- const singleflight = createSingleflight();
7
+ const defaultSingleflight = createSingleflight();
8
8
 
9
9
  /**
10
10
  * Generate a cache key from function identity and serialized args.
@@ -53,10 +53,16 @@ export function createCache<Fn extends (...args: any[]) => Promise<any>>(
53
53
  fn: Fn,
54
54
  opts: CacheOptions<Fn>,
55
55
  handler: CacheHandler
56
- ): (...args: Parameters<Fn>) => Promise<Awaited<ReturnType<Fn>>> {
56
+ ): Fn {
57
57
  const fnId = `timber-cache:${fnIdCounter++}`;
58
+ const sf = opts.timeoutMs
59
+ ? createSingleflight({ timeoutMs: opts.timeoutMs })
60
+ : defaultSingleflight;
58
61
 
59
- return async (...args: Parameters<Fn>): Promise<Awaited<ReturnType<Fn>>> => {
62
+ // Cast to Fn to preserve the original function's generic call signature.
63
+ // Without this, generic type parameters (e.g. <T> in apiFetch<T>) are
64
+ // erased and callers lose type safety on the return type.
65
+ return (async (...args: Parameters<Fn>): Promise<Awaited<ReturnType<Fn>>> => {
60
66
  const key = opts.key ? opts.key(...args) : defaultKeyGenerator(fnId, args);
61
67
 
62
68
  const cacheStart = performance.now();
@@ -80,25 +86,23 @@ export function createCache<Fn extends (...args: any[]) => Promise<any>>(
80
86
  stale: true,
81
87
  });
82
88
  // Serve stale immediately, trigger background refetch
83
- singleflight
84
- .do(`swr:${key}`, async () => {
85
- try {
86
- const fresh = await fn(...args);
87
- const tags = resolveTags(opts, args);
88
- await handler.set(key, fresh, { ttl: opts.ttl, tags });
89
- } catch {
90
- // Failed refetch stale entry continues to be served.
91
- // Error is swallowed per design doc: "Error is logged."
92
- }
93
- })
94
- .catch(() => {
95
- // Singleflight promise rejection handled — stale continues.
96
- });
89
+ sf.do(`swr:${key}`, async () => {
90
+ try {
91
+ const fresh = await fn(...args);
92
+ const tags = resolveTags(opts, args);
93
+ await handler.set(key, fresh, { ttl: opts.ttl, tags });
94
+ } catch {
95
+ // Failed refetch — stale entry continues to be served.
96
+ // Error is swallowed per design doc: "Error is logged."
97
+ }
98
+ }).catch(() => {
99
+ // Singleflight promise rejection handled — stale continues.
100
+ });
97
101
  return cached.value as Awaited<ReturnType<Fn>>;
98
102
  }
99
103
 
100
104
  // Cache miss (or stale without SWR) — execute with singleflight
101
- const result = await singleflight.do(key, () => fn(...args));
105
+ const result = await sf.do(key, () => fn(...args));
102
106
  const tags = resolveTags(opts, args);
103
107
  await handler.set(key, result, { ttl: opts.ttl, tags });
104
108
 
@@ -109,7 +113,7 @@ export function createCache<Fn extends (...args: any[]) => Promise<any>>(
109
113
  });
110
114
 
111
115
  return result as Awaited<ReturnType<Fn>>;
112
- };
116
+ }) as unknown as Fn;
113
117
  }
114
118
 
115
119
  /**
package/src/cli.ts CHANGED
@@ -63,7 +63,12 @@ export interface ViteDeps {
63
63
 
64
64
  /**
65
65
  * Start the Vite dev server.
66
- * Middleware re-runs on file change via HMR wiring in timber-routing.
66
+ *
67
+ * The timber plugin binds the port immediately with a holding page
68
+ * (in rootSync's config hook) so browsers see a "starting..." page
69
+ * instead of ERR_CONNECTION_REFUSED during initialization.
70
+ *
71
+ * See design/21-dev-server.md, TIM-665.
67
72
  */
68
73
  export async function runDev(options: CommandOptions, _deps?: ViteDeps): Promise<void> {
69
74
  const createServer = _deps?.createServer ?? (await import('vite')).createServer;
@@ -1,5 +1,6 @@
1
1
  /**
2
- * Dev-only browser helpers — server log replay and client error forwarding.
2
+ * Dev-only browser helpers — server log replay, client error forwarding,
3
+ * and compiling overlay.
3
4
  *
4
5
  * These are only active when import.meta.hot is available (Vite dev mode).
5
6
  * Extracted from browser-entry.ts to keep files under 500 lines.
@@ -100,6 +101,132 @@ export function setupServerLogReplay(hot: Pick<HotInterface, 'on'>): void {
100
101
  });
101
102
  }
102
103
 
104
+ // ─── Compiling Overlay ──────────────────────────────────────────
105
+
106
+ /**
107
+ * Compiling overlay state.
108
+ *
109
+ * A small non-blocking indicator in the bottom-left corner that shows
110
+ * "Compiling..." during HMR/module updates. Uses a 200ms debounce so
111
+ * near-instant updates never flash the indicator.
112
+ *
113
+ * The overlay is lazily created on first show and reused thereafter.
114
+ * All styles are inline — no external CSS dependency.
115
+ */
116
+
117
+ let overlayEl: HTMLDivElement | null = null;
118
+ let showTimeoutId: ReturnType<typeof setTimeout> | null = null;
119
+ let pendingUpdates = 0;
120
+
121
+ /** Debounce delay before showing the overlay (ms). */
122
+ const SHOW_DELAY_MS = 200;
123
+
124
+ /**
125
+ * Get the effective debounce delay.
126
+ * Checks for a test override on `window.__timber_compiling_debounce`
127
+ * (number) to allow E2E tests to bypass the debounce.
128
+ */
129
+ function getShowDelay(): number {
130
+ const win = window as unknown as { __timber_compiling_debounce?: number };
131
+ return typeof win.__timber_compiling_debounce === 'number'
132
+ ? win.__timber_compiling_debounce
133
+ : SHOW_DELAY_MS;
134
+ }
135
+
136
+ /**
137
+ * Lazily create the overlay DOM element.
138
+ * Appended to document.body with fixed positioning in the bottom-left.
139
+ */
140
+ function getOrCreateOverlay(): HTMLDivElement {
141
+ if (overlayEl) return overlayEl;
142
+
143
+ const el = document.createElement('div');
144
+ el.setAttribute('data-timber-compiling-overlay', '');
145
+ Object.assign(el.style, {
146
+ position: 'fixed',
147
+ bottom: '16px',
148
+ left: '16px',
149
+ padding: '6px 12px',
150
+ background: 'rgba(0, 0, 0, 0.85)',
151
+ color: '#fff',
152
+ fontSize: '13px',
153
+ fontFamily:
154
+ '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif',
155
+ fontWeight: '500',
156
+ borderRadius: '6px',
157
+ zIndex: '2147483647',
158
+ pointerEvents: 'none',
159
+ opacity: '0',
160
+ transition: 'opacity 150ms ease',
161
+ boxShadow: '0 2px 8px rgba(0, 0, 0, 0.3)',
162
+ lineHeight: '1',
163
+ });
164
+ el.textContent = 'Compiling…';
165
+ document.body.appendChild(el);
166
+ overlayEl = el;
167
+ return el;
168
+ }
169
+
170
+ /**
171
+ * Signal that an HMR/module update has started.
172
+ *
173
+ * If an update is already tracked, this increments the counter.
174
+ * The overlay is shown after SHOW_DELAY_MS to avoid flashing
175
+ * for fast updates.
176
+ *
177
+ * When the delay is 0 (E2E test override), the overlay is shown
178
+ * synchronously to eliminate the race where hideCompilingOverlay()
179
+ * clears the show timeout before it fires.
180
+ */
181
+ export function showCompilingOverlay(): void {
182
+ pendingUpdates++;
183
+ if (pendingUpdates === 1) {
184
+ const delay = getShowDelay();
185
+ if (delay <= 0) {
186
+ // Synchronous show — no setTimeout race with hideCompilingOverlay().
187
+ // Used by E2E tests to guarantee the overlay is visible before
188
+ // the async HMR refresh begins its network roundtrip.
189
+ const el = getOrCreateOverlay();
190
+ el.style.opacity = '1';
191
+ } else {
192
+ // Production path — debounce to avoid flashing for fast updates
193
+ showTimeoutId = setTimeout(() => {
194
+ showTimeoutId = null;
195
+ const el = getOrCreateOverlay();
196
+ el.style.opacity = '1';
197
+ }, delay);
198
+ }
199
+ }
200
+ }
201
+
202
+ /**
203
+ * Signal that an HMR/module update has completed.
204
+ *
205
+ * When all pending updates are resolved, the overlay is hidden.
206
+ * If the overlay was never shown (update completed within the
207
+ * debounce window), the timeout is cleared — no visual flash.
208
+ *
209
+ * Test hook: if `window.__timber_hold_compiling_overlay` is true,
210
+ * the overlay stays visible. E2E tests use this to avoid a race
211
+ * where the RSC refresh completes between polling intervals,
212
+ * making the overlay impossible to observe.
213
+ */
214
+ export function hideCompilingOverlay(): void {
215
+ const win = window as unknown as { __timber_hold_compiling_overlay?: boolean };
216
+ if (win.__timber_hold_compiling_overlay) return;
217
+
218
+ pendingUpdates = Math.max(0, pendingUpdates - 1);
219
+ if (pendingUpdates === 0) {
220
+ if (showTimeoutId !== null) {
221
+ clearTimeout(showTimeoutId);
222
+ showTimeoutId = null;
223
+ }
224
+ if (overlayEl) {
225
+ overlayEl.style.opacity = '0';
226
+ }
227
+ }
228
+ }
229
+
103
230
  // ─── Client Error Forwarding ────────────────────────────────────────
104
231
 
105
232
  /**