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

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 (571) 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-BX5gL1r-.js +64 -0
  36. package/dist/_chunks/stale-reload-BX5gL1r-.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/adapters/cloudflare-dev.d.ts +109 -0
  44. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  45. package/dist/adapters/cloudflare-dev.js +73 -0
  46. package/dist/adapters/cloudflare-dev.js.map +1 -0
  47. package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
  48. package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
  49. package/dist/adapters/cloudflare-kv-cache.js +95 -0
  50. package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
  51. package/dist/adapters/cloudflare.d.ts +148 -12
  52. package/dist/adapters/cloudflare.d.ts.map +1 -1
  53. package/dist/adapters/cloudflare.js +135 -11
  54. package/dist/adapters/cloudflare.js.map +1 -1
  55. package/dist/adapters/compress-module.d.ts.map +1 -1
  56. package/dist/adapters/nitro.d.ts +17 -1
  57. package/dist/adapters/nitro.d.ts.map +1 -1
  58. package/dist/adapters/nitro.js +56 -13
  59. package/dist/adapters/nitro.js.map +1 -1
  60. package/dist/cache/cache-api.d.ts +24 -0
  61. package/dist/cache/cache-api.d.ts.map +1 -0
  62. package/dist/cache/handler-store.d.ts +31 -0
  63. package/dist/cache/handler-store.d.ts.map +1 -0
  64. package/dist/cache/index.d.ts +23 -7
  65. package/dist/cache/index.d.ts.map +1 -1
  66. package/dist/cache/index.js +142 -80
  67. package/dist/cache/index.js.map +1 -1
  68. package/dist/cache/singleflight.d.ts +18 -1
  69. package/dist/cache/singleflight.d.ts.map +1 -1
  70. package/dist/cache/sizeof.d.ts +22 -0
  71. package/dist/cache/sizeof.d.ts.map +1 -0
  72. package/dist/cache/timber-cache.d.ts +1 -1
  73. package/dist/cache/timber-cache.d.ts.map +1 -1
  74. package/dist/cli.d.ts +6 -1
  75. package/dist/cli.d.ts.map +1 -1
  76. package/dist/cli.js +8 -3
  77. package/dist/cli.js.map +1 -1
  78. package/dist/client/browser-dev.d.ts +27 -1
  79. package/dist/client/browser-dev.d.ts.map +1 -1
  80. package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
  81. package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
  82. package/dist/client/browser-entry/hmr.d.ts +21 -0
  83. package/dist/client/browser-entry/hmr.d.ts.map +1 -0
  84. package/dist/client/browser-entry/hydrate.d.ts +46 -0
  85. package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
  86. package/dist/client/browser-entry/index.d.ts +30 -0
  87. package/dist/client/browser-entry/index.d.ts.map +1 -0
  88. package/dist/client/browser-entry/post-hydration.d.ts +26 -0
  89. package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
  90. package/dist/client/browser-entry/router-init.d.ts +23 -0
  91. package/dist/client/browser-entry/router-init.d.ts.map +1 -0
  92. package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
  93. package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
  94. package/dist/client/browser-entry/scroll.d.ts +19 -0
  95. package/dist/client/browser-entry/scroll.d.ts.map +1 -0
  96. package/dist/client/error-boundary.d.ts +12 -5
  97. package/dist/client/error-boundary.d.ts.map +1 -1
  98. package/dist/client/error-boundary.js +10 -4
  99. package/dist/client/error-boundary.js.map +1 -1
  100. package/dist/client/error-reconstituter.d.ts +54 -0
  101. package/dist/client/error-reconstituter.d.ts.map +1 -0
  102. package/dist/client/form.d.ts +3 -3
  103. package/dist/client/form.d.ts.map +1 -1
  104. package/dist/client/history.d.ts +19 -4
  105. package/dist/client/history.d.ts.map +1 -1
  106. package/dist/client/index.d.ts +7 -21
  107. package/dist/client/index.d.ts.map +1 -1
  108. package/dist/client/index.js +210 -1017
  109. package/dist/client/index.js.map +1 -1
  110. package/dist/client/internal.d.ts +18 -0
  111. package/dist/client/internal.d.ts.map +1 -0
  112. package/dist/client/internal.js +890 -0
  113. package/dist/client/internal.js.map +1 -0
  114. package/dist/client/link-pending-store.d.ts +63 -0
  115. package/dist/client/link-pending-store.d.ts.map +1 -0
  116. package/dist/client/link.d.ts +90 -32
  117. package/dist/client/link.d.ts.map +1 -1
  118. package/dist/client/nav-link-store.d.ts +36 -0
  119. package/dist/client/nav-link-store.d.ts.map +1 -0
  120. package/dist/client/navigation-api-types.d.ts +90 -0
  121. package/dist/client/navigation-api-types.d.ts.map +1 -0
  122. package/dist/client/navigation-api.d.ts +115 -0
  123. package/dist/client/navigation-api.d.ts.map +1 -0
  124. package/dist/client/navigation-context.d.ts +13 -2
  125. package/dist/client/navigation-context.d.ts.map +1 -1
  126. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  127. package/dist/client/navigation-root.d.ts.map +1 -0
  128. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  129. package/dist/client/router-ref.d.ts +1 -1
  130. package/dist/client/router.d.ts +70 -4
  131. package/dist/client/router.d.ts.map +1 -1
  132. package/dist/client/rsc-fetch.d.ts +38 -3
  133. package/dist/client/rsc-fetch.d.ts.map +1 -1
  134. package/dist/client/segment-cache.d.ts +1 -1
  135. package/dist/client/segment-cache.d.ts.map +1 -1
  136. package/dist/client/segment-outlet.d.ts +63 -0
  137. package/dist/client/segment-outlet.d.ts.map +1 -0
  138. package/dist/client/ssr-data.d.ts +13 -4
  139. package/dist/client/ssr-data.d.ts.map +1 -1
  140. package/dist/client/stale-reload.d.ts +15 -0
  141. package/dist/client/stale-reload.d.ts.map +1 -1
  142. package/dist/client/top-loader.d.ts +5 -5
  143. package/dist/client/top-loader.d.ts.map +1 -1
  144. package/dist/client/use-link-status.d.ts +5 -5
  145. package/dist/client/use-link-status.d.ts.map +1 -1
  146. package/dist/client/use-params.d.ts +6 -4
  147. package/dist/client/use-params.d.ts.map +1 -1
  148. package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
  149. package/dist/client/use-pending-navigation.d.ts.map +1 -0
  150. package/dist/client/use-query-states.d.ts +1 -1
  151. package/dist/client/use-query-states.d.ts.map +1 -1
  152. package/dist/client/use-router.d.ts +1 -1
  153. package/dist/codec.d.ts +33 -0
  154. package/dist/codec.d.ts.map +1 -0
  155. package/dist/codec.js +2 -0
  156. package/dist/config-types.d.ts +227 -0
  157. package/dist/config-types.d.ts.map +1 -0
  158. package/dist/content/index.d.ts +1 -10
  159. package/dist/content/index.d.ts.map +1 -1
  160. package/dist/content/index.js +0 -2
  161. package/dist/cookies/define-cookie.d.ts +35 -14
  162. package/dist/cookies/define-cookie.d.ts.map +1 -1
  163. package/dist/cookies/index.js +1 -83
  164. package/dist/fonts/css.d.ts +1 -0
  165. package/dist/fonts/css.d.ts.map +1 -1
  166. package/dist/index.d.ts +45 -192
  167. package/dist/index.d.ts.map +1 -1
  168. package/dist/index.js +12357 -11925
  169. package/dist/index.js.map +1 -1
  170. package/dist/plugin-context.d.ts +107 -0
  171. package/dist/plugin-context.d.ts.map +1 -0
  172. package/dist/plugins/adapter-build.d.ts +1 -1
  173. package/dist/plugins/adapter-build.d.ts.map +1 -1
  174. package/dist/plugins/build-manifest.d.ts +2 -2
  175. package/dist/plugins/build-manifest.d.ts.map +1 -1
  176. package/dist/plugins/build-report.d.ts +3 -3
  177. package/dist/plugins/build-report.d.ts.map +1 -1
  178. package/dist/plugins/client-chunks.d.ts +32 -0
  179. package/dist/plugins/client-chunks.d.ts.map +1 -0
  180. package/dist/plugins/content.d.ts +1 -1
  181. package/dist/plugins/content.d.ts.map +1 -1
  182. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  183. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  184. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  185. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  186. package/dist/plugins/dev-logs.d.ts +1 -1
  187. package/dist/plugins/dev-logs.d.ts.map +1 -1
  188. package/dist/plugins/dev-server.d.ts +1 -1
  189. package/dist/plugins/dev-server.d.ts.map +1 -1
  190. package/dist/plugins/entries.d.ts +1 -1
  191. package/dist/plugins/entries.d.ts.map +1 -1
  192. package/dist/plugins/fonts.d.ts +19 -5
  193. package/dist/plugins/fonts.d.ts.map +1 -1
  194. package/dist/plugins/mdx.d.ts +1 -1
  195. package/dist/plugins/mdx.d.ts.map +1 -1
  196. package/dist/plugins/routing.d.ts +1 -1
  197. package/dist/plugins/routing.d.ts.map +1 -1
  198. package/dist/plugins/server-bundle.d.ts.map +1 -1
  199. package/dist/plugins/shims.d.ts +6 -5
  200. package/dist/plugins/shims.d.ts.map +1 -1
  201. package/dist/plugins/static-build.d.ts +4 -4
  202. package/dist/plugins/static-build.d.ts.map +1 -1
  203. package/dist/routing/codegen.d.ts +2 -2
  204. package/dist/routing/codegen.d.ts.map +1 -1
  205. package/dist/routing/index.d.ts +2 -0
  206. package/dist/routing/index.d.ts.map +1 -1
  207. package/dist/routing/index.js +3 -2
  208. package/dist/routing/scanner.d.ts.map +1 -1
  209. package/dist/routing/segment-classify.d.ts +46 -0
  210. package/dist/routing/segment-classify.d.ts.map +1 -0
  211. package/dist/routing/status-file-lint.d.ts +2 -1
  212. package/dist/routing/status-file-lint.d.ts.map +1 -1
  213. package/dist/routing/types.d.ts +16 -4
  214. package/dist/routing/types.d.ts.map +1 -1
  215. package/dist/rsc-runtime/rsc.d.ts +1 -1
  216. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  217. package/dist/rsc-runtime/ssr.d.ts +12 -0
  218. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  219. package/dist/schema-bridge.d.ts +76 -0
  220. package/dist/schema-bridge.d.ts.map +1 -0
  221. package/dist/search-params/define.d.ts +139 -0
  222. package/dist/search-params/define.d.ts.map +1 -0
  223. package/dist/search-params/index.d.ts +4 -7
  224. package/dist/search-params/index.d.ts.map +1 -1
  225. package/dist/search-params/index.js +32 -441
  226. package/dist/search-params/index.js.map +1 -1
  227. package/dist/search-params/registry.d.ts +2 -2
  228. package/dist/search-params/registry.d.ts.map +1 -1
  229. package/dist/search-params/wrappers.d.ts +53 -0
  230. package/dist/search-params/wrappers.d.ts.map +1 -0
  231. package/dist/segment-params/define.d.ts +78 -0
  232. package/dist/segment-params/define.d.ts.map +1 -0
  233. package/dist/segment-params/index.d.ts +3 -0
  234. package/dist/segment-params/index.d.ts.map +1 -0
  235. package/dist/segment-params/index.js +2 -0
  236. package/dist/server/access-gate.d.ts +4 -0
  237. package/dist/server/access-gate.d.ts.map +1 -1
  238. package/dist/server/action-client.d.ts +25 -6
  239. package/dist/server/action-client.d.ts.map +1 -1
  240. package/dist/server/action-encryption.d.ts +76 -0
  241. package/dist/server/action-encryption.d.ts.map +1 -0
  242. package/dist/server/action-handler.d.ts.map +1 -1
  243. package/dist/server/actions.d.ts +3 -6
  244. package/dist/server/actions.d.ts.map +1 -1
  245. package/dist/server/als-registry.d.ts +32 -4
  246. package/dist/server/als-registry.d.ts.map +1 -1
  247. package/dist/server/build-manifest.d.ts +2 -2
  248. package/dist/server/build-manifest.d.ts.map +1 -1
  249. package/dist/server/debug.d.ts +1 -1
  250. package/dist/server/default-logger.d.ts +22 -0
  251. package/dist/server/default-logger.d.ts.map +1 -0
  252. package/dist/server/deny-page-resolver.d.ts +52 -0
  253. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  254. package/dist/server/deny-renderer.d.ts.map +1 -1
  255. package/dist/server/dev-holding-server.d.ts +52 -0
  256. package/dist/server/dev-holding-server.d.ts.map +1 -0
  257. package/dist/server/dev-warnings.d.ts +1 -21
  258. package/dist/server/dev-warnings.d.ts.map +1 -1
  259. package/dist/server/early-hints.d.ts +13 -5
  260. package/dist/server/early-hints.d.ts.map +1 -1
  261. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  262. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  263. package/dist/server/fallback-error.d.ts +4 -3
  264. package/dist/server/fallback-error.d.ts.map +1 -1
  265. package/dist/server/flight-injection-state.d.ts +66 -0
  266. package/dist/server/flight-injection-state.d.ts.map +1 -0
  267. package/dist/server/flight-scripts.d.ts +42 -0
  268. package/dist/server/flight-scripts.d.ts.map +1 -0
  269. package/dist/server/flush.d.ts.map +1 -1
  270. package/dist/server/form-data.d.ts +29 -0
  271. package/dist/server/form-data.d.ts.map +1 -1
  272. package/dist/server/html-injectors.d.ts +51 -11
  273. package/dist/server/html-injectors.d.ts.map +1 -1
  274. package/dist/server/index.d.ts +5 -43
  275. package/dist/server/index.d.ts.map +1 -1
  276. package/dist/server/index.js +37 -2798
  277. package/dist/server/index.js.map +1 -1
  278. package/dist/server/internal.d.ts +46 -0
  279. package/dist/server/internal.d.ts.map +1 -0
  280. package/dist/server/internal.js +2883 -0
  281. package/dist/server/internal.js.map +1 -0
  282. package/dist/server/logger.d.ts +25 -7
  283. package/dist/server/logger.d.ts.map +1 -1
  284. package/dist/server/middleware-runner.d.ts +19 -4
  285. package/dist/server/middleware-runner.d.ts.map +1 -1
  286. package/dist/server/node-stream-transforms.d.ts +113 -0
  287. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  288. package/dist/server/page-deny-boundary.d.ts +31 -0
  289. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  290. package/dist/server/pipeline-interception.d.ts +1 -1
  291. package/dist/server/pipeline-interception.d.ts.map +1 -1
  292. package/dist/server/pipeline-metadata.d.ts +6 -0
  293. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  294. package/dist/server/pipeline.d.ts +42 -10
  295. package/dist/server/pipeline.d.ts.map +1 -1
  296. package/dist/server/primitives.d.ts +69 -18
  297. package/dist/server/primitives.d.ts.map +1 -1
  298. package/dist/server/render-timeout.d.ts +51 -0
  299. package/dist/server/render-timeout.d.ts.map +1 -0
  300. package/dist/server/request-context.d.ts +112 -43
  301. package/dist/server/request-context.d.ts.map +1 -1
  302. package/dist/server/route-element-builder.d.ts +27 -1
  303. package/dist/server/route-element-builder.d.ts.map +1 -1
  304. package/dist/server/route-handler.d.ts.map +1 -1
  305. package/dist/server/route-matcher.d.ts +9 -2
  306. package/dist/server/route-matcher.d.ts.map +1 -1
  307. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  308. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  309. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  310. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  311. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  312. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  313. package/dist/server/rsc-entry/index.d.ts +8 -3
  314. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  315. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  316. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  317. package/dist/server/rsc-entry/rsc-stream.d.ts +4 -1
  318. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  319. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  320. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  321. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  322. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  323. package/dist/server/safe-load.d.ts +46 -0
  324. package/dist/server/safe-load.d.ts.map +1 -0
  325. package/dist/server/sitemap-generator.d.ts +129 -0
  326. package/dist/server/sitemap-generator.d.ts.map +1 -0
  327. package/dist/server/sitemap-handler.d.ts +22 -0
  328. package/dist/server/sitemap-handler.d.ts.map +1 -0
  329. package/dist/server/slot-resolver.d.ts +1 -1
  330. package/dist/server/slot-resolver.d.ts.map +1 -1
  331. package/dist/server/ssr-entry.d.ts +22 -0
  332. package/dist/server/ssr-entry.d.ts.map +1 -1
  333. package/dist/server/ssr-render.d.ts +39 -21
  334. package/dist/server/ssr-render.d.ts.map +1 -1
  335. package/dist/server/ssr-wrappers.d.ts +50 -0
  336. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  337. package/dist/server/status-code-resolver.d.ts +1 -1
  338. package/dist/server/status-code-resolver.d.ts.map +1 -1
  339. package/dist/server/stream-utils.d.ts +36 -0
  340. package/dist/server/stream-utils.d.ts.map +1 -0
  341. package/dist/server/tracing.d.ts +4 -4
  342. package/dist/server/tracing.d.ts.map +1 -1
  343. package/dist/server/tree-builder.d.ts +22 -19
  344. package/dist/server/tree-builder.d.ts.map +1 -1
  345. package/dist/server/types.d.ts +1 -4
  346. package/dist/server/types.d.ts.map +1 -1
  347. package/dist/server/version-skew.d.ts +61 -0
  348. package/dist/server/version-skew.d.ts.map +1 -0
  349. package/dist/shared/merge-search-params.d.ts +22 -0
  350. package/dist/shared/merge-search-params.d.ts.map +1 -0
  351. package/dist/shims/font-google.d.ts +1 -1
  352. package/dist/shims/font-google.d.ts.map +1 -1
  353. package/dist/shims/font-google.js +42 -0
  354. package/dist/shims/font-google.js.map +1 -0
  355. package/dist/shims/font-local.d.ts +26 -0
  356. package/dist/shims/font-local.d.ts.map +1 -0
  357. package/dist/shims/font-local.js +20 -0
  358. package/dist/shims/font-local.js.map +1 -0
  359. package/dist/shims/headers.d.ts +2 -1
  360. package/dist/shims/headers.d.ts.map +1 -1
  361. package/dist/shims/navigation-client.d.ts +1 -1
  362. package/dist/shims/navigation-client.d.ts.map +1 -1
  363. package/dist/shims/navigation.d.ts +3 -2
  364. package/dist/shims/navigation.d.ts.map +1 -1
  365. package/dist/utils/directive-parser.d.ts +5 -2
  366. package/dist/utils/directive-parser.d.ts.map +1 -1
  367. package/dist/utils/state-machine.d.ts +80 -0
  368. package/dist/utils/state-machine.d.ts.map +1 -0
  369. package/package.json +56 -22
  370. package/src/adapters/cloudflare-dev.ts +177 -0
  371. package/src/adapters/cloudflare-kv-cache.ts +142 -0
  372. package/src/adapters/cloudflare.ts +342 -28
  373. package/src/adapters/compress-module.ts +24 -4
  374. package/src/adapters/nitro.ts +52 -8
  375. package/src/adapters/wrangler.d.ts +7 -0
  376. package/src/cache/cache-api.ts +38 -0
  377. package/src/cache/handler-store.ts +68 -0
  378. package/src/cache/index.ts +81 -18
  379. package/src/cache/singleflight.ts +62 -4
  380. package/src/cache/sizeof.ts +31 -0
  381. package/src/cache/timber-cache.ts +24 -20
  382. package/src/cli.ts +16 -6
  383. package/src/client/browser-dev.ts +128 -1
  384. package/src/client/browser-entry/action-dispatch.ts +116 -0
  385. package/src/client/browser-entry/hmr.ts +81 -0
  386. package/src/client/browser-entry/hydrate.ts +145 -0
  387. package/src/client/browser-entry/index.ts +138 -0
  388. package/src/client/browser-entry/post-hydration.ts +119 -0
  389. package/src/client/browser-entry/router-init.ts +193 -0
  390. package/src/client/browser-entry/rsc-stream.ts +157 -0
  391. package/src/client/browser-entry/scroll.ts +27 -0
  392. package/src/client/error-boundary.tsx +48 -16
  393. package/src/client/error-reconstituter.tsx +65 -0
  394. package/src/client/form.tsx +9 -7
  395. package/src/client/history.ts +26 -4
  396. package/src/client/index.ts +19 -38
  397. package/src/client/internal.ts +57 -0
  398. package/src/client/link-pending-store.ts +111 -0
  399. package/src/client/link.tsx +329 -97
  400. package/src/client/nav-link-store.ts +47 -0
  401. package/src/client/navigation-api-types.ts +112 -0
  402. package/src/client/navigation-api.ts +332 -0
  403. package/src/client/navigation-context.ts +31 -6
  404. package/src/client/navigation-root.tsx +342 -0
  405. package/src/client/nuqs-adapter.tsx +16 -3
  406. package/src/client/router-ref.ts +1 -1
  407. package/src/client/router.ts +299 -72
  408. package/src/client/rsc-fetch.ts +97 -8
  409. package/src/client/segment-cache.ts +1 -1
  410. package/src/client/segment-outlet.tsx +86 -0
  411. package/src/client/ssr-data.ts +13 -5
  412. package/src/client/stale-reload.ts +72 -3
  413. package/src/client/top-loader.tsx +16 -8
  414. package/src/client/use-link-status.ts +7 -7
  415. package/src/client/use-params.ts +7 -5
  416. package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
  417. package/src/client/use-query-states.ts +3 -3
  418. package/src/client/use-router.ts +1 -1
  419. package/src/codec.ts +49 -0
  420. package/src/config-types.ts +225 -0
  421. package/src/content/index.ts +5 -13
  422. package/src/cookies/define-cookie.ts +78 -25
  423. package/src/cookies/index.ts +8 -0
  424. package/src/fonts/css.ts +2 -1
  425. package/src/index.ts +295 -354
  426. package/src/plugin-context.ts +240 -0
  427. package/src/plugins/adapter-build.ts +9 -3
  428. package/src/plugins/build-manifest.ts +13 -2
  429. package/src/plugins/build-report.ts +3 -3
  430. package/src/plugins/client-chunks.ts +65 -0
  431. package/src/plugins/content.ts +1 -1
  432. package/src/plugins/dev-browser-logs.ts +288 -0
  433. package/src/plugins/dev-error-overlay.ts +70 -1
  434. package/src/plugins/dev-logs.ts +1 -1
  435. package/src/plugins/dev-server.ts +70 -9
  436. package/src/plugins/entries.ts +71 -10
  437. package/src/plugins/fonts.ts +168 -61
  438. package/src/plugins/mdx.ts +1 -1
  439. package/src/plugins/routing.ts +57 -17
  440. package/src/plugins/server-action-exports.ts +1 -1
  441. package/src/plugins/server-bundle.ts +32 -1
  442. package/src/plugins/shims.ts +135 -35
  443. package/src/plugins/static-build.ts +17 -11
  444. package/src/routing/codegen.ts +165 -105
  445. package/src/routing/index.ts +2 -0
  446. package/src/routing/scanner.ts +93 -23
  447. package/src/routing/segment-classify.ts +89 -0
  448. package/src/routing/status-file-lint.ts +3 -2
  449. package/src/routing/types.ts +17 -4
  450. package/src/rsc-runtime/rsc.ts +2 -0
  451. package/src/rsc-runtime/ssr.ts +50 -0
  452. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  453. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  454. package/src/search-params/define.ts +482 -0
  455. package/src/search-params/index.ts +14 -20
  456. package/src/search-params/registry.ts +2 -2
  457. package/src/search-params/wrappers.ts +85 -0
  458. package/src/segment-params/define.ts +279 -0
  459. package/src/segment-params/index.ts +9 -0
  460. package/src/server/access-gate.tsx +70 -29
  461. package/src/server/action-client.ts +46 -11
  462. package/src/server/action-encryption.ts +144 -0
  463. package/src/server/action-handler.ts +21 -4
  464. package/src/server/actions.ts +10 -9
  465. package/src/server/als-registry.ts +34 -6
  466. package/src/server/build-manifest.ts +10 -4
  467. package/src/server/compress.ts +25 -7
  468. package/src/server/debug.ts +1 -1
  469. package/src/server/default-logger.ts +99 -0
  470. package/src/server/deny-page-resolver.ts +154 -0
  471. package/src/server/deny-renderer.ts +24 -38
  472. package/src/server/dev-holding-server.ts +185 -0
  473. package/src/server/dev-warnings.ts +4 -49
  474. package/src/server/early-hints.ts +36 -15
  475. package/src/server/error-boundary-wrapper.ts +74 -22
  476. package/src/server/fallback-error.ts +31 -15
  477. package/src/server/flight-injection-state.ts +113 -0
  478. package/src/server/flight-scripts.ts +62 -0
  479. package/src/server/flush.ts +2 -1
  480. package/src/server/form-data.ts +76 -0
  481. package/src/server/html-injectors.ts +280 -120
  482. package/src/server/index.ts +25 -177
  483. package/src/server/internal.ts +169 -0
  484. package/src/server/logger.ts +44 -36
  485. package/src/server/middleware-runner.ts +31 -4
  486. package/src/server/node-stream-transforms.ts +509 -0
  487. package/src/server/page-deny-boundary.tsx +56 -0
  488. package/src/server/pipeline-interception.ts +17 -16
  489. package/src/server/pipeline-metadata.ts +13 -0
  490. package/src/server/pipeline.ts +227 -62
  491. package/src/server/primitives.ts +111 -28
  492. package/src/server/render-timeout.ts +108 -0
  493. package/src/server/request-context.ts +293 -132
  494. package/src/server/route-element-builder.ts +283 -191
  495. package/src/server/route-handler.ts +24 -4
  496. package/src/server/route-matcher.ts +24 -20
  497. package/src/server/rsc-entry/api-handler.ts +15 -16
  498. package/src/server/rsc-entry/error-renderer.ts +300 -89
  499. package/src/server/rsc-entry/helpers.ts +134 -5
  500. package/src/server/rsc-entry/index.ts +200 -112
  501. package/src/server/rsc-entry/rsc-payload.ts +65 -18
  502. package/src/server/rsc-entry/rsc-stream.ts +65 -13
  503. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  504. package/src/server/rsc-entry/ssr-renderer.ts +168 -38
  505. package/src/server/safe-load.ts +60 -0
  506. package/src/server/sitemap-generator.ts +338 -0
  507. package/src/server/sitemap-handler.ts +126 -0
  508. package/src/server/slot-resolver.ts +244 -229
  509. package/src/server/ssr-entry.ts +211 -32
  510. package/src/server/ssr-render.ts +289 -67
  511. package/src/server/ssr-wrappers.tsx +139 -0
  512. package/src/server/status-code-resolver.ts +1 -1
  513. package/src/server/stream-utils.ts +213 -0
  514. package/src/server/tracing.ts +20 -9
  515. package/src/server/tree-builder.ts +92 -58
  516. package/src/server/types.ts +3 -6
  517. package/src/server/version-skew.ts +104 -0
  518. package/src/shared/merge-search-params.ts +55 -0
  519. package/src/shims/font-google.ts +1 -1
  520. package/src/shims/font-local.ts +34 -0
  521. package/src/shims/headers.ts +5 -1
  522. package/src/shims/navigation-client.ts +1 -1
  523. package/src/shims/navigation.ts +7 -2
  524. package/src/utils/directive-parser.ts +5 -2
  525. package/src/utils/state-machine.ts +111 -0
  526. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  527. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  528. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  529. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  530. package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
  531. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  532. package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
  533. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  534. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  535. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  536. package/dist/cache/register-cached-function.d.ts +0 -17
  537. package/dist/cache/register-cached-function.d.ts.map +0 -1
  538. package/dist/client/browser-entry.d.ts +0 -21
  539. package/dist/client/browser-entry.d.ts.map +0 -1
  540. package/dist/client/link-status-provider.d.ts +0 -11
  541. package/dist/client/link-status-provider.d.ts.map +0 -1
  542. package/dist/client/transition-root.d.ts.map +0 -1
  543. package/dist/client/use-navigation-pending.d.ts.map +0 -1
  544. package/dist/cookies/index.js.map +0 -1
  545. package/dist/plugins/cache-transform.d.ts +0 -36
  546. package/dist/plugins/cache-transform.d.ts.map +0 -1
  547. package/dist/plugins/dynamic-transform.d.ts +0 -72
  548. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  549. package/dist/search-params/analyze.d.ts +0 -54
  550. package/dist/search-params/analyze.d.ts.map +0 -1
  551. package/dist/search-params/builtin-codecs.d.ts +0 -105
  552. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  553. package/dist/search-params/codecs.d.ts +0 -53
  554. package/dist/search-params/codecs.d.ts.map +0 -1
  555. package/dist/search-params/create.d.ts +0 -106
  556. package/dist/search-params/create.d.ts.map +0 -1
  557. package/dist/server/prerender.d.ts +0 -77
  558. package/dist/server/prerender.d.ts.map +0 -1
  559. package/dist/server/response-cache.d.ts +0 -54
  560. package/dist/server/response-cache.d.ts.map +0 -1
  561. package/src/cache/register-cached-function.ts +0 -103
  562. package/src/client/browser-entry.ts +0 -678
  563. package/src/client/link-status-provider.tsx +0 -30
  564. package/src/client/transition-root.tsx +0 -166
  565. package/src/plugins/cache-transform.ts +0 -199
  566. package/src/plugins/dynamic-transform.ts +0 -161
  567. package/src/search-params/analyze.ts +0 -192
  568. package/src/search-params/builtin-codecs.ts +0 -228
  569. package/src/search-params/create.ts +0 -321
  570. package/src/server/prerender.ts +0 -139
  571. package/src/server/response-cache.ts +0 -410
@@ -0,0 +1,116 @@
1
+ /**
2
+ * Server Action Dispatch — registers the callServer callback.
3
+ *
4
+ * When React encounters a server reference (from `'use server'` modules),
5
+ * it calls `callServer(id, args)` to dispatch the action to the server.
6
+ * The RSC plugin delegates to `globalThis.__viteRscCallServer` which is
7
+ * set by `setServerCallback`.
8
+ *
9
+ * The callback:
10
+ * 1. Serializes args via `encodeReply` (RSC wire format)
11
+ * 2. POSTs to the current URL with `Accept: text/x-component`
12
+ * 3. Decodes the RSC response stream
13
+ *
14
+ * See design/08-forms-and-actions.md §"Client-Side Form Mechanics"
15
+ */
16
+
17
+ import { setServerCallback, encodeReply, createFromFetch } from '../../rsc-runtime/browser.js';
18
+ import { getRouter } from '#client-internal';
19
+ import { setHardNavigating } from '../navigation-root.js';
20
+ import { isStaleClientReference, triggerStaleReload } from '../stale-reload.js';
21
+ import { getClientDeploymentId, DEPLOYMENT_ID_HEADER, RELOAD_HEADER } from '../rsc-fetch.js';
22
+
23
+ export function setupServerActions(): void {
24
+ setServerCallback(async (id: string, args: unknown[]) => {
25
+ const body = await encodeReply(args);
26
+
27
+ // Track the X-Timber-Revalidation header from the response.
28
+ // We intercept the fetch promise to read headers before createFromFetch
29
+ // consumes the body stream.
30
+ let hasRevalidation = false;
31
+ let hasRedirect = false;
32
+ let headElementsJson: string | null = null;
33
+
34
+ // Build action request headers. Include deployment ID for version
35
+ // skew detection (TIM-446) — the server rejects stale actions gracefully.
36
+ const actionHeaders: Record<string, string> = {
37
+ 'Accept': 'text/x-component',
38
+ 'x-rsc-action': id,
39
+ };
40
+ const actionDeploymentId = getClientDeploymentId();
41
+ if (actionDeploymentId) {
42
+ actionHeaders[DEPLOYMENT_ID_HEADER] = actionDeploymentId;
43
+ }
44
+
45
+ const response = fetch(window.location.href, {
46
+ method: 'POST',
47
+ headers: actionHeaders,
48
+ body,
49
+ }).then((res) => {
50
+ // Version skew detection (TIM-446): if the server signals a reload,
51
+ // trigger a full page load to pick up the new deployment.
52
+ if (res.headers.get(RELOAD_HEADER) === '1') {
53
+ window.location.reload();
54
+ throw new Error('Version skew detected — reloading page');
55
+ }
56
+ hasRevalidation = res.headers.get('X-Timber-Revalidation') === '1';
57
+ hasRedirect = res.headers.get('X-Timber-Redirect') != null;
58
+ headElementsJson = res.headers.get('X-Timber-Head');
59
+ return res;
60
+ });
61
+
62
+ let decoded: unknown;
63
+ try {
64
+ decoded = await createFromFetch(response);
65
+ } catch (error) {
66
+ if (isStaleClientReference(error)) {
67
+ triggerStaleReload();
68
+ // Return a never-resolving promise to prevent further processing
69
+ return new Promise(() => {});
70
+ }
71
+ throw error;
72
+ }
73
+
74
+ // Handle redirect — server encoded the redirect location in the RSC stream
75
+ // instead of returning HTTP 302. Perform a client-side SPA navigation.
76
+ if (hasRedirect) {
77
+ const wrapper = decoded as { _redirect: string; _status: number };
78
+ try {
79
+ const router = getRouter();
80
+ void router.navigate(wrapper._redirect);
81
+ } catch {
82
+ // Router not yet initialized — fall back to full navigation.
83
+ // Set hard-navigating flag to prevent Navigation API interception
84
+ // and React from rendering during page teardown. See TIM-626.
85
+ setHardNavigating(true);
86
+ window.location.href = wrapper._redirect;
87
+ }
88
+ return undefined;
89
+ }
90
+
91
+ if (hasRevalidation) {
92
+ // Piggybacked response: wrapper object { _action, _tree }
93
+ // Apply the revalidated tree directly — no separate router.refresh() needed.
94
+ const wrapper = decoded as { _action: unknown; _tree: unknown };
95
+ try {
96
+ const router = getRouter();
97
+ const headElements = headElementsJson ? JSON.parse(headElementsJson) : null;
98
+ router.applyRevalidation(wrapper._tree, headElements);
99
+ } catch {
100
+ // Router not yet initialized — fall through
101
+ }
102
+ return wrapper._action;
103
+ }
104
+
105
+ // No piggybacked revalidation — refresh to pick up any mutations.
106
+ // This covers actions that don't call revalidatePath().
107
+ try {
108
+ const router = getRouter();
109
+ void router.refresh();
110
+ } catch {
111
+ // Router not yet initialized (rare edge case during bootstrap)
112
+ }
113
+
114
+ return decoded;
115
+ });
116
+ }
@@ -0,0 +1,81 @@
1
+ /**
2
+ * HMR & Dev Tooling — dev-only wiring for hot module replacement.
3
+ *
4
+ * Handles:
5
+ * - RSC module invalidation (rsc:update → router.refresh())
6
+ * - Dev warnings forwarded from server via WebSocket
7
+ * - Server console log replay in browser
8
+ * - Client error forwarding to Vite error overlay
9
+ *
10
+ * See design/21-dev-server.md §"HMR Wiring"
11
+ */
12
+
13
+ import type { RouterInstance } from '#client-internal';
14
+ import {
15
+ setupServerLogReplay,
16
+ setupClientErrorForwarding,
17
+ showCompilingOverlay,
18
+ hideCompilingOverlay,
19
+ } from '../browser-dev.js';
20
+
21
+ type HotApi = {
22
+ on(event: string, cb: (...args: unknown[]) => void): void;
23
+ send(event: string, data: unknown): void;
24
+ };
25
+
26
+ /**
27
+ * Set up HMR and dev tool integration.
28
+ *
29
+ * Reads `import.meta.hot` to register dev-only event listeners.
30
+ * In production builds, `import.meta.hot` is undefined and this
31
+ * function is a no-op.
32
+ */
33
+ export function setupHmr(router: RouterInstance): void {
34
+ // Vite injects import.meta.hot in dev mode. Cast to access it without
35
+ // requiring vite/client types in the package tsconfig.
36
+ const hot = (import.meta as unknown as { hot?: HotApi }).hot;
37
+ if (!hot) return;
38
+
39
+ // ─── Compiling overlay: Vite client HMR updates ──────────────
40
+ // vite:beforeUpdate fires when Vite begins applying client-side
41
+ // module updates (React Fast Refresh, CSS hot update, etc.).
42
+ // vite:afterUpdate fires once all updates are applied.
43
+ hot.on('vite:beforeUpdate', () => {
44
+ showCompilingOverlay();
45
+ });
46
+ hot.on('vite:afterUpdate', () => {
47
+ hideCompilingOverlay();
48
+ });
49
+
50
+ // ─── RSC module invalidation ─────────────────────────────────
51
+ // rsc:update fires when a server component is edited.
52
+ // Show the compiling overlay while the RSC refresh is in flight.
53
+ hot.on('rsc:update', () => {
54
+ showCompilingOverlay();
55
+ void router.refresh().finally(() => {
56
+ hideCompilingOverlay();
57
+ });
58
+ });
59
+
60
+ // Listen for dev warnings forwarded from the server via WebSocket.
61
+ // See dev-warnings.ts — emitOnce() sends these via server.hot.send().
62
+ hot.on('timber:dev-warning', (data: unknown) => {
63
+ const warning = data as { level: string; message: string };
64
+ if (warning.level === 'error') {
65
+ console.error(warning.message);
66
+ } else {
67
+ console.warn(warning.message);
68
+ }
69
+ });
70
+
71
+ // Listen for server console logs forwarded via WebSocket.
72
+ // Replays them in the browser console with a [SERVER] prefix
73
+ // so developers can see server output without switching to the terminal.
74
+ // See plugins/dev-logs.ts.
75
+ setupServerLogReplay(hot);
76
+
77
+ // Forward uncaught client errors to the server for the dev overlay.
78
+ // The server source-maps the stack and sends it back via Vite's
79
+ // error overlay protocol. See dev-server.ts §client error listener.
80
+ setupClientErrorForwarding(hot);
81
+ }
@@ -0,0 +1,145 @@
1
+ /**
2
+ * Hydration — pre-hydration sequence and React root creation.
3
+ *
4
+ * Handles two paths:
5
+ * 1. RSC payload available: hydrateRoot with NavigationProvider wrapping
6
+ * 2. No RSC payload: deferred root creation via installDeferredNavigation
7
+ *
8
+ * Pre-hydration ordering contract (MUST execute in this order):
9
+ * 1. initRouter() — creates the global router so useRouter()
10
+ * works during render
11
+ * 2. setCurrentParams() — populates params snapshot so
12
+ * + setNavigationState() useSegmentParams() and usePathname()
13
+ * return correct values during hydration
14
+ * 3. hydrateRoot() — synchronously executes component render
15
+ * functions that depend on steps 1-2
16
+ *
17
+ * See design/19-client-navigation.md §"NavigationContext"
18
+ */
19
+
20
+ import { createElement } from 'react';
21
+ import { hydrateRoot, createRoot } from 'react-dom/client';
22
+ import { getRouterOrNull, setCurrentParams } from '#client-internal';
23
+ import {
24
+ NavigationProvider,
25
+ getNavigationState,
26
+ setNavigationState,
27
+ } from '../navigation-context.js';
28
+ import { TimberNuqsAdapter } from '../nuqs-adapter.js';
29
+ import { isPageUnloading } from '../unload-guard.js';
30
+ import { NavigationRoot, installDeferredNavigation } from '../navigation-root.js';
31
+ import type { TopLoaderConfig } from '../top-loader.js';
32
+
33
+ import type { RscStreamResult } from './rsc-stream.js';
34
+
35
+ interface HydrateOptions {
36
+ /** RSC stream result (null if no inlined payload) */
37
+ rscResult: RscStreamResult | null;
38
+ /** Runtime config from virtual:timber-config */
39
+ config: { topLoader?: TopLoaderConfig };
40
+ }
41
+
42
+ /**
43
+ * Run the pre-hydration sequence: read server-embedded params and
44
+ * set navigation state. Must be called AFTER initRouter().
45
+ */
46
+ export function runPreHydration(): void {
47
+ const earlyParams = (self as unknown as Record<string, unknown>).__timber_params;
48
+ if (earlyParams && typeof earlyParams === 'object') {
49
+ setCurrentParams(earlyParams as Record<string, string | string[]>);
50
+ setNavigationState({
51
+ params: earlyParams as Record<string, string | string[]>,
52
+ pathname: window.location.pathname,
53
+ });
54
+ delete (self as unknown as Record<string, unknown>).__timber_params;
55
+ } else {
56
+ setNavigationState({
57
+ params: {},
58
+ pathname: window.location.pathname,
59
+ });
60
+ }
61
+ }
62
+
63
+ /**
64
+ * Hydrate the React tree or set up deferred root creation.
65
+ *
66
+ * When an RSC payload is available, wraps it with NavigationProvider +
67
+ * TimberNuqsAdapter + NavigationRoot and calls hydrateRoot on the
68
+ * document.
69
+ *
70
+ * When no RSC payload is available (JS-only client), sets up deferred
71
+ * navigation so the first client navigation creates the React root.
72
+ */
73
+ export function hydrateApp({ rscResult, config }: HydrateOptions): void {
74
+ if (rscResult) {
75
+ const element = rscResult.element;
76
+
77
+ // Wrap with NavigationProvider (for atomic useParams/usePathname),
78
+ // TimberNuqsAdapter (for nuqs context), and NavigationRoot (for
79
+ // transition-based rendering during client navigation).
80
+ //
81
+ // NavigationRoot holds the element in React state and updates via
82
+ // startTransition, so React keeps old UI visible while new Suspense
83
+ // boundaries resolve during navigation. See design/05-streaming.md.
84
+ const navState = getNavigationState();
85
+ const withNav = createElement(
86
+ NavigationProvider,
87
+ { value: navState },
88
+ element as React.ReactNode
89
+ );
90
+ const wrapped = createElement(TimberNuqsAdapter, null, withNav);
91
+ const rootElement = createElement(NavigationRoot, {
92
+ initial: wrapped,
93
+ topLoaderConfig: config.topLoader,
94
+ });
95
+
96
+ if (process.env.NODE_ENV !== 'production') {
97
+ if (!getRouterOrNull()) {
98
+ throw new Error(
99
+ '[timber] hydrateRoot called before initRouter() — bootstrap order violated'
100
+ );
101
+ }
102
+ }
103
+
104
+ hydrateRoot(document, rootElement, {
105
+ // Suppress recoverable hydration errors from deny/error signals
106
+ // inside Suspense boundaries. The server already handled these
107
+ // (wrapStreamWithErrorHandling closes the stream cleanly after
108
+ // the shell is flushed). React replays the error during hydration
109
+ // but the server HTML is already correct — no recovery needed.
110
+ onRecoverableError(error: unknown) {
111
+ // Suppress errors during page unload (refresh/navigate away).
112
+ // The aborted stream causes incomplete HTML which React flags
113
+ // as a recoverable error — but the page is being replaced.
114
+ if (isPageUnloading()) return;
115
+ // Only log in dev — in production these are expected for
116
+ // deny() inside Suspense and streaming error boundaries.
117
+ if (process.env.NODE_ENV === 'development') {
118
+ console.debug('[timber] Hydration recoverable error:', error);
119
+ }
120
+ },
121
+ });
122
+ } else {
123
+ // No RSC payload available — defer React root creation until the
124
+ // first client navigation (TIM-600).
125
+ //
126
+ // We must NOT call createRoot(document).render() here — that would
127
+ // take React ownership of the entire document and blank the SSR HTML.
128
+ // Instead, installDeferredNavigation sets up one-shot callbacks so
129
+ // the first navigateTransition/transitionRender call creates the root
130
+ // on `document` with the navigated content. After that initial render,
131
+ // NavigationRoot's real startTransition-based callbacks take over.
132
+ //
133
+ // This also fixes TIM-580 (navigation from SSR-only pages) because
134
+ // the deferred callbacks ensure NavigationRoot is mounted before the
135
+ // first navigation completes.
136
+ installDeferredNavigation((initial) => {
137
+ const rootElement = createElement(NavigationRoot, {
138
+ initial,
139
+ topLoaderConfig: config.topLoader,
140
+ });
141
+ const root = createRoot(document);
142
+ root.render(rootElement);
143
+ });
144
+ }
145
+ }
@@ -0,0 +1,138 @@
1
+ /**
2
+ * Browser Entry — Client-side hydration and navigation bootstrap.
3
+ *
4
+ * This is the thin orchestrator that coordinates the bootstrap sequence.
5
+ * Each responsibility is extracted into a focused module:
6
+ *
7
+ * action-dispatch.ts — server action callServer callback
8
+ * rsc-stream.ts — __timber_f chunk handling + ReadableStream
9
+ * router-init.ts — createRouter + Navigation API setup
10
+ * hydrate.ts — pre-hydration sequence + hydrateRoot/createRoot
11
+ * post-hydration.ts — history stack, segment cache, popstate, scroll
12
+ * hmr.ts — dev-only HMR + error forwarding
13
+ * scroll.ts — getScrollY helper
14
+ *
15
+ * Bootstrap call order contract:
16
+ *
17
+ * 1. setupServerActions() — register callServer (independent)
18
+ * 2. createRscPayloadStream() — decode inlined RSC payload
19
+ * 3. createTimberRouter() — create router + Navigation API
20
+ * 4. runPreHydration() — set params + navigation state
21
+ * 5. hydrateApp() — hydrateRoot or deferred createRoot
22
+ * 6. setupPostHydration() — history stack, popstate, scroll
23
+ * 7. setupHmr() — dev-only HMR wiring
24
+ * 8. stale reload handlers — global error listeners
25
+ * 9. timber-ready signal — E2E test readiness indicator
26
+ *
27
+ * Design docs: 18-build-system.md §"Entry Files", 19-client-navigation.md
28
+ */
29
+
30
+ // @ts-expect-error — virtual module provided by timber-entries plugin
31
+ import config from 'virtual:timber-config';
32
+
33
+ import { setClientDeploymentId } from '../rsc-fetch.js';
34
+ import {
35
+ isStaleClientReference,
36
+ isChunkLoadError,
37
+ triggerStaleReload,
38
+ clearStaleReloadFlag,
39
+ } from '../stale-reload.js';
40
+
41
+ import { setupServerActions } from './action-dispatch.js';
42
+ import { createRscPayloadStream } from './rsc-stream.js';
43
+ import { createTimberRouter } from './router-init.js';
44
+ import { runPreHydration, hydrateApp } from './hydrate.js';
45
+ import { setupPostHydration } from './post-hydration.js';
46
+ import { setupHmr } from './hmr.js';
47
+
48
+ // ─── 1. Server Action Dispatch (independent) ────────────────────
49
+
50
+ setupServerActions();
51
+
52
+ // ─── 2–7. Bootstrap ─────────────────────────────────────────────
53
+
54
+ function bootstrap(runtimeConfig: typeof config): void {
55
+ // Initialize deployment ID for version skew detection (TIM-446).
56
+ // In dev mode this is null — skew checks are skipped.
57
+ const deploymentId = (runtimeConfig as Record<string, unknown>).deploymentId as string | null;
58
+ if (deploymentId) {
59
+ setClientDeploymentId(deploymentId);
60
+ }
61
+
62
+ // Take manual control of scroll restoration. Even though segment tree
63
+ // merging preserves shared layout DOM via cloneElement (so React doesn't
64
+ // reset scroll on those elements), the root-level reactRoot.render() with
65
+ // a new element tree can still cause scroll resets on the document during
66
+ // reconciliation. Manual control ensures consistent behavior.
67
+ window.history.scrollRestoration = 'manual';
68
+
69
+ // Step 2: Decode inlined RSC payload (may be null for JS-only clients)
70
+ const rscResult = createRscPayloadStream();
71
+
72
+ // Step 3: Create router + Navigation API integration
73
+ const { router, navApiController } = createTimberRouter();
74
+
75
+ // Step 4: Pre-hydration — set params + navigation state (MUST run after router init)
76
+ runPreHydration();
77
+
78
+ // Step 5: Hydrate or set up deferred root creation
79
+ hydrateApp({ rscResult, config: runtimeConfig });
80
+
81
+ // Step 6: Post-hydration wiring
82
+ setupPostHydration({
83
+ router,
84
+ navApiController,
85
+ initialElement: rscResult?.element ?? null,
86
+ });
87
+
88
+ // Step 7: HMR (dev-only, no-op in production)
89
+ setupHmr(router);
90
+ }
91
+
92
+ bootstrap(config);
93
+
94
+ // ─── 8. Stale Reload Handlers ───────────────────────────────────
95
+
96
+ // Clear the stale reload flag on successful bootstrap. If the page
97
+ // loaded and bootstrapped without hitting a stale reference error,
98
+ // the loop guard should reset so the next stale error gets a fresh
99
+ // reload attempt.
100
+ clearStaleReloadFlag();
101
+
102
+ // Global error handler for stale client reference errors during hydration.
103
+ // The initial RSC payload is decoded lazily by React via createFromReadableStream.
104
+ // If the payload references a module ID from a newer deployment, the error
105
+ // surfaces as an unhandled rejection during React's render/hydration cycle.
106
+ // This handler catches those errors and triggers a full page reload.
107
+ //
108
+ // Also catches chunk load failures (dynamic import of missing assets after
109
+ // a deployment) — these surface as "Failed to fetch dynamically imported module"
110
+ // or "Loading chunk <name> failed" errors. See TIM-446.
111
+ window.addEventListener('unhandledrejection', (event) => {
112
+ if (isStaleClientReference(event.reason) || isChunkLoadError(event.reason)) {
113
+ event.preventDefault();
114
+ triggerStaleReload();
115
+ }
116
+ });
117
+
118
+ // Also catch synchronous errors from chunk loads (some browsers throw
119
+ // TypeError synchronously instead of via unhandled rejection).
120
+ window.addEventListener('error', (event) => {
121
+ if (isChunkLoadError(event.error)) {
122
+ event.preventDefault();
123
+ triggerStaleReload();
124
+ }
125
+ });
126
+
127
+ // ─── 9. Ready Signal ────────────────────────────────────────────
128
+
129
+ // Signal that the client runtime has been initialized.
130
+ // Used by E2E tests to wait for hydration before interacting.
131
+ // We append a <meta name="timber-ready"> tag rather than setting a
132
+ // data attribute on <html>. Since React owns the entire document
133
+ // via hydrateRoot(document, ...), mutating <html> attributes causes
134
+ // hydration mismatch warnings. Dynamically-added <meta> tags don't
135
+ // conflict because React doesn't reconcile them.
136
+ const readyMeta = document.createElement('meta');
137
+ readyMeta.name = 'timber-ready';
138
+ document.head.appendChild(readyMeta);
@@ -0,0 +1,119 @@
1
+ /**
2
+ * Post-Hydration Wiring — history stack, segment cache, popstate, scroll.
3
+ *
4
+ * Sets up everything that needs to happen after the React root exists:
5
+ * - Stores initial page in history stack for instant back navigation
6
+ * - Initializes scroll state for the initial entry
7
+ * - Populates segment cache from server-embedded metadata
8
+ * - Registers popstate handler for back/forward navigation
9
+ * - Sets up debounced scroll position saving
10
+ *
11
+ * See design/19-client-navigation.md §"History Stack"
12
+ */
13
+
14
+ import { setCurrentParams } from '#client-internal';
15
+ import type { RouterInstance } from '#client-internal';
16
+ import { setNavigationState } from '../navigation-context.js';
17
+ import type { NavigationApiController } from '../navigation-api.js';
18
+ import { getScrollY } from './scroll.js';
19
+
20
+ interface PostHydrationOptions {
21
+ router: RouterInstance;
22
+ navApiController: NavigationApiController | null;
23
+ /** Decoded RSC element from initial SSR (null if no RSC payload) */
24
+ initialElement: unknown;
25
+ }
26
+
27
+ /**
28
+ * Wire up post-hydration event handlers and state.
29
+ */
30
+ export function setupPostHydration({
31
+ router,
32
+ navApiController,
33
+ initialElement,
34
+ }: PostHydrationOptions): void {
35
+ // Store the initial page in the history stack so back-button works
36
+ // after the first navigation. We store the decoded RSC element so
37
+ // back navigation can replay it instantly without a server fetch.
38
+ router.historyStack.push(window.location.pathname + window.location.search, {
39
+ payload: initialElement,
40
+ headElements: null, // SSR already set the correct head
41
+ });
42
+
43
+ // Initialize scroll state for the initial entry.
44
+ // When Navigation API is available, use per-entry state.
45
+ // Otherwise fall back to history.state.
46
+ if (navApiController) {
47
+ navApiController.saveScrollPosition(0);
48
+ } else {
49
+ window.history.replaceState({ timber: true, scrollY: 0 }, '');
50
+ }
51
+
52
+ // Populate the segment cache from server-embedded segment metadata.
53
+ // This enables state tree diffing from the very first client navigation.
54
+ // See design/19-client-navigation.md §"X-Timber-State-Tree Header"
55
+ const timberSegments = (self as unknown as Record<string, unknown>).__timber_segments;
56
+ if (Array.isArray(timberSegments)) {
57
+ router.initSegmentCache(timberSegments);
58
+ delete (self as unknown as Record<string, unknown>).__timber_segments;
59
+ }
60
+
61
+ // NOTE: We do NOT cache segment elements from the initial RSC payload here.
62
+ // The decoded element from createFromReadableStream is a thenable/lazy
63
+ // element that React resolves during render — the segment walker can't
64
+ // traverse it. The element cache is populated lazily after the first SPA
65
+ // navigation (via mergeAndCachePayload in renderViaTransition), when
66
+ // the decoded payload is a fully resolved React element tree.
67
+
68
+ // If the hydration path was skipped (no RSC payload), populate the
69
+ // fallback params from server embed here.
70
+ const lateTimberParams = (self as unknown as Record<string, unknown>).__timber_params;
71
+ if (lateTimberParams && typeof lateTimberParams === 'object') {
72
+ setCurrentParams(lateTimberParams as Record<string, string | string[]>);
73
+ setNavigationState({
74
+ params: lateTimberParams as Record<string, string | string[]>,
75
+ pathname: window.location.pathname,
76
+ });
77
+ delete (self as unknown as Record<string, unknown>).__timber_params;
78
+ }
79
+
80
+ // Register popstate handler for back/forward navigation.
81
+ // When Navigation API is active, the navigate event covers traversals —
82
+ // popstate is a no-op. When unavailable, popstate handles back/forward.
83
+ //
84
+ // Use pathname+search (not full href) to match the URL format used by
85
+ // navigate() — Link hrefs are relative paths like "/scroll-test/page-a".
86
+ // Read scrollY from history.state — the browser maintains per-entry state
87
+ // so duplicate URLs in history each have their own scroll position.
88
+ window.addEventListener('popstate', () => {
89
+ // Navigation API handles traversals via the navigate event.
90
+ if (navApiController) return;
91
+
92
+ const state = window.history.state;
93
+ const scrollY = state && typeof state.scrollY === 'number' ? state.scrollY : 0;
94
+ void router.handlePopState(window.location.pathname + window.location.search, scrollY);
95
+ });
96
+
97
+ // Keep scroll position up to date as the user scrolls.
98
+ // This ensures that when the user presses back/forward, the departing
99
+ // page's scroll position is already saved in its history entry.
100
+ // When Navigation API is available, uses per-entry state via
101
+ // navigation.updateCurrentEntry(). Otherwise falls back to history.state.
102
+ // Debounced to avoid excessive state updates during smooth scrolling.
103
+ let scrollTimer: ReturnType<typeof setTimeout>;
104
+ function saveScrollPosition(): void {
105
+ clearTimeout(scrollTimer);
106
+ scrollTimer = setTimeout(() => {
107
+ const y = getScrollY();
108
+ if (navApiController) {
109
+ navApiController.saveScrollPosition(y);
110
+ } else {
111
+ const state = window.history.state;
112
+ if (state && typeof state === 'object') {
113
+ window.history.replaceState({ ...state, scrollY: y }, '');
114
+ }
115
+ }
116
+ }, 100);
117
+ }
118
+ window.addEventListener('scroll', saveScrollPosition, { passive: true });
119
+ }