@timber-js/app 0.2.0-alpha.97 → 0.2.0-alpha.99

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 (372) hide show
  1. package/dist/_chunks/actions-CQ8Z8VGL.js +1061 -0
  2. package/dist/_chunks/actions-CQ8Z8VGL.js.map +1 -0
  3. package/dist/_chunks/build-output-helper-DXnW0qjz.js +61 -0
  4. package/dist/_chunks/build-output-helper-DXnW0qjz.js.map +1 -0
  5. package/dist/_chunks/{define-Itxvcd7F.js → define-B-Q_UMOD.js} +19 -23
  6. package/dist/_chunks/define-B-Q_UMOD.js.map +1 -0
  7. package/dist/_chunks/{define-C77ScO0m.js → define-CfBPoJb0.js} +24 -7
  8. package/dist/_chunks/define-CfBPoJb0.js.map +1 -0
  9. package/dist/_chunks/define-cookie-BjpIt4UC.js +194 -0
  10. package/dist/_chunks/define-cookie-BjpIt4UC.js.map +1 -0
  11. package/dist/_chunks/{format-CYBGxKtc.js → format-Bcn-Iv1x.js} +1 -1
  12. package/dist/_chunks/{format-CYBGxKtc.js.map → format-Bcn-Iv1x.js.map} +1 -1
  13. package/dist/_chunks/handler-store-B-lqaGyh.js +54 -0
  14. package/dist/_chunks/handler-store-B-lqaGyh.js.map +1 -0
  15. package/dist/_chunks/logger-0m8MsKdc.js +291 -0
  16. package/dist/_chunks/logger-0m8MsKdc.js.map +1 -0
  17. package/dist/_chunks/merge-search-params-BphMdht_.js +122 -0
  18. package/dist/_chunks/merge-search-params-BphMdht_.js.map +1 -0
  19. package/dist/_chunks/{metadata-routes-DS3eKNmf.js → metadata-routes-BU684ls2.js} +1 -1
  20. package/dist/_chunks/{metadata-routes-DS3eKNmf.js.map → metadata-routes-BU684ls2.js.map} +1 -1
  21. package/dist/_chunks/navigation-root-BCYczjml.js +96 -0
  22. package/dist/_chunks/navigation-root-BCYczjml.js.map +1 -0
  23. package/dist/_chunks/registry-I2ss-lvy.js +20 -0
  24. package/dist/_chunks/registry-I2ss-lvy.js.map +1 -0
  25. package/dist/_chunks/router-ref-h3-UaCQv.js +28 -0
  26. package/dist/_chunks/router-ref-h3-UaCQv.js.map +1 -0
  27. package/dist/_chunks/{schema-bridge-C3xl_vfb.js → schema-bridge-Cxu4l-7p.js} +1 -1
  28. package/dist/_chunks/{schema-bridge-C3xl_vfb.js.map → schema-bridge-Cxu4l-7p.js.map} +1 -1
  29. package/dist/_chunks/segment-classify-BjfuctV2.js +137 -0
  30. package/dist/_chunks/segment-classify-BjfuctV2.js.map +1 -0
  31. package/dist/_chunks/{segment-context-fHFLF1PE.js → segment-context-Dx_OizxD.js} +1 -1
  32. package/dist/_chunks/{segment-context-fHFLF1PE.js.map → segment-context-Dx_OizxD.js.map} +1 -1
  33. package/dist/_chunks/{router-ref-C8OCm7g7.js → ssr-data-B4CdH7rE.js} +2 -26
  34. package/dist/_chunks/ssr-data-B4CdH7rE.js.map +1 -0
  35. package/dist/_chunks/{stale-reload-BX5gL1r-.js → stale-reload-Bab885FO.js} +1 -1
  36. package/dist/_chunks/{stale-reload-BX5gL1r-.js.map → stale-reload-Bab885FO.js.map} +1 -1
  37. package/dist/_chunks/tracing-C8V-YGsP.js +329 -0
  38. package/dist/_chunks/tracing-C8V-YGsP.js.map +1 -0
  39. package/dist/_chunks/{use-query-states-BiV5GJgm.js → use-query-states-B2XTqxDR.js} +3 -19
  40. package/dist/_chunks/use-query-states-B2XTqxDR.js.map +1 -0
  41. package/dist/_chunks/{use-params-IOPu7E8t.js → use-segment-params-BkpKAQ7D.js} +9 -95
  42. package/dist/_chunks/use-segment-params-BkpKAQ7D.js.map +1 -0
  43. package/dist/_chunks/{interception-BbqMCVXa.js → walkers-Tg0Alwcg.js} +66 -87
  44. package/dist/_chunks/walkers-Tg0Alwcg.js.map +1 -0
  45. package/dist/_chunks/{dev-warnings-DpGRGoDi.js → warnings-Cg47l5sk.js} +3 -3
  46. package/dist/_chunks/warnings-Cg47l5sk.js.map +1 -0
  47. package/dist/adapters/build-output-helper.d.ts +28 -0
  48. package/dist/adapters/build-output-helper.d.ts.map +1 -0
  49. package/dist/adapters/cloudflare.d.ts.map +1 -1
  50. package/dist/adapters/cloudflare.js +8 -28
  51. package/dist/adapters/cloudflare.js.map +1 -1
  52. package/dist/adapters/nitro.d.ts.map +1 -1
  53. package/dist/adapters/nitro.js +63 -31
  54. package/dist/adapters/nitro.js.map +1 -1
  55. package/dist/adapters/shared.d.ts +16 -0
  56. package/dist/adapters/shared.d.ts.map +1 -0
  57. package/dist/cache/index.js +9 -2
  58. package/dist/cache/index.js.map +1 -1
  59. package/dist/cache/timber-cache.d.ts.map +1 -1
  60. package/dist/client/error-boundary.js +2 -1
  61. package/dist/client/error-boundary.js.map +1 -1
  62. package/dist/client/form.d.ts +10 -24
  63. package/dist/client/form.d.ts.map +1 -1
  64. package/dist/client/index.d.ts +1 -5
  65. package/dist/client/index.d.ts.map +1 -1
  66. package/dist/client/index.js +41 -91
  67. package/dist/client/index.js.map +1 -1
  68. package/dist/client/internal.d.ts +2 -1
  69. package/dist/client/internal.d.ts.map +1 -1
  70. package/dist/client/internal.js +81 -7
  71. package/dist/client/internal.js.map +1 -1
  72. package/dist/client/rsc-fetch.d.ts.map +1 -1
  73. package/dist/client/state.d.ts +1 -1
  74. package/dist/client/use-cookie.d.ts +8 -0
  75. package/dist/client/use-cookie.d.ts.map +1 -1
  76. package/dist/client/{use-params.d.ts → use-segment-params.d.ts} +1 -1
  77. package/dist/client/use-segment-params.d.ts.map +1 -0
  78. package/dist/codec.d.ts +1 -1
  79. package/dist/codec.d.ts.map +1 -1
  80. package/dist/codec.js +2 -2
  81. package/dist/config-types.d.ts +28 -0
  82. package/dist/config-types.d.ts.map +1 -1
  83. package/dist/cookies/define-cookie.d.ts +87 -35
  84. package/dist/cookies/define-cookie.d.ts.map +1 -1
  85. package/dist/cookies/index.d.ts +2 -1
  86. package/dist/cookies/index.d.ts.map +1 -1
  87. package/dist/cookies/index.js +48 -2
  88. package/dist/cookies/index.js.map +1 -0
  89. package/dist/cookies/json-cookie.d.ts +64 -0
  90. package/dist/cookies/json-cookie.d.ts.map +1 -0
  91. package/dist/cookies/validation.d.ts +46 -0
  92. package/dist/cookies/validation.d.ts.map +1 -0
  93. package/dist/{plugins/dev-404-page.d.ts → dev-tools/404-page.d.ts} +9 -19
  94. package/dist/dev-tools/404-page.d.ts.map +1 -0
  95. package/dist/{plugins/dev-browser-logs.d.ts → dev-tools/browser-logs.d.ts} +1 -1
  96. package/dist/dev-tools/browser-logs.d.ts.map +1 -0
  97. package/dist/{plugins/dev-error-page.d.ts → dev-tools/error-page.d.ts} +2 -2
  98. package/dist/dev-tools/error-page.d.ts.map +1 -0
  99. package/dist/{server/dev-holding-server.d.ts → dev-tools/holding-server.d.ts} +5 -3
  100. package/dist/dev-tools/holding-server.d.ts.map +1 -0
  101. package/dist/dev-tools/index.d.ts +31 -0
  102. package/dist/dev-tools/index.d.ts.map +1 -0
  103. package/dist/{server/dev-span-processor.d.ts → dev-tools/instrumentation.d.ts} +26 -6
  104. package/dist/dev-tools/instrumentation.d.ts.map +1 -0
  105. package/dist/{server/dev-logger.d.ts → dev-tools/logger.d.ts} +1 -1
  106. package/dist/dev-tools/logger.d.ts.map +1 -0
  107. package/dist/{plugins/dev-logs.d.ts → dev-tools/logs.d.ts} +1 -1
  108. package/dist/dev-tools/logs.d.ts.map +1 -0
  109. package/dist/{plugins/dev-error-overlay.d.ts → dev-tools/overlay.d.ts} +3 -12
  110. package/dist/dev-tools/overlay.d.ts.map +1 -0
  111. package/dist/dev-tools/stack-classifier.d.ts +34 -0
  112. package/dist/dev-tools/stack-classifier.d.ts.map +1 -0
  113. package/dist/{plugins/dev-terminal-error.d.ts → dev-tools/terminal.d.ts} +2 -2
  114. package/dist/dev-tools/terminal.d.ts.map +1 -0
  115. package/dist/{server/dev-warnings.d.ts → dev-tools/warnings.d.ts} +1 -1
  116. package/dist/dev-tools/warnings.d.ts.map +1 -0
  117. package/dist/index.d.ts +1 -0
  118. package/dist/index.d.ts.map +1 -1
  119. package/dist/index.js +285 -133
  120. package/dist/index.js.map +1 -1
  121. package/dist/plugin-context.d.ts +1 -1
  122. package/dist/plugin-context.d.ts.map +1 -1
  123. package/dist/plugins/adapter-build.d.ts.map +1 -1
  124. package/dist/plugins/build-report.d.ts +6 -4
  125. package/dist/plugins/build-report.d.ts.map +1 -1
  126. package/dist/routing/convention-lint.d.ts.map +1 -1
  127. package/dist/routing/index.d.ts +5 -3
  128. package/dist/routing/index.d.ts.map +1 -1
  129. package/dist/routing/index.js +3 -3
  130. package/dist/routing/scanner.d.ts +1 -10
  131. package/dist/routing/scanner.d.ts.map +1 -1
  132. package/dist/routing/segment-classify.d.ts +37 -8
  133. package/dist/routing/segment-classify.d.ts.map +1 -1
  134. package/dist/routing/status-file-lint.d.ts.map +1 -1
  135. package/dist/routing/types.d.ts +63 -23
  136. package/dist/routing/types.d.ts.map +1 -1
  137. package/dist/routing/walkers.d.ts +51 -0
  138. package/dist/routing/walkers.d.ts.map +1 -0
  139. package/dist/search-params/define.d.ts +25 -7
  140. package/dist/search-params/define.d.ts.map +1 -1
  141. package/dist/search-params/index.js +5 -3
  142. package/dist/search-params/index.js.map +1 -1
  143. package/dist/search-params/wrappers.d.ts +2 -2
  144. package/dist/search-params/wrappers.d.ts.map +1 -1
  145. package/dist/segment-params/define.d.ts +23 -6
  146. package/dist/segment-params/define.d.ts.map +1 -1
  147. package/dist/segment-params/index.js +1 -1
  148. package/dist/server/access-gate.d.ts +4 -3
  149. package/dist/server/access-gate.d.ts.map +1 -1
  150. package/dist/server/action-handler.d.ts +15 -6
  151. package/dist/server/action-handler.d.ts.map +1 -1
  152. package/dist/server/als-registry.d.ts +5 -5
  153. package/dist/server/als-registry.d.ts.map +1 -1
  154. package/dist/server/asset-headers.d.ts +1 -15
  155. package/dist/server/asset-headers.d.ts.map +1 -1
  156. package/dist/server/cookie-context.d.ts +170 -0
  157. package/dist/server/cookie-context.d.ts.map +1 -0
  158. package/dist/server/cookie-parsing.d.ts +51 -0
  159. package/dist/server/cookie-parsing.d.ts.map +1 -0
  160. package/dist/server/deny-boundary.d.ts +90 -0
  161. package/dist/server/deny-boundary.d.ts.map +1 -0
  162. package/dist/server/deny-renderer.d.ts.map +1 -1
  163. package/dist/server/early-hints-sender.d.ts.map +1 -1
  164. package/dist/server/html-injector-core.d.ts +212 -0
  165. package/dist/server/html-injector-core.d.ts.map +1 -0
  166. package/dist/server/html-injectors.d.ts +59 -59
  167. package/dist/server/html-injectors.d.ts.map +1 -1
  168. package/dist/server/index.d.ts +5 -4
  169. package/dist/server/index.d.ts.map +1 -1
  170. package/dist/server/index.js +4 -149
  171. package/dist/server/index.js.map +1 -1
  172. package/dist/server/internal.d.ts +6 -4
  173. package/dist/server/internal.d.ts.map +1 -1
  174. package/dist/server/internal.js +852 -852
  175. package/dist/server/internal.js.map +1 -1
  176. package/dist/server/logger.d.ts +14 -0
  177. package/dist/server/logger.d.ts.map +1 -1
  178. package/dist/server/middleware-runner.d.ts +17 -0
  179. package/dist/server/middleware-runner.d.ts.map +1 -1
  180. package/dist/server/node-stream-transforms.d.ts +46 -49
  181. package/dist/server/node-stream-transforms.d.ts.map +1 -1
  182. package/dist/server/param-coercion.d.ts +26 -0
  183. package/dist/server/param-coercion.d.ts.map +1 -0
  184. package/dist/server/pipeline-helpers.d.ts +95 -0
  185. package/dist/server/pipeline-helpers.d.ts.map +1 -0
  186. package/dist/server/pipeline-outcome.d.ts +49 -0
  187. package/dist/server/pipeline-outcome.d.ts.map +1 -0
  188. package/dist/server/pipeline-phases.d.ts +52 -0
  189. package/dist/server/pipeline-phases.d.ts.map +1 -0
  190. package/dist/server/pipeline.d.ts +51 -32
  191. package/dist/server/pipeline.d.ts.map +1 -1
  192. package/dist/server/port-resolution.d.ts +117 -0
  193. package/dist/server/port-resolution.d.ts.map +1 -0
  194. package/dist/server/request-context.d.ts +22 -159
  195. package/dist/server/request-context.d.ts.map +1 -1
  196. package/dist/server/route-element-builder.d.ts.map +1 -1
  197. package/dist/server/route-matcher.d.ts +20 -47
  198. package/dist/server/route-matcher.d.ts.map +1 -1
  199. package/dist/server/rsc-entry/action-middleware-runner.d.ts +66 -0
  200. package/dist/server/rsc-entry/action-middleware-runner.d.ts.map +1 -0
  201. package/dist/server/rsc-entry/helpers.d.ts +1 -1
  202. package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
  203. package/dist/server/rsc-entry/index.d.ts.map +1 -1
  204. package/dist/server/rsc-entry/render-route.d.ts +50 -0
  205. package/dist/server/rsc-entry/render-route.d.ts.map +1 -0
  206. package/dist/server/rsc-entry/wrap-action-dispatch.d.ts +119 -0
  207. package/dist/server/rsc-entry/wrap-action-dispatch.d.ts.map +1 -0
  208. package/dist/server/state-tree-diff.d.ts.map +1 -1
  209. package/dist/server/status-code-resolver.d.ts +16 -11
  210. package/dist/server/status-code-resolver.d.ts.map +1 -1
  211. package/dist/server/tracing.d.ts +1 -1
  212. package/dist/server/tracing.d.ts.map +1 -1
  213. package/dist/server/tree-builder.d.ts +45 -16
  214. package/dist/server/tree-builder.d.ts.map +1 -1
  215. package/dist/server/types.d.ts +48 -0
  216. package/dist/server/types.d.ts.map +1 -1
  217. package/dist/server/utils/escape-html.d.ts +14 -0
  218. package/dist/server/utils/escape-html.d.ts.map +1 -0
  219. package/dist/shims/headers.d.ts +2 -2
  220. package/dist/shims/headers.d.ts.map +1 -1
  221. package/dist/shims/navigation-client.d.ts +3 -1
  222. package/dist/shims/navigation-client.d.ts.map +1 -1
  223. package/dist/shims/navigation.d.ts +9 -4
  224. package/dist/shims/navigation.d.ts.map +1 -1
  225. package/dist/utils/directive-parser.d.ts +0 -45
  226. package/dist/utils/directive-parser.d.ts.map +1 -1
  227. package/package.json +1 -1
  228. package/src/adapters/build-output-helper.ts +77 -0
  229. package/src/adapters/cloudflare.ts +10 -50
  230. package/src/adapters/nitro.ts +66 -50
  231. package/src/adapters/shared.ts +40 -0
  232. package/src/cache/timber-cache.ts +3 -2
  233. package/src/client/form.tsx +17 -25
  234. package/src/client/index.ts +16 -9
  235. package/src/client/internal.ts +3 -2
  236. package/src/client/router.ts +1 -1
  237. package/src/client/rsc-fetch.ts +15 -0
  238. package/src/client/state.ts +2 -2
  239. package/src/client/use-cookie.ts +29 -0
  240. package/src/codec.ts +3 -7
  241. package/src/config-types.ts +28 -0
  242. package/src/cookies/define-cookie.ts +271 -78
  243. package/src/cookies/index.ts +11 -8
  244. package/src/cookies/json-cookie.ts +105 -0
  245. package/src/cookies/validation.ts +134 -0
  246. package/src/{plugins/dev-404-page.ts → dev-tools/404-page.ts} +17 -48
  247. package/src/{plugins/dev-error-page.ts → dev-tools/error-page.ts} +5 -32
  248. package/src/{server/dev-holding-server.ts → dev-tools/holding-server.ts} +4 -2
  249. package/src/dev-tools/index.ts +90 -0
  250. package/src/dev-tools/instrumentation.ts +176 -0
  251. package/src/{plugins/dev-logs.ts → dev-tools/logs.ts} +2 -2
  252. package/src/{plugins/dev-error-overlay.ts → dev-tools/overlay.ts} +5 -23
  253. package/src/dev-tools/stack-classifier.ts +75 -0
  254. package/src/{plugins/dev-terminal-error.ts → dev-tools/terminal.ts} +4 -38
  255. package/src/{server/dev-warnings.ts → dev-tools/warnings.ts} +1 -1
  256. package/src/index.ts +95 -34
  257. package/src/plugin-context.ts +1 -1
  258. package/src/plugins/adapter-build.ts +3 -1
  259. package/src/plugins/build-report.ts +13 -22
  260. package/src/plugins/dev-server.ts +3 -3
  261. package/src/plugins/routing.ts +14 -12
  262. package/src/plugins/shims.ts +1 -1
  263. package/src/plugins/static-build.ts +1 -1
  264. package/src/routing/codegen.ts +1 -1
  265. package/src/routing/convention-lint.ts +9 -8
  266. package/src/routing/index.ts +5 -3
  267. package/src/routing/interception.ts +1 -1
  268. package/src/routing/scanner.ts +22 -95
  269. package/src/routing/segment-classify.ts +107 -8
  270. package/src/routing/status-file-lint.ts +7 -5
  271. package/src/routing/types.ts +63 -23
  272. package/src/routing/walkers.ts +90 -0
  273. package/src/search-params/define.ts +71 -15
  274. package/src/search-params/wrappers.ts +9 -2
  275. package/src/segment-params/define.ts +66 -13
  276. package/src/server/access-gate.tsx +9 -8
  277. package/src/server/action-handler.ts +34 -38
  278. package/src/server/als-registry.ts +5 -5
  279. package/src/server/asset-headers.ts +8 -34
  280. package/src/server/cookie-context.ts +468 -0
  281. package/src/server/cookie-parsing.ts +135 -0
  282. package/src/server/{deny-page-resolver.ts → deny-boundary.ts} +78 -14
  283. package/src/server/deny-renderer.ts +7 -12
  284. package/src/server/early-hints-sender.ts +3 -2
  285. package/src/server/fallback-error.ts +2 -2
  286. package/src/server/html-injector-core.ts +403 -0
  287. package/src/server/html-injectors.ts +158 -297
  288. package/src/server/index.ts +13 -14
  289. package/src/server/internal.ts +10 -3
  290. package/src/server/logger.ts +23 -0
  291. package/src/server/middleware-runner.ts +44 -0
  292. package/src/server/node-stream-transforms.ts +108 -248
  293. package/src/server/param-coercion.ts +76 -0
  294. package/src/server/pipeline-helpers.ts +204 -0
  295. package/src/server/pipeline-outcome.ts +167 -0
  296. package/src/server/pipeline-phases.ts +409 -0
  297. package/src/server/pipeline.ts +70 -540
  298. package/src/server/port-resolution.ts +215 -0
  299. package/src/server/request-context.ts +46 -451
  300. package/src/server/route-element-builder.ts +8 -4
  301. package/src/server/route-matcher.ts +28 -60
  302. package/src/server/rsc-entry/action-middleware-runner.ts +167 -0
  303. package/src/server/rsc-entry/api-handler.ts +2 -2
  304. package/src/server/rsc-entry/error-renderer.ts +2 -2
  305. package/src/server/rsc-entry/helpers.ts +2 -7
  306. package/src/server/rsc-entry/index.ts +81 -366
  307. package/src/server/rsc-entry/render-route.ts +304 -0
  308. package/src/server/rsc-entry/rsc-payload.ts +1 -1
  309. package/src/server/rsc-entry/ssr-renderer.ts +2 -2
  310. package/src/server/rsc-entry/wrap-action-dispatch.ts +449 -0
  311. package/src/server/sitemap-generator.ts +1 -1
  312. package/src/server/slot-resolver.ts +1 -1
  313. package/src/server/ssr-entry.ts +1 -1
  314. package/src/server/state-tree-diff.ts +4 -1
  315. package/src/server/status-code-resolver.ts +112 -128
  316. package/src/server/tracing.ts +3 -3
  317. package/src/server/tree-builder.ts +134 -56
  318. package/src/server/types.ts +52 -0
  319. package/src/server/utils/escape-html.ts +20 -0
  320. package/src/shims/headers.ts +3 -3
  321. package/src/shims/navigation-client.ts +4 -3
  322. package/src/shims/navigation.ts +9 -7
  323. package/src/utils/directive-parser.ts +0 -392
  324. package/dist/_chunks/actions-DLnUaR65.js +0 -421
  325. package/dist/_chunks/actions-DLnUaR65.js.map +0 -1
  326. package/dist/_chunks/als-registry-HS0LGUl2.js +0 -41
  327. package/dist/_chunks/als-registry-HS0LGUl2.js.map +0 -1
  328. package/dist/_chunks/debug-ECi_61pb.js +0 -108
  329. package/dist/_chunks/debug-ECi_61pb.js.map +0 -1
  330. package/dist/_chunks/define-C77ScO0m.js.map +0 -1
  331. package/dist/_chunks/define-Itxvcd7F.js.map +0 -1
  332. package/dist/_chunks/define-cookie-BowvzoP0.js +0 -94
  333. package/dist/_chunks/define-cookie-BowvzoP0.js.map +0 -1
  334. package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +0 -1
  335. package/dist/_chunks/interception-BbqMCVXa.js.map +0 -1
  336. package/dist/_chunks/merge-search-params-Cm_KIWDX.js +0 -41
  337. package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +0 -1
  338. package/dist/_chunks/request-context-CK5tZqIP.js +0 -478
  339. package/dist/_chunks/request-context-CK5tZqIP.js.map +0 -1
  340. package/dist/_chunks/router-ref-C8OCm7g7.js.map +0 -1
  341. package/dist/_chunks/segment-classify-BDNn6EzD.js +0 -65
  342. package/dist/_chunks/segment-classify-BDNn6EzD.js.map +0 -1
  343. package/dist/_chunks/tracing-CCYbKn5n.js +0 -238
  344. package/dist/_chunks/tracing-CCYbKn5n.js.map +0 -1
  345. package/dist/_chunks/use-params-IOPu7E8t.js.map +0 -1
  346. package/dist/_chunks/use-query-states-BiV5GJgm.js.map +0 -1
  347. package/dist/client/use-params.d.ts.map +0 -1
  348. package/dist/plugins/dev-404-page.d.ts.map +0 -1
  349. package/dist/plugins/dev-browser-logs.d.ts.map +0 -1
  350. package/dist/plugins/dev-error-overlay.d.ts.map +0 -1
  351. package/dist/plugins/dev-error-page.d.ts.map +0 -1
  352. package/dist/plugins/dev-logs.d.ts.map +0 -1
  353. package/dist/plugins/dev-terminal-error.d.ts.map +0 -1
  354. package/dist/server/deny-page-resolver.d.ts +0 -52
  355. package/dist/server/deny-page-resolver.d.ts.map +0 -1
  356. package/dist/server/dev-fetch-instrumentation.d.ts +0 -22
  357. package/dist/server/dev-fetch-instrumentation.d.ts.map +0 -1
  358. package/dist/server/dev-holding-server.d.ts.map +0 -1
  359. package/dist/server/dev-logger.d.ts.map +0 -1
  360. package/dist/server/dev-span-processor.d.ts.map +0 -1
  361. package/dist/server/dev-warnings.d.ts.map +0 -1
  362. package/dist/server/manifest-status-resolver.d.ts +0 -58
  363. package/dist/server/manifest-status-resolver.d.ts.map +0 -1
  364. package/dist/server/page-deny-boundary.d.ts +0 -31
  365. package/dist/server/page-deny-boundary.d.ts.map +0 -1
  366. package/src/server/dev-fetch-instrumentation.ts +0 -96
  367. package/src/server/dev-span-processor.ts +0 -78
  368. package/src/server/manifest-status-resolver.ts +0 -215
  369. package/src/server/page-deny-boundary.tsx +0 -56
  370. /package/src/client/{use-params.ts → use-segment-params.ts} +0 -0
  371. /package/src/{plugins/dev-browser-logs.ts → dev-tools/browser-logs.ts} +0 -0
  372. /package/src/{server/dev-logger.ts → dev-tools/logger.ts} +0 -0
@@ -0,0 +1,76 @@
1
+ /**
2
+ * Segment param coercion — runs the matched route's `params.ts` codecs
3
+ * over the raw matcher output before middleware and rendering.
4
+ *
5
+ * Lifted out of `pipeline-phases.ts` (TIM-853) so the coercer can be
6
+ * imported directly by other entry points (the action-dispatch wrapper,
7
+ * the revalidation renderer in `rsc-entry/index.ts`) without pulling
8
+ * the entire pipeline phase module along with it.
9
+ *
10
+ * The function throws `ParamCoercionError` from `route-element-builder.ts`
11
+ * on any codec failure; the pipeline catches that and dispatches to the
12
+ * 404 page. See design/07-routing.md §"Where Coercion Runs".
13
+ */
14
+
15
+ import type { RouteMatch } from './pipeline.js';
16
+ import { sanitizeParamValue } from './pipeline-helpers.js';
17
+ import { loadModule } from './safe-load.js';
18
+ import { ParamCoercionError } from './route-element-builder.js';
19
+
20
+ /**
21
+ * Run segment param coercion on the matched route's segments.
22
+ *
23
+ * Loads params.ts modules from segments that have them, extracts the
24
+ * segmentParams definition, and coerces raw string params through codecs.
25
+ * Throws ParamCoercionError if any codec fails (→ 404).
26
+ *
27
+ * This runs BEFORE middleware, so ctx.segmentParams is already typed.
28
+ * See design/07-routing.md §"Where Coercion Runs"
29
+ */
30
+ export async function coerceSegmentParams(match: RouteMatch): Promise<void> {
31
+ // Unconditionally install a null-prototype target so the invariant
32
+ // "match.segmentParams is null-prototype" holds from the first line,
33
+ // regardless of whether any segment has a codec.
34
+ const mergeTarget: Record<string, unknown> = Object.create(null);
35
+ for (const key of Object.keys(match.segmentParams)) {
36
+ if (key !== '__proto__') {
37
+ mergeTarget[key] = match.segmentParams[key as keyof typeof match.segmentParams];
38
+ }
39
+ }
40
+ match.segmentParams = mergeTarget as RouteMatch['segmentParams'];
41
+
42
+ for (const segment of match.segments) {
43
+ // Only process segments that have a params.ts convention file
44
+ if (!segment.params) continue;
45
+
46
+ let mod: Record<string, unknown>;
47
+ try {
48
+ mod = await loadModule(segment.params);
49
+ } catch (err) {
50
+ throw new ParamCoercionError(
51
+ `Failed to load params module for segment "${segment.segmentName}": ${err instanceof Error ? err.message : String(err)}`
52
+ );
53
+ }
54
+
55
+ const segmentParamsDef = mod.segmentParams as
56
+ | { parse(raw: Record<string, string | string[]>): Record<string, unknown> }
57
+ | undefined;
58
+
59
+ if (!segmentParamsDef || typeof segmentParamsDef.parse !== 'function') continue;
60
+
61
+ try {
62
+ const coerced = segmentParamsDef.parse(match.segmentParams);
63
+
64
+ // Deep-sanitize codec output: every nested plain object becomes
65
+ // null-prototype with dangerous keys stripped at every depth.
66
+ // See TIM-873, design/13-security.md
67
+ for (const key of Object.keys(coerced as Record<string, unknown>)) {
68
+ if (key !== '__proto__') {
69
+ mergeTarget[key] = sanitizeParamValue((coerced as Record<string, unknown>)[key]);
70
+ }
71
+ }
72
+ } catch (err) {
73
+ throw new ParamCoercionError(err instanceof Error ? err.message : String(err));
74
+ }
75
+ }
76
+ }
@@ -0,0 +1,204 @@
1
+ /**
2
+ * Pipeline helpers — small utility functions used by `pipeline.ts` and
3
+ * `pipeline-phases.ts`. Lifted out of `pipeline.ts` to keep that file
4
+ * focused on the request handler entry point.
5
+ *
6
+ * Each helper is intentionally a free function with no closure capture, so
7
+ * it can be unit-tested in isolation.
8
+ *
9
+ * See design/07-routing.md §"Request Lifecycle".
10
+ */
11
+
12
+ import type { ProxyExport } from './proxy.js';
13
+ import { getSetCookieHeaders } from './cookie-context.js';
14
+ import { callOnRequestError } from './instrumentation.js';
15
+ import { getTraceId } from './tracing.js';
16
+ import { RedirectSignal } from './primitives.js';
17
+ import type { ProxyConfig } from './pipeline.js';
18
+
19
+ // ─── Prototype-Pollution-Safe Sanitizer ────────────────────────────────────
20
+
21
+ /**
22
+ * Only __proto__ needs stripping — it has a language-level setter that
23
+ * changes the prototype chain of spread copies. constructor and prototype
24
+ * are harmless own properties on null-prototype objects.
25
+ */
26
+ const DANGEROUS_KEYS = new Set(['__proto__']);
27
+
28
+ /**
29
+ * Deep-walk a value returned by a segment param codec, producing a
30
+ * sanitized copy where every plain object is null-prototype and
31
+ * dangerous keys (__proto__, constructor, prototype) are stripped at
32
+ * every depth.
33
+ *
34
+ * Non-plain objects (Date, Map, class instances, etc.) are returned
35
+ * as-is — they cannot be poisoned by `{...x}` spread and may carry
36
+ * author-intended prototype methods.
37
+ *
38
+ * Arrays are walked element-wise.
39
+ *
40
+ * Performance: URL params are bounded by URL length (~8 KB). Realistic
41
+ * trees are <1 KB. The recursive walk is sub-microsecond.
42
+ *
43
+ * See TIM-655, TIM-855, TIM-873, design/13-security.md
44
+ */
45
+ export function sanitizeParamValue(value: unknown): unknown {
46
+ if (value === null || typeof value !== 'object') return value;
47
+
48
+ if (Array.isArray(value)) {
49
+ return value.map(sanitizeParamValue);
50
+ }
51
+
52
+ // Only walk plain objects — anything with a custom prototype (Date, Map,
53
+ // class instances) is left untouched.
54
+ const proto = Object.getPrototypeOf(value);
55
+ if (proto !== Object.prototype && proto !== null) return value;
56
+
57
+ const out: Record<string, unknown> = Object.create(null);
58
+ for (const key of Object.keys(value as Record<string, unknown>)) {
59
+ if (!DANGEROUS_KEYS.has(key)) {
60
+ out[key] = sanitizeParamValue((value as Record<string, unknown>)[key]);
61
+ }
62
+ }
63
+ return out;
64
+ }
65
+
66
+ // ─── Proxy Resolver ────────────────────────────────────────────────────────
67
+
68
+ /**
69
+ * Resolver closure produced once at pipeline construction. The lazy variant
70
+ * still calls `loader()` per-request (HMR relies on re-importing), but the
71
+ * choice of which branch to take is made once, not on every request.
72
+ */
73
+ export type ProxyResolver = () => ProxyExport | Promise<ProxyExport>;
74
+
75
+ /**
76
+ * Build a proxy resolver closure from the declared source. Called exactly
77
+ * once at `createPipeline` setup time, so the hot path sees only the branch
78
+ * that corresponds to this pipeline's configured variant.
79
+ *
80
+ * Returns `null` when the app has no proxy.ts — the hot path short-circuits
81
+ * around `runProxyPhase` entirely in that case.
82
+ *
83
+ * Accepts the sugar form (a bare `ProxyExport` — function or function array)
84
+ * and normalises it to the static variant. Functions and arrays are
85
+ * structurally distinct from the tagged `{ kind: 'lazy', loader }` object,
86
+ * so discrimination is unambiguous.
87
+ */
88
+ export function makeProxyResolver(
89
+ proxy: ProxyConfig | ProxyExport | undefined
90
+ ): ProxyResolver | null {
91
+ if (proxy === undefined) return null;
92
+ // Sugar: a bare ProxyExport (function or function array) — treat as static.
93
+ if (typeof proxy === 'function' || Array.isArray(proxy)) {
94
+ const exp = proxy;
95
+ return () => exp;
96
+ }
97
+ if (proxy.kind === 'static') {
98
+ const exp = proxy.export;
99
+ return () => exp;
100
+ }
101
+ const loader = proxy.loader;
102
+ return async () => (await loader()).default;
103
+ }
104
+
105
+ // ─── Cookie / Header Helpers ───────────────────────────────────────────────
106
+
107
+ /**
108
+ * Apply all Set-Cookie headers from the cookie jar to a Headers object.
109
+ * Each cookie gets its own Set-Cookie header per RFC 6265 §4.1.
110
+ */
111
+ export function applyCookieJar(headers: Headers): void {
112
+ for (const value of getSetCookieHeaders()) {
113
+ headers.append('Set-Cookie', value);
114
+ }
115
+ }
116
+
117
+ /**
118
+ * Merge framework-managed response headers onto a terminal response without
119
+ * overwriting headers the terminal response already set itself.
120
+ */
121
+ export function mergeMissingHeaders(target: Headers, source: Headers): void {
122
+ const existingKeys = new Set([...target.keys()].map((key) => key.toLowerCase()));
123
+ for (const [key, value] of source.entries()) {
124
+ if (!existingKeys.has(key.toLowerCase())) {
125
+ target.append(key, value);
126
+ }
127
+ }
128
+ }
129
+
130
+ // ─── Mutable Response Cloning ──────────────────────────────────────────────
131
+
132
+ /**
133
+ * Clone a Response into a fresh one whose header bag is guaranteed mutable.
134
+ *
135
+ * `Response.redirect()` and some platform-level passthrough responses (notably
136
+ * on Cloudflare Workers) return objects with frozen header bags. Calling
137
+ * `.set()` or `.append()` on them throws `TypeError: immutable`, which the
138
+ * pipeline can hit when it appends Set-Cookie or Server-Timing entries.
139
+ *
140
+ * The pipeline calls this at the producer sites where user-controlled
141
+ * responses enter the framework — `outcomeToResponse` for all phase outcomes,
142
+ * and `handleRequest` for metadata-route and auto-sitemap user handlers — so
143
+ * downstream code can write headers without runtime feature-detection.
144
+ *
145
+ * The clone is unconditional. This is a deliberate trade: we avoid a
146
+ * try/catch + thrown `TypeError` on every request (the previous probe-based
147
+ * approach paid that cost on the hot path) and accept one cheap Response
148
+ * rewrap at the framework boundary instead.
149
+ */
150
+ export function cloneWithMutableHeaders(response: Response): Response {
151
+ return new Response(response.body, {
152
+ status: response.status,
153
+ statusText: response.statusText,
154
+ headers: new Headers(response.headers),
155
+ });
156
+ }
157
+
158
+ // ─── Redirect Builder ──────────────────────────────────────────────────────
159
+
160
+ /**
161
+ * Build a redirect Response from a RedirectSignal.
162
+ *
163
+ * For RSC payload requests (client navigation), returns 204 + X-Timber-Redirect
164
+ * so the client router can perform a soft SPA redirect. A raw 302 would be
165
+ * turned into an opaque redirect by fetch({redirect:'manual'}), crashing
166
+ * createFromFetch. See design/19-client-navigation.md.
167
+ */
168
+ export function buildRedirectResponse(
169
+ signal: RedirectSignal,
170
+ req: Request,
171
+ headers: Headers
172
+ ): Response {
173
+ const isRsc = (req.headers.get('Accept') ?? '').includes('text/x-component');
174
+ if (isRsc) {
175
+ headers.set('X-Timber-Redirect', signal.location);
176
+ return new Response(null, { status: 204, headers });
177
+ }
178
+ headers.set('Location', signal.location);
179
+ return new Response(null, { status: signal.status, headers });
180
+ }
181
+
182
+ // ─── Instrumentation ───────────────────────────────────────────────────────
183
+
184
+ /**
185
+ * Fire the user's onRequestError hook with request context.
186
+ * Extracts request info from the Request object and calls the instrumentation hook.
187
+ */
188
+ export async function fireOnRequestError(
189
+ error: unknown,
190
+ req: Request,
191
+ phase: 'proxy' | 'handler' | 'render' | 'action' | 'route'
192
+ ): Promise<void> {
193
+ const url = new URL(req.url);
194
+ const headersObj: Record<string, string> = {};
195
+ req.headers.forEach((v, k) => {
196
+ headersObj[k] = v;
197
+ });
198
+
199
+ await callOnRequestError(
200
+ error,
201
+ { method: req.method, path: url.pathname, headers: headersObj },
202
+ { phase, routePath: url.pathname, routeType: 'page', traceId: getTraceId() }
203
+ );
204
+ }
@@ -0,0 +1,167 @@
1
+ /**
2
+ * Pipeline outcome translator — converts a `PhaseOutcome` (the value
3
+ * each phase function returns) into a final `Response`.
4
+ *
5
+ * Lifted out of `pipeline-phases.ts` (TIM-853) so the per-phase try /
6
+ * catch logic and the terminal Response-building logic each live in
7
+ * their own file. The phases produce values; this module is the single
8
+ * source of truth for how those values become wire responses.
9
+ *
10
+ * See design/07-routing.md §"Request Lifecycle".
11
+ */
12
+
13
+ import {
14
+ applyCookieJar,
15
+ buildRedirectResponse,
16
+ cloneWithMutableHeaders,
17
+ fireOnRequestError,
18
+ mergeMissingHeaders,
19
+ } from './pipeline-helpers.js';
20
+ import {
21
+ logProxyError,
22
+ logMiddlewareError,
23
+ logMiddlewareShortCircuit,
24
+ logRenderError,
25
+ } from './logger.js';
26
+ import { markResponseFlushed } from './request-context.js';
27
+ import { RedirectSignal, DenySignal } from './primitives.js';
28
+ import type { PipelineConfig, RouteMatch } from './pipeline.js';
29
+
30
+ // ─── Phase Outcome ─────────────────────────────────────────────────────────
31
+
32
+ export type PhaseName = 'proxy' | 'middleware' | 'render';
33
+
34
+ export type PhaseOutcome =
35
+ | { kind: 'response'; phase: PhaseName; response: Response }
36
+ | { kind: 'redirect'; phase: PhaseName; signal: RedirectSignal }
37
+ | { kind: 'deny'; phase: PhaseName; signal: DenySignal }
38
+ | { kind: 'error'; phase: PhaseName; error: unknown };
39
+
40
+ export interface OutcomeContext {
41
+ req: Request;
42
+ method: string;
43
+ path: string;
44
+ responseHeaders?: Headers;
45
+ match?: RouteMatch;
46
+ }
47
+
48
+ // ─── Translator ────────────────────────────────────────────────────────────
49
+
50
+ /**
51
+ * Terminal outcome handler — converts a `PhaseOutcome` into a final
52
+ * `Response`, applying cookies, building redirects, rendering deny pages
53
+ * and fallback error pages, and firing instrumentation hooks.
54
+ *
55
+ * This is the single source of truth for how phase outputs become wire
56
+ * responses; the per-phase try/catch blocks now produce values, not
57
+ * Responses, so the conversion logic lives in exactly one place.
58
+ */
59
+ export async function outcomeToResponse(
60
+ config: PipelineConfig,
61
+ outcome: PhaseOutcome,
62
+ ctx: OutcomeContext
63
+ ): Promise<Response> {
64
+ switch (outcome.kind) {
65
+ case 'response': {
66
+ // Clone unconditionally so downstream code (cookie/header merge,
67
+ // Server-Timing in createPipeline) can write headers without paying
68
+ // for a try/catch immutability probe per request. User middleware,
69
+ // proxy, and route code may all return `Response.redirect()` or
70
+ // platform-level responses with frozen header bags. See TIM-866.
71
+ const finalResponse = cloneWithMutableHeaders(outcome.response);
72
+
73
+ if (outcome.phase === 'proxy') return finalResponse;
74
+
75
+ if (outcome.phase === 'middleware' && ctx.responseHeaders) {
76
+ applyCookieJar(finalResponse.headers);
77
+ mergeMissingHeaders(finalResponse.headers, ctx.responseHeaders);
78
+ logMiddlewareShortCircuit({
79
+ method: ctx.method,
80
+ path: ctx.path,
81
+ status: finalResponse.status,
82
+ });
83
+ }
84
+
85
+ if (outcome.phase === 'render') {
86
+ markResponseFlushed();
87
+ }
88
+
89
+ return finalResponse;
90
+ }
91
+
92
+ case 'redirect': {
93
+ const headers = ctx.responseHeaders ?? new Headers();
94
+ applyCookieJar(headers);
95
+ return buildRedirectResponse(outcome.signal, ctx.req, headers);
96
+ }
97
+
98
+ case 'deny': {
99
+ const headers = ctx.responseHeaders ?? new Headers();
100
+ applyCookieJar(headers);
101
+ if (config.renderDenyFallback) {
102
+ try {
103
+ // Clone user-supplied deny-page responses so downstream
104
+ // Server-Timing writes are safe against frozen header bags
105
+ // (e.g. user returned Response.redirect from the hook).
106
+ return cloneWithMutableHeaders(
107
+ await config.renderDenyFallback(outcome.signal, ctx.req, headers, ctx.match)
108
+ );
109
+ } catch (denyRenderError) {
110
+ // Deny page rendering failed — log before falling through to bare response.
111
+ // Without this, a crashing deny page produces a blank response with zero
112
+ // server-side signal. See TIM-876.
113
+ logRenderError({ method: ctx.method, path: ctx.path, error: denyRenderError });
114
+ await fireOnRequestError(denyRenderError, ctx.req, 'render');
115
+ if (config.onPipelineError && denyRenderError instanceof Error)
116
+ config.onPipelineError(denyRenderError, 'render');
117
+ }
118
+ }
119
+ return new Response(null, { status: outcome.signal.status, headers });
120
+ }
121
+
122
+ case 'error': {
123
+ if (outcome.phase === 'proxy') {
124
+ logProxyError({ error: outcome.error });
125
+ await fireOnRequestError(outcome.error, ctx.req, 'proxy');
126
+ if (config.onPipelineError && outcome.error instanceof Error)
127
+ config.onPipelineError(outcome.error, 'proxy');
128
+ return new Response(null, { status: 500 });
129
+ }
130
+
131
+ if (outcome.phase === 'middleware') {
132
+ logMiddlewareError({ method: ctx.method, path: ctx.path, error: outcome.error });
133
+ await fireOnRequestError(outcome.error, ctx.req, 'handler');
134
+ if (config.onPipelineError && outcome.error instanceof Error) {
135
+ config.onPipelineError(outcome.error, 'middleware');
136
+ }
137
+ return new Response(null, { status: 500 });
138
+ }
139
+
140
+ const headers = ctx.responseHeaders ?? new Headers();
141
+ applyCookieJar(headers);
142
+ logRenderError({ method: ctx.method, path: ctx.path, error: outcome.error });
143
+ await fireOnRequestError(outcome.error, ctx.req, 'render');
144
+ if (config.onPipelineError && outcome.error instanceof Error)
145
+ config.onPipelineError(outcome.error, 'render');
146
+ if (config.renderFallbackError) {
147
+ try {
148
+ // Clone user-supplied fallback error responses so downstream
149
+ // Server-Timing writes are safe against frozen header bags.
150
+ return cloneWithMutableHeaders(
151
+ await config.renderFallbackError(outcome.error, ctx.req, headers)
152
+ );
153
+ } catch (fallbackRenderError) {
154
+ // Fallback rendering itself failed — log the secondary error before
155
+ // falling through to bare 500. The original render error was already
156
+ // logged above; this captures the fallback renderer's own crash so it
157
+ // doesn't vanish silently. See TIM-876.
158
+ logRenderError({ method: ctx.method, path: ctx.path, error: fallbackRenderError });
159
+ await fireOnRequestError(fallbackRenderError, ctx.req, 'render');
160
+ if (config.onPipelineError && fallbackRenderError instanceof Error)
161
+ config.onPipelineError(fallbackRenderError, 'render');
162
+ }
163
+ }
164
+ return new Response(null, { status: 500 });
165
+ }
166
+ }
167
+ }