@timber-js/app 0.2.0-alpha.7 → 0.2.0-alpha.71

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (500) hide show
  1. package/LICENSE +8 -0
  2. package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-BJARkOcu.js} +1 -1
  3. package/dist/_chunks/als-registry-BJARkOcu.js.map +1 -0
  4. package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
  5. package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
  6. package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
  7. package/dist/_chunks/define-CGuYoRHU.js +199 -0
  8. package/dist/_chunks/define-CGuYoRHU.js.map +1 -0
  9. package/dist/_chunks/define-Dz1bqwaS.js +106 -0
  10. package/dist/_chunks/define-Dz1bqwaS.js.map +1 -0
  11. package/dist/_chunks/define-cookie-B5mewxwM.js +93 -0
  12. package/dist/_chunks/define-cookie-B5mewxwM.js.map +1 -0
  13. package/dist/_chunks/error-boundary-D9hzsveV.js +216 -0
  14. package/dist/_chunks/error-boundary-D9hzsveV.js.map +1 -0
  15. package/dist/_chunks/{format-DviM89f0.js → format-Rn922VH2.js} +3 -20
  16. package/dist/_chunks/format-Rn922VH2.js.map +1 -0
  17. package/dist/_chunks/{tracing-Cwn7697K.js → handler-store-BVePM7hp.js} +68 -3
  18. package/dist/_chunks/handler-store-BVePM7hp.js.map +1 -0
  19. package/dist/_chunks/{interception-BOoWmLUA.js → interception-CEdHHviP.js} +171 -97
  20. package/dist/_chunks/interception-CEdHHviP.js.map +1 -0
  21. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
  22. package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
  23. package/dist/_chunks/{request-context-DIkVh_jG.js → request-context-CywiO4jV.js} +181 -69
  24. package/dist/_chunks/request-context-CywiO4jV.js.map +1 -0
  25. package/dist/_chunks/schema-bridge-C4SwjCQD.js +86 -0
  26. package/dist/_chunks/schema-bridge-C4SwjCQD.js.map +1 -0
  27. package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
  28. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
  29. package/dist/_chunks/segment-context-hzuJ048X.js +72 -0
  30. package/dist/_chunks/segment-context-hzuJ048X.js.map +1 -0
  31. package/dist/_chunks/stale-reload-BLUC_Pl_.js +64 -0
  32. package/dist/_chunks/stale-reload-BLUC_Pl_.js.map +1 -0
  33. package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-DAhgj8Gx.js} +1 -1
  34. package/dist/_chunks/use-query-states-DAhgj8Gx.js.map +1 -0
  35. package/dist/_chunks/wrappers-LZbghvn0.js +63 -0
  36. package/dist/_chunks/wrappers-LZbghvn0.js.map +1 -0
  37. package/dist/adapters/cloudflare-dev.d.ts +109 -0
  38. package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
  39. package/dist/adapters/cloudflare-dev.js +73 -0
  40. package/dist/adapters/cloudflare-dev.js.map +1 -0
  41. package/dist/adapters/cloudflare.d.ts +148 -12
  42. package/dist/adapters/cloudflare.d.ts.map +1 -1
  43. package/dist/adapters/cloudflare.js +135 -11
  44. package/dist/adapters/cloudflare.js.map +1 -1
  45. package/dist/adapters/compress-module.d.ts.map +1 -1
  46. package/dist/adapters/nitro.d.ts +17 -1
  47. package/dist/adapters/nitro.d.ts.map +1 -1
  48. package/dist/adapters/nitro.js +56 -13
  49. package/dist/adapters/nitro.js.map +1 -1
  50. package/dist/cache/cache-api.d.ts +24 -0
  51. package/dist/cache/cache-api.d.ts.map +1 -0
  52. package/dist/cache/fast-hash.d.ts +22 -0
  53. package/dist/cache/fast-hash.d.ts.map +1 -0
  54. package/dist/cache/handler-store.d.ts +31 -0
  55. package/dist/cache/handler-store.d.ts.map +1 -0
  56. package/dist/cache/index.d.ts +7 -5
  57. package/dist/cache/index.d.ts.map +1 -1
  58. package/dist/cache/index.js +111 -73
  59. package/dist/cache/index.js.map +1 -1
  60. package/dist/cache/singleflight.d.ts +18 -1
  61. package/dist/cache/singleflight.d.ts.map +1 -1
  62. package/dist/cache/timber-cache.d.ts +1 -1
  63. package/dist/cache/timber-cache.d.ts.map +1 -1
  64. package/dist/client/error-boundary.d.ts +12 -5
  65. package/dist/client/error-boundary.d.ts.map +1 -1
  66. package/dist/client/error-boundary.js +1 -125
  67. package/dist/client/error-reconstituter.d.ts +54 -0
  68. package/dist/client/error-reconstituter.d.ts.map +1 -0
  69. package/dist/client/form.d.ts +2 -2
  70. package/dist/client/form.d.ts.map +1 -1
  71. package/dist/client/history.d.ts +19 -4
  72. package/dist/client/history.d.ts.map +1 -1
  73. package/dist/client/index.d.ts +6 -5
  74. package/dist/client/index.d.ts.map +1 -1
  75. package/dist/client/index.js +537 -166
  76. package/dist/client/index.js.map +1 -1
  77. package/dist/client/link-pending-store.d.ts +78 -0
  78. package/dist/client/link-pending-store.d.ts.map +1 -0
  79. package/dist/client/link.d.ts +90 -32
  80. package/dist/client/link.d.ts.map +1 -1
  81. package/dist/client/nav-link-store.d.ts +36 -0
  82. package/dist/client/nav-link-store.d.ts.map +1 -0
  83. package/dist/client/navigation-api-types.d.ts +90 -0
  84. package/dist/client/navigation-api-types.d.ts.map +1 -0
  85. package/dist/client/navigation-api.d.ts +115 -0
  86. package/dist/client/navigation-api.d.ts.map +1 -0
  87. package/dist/client/navigation-context.d.ts +13 -2
  88. package/dist/client/navigation-context.d.ts.map +1 -1
  89. package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
  90. package/dist/client/navigation-root.d.ts.map +1 -0
  91. package/dist/client/nuqs-adapter.d.ts.map +1 -1
  92. package/dist/client/router.d.ts +70 -4
  93. package/dist/client/router.d.ts.map +1 -1
  94. package/dist/client/rsc-fetch.d.ts +38 -3
  95. package/dist/client/rsc-fetch.d.ts.map +1 -1
  96. package/dist/client/segment-cache.d.ts +1 -1
  97. package/dist/client/segment-cache.d.ts.map +1 -1
  98. package/dist/client/segment-context.d.ts +1 -1
  99. package/dist/client/segment-context.d.ts.map +1 -1
  100. package/dist/client/segment-merger.d.ts.map +1 -1
  101. package/dist/client/segment-outlet.d.ts +63 -0
  102. package/dist/client/segment-outlet.d.ts.map +1 -0
  103. package/dist/client/ssr-data.d.ts +13 -4
  104. package/dist/client/ssr-data.d.ts.map +1 -1
  105. package/dist/client/stale-reload.d.ts +15 -0
  106. package/dist/client/stale-reload.d.ts.map +1 -1
  107. package/dist/client/top-loader.d.ts +3 -3
  108. package/dist/client/top-loader.d.ts.map +1 -1
  109. package/dist/client/use-params.d.ts +6 -4
  110. package/dist/client/use-params.d.ts.map +1 -1
  111. package/dist/client/use-query-states.d.ts +1 -1
  112. package/dist/client/use-query-states.d.ts.map +1 -1
  113. package/dist/codec.d.ts +23 -0
  114. package/dist/codec.d.ts.map +1 -0
  115. package/dist/codec.js +2 -0
  116. package/dist/cookies/define-cookie.d.ts +35 -14
  117. package/dist/cookies/define-cookie.d.ts.map +1 -1
  118. package/dist/cookies/index.d.ts +2 -0
  119. package/dist/cookies/index.d.ts.map +1 -1
  120. package/dist/cookies/index.js +3 -84
  121. package/dist/fonts/css.d.ts +1 -0
  122. package/dist/fonts/css.d.ts.map +1 -1
  123. package/dist/index.d.ts +154 -38
  124. package/dist/index.d.ts.map +1 -1
  125. package/dist/index.js +12092 -11916
  126. package/dist/index.js.map +1 -1
  127. package/dist/plugins/adapter-build.d.ts +1 -1
  128. package/dist/plugins/adapter-build.d.ts.map +1 -1
  129. package/dist/plugins/build-manifest.d.ts +2 -2
  130. package/dist/plugins/build-manifest.d.ts.map +1 -1
  131. package/dist/plugins/build-report.d.ts +3 -3
  132. package/dist/plugins/build-report.d.ts.map +1 -1
  133. package/dist/plugins/client-chunks.d.ts +32 -0
  134. package/dist/plugins/client-chunks.d.ts.map +1 -0
  135. package/dist/plugins/content.d.ts +1 -1
  136. package/dist/plugins/content.d.ts.map +1 -1
  137. package/dist/plugins/dev-browser-logs.d.ts +84 -0
  138. package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
  139. package/dist/plugins/dev-error-overlay.d.ts +26 -1
  140. package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
  141. package/dist/plugins/dev-logs.d.ts +1 -1
  142. package/dist/plugins/dev-logs.d.ts.map +1 -1
  143. package/dist/plugins/dev-server.d.ts +1 -1
  144. package/dist/plugins/dev-server.d.ts.map +1 -1
  145. package/dist/plugins/entries.d.ts +1 -1
  146. package/dist/plugins/entries.d.ts.map +1 -1
  147. package/dist/plugins/fonts.d.ts +19 -5
  148. package/dist/plugins/fonts.d.ts.map +1 -1
  149. package/dist/plugins/mdx.d.ts +1 -1
  150. package/dist/plugins/mdx.d.ts.map +1 -1
  151. package/dist/plugins/routing.d.ts +1 -1
  152. package/dist/plugins/routing.d.ts.map +1 -1
  153. package/dist/plugins/server-bundle.d.ts.map +1 -1
  154. package/dist/plugins/shims.d.ts +6 -5
  155. package/dist/plugins/shims.d.ts.map +1 -1
  156. package/dist/plugins/static-build.d.ts +1 -1
  157. package/dist/plugins/static-build.d.ts.map +1 -1
  158. package/dist/routing/codegen.d.ts +2 -2
  159. package/dist/routing/codegen.d.ts.map +1 -1
  160. package/dist/routing/index.d.ts +2 -0
  161. package/dist/routing/index.d.ts.map +1 -1
  162. package/dist/routing/index.js +3 -2
  163. package/dist/routing/scanner.d.ts.map +1 -1
  164. package/dist/routing/segment-classify.d.ts +46 -0
  165. package/dist/routing/segment-classify.d.ts.map +1 -0
  166. package/dist/routing/status-file-lint.d.ts +2 -1
  167. package/dist/routing/status-file-lint.d.ts.map +1 -1
  168. package/dist/routing/types.d.ts +16 -4
  169. package/dist/routing/types.d.ts.map +1 -1
  170. package/dist/rsc-runtime/rsc.d.ts +1 -1
  171. package/dist/rsc-runtime/rsc.d.ts.map +1 -1
  172. package/dist/rsc-runtime/ssr.d.ts +12 -0
  173. package/dist/rsc-runtime/ssr.d.ts.map +1 -1
  174. package/dist/schema-bridge.d.ts +76 -0
  175. package/dist/schema-bridge.d.ts.map +1 -0
  176. package/dist/search-params/define.d.ts +139 -0
  177. package/dist/search-params/define.d.ts.map +1 -0
  178. package/dist/search-params/index.d.ts +4 -6
  179. package/dist/search-params/index.d.ts.map +1 -1
  180. package/dist/search-params/index.js +4 -474
  181. package/dist/search-params/registry.d.ts +1 -1
  182. package/dist/search-params/wrappers.d.ts +53 -0
  183. package/dist/search-params/wrappers.d.ts.map +1 -0
  184. package/dist/segment-params/define.d.ts +78 -0
  185. package/dist/segment-params/define.d.ts.map +1 -0
  186. package/dist/segment-params/index.d.ts +7 -0
  187. package/dist/segment-params/index.d.ts.map +1 -0
  188. package/dist/segment-params/index.js +4 -0
  189. package/dist/server/access-gate.d.ts +4 -0
  190. package/dist/server/access-gate.d.ts.map +1 -1
  191. package/dist/server/action-client.d.ts +12 -1
  192. package/dist/server/action-client.d.ts.map +1 -1
  193. package/dist/server/action-encryption.d.ts +76 -0
  194. package/dist/server/action-encryption.d.ts.map +1 -0
  195. package/dist/server/action-handler.d.ts.map +1 -1
  196. package/dist/server/actions.d.ts +3 -6
  197. package/dist/server/actions.d.ts.map +1 -1
  198. package/dist/server/als-registry.d.ts +32 -4
  199. package/dist/server/als-registry.d.ts.map +1 -1
  200. package/dist/server/build-manifest.d.ts +2 -2
  201. package/dist/server/build-manifest.d.ts.map +1 -1
  202. package/dist/server/debug.d.ts +1 -1
  203. package/dist/server/default-logger.d.ts +22 -0
  204. package/dist/server/default-logger.d.ts.map +1 -0
  205. package/dist/server/deny-page-resolver.d.ts +52 -0
  206. package/dist/server/deny-page-resolver.d.ts.map +1 -0
  207. package/dist/server/deny-renderer.d.ts.map +1 -1
  208. package/dist/server/dev-warnings.d.ts +0 -14
  209. package/dist/server/dev-warnings.d.ts.map +1 -1
  210. package/dist/server/early-hints.d.ts +13 -5
  211. package/dist/server/early-hints.d.ts.map +1 -1
  212. package/dist/server/error-boundary-wrapper.d.ts +7 -1
  213. package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
  214. package/dist/server/fallback-error.d.ts +4 -3
  215. package/dist/server/fallback-error.d.ts.map +1 -1
  216. package/dist/server/flight-injection-state.d.ts +66 -0
  217. package/dist/server/flight-injection-state.d.ts.map +1 -0
  218. package/dist/server/flight-scripts.d.ts +42 -0
  219. package/dist/server/flight-scripts.d.ts.map +1 -0
  220. package/dist/server/flush.d.ts.map +1 -1
  221. package/dist/server/form-data.d.ts +29 -0
  222. package/dist/server/form-data.d.ts.map +1 -1
  223. package/dist/server/html-injectors.d.ts +51 -11
  224. package/dist/server/html-injectors.d.ts.map +1 -1
  225. package/dist/server/index.d.ts +5 -3
  226. package/dist/server/index.d.ts.map +1 -1
  227. package/dist/server/index.js +2176 -1663
  228. package/dist/server/index.js.map +1 -1
  229. package/dist/server/logger.d.ts +25 -7
  230. package/dist/server/logger.d.ts.map +1 -1
  231. package/dist/server/middleware-runner.d.ts +19 -4
  232. package/dist/server/middleware-runner.d.ts.map +1 -1
  233. package/dist/server/node-stream-transforms.d.ts +113 -0
  234. package/dist/server/node-stream-transforms.d.ts.map +1 -0
  235. package/dist/server/page-deny-boundary.d.ts +31 -0
  236. package/dist/server/page-deny-boundary.d.ts.map +1 -0
  237. package/dist/server/pipeline-interception.d.ts +1 -1
  238. package/dist/server/pipeline-interception.d.ts.map +1 -1
  239. package/dist/server/pipeline-metadata.d.ts +6 -0
  240. package/dist/server/pipeline-metadata.d.ts.map +1 -1
  241. package/dist/server/pipeline.d.ts +32 -10
  242. package/dist/server/pipeline.d.ts.map +1 -1
  243. package/dist/server/primitives.d.ts +30 -3
  244. package/dist/server/primitives.d.ts.map +1 -1
  245. package/dist/server/render-timeout.d.ts +51 -0
  246. package/dist/server/render-timeout.d.ts.map +1 -0
  247. package/dist/server/request-context.d.ts +76 -37
  248. package/dist/server/request-context.d.ts.map +1 -1
  249. package/dist/server/route-element-builder.d.ts +27 -1
  250. package/dist/server/route-element-builder.d.ts.map +1 -1
  251. package/dist/server/route-handler.d.ts.map +1 -1
  252. package/dist/server/route-matcher.d.ts +9 -2
  253. package/dist/server/route-matcher.d.ts.map +1 -1
  254. package/dist/server/rsc-entry/api-handler.d.ts +2 -2
  255. package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
  256. package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
  257. package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
  258. package/dist/server/rsc-entry/helpers.d.ts +48 -5
  259. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  260. package/dist/server/rsc-entry/index.d.ts +8 -3
  261. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  262. package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
  263. package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
  264. package/dist/server/rsc-entry/rsc-stream.d.ts +10 -1
  265. package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
  266. package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
  267. package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
  268. package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
  269. package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
  270. package/dist/server/safe-load.d.ts +46 -0
  271. package/dist/server/safe-load.d.ts.map +1 -0
  272. package/dist/server/sitemap-generator.d.ts +129 -0
  273. package/dist/server/sitemap-generator.d.ts.map +1 -0
  274. package/dist/server/sitemap-handler.d.ts +22 -0
  275. package/dist/server/sitemap-handler.d.ts.map +1 -0
  276. package/dist/server/slot-resolver.d.ts +1 -1
  277. package/dist/server/slot-resolver.d.ts.map +1 -1
  278. package/dist/server/ssr-entry.d.ts +22 -0
  279. package/dist/server/ssr-entry.d.ts.map +1 -1
  280. package/dist/server/ssr-render.d.ts +39 -21
  281. package/dist/server/ssr-render.d.ts.map +1 -1
  282. package/dist/server/ssr-wrappers.d.ts +50 -0
  283. package/dist/server/ssr-wrappers.d.ts.map +1 -0
  284. package/dist/server/status-code-resolver.d.ts +1 -1
  285. package/dist/server/status-code-resolver.d.ts.map +1 -1
  286. package/dist/server/stream-utils.d.ts +36 -0
  287. package/dist/server/stream-utils.d.ts.map +1 -0
  288. package/dist/server/tracing.d.ts +10 -0
  289. package/dist/server/tracing.d.ts.map +1 -1
  290. package/dist/server/tree-builder.d.ts +22 -19
  291. package/dist/server/tree-builder.d.ts.map +1 -1
  292. package/dist/server/types.d.ts +1 -4
  293. package/dist/server/types.d.ts.map +1 -1
  294. package/dist/server/version-skew.d.ts +61 -0
  295. package/dist/server/version-skew.d.ts.map +1 -0
  296. package/dist/server/waituntil-bridge.d.ts.map +1 -1
  297. package/dist/shared/merge-search-params.d.ts +22 -0
  298. package/dist/shared/merge-search-params.d.ts.map +1 -0
  299. package/dist/shims/font-google.d.ts +1 -1
  300. package/dist/shims/font-google.d.ts.map +1 -1
  301. package/dist/shims/font-google.js +42 -0
  302. package/dist/shims/font-google.js.map +1 -0
  303. package/dist/shims/font-local.d.ts +26 -0
  304. package/dist/shims/font-local.d.ts.map +1 -0
  305. package/dist/shims/font-local.js +20 -0
  306. package/dist/shims/font-local.js.map +1 -0
  307. package/dist/shims/navigation-client.d.ts +1 -1
  308. package/dist/shims/navigation-client.d.ts.map +1 -1
  309. package/dist/shims/navigation.d.ts +1 -1
  310. package/dist/shims/navigation.d.ts.map +1 -1
  311. package/dist/utils/directive-parser.d.ts +5 -2
  312. package/dist/utils/directive-parser.d.ts.map +1 -1
  313. package/dist/utils/state-machine.d.ts +80 -0
  314. package/dist/utils/state-machine.d.ts.map +1 -0
  315. package/package.json +37 -17
  316. package/src/adapters/cloudflare-dev.ts +177 -0
  317. package/src/adapters/cloudflare.ts +342 -28
  318. package/src/adapters/compress-module.ts +24 -4
  319. package/src/adapters/nitro.ts +58 -9
  320. package/src/adapters/wrangler.d.ts +7 -0
  321. package/src/cache/cache-api.ts +38 -0
  322. package/src/cache/fast-hash.ts +34 -0
  323. package/src/cache/handler-store.ts +68 -0
  324. package/src/cache/index.ts +9 -5
  325. package/src/cache/singleflight.ts +62 -4
  326. package/src/cache/timber-cache.ts +40 -29
  327. package/src/cli.ts +0 -0
  328. package/src/client/browser-entry.ts +314 -142
  329. package/src/client/error-boundary.tsx +48 -16
  330. package/src/client/error-reconstituter.tsx +65 -0
  331. package/src/client/form.tsx +2 -2
  332. package/src/client/history.ts +26 -4
  333. package/src/client/index.ts +13 -4
  334. package/src/client/link-pending-store.ts +136 -0
  335. package/src/client/link.tsx +346 -105
  336. package/src/client/nav-link-store.ts +47 -0
  337. package/src/client/navigation-api-types.ts +112 -0
  338. package/src/client/navigation-api.ts +332 -0
  339. package/src/client/navigation-context.ts +27 -6
  340. package/src/client/navigation-root.tsx +346 -0
  341. package/src/client/nuqs-adapter.tsx +16 -3
  342. package/src/client/router.ts +302 -77
  343. package/src/client/rsc-fetch.ts +93 -5
  344. package/src/client/segment-cache.ts +1 -1
  345. package/src/client/segment-context.ts +6 -1
  346. package/src/client/segment-merger.ts +2 -8
  347. package/src/client/segment-outlet.tsx +86 -0
  348. package/src/client/ssr-data.ts +13 -5
  349. package/src/client/stale-reload.ts +73 -6
  350. package/src/client/top-loader.tsx +22 -13
  351. package/src/client/use-navigation-pending.ts +1 -1
  352. package/src/client/use-params.ts +7 -5
  353. package/src/client/use-query-states.ts +2 -2
  354. package/src/codec.ts +34 -0
  355. package/src/cookies/define-cookie.ts +72 -21
  356. package/src/cookies/index.ts +7 -0
  357. package/src/fonts/css.ts +2 -1
  358. package/src/index.ts +328 -92
  359. package/src/plugins/adapter-build.ts +8 -2
  360. package/src/plugins/build-manifest.ts +13 -2
  361. package/src/plugins/build-report.ts +3 -3
  362. package/src/plugins/client-chunks.ts +65 -0
  363. package/src/plugins/content.ts +1 -1
  364. package/src/plugins/dev-browser-logs.ts +288 -0
  365. package/src/plugins/dev-error-overlay.ts +70 -1
  366. package/src/plugins/dev-logs.ts +1 -1
  367. package/src/plugins/dev-server.ts +55 -9
  368. package/src/plugins/entries.ts +70 -9
  369. package/src/plugins/fonts.ts +167 -61
  370. package/src/plugins/mdx.ts +1 -1
  371. package/src/plugins/routing.ts +57 -17
  372. package/src/plugins/server-action-exports.ts +1 -1
  373. package/src/plugins/server-bundle.ts +32 -1
  374. package/src/plugins/shims.ts +76 -33
  375. package/src/plugins/static-build.ts +10 -6
  376. package/src/routing/codegen.ts +165 -105
  377. package/src/routing/index.ts +2 -0
  378. package/src/routing/scanner.ts +93 -23
  379. package/src/routing/segment-classify.ts +89 -0
  380. package/src/routing/status-file-lint.ts +3 -2
  381. package/src/routing/types.ts +17 -4
  382. package/src/rsc-runtime/rsc.ts +2 -0
  383. package/src/rsc-runtime/ssr.ts +50 -0
  384. package/src/rsc-runtime/vendor-types.d.ts +7 -0
  385. package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
  386. package/src/search-params/define.ts +482 -0
  387. package/src/search-params/index.ts +13 -19
  388. package/src/search-params/registry.ts +1 -1
  389. package/src/search-params/wrappers.ts +85 -0
  390. package/src/segment-params/define.ts +279 -0
  391. package/src/segment-params/index.ts +28 -0
  392. package/src/server/access-gate.tsx +70 -29
  393. package/src/server/action-client.ts +28 -3
  394. package/src/server/action-encryption.ts +144 -0
  395. package/src/server/action-handler.ts +20 -3
  396. package/src/server/actions.ts +10 -9
  397. package/src/server/als-registry.ts +32 -4
  398. package/src/server/build-manifest.ts +10 -4
  399. package/src/server/compress.ts +25 -7
  400. package/src/server/debug.ts +1 -1
  401. package/src/server/default-logger.ts +99 -0
  402. package/src/server/deny-page-resolver.ts +154 -0
  403. package/src/server/deny-renderer.ts +24 -38
  404. package/src/server/dev-warnings.ts +2 -28
  405. package/src/server/early-hints.ts +36 -15
  406. package/src/server/error-boundary-wrapper.ts +74 -22
  407. package/src/server/fallback-error.ts +31 -15
  408. package/src/server/flight-injection-state.ts +113 -0
  409. package/src/server/flight-scripts.ts +62 -0
  410. package/src/server/flush.ts +2 -1
  411. package/src/server/form-data.ts +76 -0
  412. package/src/server/html-injectors.ts +277 -117
  413. package/src/server/index.ts +9 -5
  414. package/src/server/logger.ts +44 -36
  415. package/src/server/middleware-runner.ts +31 -4
  416. package/src/server/node-stream-transforms.ts +509 -0
  417. package/src/server/page-deny-boundary.tsx +56 -0
  418. package/src/server/pipeline-interception.ts +17 -16
  419. package/src/server/pipeline-metadata.ts +13 -0
  420. package/src/server/pipeline.ts +195 -51
  421. package/src/server/primitives.ts +47 -5
  422. package/src/server/render-timeout.ts +108 -0
  423. package/src/server/request-context.ts +240 -117
  424. package/src/server/route-element-builder.ts +284 -197
  425. package/src/server/route-handler.ts +24 -4
  426. package/src/server/route-matcher.ts +24 -20
  427. package/src/server/rsc-entry/api-handler.ts +15 -16
  428. package/src/server/rsc-entry/error-renderer.ts +300 -89
  429. package/src/server/rsc-entry/helpers.ts +134 -5
  430. package/src/server/rsc-entry/index.ts +202 -113
  431. package/src/server/rsc-entry/rsc-payload.ts +100 -21
  432. package/src/server/rsc-entry/rsc-stream.ts +74 -18
  433. package/src/server/rsc-entry/ssr-bridge.ts +14 -5
  434. package/src/server/rsc-entry/ssr-renderer.ts +173 -40
  435. package/src/server/safe-load.ts +60 -0
  436. package/src/server/sitemap-generator.ts +338 -0
  437. package/src/server/sitemap-handler.ts +126 -0
  438. package/src/server/slot-resolver.ts +243 -228
  439. package/src/server/ssr-entry.ts +211 -32
  440. package/src/server/ssr-render.ts +289 -67
  441. package/src/server/ssr-wrappers.tsx +139 -0
  442. package/src/server/status-code-resolver.ts +1 -1
  443. package/src/server/stream-utils.ts +213 -0
  444. package/src/server/tracing.ts +37 -3
  445. package/src/server/tree-builder.ts +92 -58
  446. package/src/server/types.ts +3 -6
  447. package/src/server/version-skew.ts +104 -0
  448. package/src/server/waituntil-bridge.ts +4 -1
  449. package/src/shared/merge-search-params.ts +55 -0
  450. package/src/shims/font-google.ts +1 -1
  451. package/src/shims/font-local.ts +34 -0
  452. package/src/shims/navigation-client.ts +1 -1
  453. package/src/shims/navigation.ts +2 -1
  454. package/src/utils/directive-parser.ts +5 -2
  455. package/src/utils/state-machine.ts +111 -0
  456. package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
  457. package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
  458. package/dist/_chunks/format-DviM89f0.js.map +0 -1
  459. package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
  460. package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
  461. package/dist/_chunks/ssr-data-MjmprTmO.js +0 -88
  462. package/dist/_chunks/ssr-data-MjmprTmO.js.map +0 -1
  463. package/dist/_chunks/tracing-Cwn7697K.js.map +0 -1
  464. package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
  465. package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
  466. package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
  467. package/dist/cache/register-cached-function.d.ts +0 -17
  468. package/dist/cache/register-cached-function.d.ts.map +0 -1
  469. package/dist/client/error-boundary.js.map +0 -1
  470. package/dist/client/link-status-provider.d.ts +0 -11
  471. package/dist/client/link-status-provider.d.ts.map +0 -1
  472. package/dist/client/transition-root.d.ts.map +0 -1
  473. package/dist/cookies/index.js.map +0 -1
  474. package/dist/plugins/cache-transform.d.ts +0 -36
  475. package/dist/plugins/cache-transform.d.ts.map +0 -1
  476. package/dist/plugins/dynamic-transform.d.ts +0 -72
  477. package/dist/plugins/dynamic-transform.d.ts.map +0 -1
  478. package/dist/search-params/analyze.d.ts +0 -54
  479. package/dist/search-params/analyze.d.ts.map +0 -1
  480. package/dist/search-params/builtin-codecs.d.ts +0 -105
  481. package/dist/search-params/builtin-codecs.d.ts.map +0 -1
  482. package/dist/search-params/codecs.d.ts +0 -53
  483. package/dist/search-params/codecs.d.ts.map +0 -1
  484. package/dist/search-params/create.d.ts +0 -106
  485. package/dist/search-params/create.d.ts.map +0 -1
  486. package/dist/search-params/index.js.map +0 -1
  487. package/dist/server/prerender.d.ts +0 -77
  488. package/dist/server/prerender.d.ts.map +0 -1
  489. package/dist/server/response-cache.d.ts +0 -53
  490. package/dist/server/response-cache.d.ts.map +0 -1
  491. package/src/cache/register-cached-function.ts +0 -99
  492. package/src/client/link-status-provider.tsx +0 -30
  493. package/src/client/transition-root.tsx +0 -160
  494. package/src/plugins/cache-transform.ts +0 -199
  495. package/src/plugins/dynamic-transform.ts +0 -161
  496. package/src/search-params/analyze.ts +0 -192
  497. package/src/search-params/builtin-codecs.ts +0 -228
  498. package/src/search-params/create.ts +0 -321
  499. package/src/server/prerender.ts +0 -139
  500. package/src/server/response-cache.ts +0 -277
@@ -1,199 +0,0 @@
1
- import type { Plugin } from 'vite';
2
- import { findFunctionsWithDirective, containsDirective } from '#/utils/directive-parser.js';
3
-
4
- /**
5
- * Parse a cacheLife duration string to seconds.
6
- * Supports: '30s', '5m', '1h', '2d', '1w', or a plain number (seconds).
7
- */
8
- export function parseCacheLife(value: string | number): number {
9
- if (typeof value === 'number') return value;
10
-
11
- const match = value.match(/^(\d+)(s|m|h|d|w)$/);
12
- if (!match) {
13
- throw new Error(
14
- `Invalid cacheLife value: "${value}". Expected format: "30s", "5m", "1h", "2d", "1w", or a number.`
15
- );
16
- }
17
-
18
- const amount = parseInt(match[1], 10);
19
- const unit = match[2];
20
- const multipliers: Record<string, number> = { s: 1, m: 60, h: 3600, d: 86400, w: 604800 };
21
- return amount * multipliers[unit];
22
- }
23
-
24
- // Default TTL when no cacheLife() is specified (Infinity means cache until explicit invalidation).
25
- const DEFAULT_TTL = Infinity;
26
-
27
- export interface CacheTransformWarning {
28
- message: string;
29
- functionName: string;
30
- }
31
-
32
- interface TransformResult {
33
- code: string;
34
- map?: null;
35
- warnings?: CacheTransformWarning[];
36
- }
37
-
38
- /**
39
- * Match cacheLife() calls: cacheLife('1h'), cacheLife("5m"), cacheLife(300)
40
- */
41
- const CACHE_LIFE_PATTERN = /cacheLife\(\s*(?:'([^']+)'|"([^"]+)"|(\d+))\s*\)/;
42
-
43
- /**
44
- * Strip the 'use cache' directive and cacheLife() call from a function body.
45
- * Returns the cleaned body and the extracted TTL.
46
- */
47
- function extractCacheDirectives(body: string): { cleanBody: string; ttl: number } {
48
- let ttl = DEFAULT_TTL;
49
-
50
- // Remove 'use cache' / "use cache" directive (including optional semicolon and newline)
51
- let cleanBody = body.replace(/\s*['"]use cache['"];?\s*\n?/, '\n');
52
-
53
- // Extract and remove cacheLife() calls
54
- const lifeMatch = cleanBody.match(CACHE_LIFE_PATTERN);
55
- if (lifeMatch) {
56
- const value = lifeMatch[1] || lifeMatch[2] || parseInt(lifeMatch[3], 10);
57
- ttl = parseCacheLife(value);
58
- cleanBody = cleanBody.replace(/\s*cacheLife\([^)]*\);?\s*\n?/, '\n');
59
- }
60
-
61
- return { cleanBody, ttl };
62
- }
63
-
64
- /**
65
- * Determine if a function name is a React component (PascalCase).
66
- */
67
- function isComponentName(name: string): boolean {
68
- return /^[A-Z]/.test(name);
69
- }
70
-
71
- /**
72
- * Pattern matching page/layout file conventions in a dynamic route segment.
73
- * Matches paths like: app/[slug]/page.tsx, app/[id]/layout.ts, etc.
74
- */
75
- const DYNAMIC_ROUTE_PAGE_PATTERN = /\/\[[^\]]+\].*\/(page|layout)\.[jt]sx?$/;
76
-
77
- /**
78
- * Detect if a function declaration has Promise-typed parameters.
79
- *
80
- * Checks for common patterns:
81
- * - `params: Promise<...>`
82
- * - `{ params }: { params: Promise<...> }`
83
- * - `props: { params: Promise<...> }`
84
- */
85
- const PROMISE_PARAMS_PATTERN = /params\s*(?::|.*?:)\s*Promise\s*</;
86
-
87
- /**
88
- * Check if a 'use cache' function in a dynamic route page/layout receives
89
- * Promise-typed params, which are not serializable as cache keys.
90
- */
91
- export function detectPromiseParamsWarning(
92
- declaration: string,
93
- functionName: string,
94
- fileId: string
95
- ): CacheTransformWarning | null {
96
- if (!DYNAMIC_ROUTE_PAGE_PATTERN.test(fileId)) return null;
97
- if (!PROMISE_PARAMS_PATTERN.test(declaration)) return null;
98
-
99
- return {
100
- message:
101
- `'use cache' on "${functionName}" in "${fileId}" receives Promise params. ` +
102
- `Promise is not serializable as a cache key and will cause runtime errors. ` +
103
- `Either remove 'use cache' or await the params before using them in a separate cached function.`,
104
- functionName,
105
- };
106
- }
107
-
108
- /**
109
- * Transform source code containing 'use cache' directives into
110
- * registerCachedFunction() calls.
111
- *
112
- * Returns null if no transformations were made.
113
- */
114
- export function transformUseCache(code: string, fileId: string): TransformResult | null {
115
- if (!containsDirective(code, 'use cache')) return null;
116
-
117
- const functions = findFunctionsWithDirective(code, 'use cache');
118
- if (functions.length === 0) return null;
119
-
120
- let result = code;
121
- let needsImport = false;
122
- const warnings: CacheTransformWarning[] = [];
123
-
124
- // Process functions from end to start (sorted descending by start position)
125
- for (const fn of functions) {
126
- // Warn if function receives Promise params in a dynamic route page/layout
127
- const promiseWarning = detectPromiseParamsWarning(fn.declaration, fn.name, fileId);
128
- if (promiseWarning) warnings.push(promiseWarning);
129
- const { cleanBody, ttl } = extractCacheDirectives(fn.bodyContent);
130
- const stableId = `${fileId}#${fn.name}`;
131
- const isComponent = isComponentName(fn.name);
132
-
133
- // Build the options object
134
- const optsParts = [`ttl: ${ttl === Infinity ? 'Infinity' : ttl}`];
135
- optsParts.push(`id: '${stableId}'`);
136
- if (isComponent) {
137
- optsParts.push('isComponent: true');
138
- }
139
- const optsStr = `{ ${optsParts.join(', ')} }`;
140
-
141
- // Build the replacement
142
- let replacement: string;
143
- if (fn.isArrow) {
144
- // const Name = async (...) => { body } → const Name = registerCachedFunction(async (...) => { body }, opts)
145
- const arrowSig = fn.declaration.replace(/^(?:const|let|var)\s+\w+\s*=\s*/, '');
146
- replacement = `const ${fn.name} = registerCachedFunction(${arrowSig} {${cleanBody}}, ${optsStr})`;
147
- } else {
148
- // async function Name(...) { body } → const Name = registerCachedFunction(async function Name(...) { body }, opts)
149
- const fnDecl = fn.declaration.replace(/^(?:export\s+default\s+|export\s+)?/, '');
150
- const exportPrefix = fn.prefix.includes('default')
151
- ? 'export default '
152
- : fn.prefix.includes('export')
153
- ? 'export '
154
- : '';
155
-
156
- replacement = `${exportPrefix}const ${fn.name} = registerCachedFunction(${fnDecl} {${cleanBody}}, ${optsStr})`;
157
- }
158
-
159
- result = result.slice(0, fn.start) + replacement + result.slice(fn.end);
160
- needsImport = true;
161
- }
162
-
163
- if (needsImport) {
164
- // Add the import at the top of the file
165
- result = `import { registerCachedFunction } from '@timber-js/app/cache';\n` + result;
166
- }
167
-
168
- return { code: result, map: null, warnings: warnings.length > 0 ? warnings : undefined };
169
- }
170
-
171
- /**
172
- * Vite plugin: timber-cache
173
- *
174
- * Transforms 'use cache' directives into registerCachedFunction() calls.
175
- * Only runs in the RSC environment.
176
- */
177
- export function cacheTransformPlugin(): Plugin {
178
- return {
179
- name: 'timber-cache',
180
-
181
- transform(code, id) {
182
- // Only transform in RSC environment
183
- // Skip node_modules and non-JS/TS files
184
- if (id.includes('node_modules')) return null;
185
- if (!/\.[jt]sx?$/.test(id)) return null;
186
-
187
- // Quick bail-out: no 'use cache' directive in this file
188
- if (!containsDirective(code, 'use cache')) return null;
189
-
190
- const result = transformUseCache(code, id);
191
- if (result?.warnings) {
192
- for (const w of result.warnings) {
193
- this.warn(`[timber] ${w.message}`);
194
- }
195
- }
196
- return result;
197
- },
198
- };
199
- }
@@ -1,161 +0,0 @@
1
- /**
2
- * timber-dynamic-transform — Vite sub-plugin for 'use dynamic' directive.
3
- *
4
- * Detects `'use dynamic'` directives in server component function bodies
5
- * and transforms them into `markDynamic()` runtime calls. The directive
6
- * declares a dynamic boundary — the component and its subtree opt out of
7
- * the pre-rendered shell and render per-request.
8
- *
9
- * - In `output: 'static'` mode, `'use dynamic'` is a build error.
10
- * - In standard SSR routes (no prerender.ts), the directive is a no-op
11
- * (everything is already per-request), but the transform still runs
12
- * so the runtime can skip unnecessary work.
13
- *
14
- * Design doc: design/15-future-prerendering.md §"'use dynamic'"
15
- */
16
-
17
- import type { Plugin } from 'vite';
18
- import type { PluginContext } from '#/index.js';
19
- import { findFunctionsWithDirective, containsDirective } from '#/utils/directive-parser.js';
20
-
21
- // ---------------------------------------------------------------------------
22
- // Detection
23
- // ---------------------------------------------------------------------------
24
-
25
- /**
26
- * Quick check: does this source file contain 'use dynamic' anywhere?
27
- * Used as a fast bail-out before doing expensive AST parsing.
28
- */
29
- export function containsUseDynamic(code: string): boolean {
30
- return containsDirective(code, 'use dynamic');
31
- }
32
-
33
- // ---------------------------------------------------------------------------
34
- // Transform
35
- // ---------------------------------------------------------------------------
36
-
37
- interface TransformResult {
38
- code: string;
39
- map?: null;
40
- }
41
-
42
- /**
43
- * Find function declarations/expressions containing 'use dynamic' and
44
- * transform them into markDynamic() calls.
45
- *
46
- * Input:
47
- * ```tsx
48
- * export default async function AddToCartButton({ productId }) {
49
- * 'use dynamic'
50
- * const user = await getUser()
51
- * return <button>Add to cart</button>
52
- * }
53
- * ```
54
- *
55
- * Output:
56
- * ```tsx
57
- * import { markDynamic as __markDynamic } from '@timber-js/app/runtime';
58
- * export default async function AddToCartButton({ productId }) {
59
- * __markDynamic();
60
- * const user = await getUser()
61
- * return <button>Add to cart</button>
62
- * }
63
- * ```
64
- *
65
- * The markDynamic() call registers the component boundary as dynamic
66
- * at render time. The pre-render pass uses this to know which subtrees
67
- * to skip and leave as holes for per-request rendering.
68
- */
69
- export function transformUseDynamic(code: string): TransformResult | null {
70
- if (!containsUseDynamic(code)) return null;
71
-
72
- const functions = findFunctionsWithDirective(code, 'use dynamic');
73
- if (functions.length === 0) return null;
74
-
75
- // Replace directive strings with __markDynamic() calls, processing
76
- // from end to start to preserve source offsets
77
- let result = code;
78
- for (const fn of functions) {
79
- // Replace the directive in the body content
80
- const cleanBody = fn.bodyContent.replace(/['"]use dynamic['"];?/, '__markDynamic();');
81
- // Reconstruct: replace the body content between braces
82
- result = result.slice(0, fn.bodyStart) + cleanBody + result.slice(fn.bodyEnd);
83
- }
84
-
85
- // Add the import at the top
86
- result = `import { markDynamic as __markDynamic } from '@timber-js/app/runtime';\n` + result;
87
-
88
- return { code: result, map: null };
89
- }
90
-
91
- // ---------------------------------------------------------------------------
92
- // Static mode validation
93
- // ---------------------------------------------------------------------------
94
-
95
- /**
96
- * In `output: 'static'` mode, `'use dynamic'` is a build error.
97
- * Static mode renders everything at build time — there is no per-request
98
- * rendering to opt into.
99
- */
100
- export function validateNoDynamicInStaticMode(
101
- code: string
102
- ): { message: string; line?: number } | null {
103
- if (!containsUseDynamic(code)) return null;
104
-
105
- const functions = findFunctionsWithDirective(code, 'use dynamic');
106
- if (functions.length === 0) return null;
107
-
108
- return {
109
- message:
110
- `'use dynamic' cannot be used in static mode (output: 'static'). ` +
111
- `Static mode renders all content at build time — there is no per-request rendering. ` +
112
- `Remove the directive or switch to output: 'server'.`,
113
- line: functions[functions.length - 1].directiveLine, // First occurrence (sorted descending)
114
- };
115
- }
116
-
117
- // ---------------------------------------------------------------------------
118
- // Vite Plugin
119
- // ---------------------------------------------------------------------------
120
-
121
- /**
122
- * Create the timber-dynamic-transform Vite plugin.
123
- *
124
- * In server mode: transforms 'use dynamic' into markDynamic() calls.
125
- * In static mode: rejects 'use dynamic' as a build error.
126
- */
127
- export function timberDynamicTransform(ctx: PluginContext): Plugin {
128
- const isStatic = ctx.config.output === 'static';
129
-
130
- return {
131
- name: 'timber-dynamic-transform',
132
-
133
- transform(code: string, id: string) {
134
- // Skip node_modules
135
- if (id.includes('node_modules')) return null;
136
-
137
- // Only check files in the app directory
138
- if (!id.includes('/app/') && !id.startsWith('app/')) return null;
139
-
140
- // Only check JS/TS files
141
- if (!/\.[jt]sx?$/.test(id)) return null;
142
-
143
- // Quick bail-out
144
- if (!containsUseDynamic(code)) return null;
145
-
146
- // In static mode, 'use dynamic' is a build error
147
- if (isStatic) {
148
- const error = validateNoDynamicInStaticMode(code);
149
- if (error) {
150
- this.error(
151
- `[timber] Static mode error in ${id}${error.line ? `:${error.line}` : ''}: ${error.message}`
152
- );
153
- }
154
- return null;
155
- }
156
-
157
- // In server mode, transform the directive
158
- return transformUseDynamic(code);
159
- },
160
- };
161
- }
@@ -1,192 +0,0 @@
1
- /**
2
- * Static analyzability checker for search-params.ts files.
3
- *
4
- * Validates that a search-params.ts file's default export is statically
5
- * analyzable — a createSearchParams() call or a chain of .extend()/.pick()
6
- * calls on a SearchParamsDefinition.
7
- *
8
- * Non-analyzable files produce a hard build error with a diagnostic.
9
- *
10
- * Design doc: design/09-typescript.md §"Static Analyzability"
11
- */
12
-
13
- // ---------------------------------------------------------------------------
14
- // Types
15
- // ---------------------------------------------------------------------------
16
-
17
- /** Result of analyzing a search-params.ts file. */
18
- export interface AnalyzeResult {
19
- /** Whether the file is statically analyzable. */
20
- valid: boolean;
21
- /** Error details when valid is false. */
22
- error?: AnalyzeError;
23
- }
24
-
25
- /** Diagnostic error for non-analyzable search-params.ts. */
26
- export interface AnalyzeError {
27
- /** Absolute file path. */
28
- filePath: string;
29
- /** Description of the non-analyzable expression. */
30
- expression: string;
31
- /** Suggested fix. */
32
- suggestion: string;
33
- }
34
-
35
- // ---------------------------------------------------------------------------
36
- // AST-free source analysis
37
- //
38
- // We use a lightweight regex-based approach to validate the structure of the
39
- // default export. This avoids requiring a TypeScript compiler instance at
40
- // build time for the initial validation pass. The full type extraction
41
- // (reading T from SearchParamsDefinition<T>) still happens via the TypeScript
42
- // compiler in the codegen step — this module just validates the *shape*.
43
- // ---------------------------------------------------------------------------
44
-
45
- /**
46
- * Patterns that indicate a valid default export:
47
- *
48
- * 1. `export default createSearchParams(...)`
49
- * 2. `export default someVar.extend(...)`
50
- * 3. `export default someVar.pick(...)`
51
- * 4. `export default someVar.extend(...).extend(...)` (chained)
52
- * 5. `export default someVar.extend(...).pick(...)` (chained)
53
- * 6. `export default createSearchParams(...).extend(...)`
54
- *
55
- * Invalid patterns:
56
- * - `export default someFunction(...)` (arbitrary factory)
57
- * - `export default condition ? a : b` (runtime conditional)
58
- * - `export default variable` (opaque reference without call)
59
- */
60
-
61
- /**
62
- * Analyze a search-params.ts file source for static analyzability.
63
- *
64
- * @param source - The file content as a string
65
- * @param filePath - Absolute path to the file (for diagnostics)
66
- */
67
- export function analyzeSearchParams(source: string, filePath: string): AnalyzeResult {
68
- // Strip comments to avoid false matches
69
- const stripped = stripComments(source);
70
-
71
- // Find the default export
72
- const defaultExport = extractDefaultExport(stripped);
73
-
74
- if (!defaultExport) {
75
- return {
76
- valid: false,
77
- error: {
78
- filePath,
79
- expression: '(no default export found)',
80
- suggestion:
81
- 'search-params.ts must have a default export. Use: export default createSearchParams({ ... })',
82
- },
83
- };
84
- }
85
-
86
- // Validate the expression
87
- if (isValidExpression(defaultExport.trim())) {
88
- return { valid: true };
89
- }
90
-
91
- return {
92
- valid: false,
93
- error: {
94
- filePath,
95
- expression: defaultExport.trim(),
96
- suggestion:
97
- 'The default export must be a createSearchParams() call, or a chain of ' +
98
- '.extend() / .pick() calls on a SearchParamsDefinition. Arbitrary factory ' +
99
- 'functions and runtime conditionals are not supported.',
100
- },
101
- };
102
- }
103
-
104
- // ---------------------------------------------------------------------------
105
- // Internal helpers
106
- // ---------------------------------------------------------------------------
107
-
108
- /** Strip single-line and multi-line comments from source. */
109
- function stripComments(source: string): string {
110
- // Remove multi-line comments
111
- let result = source.replace(/\/\*[\s\S]*?\*\//g, '');
112
- // Remove single-line comments
113
- result = result.replace(/\/\/.*$/gm, '');
114
- return result;
115
- }
116
-
117
- /**
118
- * Extract the expression from `export default <expr>`.
119
- *
120
- * Handles both:
121
- * export default createSearchParams(...)
122
- * export default expr\n (terminated by newline or semicolon before next statement)
123
- */
124
- function extractDefaultExport(source: string): string | undefined {
125
- // Match `export default` followed by the expression
126
- const match = source.match(
127
- /export\s+default\s+([\s\S]+?)(?:;|\n(?=export|import|const|let|var|function|class|type|interface|declare))/
128
- );
129
- if (match) {
130
- return match[1];
131
- }
132
-
133
- // Fallback: match everything after `export default` to end of file
134
- const fallback = source.match(/export\s+default\s+([\s\S]+)$/);
135
- if (fallback) {
136
- return fallback[1].replace(/;\s*$/, '');
137
- }
138
-
139
- return undefined;
140
- }
141
-
142
- /**
143
- * Check if an expression is a valid statically-analyzable pattern.
144
- *
145
- * Valid patterns:
146
- * - Starts with `createSearchParams(`
147
- * - Contains `.extend(` or `.pick(` chains (possibly starting with createSearchParams or a variable)
148
- * - A variable identifier followed by chaining
149
- */
150
- function isValidExpression(expr: string): boolean {
151
- // Normalize whitespace
152
- const normalized = expr.replace(/\s+/g, ' ').trim();
153
-
154
- // Pattern 1: starts with createSearchParams(
155
- if (normalized.startsWith('createSearchParams(')) {
156
- return true;
157
- }
158
-
159
- // Pattern 2: chain ending with .extend(...) or .pick(...)
160
- // This covers: someVar.extend(...), createSearchParams(...).extend(...).pick(...), etc.
161
- if (/\.(extend|pick)\s*\(/.test(normalized)) {
162
- // Reject ternaries and other conditional patterns
163
- if (/\?/.test(normalized) && /:/.test(normalized)) {
164
- return false;
165
- }
166
- // Reject function declarations/expressions
167
- if (/^\s*(function|=>|\()/.test(normalized)) {
168
- return false;
169
- }
170
- return true;
171
- }
172
-
173
- return false;
174
- }
175
-
176
- /**
177
- * Format an AnalyzeError into a human-readable build error message.
178
- */
179
- export function formatAnalyzeError(error: AnalyzeError): string {
180
- return [
181
- `[timber] Non-analyzable search-params.ts`,
182
- ``,
183
- ` File: ${error.filePath}`,
184
- ` Expression: ${error.expression}`,
185
- ``,
186
- ` ${error.suggestion}`,
187
- ``,
188
- ` The framework must be able to statically extract the type from your`,
189
- ` search-params.ts at build time. Dynamic values, conditionals, and`,
190
- ` arbitrary factory functions prevent this analysis.`,
191
- ].join('\n');
192
- }