@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.
- package/LICENSE +8 -0
- package/dist/_chunks/actions-Dg-ANYHb.js +421 -0
- package/dist/_chunks/actions-Dg-ANYHb.js.map +1 -0
- package/dist/_chunks/{als-registry-B7DbZ2hS.js → als-registry-HS0LGUl2.js} +1 -1
- package/dist/_chunks/als-registry-HS0LGUl2.js.map +1 -0
- package/dist/_chunks/chunk-DYhsFzuS.js +33 -0
- package/dist/_chunks/{debug-gwlJkDuf.js → debug-ECi_61pb.js} +2 -2
- package/dist/_chunks/debug-ECi_61pb.js.map +1 -0
- package/dist/_chunks/define-C77ScO0m.js +106 -0
- package/dist/_chunks/define-C77ScO0m.js.map +1 -0
- package/dist/_chunks/define-CZqDwhSu.js +199 -0
- package/dist/_chunks/define-CZqDwhSu.js.map +1 -0
- package/dist/_chunks/define-cookie-C2IkoFGN.js +94 -0
- package/dist/_chunks/define-cookie-C2IkoFGN.js.map +1 -0
- package/dist/_chunks/{format-DviM89f0.js → dev-warnings-DpGRGoDi.js} +5 -44
- package/dist/_chunks/dev-warnings-DpGRGoDi.js.map +1 -0
- package/dist/_chunks/format-CYBGxKtc.js +14 -0
- package/dist/_chunks/format-CYBGxKtc.js.map +1 -0
- package/dist/_chunks/{interception-BOoWmLUA.js → interception-Dpn_UfAD.js} +171 -97
- package/dist/_chunks/interception-Dpn_UfAD.js.map +1 -0
- package/dist/_chunks/merge-search-params-Cm_KIWDX.js +41 -0
- package/dist/_chunks/merge-search-params-Cm_KIWDX.js.map +1 -0
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js → metadata-routes-DS3eKNmf.js} +1 -1
- package/dist/_chunks/{metadata-routes-Cjmvi3rQ.js.map → metadata-routes-DS3eKNmf.js.map} +1 -1
- package/dist/_chunks/request-context-qMsWgy9C.js +478 -0
- package/dist/_chunks/request-context-qMsWgy9C.js.map +1 -0
- package/dist/_chunks/schema-bridge-C3xl_vfb.js +86 -0
- package/dist/_chunks/schema-bridge-C3xl_vfb.js.map +1 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js +65 -0
- package/dist/_chunks/segment-classify-BDNn6EzD.js.map +1 -0
- package/dist/_chunks/segment-context-fHFLF1PE.js +34 -0
- package/dist/_chunks/segment-context-fHFLF1PE.js.map +1 -0
- package/dist/_chunks/{ssr-data-MjmprTmO.js → ssr-data-DzuI0bIV.js} +1 -1
- package/dist/_chunks/{ssr-data-MjmprTmO.js.map → ssr-data-DzuI0bIV.js.map} +1 -1
- package/dist/_chunks/stale-reload-BX5gL1r-.js +64 -0
- package/dist/_chunks/stale-reload-BX5gL1r-.js.map +1 -0
- package/dist/_chunks/{tracing-CemImE6h.js → tracing-CCYbKn5n.js} +60 -9
- package/dist/_chunks/tracing-CCYbKn5n.js.map +1 -0
- package/dist/_chunks/use-params-Br9YSUFV.js +295 -0
- package/dist/_chunks/use-params-Br9YSUFV.js.map +1 -0
- package/dist/_chunks/{use-query-states-D5KaffOK.js → use-query-states-Lo_s_pw2.js} +4 -4
- package/dist/_chunks/use-query-states-Lo_s_pw2.js.map +1 -0
- package/dist/adapters/cloudflare-dev.d.ts +109 -0
- package/dist/adapters/cloudflare-dev.d.ts.map +1 -0
- package/dist/adapters/cloudflare-dev.js +73 -0
- package/dist/adapters/cloudflare-dev.js.map +1 -0
- package/dist/adapters/cloudflare-kv-cache.d.ts +64 -0
- package/dist/adapters/cloudflare-kv-cache.d.ts.map +1 -0
- package/dist/adapters/cloudflare-kv-cache.js +95 -0
- package/dist/adapters/cloudflare-kv-cache.js.map +1 -0
- package/dist/adapters/cloudflare.d.ts +148 -12
- package/dist/adapters/cloudflare.d.ts.map +1 -1
- package/dist/adapters/cloudflare.js +135 -11
- package/dist/adapters/cloudflare.js.map +1 -1
- package/dist/adapters/compress-module.d.ts.map +1 -1
- package/dist/adapters/nitro.d.ts +17 -1
- package/dist/adapters/nitro.d.ts.map +1 -1
- package/dist/adapters/nitro.js +56 -13
- package/dist/adapters/nitro.js.map +1 -1
- package/dist/cache/cache-api.d.ts +24 -0
- package/dist/cache/cache-api.d.ts.map +1 -0
- package/dist/cache/handler-store.d.ts +31 -0
- package/dist/cache/handler-store.d.ts.map +1 -0
- package/dist/cache/index.d.ts +23 -7
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +142 -80
- package/dist/cache/index.js.map +1 -1
- package/dist/cache/singleflight.d.ts +18 -1
- package/dist/cache/singleflight.d.ts.map +1 -1
- package/dist/cache/sizeof.d.ts +22 -0
- package/dist/cache/sizeof.d.ts.map +1 -0
- package/dist/cache/timber-cache.d.ts +1 -1
- package/dist/cache/timber-cache.d.ts.map +1 -1
- package/dist/cli.d.ts +6 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +8 -3
- package/dist/cli.js.map +1 -1
- package/dist/client/browser-dev.d.ts +27 -1
- package/dist/client/browser-dev.d.ts.map +1 -1
- package/dist/client/browser-entry/action-dispatch.d.ts +17 -0
- package/dist/client/browser-entry/action-dispatch.d.ts.map +1 -0
- package/dist/client/browser-entry/hmr.d.ts +21 -0
- package/dist/client/browser-entry/hmr.d.ts.map +1 -0
- package/dist/client/browser-entry/hydrate.d.ts +46 -0
- package/dist/client/browser-entry/hydrate.d.ts.map +1 -0
- package/dist/client/browser-entry/index.d.ts +30 -0
- package/dist/client/browser-entry/index.d.ts.map +1 -0
- package/dist/client/browser-entry/post-hydration.d.ts +26 -0
- package/dist/client/browser-entry/post-hydration.d.ts.map +1 -0
- package/dist/client/browser-entry/router-init.d.ts +23 -0
- package/dist/client/browser-entry/router-init.d.ts.map +1 -0
- package/dist/client/browser-entry/rsc-stream.d.ts +24 -0
- package/dist/client/browser-entry/rsc-stream.d.ts.map +1 -0
- package/dist/client/browser-entry/scroll.d.ts +19 -0
- package/dist/client/browser-entry/scroll.d.ts.map +1 -0
- package/dist/client/error-boundary.d.ts +12 -5
- package/dist/client/error-boundary.d.ts.map +1 -1
- package/dist/client/error-boundary.js +10 -4
- package/dist/client/error-boundary.js.map +1 -1
- package/dist/client/error-reconstituter.d.ts +54 -0
- package/dist/client/error-reconstituter.d.ts.map +1 -0
- package/dist/client/form.d.ts +3 -3
- package/dist/client/form.d.ts.map +1 -1
- package/dist/client/history.d.ts +19 -4
- package/dist/client/history.d.ts.map +1 -1
- package/dist/client/index.d.ts +7 -21
- package/dist/client/index.d.ts.map +1 -1
- package/dist/client/index.js +210 -1017
- package/dist/client/index.js.map +1 -1
- package/dist/client/internal.d.ts +18 -0
- package/dist/client/internal.d.ts.map +1 -0
- package/dist/client/internal.js +890 -0
- package/dist/client/internal.js.map +1 -0
- package/dist/client/link-pending-store.d.ts +63 -0
- package/dist/client/link-pending-store.d.ts.map +1 -0
- package/dist/client/link.d.ts +90 -32
- package/dist/client/link.d.ts.map +1 -1
- package/dist/client/nav-link-store.d.ts +36 -0
- package/dist/client/nav-link-store.d.ts.map +1 -0
- package/dist/client/navigation-api-types.d.ts +90 -0
- package/dist/client/navigation-api-types.d.ts.map +1 -0
- package/dist/client/navigation-api.d.ts +115 -0
- package/dist/client/navigation-api.d.ts.map +1 -0
- package/dist/client/navigation-context.d.ts +13 -2
- package/dist/client/navigation-context.d.ts.map +1 -1
- package/dist/client/{transition-root.d.ts → navigation-root.d.ts} +42 -8
- package/dist/client/navigation-root.d.ts.map +1 -0
- package/dist/client/nuqs-adapter.d.ts.map +1 -1
- package/dist/client/router-ref.d.ts +1 -1
- package/dist/client/router.d.ts +70 -4
- package/dist/client/router.d.ts.map +1 -1
- package/dist/client/rsc-fetch.d.ts +38 -3
- package/dist/client/rsc-fetch.d.ts.map +1 -1
- package/dist/client/segment-cache.d.ts +1 -1
- package/dist/client/segment-cache.d.ts.map +1 -1
- package/dist/client/segment-outlet.d.ts +63 -0
- package/dist/client/segment-outlet.d.ts.map +1 -0
- package/dist/client/ssr-data.d.ts +13 -4
- package/dist/client/ssr-data.d.ts.map +1 -1
- package/dist/client/stale-reload.d.ts +15 -0
- package/dist/client/stale-reload.d.ts.map +1 -1
- package/dist/client/top-loader.d.ts +5 -5
- package/dist/client/top-loader.d.ts.map +1 -1
- package/dist/client/use-link-status.d.ts +5 -5
- package/dist/client/use-link-status.d.ts.map +1 -1
- package/dist/client/use-params.d.ts +6 -4
- package/dist/client/use-params.d.ts.map +1 -1
- package/dist/client/{use-navigation-pending.d.ts → use-pending-navigation.d.ts} +4 -4
- package/dist/client/use-pending-navigation.d.ts.map +1 -0
- package/dist/client/use-query-states.d.ts +1 -1
- package/dist/client/use-query-states.d.ts.map +1 -1
- package/dist/client/use-router.d.ts +1 -1
- package/dist/codec.d.ts +33 -0
- package/dist/codec.d.ts.map +1 -0
- package/dist/codec.js +2 -0
- package/dist/config-types.d.ts +227 -0
- package/dist/config-types.d.ts.map +1 -0
- package/dist/content/index.d.ts +1 -10
- package/dist/content/index.d.ts.map +1 -1
- package/dist/content/index.js +0 -2
- package/dist/cookies/define-cookie.d.ts +35 -14
- package/dist/cookies/define-cookie.d.ts.map +1 -1
- package/dist/cookies/index.js +1 -83
- package/dist/fonts/css.d.ts +1 -0
- package/dist/fonts/css.d.ts.map +1 -1
- package/dist/index.d.ts +45 -192
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +12357 -11925
- package/dist/index.js.map +1 -1
- package/dist/plugin-context.d.ts +107 -0
- package/dist/plugin-context.d.ts.map +1 -0
- package/dist/plugins/adapter-build.d.ts +1 -1
- package/dist/plugins/adapter-build.d.ts.map +1 -1
- package/dist/plugins/build-manifest.d.ts +2 -2
- package/dist/plugins/build-manifest.d.ts.map +1 -1
- package/dist/plugins/build-report.d.ts +3 -3
- package/dist/plugins/build-report.d.ts.map +1 -1
- package/dist/plugins/client-chunks.d.ts +32 -0
- package/dist/plugins/client-chunks.d.ts.map +1 -0
- package/dist/plugins/content.d.ts +1 -1
- package/dist/plugins/content.d.ts.map +1 -1
- package/dist/plugins/dev-browser-logs.d.ts +84 -0
- package/dist/plugins/dev-browser-logs.d.ts.map +1 -0
- package/dist/plugins/dev-error-overlay.d.ts +26 -1
- package/dist/plugins/dev-error-overlay.d.ts.map +1 -1
- package/dist/plugins/dev-logs.d.ts +1 -1
- package/dist/plugins/dev-logs.d.ts.map +1 -1
- package/dist/plugins/dev-server.d.ts +1 -1
- package/dist/plugins/dev-server.d.ts.map +1 -1
- package/dist/plugins/entries.d.ts +1 -1
- package/dist/plugins/entries.d.ts.map +1 -1
- package/dist/plugins/fonts.d.ts +19 -5
- package/dist/plugins/fonts.d.ts.map +1 -1
- package/dist/plugins/mdx.d.ts +1 -1
- package/dist/plugins/mdx.d.ts.map +1 -1
- package/dist/plugins/routing.d.ts +1 -1
- package/dist/plugins/routing.d.ts.map +1 -1
- package/dist/plugins/server-bundle.d.ts.map +1 -1
- package/dist/plugins/shims.d.ts +6 -5
- package/dist/plugins/shims.d.ts.map +1 -1
- package/dist/plugins/static-build.d.ts +4 -4
- package/dist/plugins/static-build.d.ts.map +1 -1
- package/dist/routing/codegen.d.ts +2 -2
- package/dist/routing/codegen.d.ts.map +1 -1
- package/dist/routing/index.d.ts +2 -0
- package/dist/routing/index.d.ts.map +1 -1
- package/dist/routing/index.js +3 -2
- package/dist/routing/scanner.d.ts.map +1 -1
- package/dist/routing/segment-classify.d.ts +46 -0
- package/dist/routing/segment-classify.d.ts.map +1 -0
- package/dist/routing/status-file-lint.d.ts +2 -1
- package/dist/routing/status-file-lint.d.ts.map +1 -1
- package/dist/routing/types.d.ts +16 -4
- package/dist/routing/types.d.ts.map +1 -1
- package/dist/rsc-runtime/rsc.d.ts +1 -1
- package/dist/rsc-runtime/rsc.d.ts.map +1 -1
- package/dist/rsc-runtime/ssr.d.ts +12 -0
- package/dist/rsc-runtime/ssr.d.ts.map +1 -1
- package/dist/schema-bridge.d.ts +76 -0
- package/dist/schema-bridge.d.ts.map +1 -0
- package/dist/search-params/define.d.ts +139 -0
- package/dist/search-params/define.d.ts.map +1 -0
- package/dist/search-params/index.d.ts +4 -7
- package/dist/search-params/index.d.ts.map +1 -1
- package/dist/search-params/index.js +32 -441
- package/dist/search-params/index.js.map +1 -1
- package/dist/search-params/registry.d.ts +2 -2
- package/dist/search-params/registry.d.ts.map +1 -1
- package/dist/search-params/wrappers.d.ts +53 -0
- package/dist/search-params/wrappers.d.ts.map +1 -0
- package/dist/segment-params/define.d.ts +78 -0
- package/dist/segment-params/define.d.ts.map +1 -0
- package/dist/segment-params/index.d.ts +3 -0
- package/dist/segment-params/index.d.ts.map +1 -0
- package/dist/segment-params/index.js +2 -0
- package/dist/server/access-gate.d.ts +4 -0
- package/dist/server/access-gate.d.ts.map +1 -1
- package/dist/server/action-client.d.ts +25 -6
- package/dist/server/action-client.d.ts.map +1 -1
- package/dist/server/action-encryption.d.ts +76 -0
- package/dist/server/action-encryption.d.ts.map +1 -0
- package/dist/server/action-handler.d.ts.map +1 -1
- package/dist/server/actions.d.ts +3 -6
- package/dist/server/actions.d.ts.map +1 -1
- package/dist/server/als-registry.d.ts +32 -4
- package/dist/server/als-registry.d.ts.map +1 -1
- package/dist/server/build-manifest.d.ts +2 -2
- package/dist/server/build-manifest.d.ts.map +1 -1
- package/dist/server/debug.d.ts +1 -1
- package/dist/server/default-logger.d.ts +22 -0
- package/dist/server/default-logger.d.ts.map +1 -0
- package/dist/server/deny-page-resolver.d.ts +52 -0
- package/dist/server/deny-page-resolver.d.ts.map +1 -0
- package/dist/server/deny-renderer.d.ts.map +1 -1
- package/dist/server/dev-holding-server.d.ts +52 -0
- package/dist/server/dev-holding-server.d.ts.map +1 -0
- package/dist/server/dev-warnings.d.ts +1 -21
- package/dist/server/dev-warnings.d.ts.map +1 -1
- package/dist/server/early-hints.d.ts +13 -5
- package/dist/server/early-hints.d.ts.map +1 -1
- package/dist/server/error-boundary-wrapper.d.ts +7 -1
- package/dist/server/error-boundary-wrapper.d.ts.map +1 -1
- package/dist/server/fallback-error.d.ts +4 -3
- package/dist/server/fallback-error.d.ts.map +1 -1
- package/dist/server/flight-injection-state.d.ts +66 -0
- package/dist/server/flight-injection-state.d.ts.map +1 -0
- package/dist/server/flight-scripts.d.ts +42 -0
- package/dist/server/flight-scripts.d.ts.map +1 -0
- package/dist/server/flush.d.ts.map +1 -1
- package/dist/server/form-data.d.ts +29 -0
- package/dist/server/form-data.d.ts.map +1 -1
- package/dist/server/html-injectors.d.ts +51 -11
- package/dist/server/html-injectors.d.ts.map +1 -1
- package/dist/server/index.d.ts +5 -43
- package/dist/server/index.d.ts.map +1 -1
- package/dist/server/index.js +37 -2798
- package/dist/server/index.js.map +1 -1
- package/dist/server/internal.d.ts +46 -0
- package/dist/server/internal.d.ts.map +1 -0
- package/dist/server/internal.js +2883 -0
- package/dist/server/internal.js.map +1 -0
- package/dist/server/logger.d.ts +25 -7
- package/dist/server/logger.d.ts.map +1 -1
- package/dist/server/middleware-runner.d.ts +19 -4
- package/dist/server/middleware-runner.d.ts.map +1 -1
- package/dist/server/node-stream-transforms.d.ts +113 -0
- package/dist/server/node-stream-transforms.d.ts.map +1 -0
- package/dist/server/page-deny-boundary.d.ts +31 -0
- package/dist/server/page-deny-boundary.d.ts.map +1 -0
- package/dist/server/pipeline-interception.d.ts +1 -1
- package/dist/server/pipeline-interception.d.ts.map +1 -1
- package/dist/server/pipeline-metadata.d.ts +6 -0
- package/dist/server/pipeline-metadata.d.ts.map +1 -1
- package/dist/server/pipeline.d.ts +42 -10
- package/dist/server/pipeline.d.ts.map +1 -1
- package/dist/server/primitives.d.ts +69 -18
- package/dist/server/primitives.d.ts.map +1 -1
- package/dist/server/render-timeout.d.ts +51 -0
- package/dist/server/render-timeout.d.ts.map +1 -0
- package/dist/server/request-context.d.ts +112 -43
- package/dist/server/request-context.d.ts.map +1 -1
- package/dist/server/route-element-builder.d.ts +27 -1
- package/dist/server/route-element-builder.d.ts.map +1 -1
- package/dist/server/route-handler.d.ts.map +1 -1
- package/dist/server/route-matcher.d.ts +9 -2
- package/dist/server/route-matcher.d.ts.map +1 -1
- package/dist/server/rsc-entry/api-handler.d.ts +2 -2
- package/dist/server/rsc-entry/api-handler.d.ts.map +1 -1
- package/dist/server/rsc-entry/error-renderer.d.ts +26 -13
- package/dist/server/rsc-entry/error-renderer.d.ts.map +1 -1
- package/dist/server/rsc-entry/helpers.d.ts +48 -5
- package/dist/server/rsc-entry/helpers.d.ts.map +1 -1
- package/dist/server/rsc-entry/index.d.ts +8 -3
- package/dist/server/rsc-entry/index.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-payload.d.ts +3 -3
- package/dist/server/rsc-entry/rsc-payload.d.ts.map +1 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts +4 -1
- package/dist/server/rsc-entry/rsc-stream.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts +1 -1
- package/dist/server/rsc-entry/ssr-bridge.d.ts.map +1 -1
- package/dist/server/rsc-entry/ssr-renderer.d.ts +19 -4
- package/dist/server/rsc-entry/ssr-renderer.d.ts.map +1 -1
- package/dist/server/safe-load.d.ts +46 -0
- package/dist/server/safe-load.d.ts.map +1 -0
- package/dist/server/sitemap-generator.d.ts +129 -0
- package/dist/server/sitemap-generator.d.ts.map +1 -0
- package/dist/server/sitemap-handler.d.ts +22 -0
- package/dist/server/sitemap-handler.d.ts.map +1 -0
- package/dist/server/slot-resolver.d.ts +1 -1
- package/dist/server/slot-resolver.d.ts.map +1 -1
- package/dist/server/ssr-entry.d.ts +22 -0
- package/dist/server/ssr-entry.d.ts.map +1 -1
- package/dist/server/ssr-render.d.ts +39 -21
- package/dist/server/ssr-render.d.ts.map +1 -1
- package/dist/server/ssr-wrappers.d.ts +50 -0
- package/dist/server/ssr-wrappers.d.ts.map +1 -0
- package/dist/server/status-code-resolver.d.ts +1 -1
- package/dist/server/status-code-resolver.d.ts.map +1 -1
- package/dist/server/stream-utils.d.ts +36 -0
- package/dist/server/stream-utils.d.ts.map +1 -0
- package/dist/server/tracing.d.ts +4 -4
- package/dist/server/tracing.d.ts.map +1 -1
- package/dist/server/tree-builder.d.ts +22 -19
- package/dist/server/tree-builder.d.ts.map +1 -1
- package/dist/server/types.d.ts +1 -4
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/version-skew.d.ts +61 -0
- package/dist/server/version-skew.d.ts.map +1 -0
- package/dist/shared/merge-search-params.d.ts +22 -0
- package/dist/shared/merge-search-params.d.ts.map +1 -0
- package/dist/shims/font-google.d.ts +1 -1
- package/dist/shims/font-google.d.ts.map +1 -1
- package/dist/shims/font-google.js +42 -0
- package/dist/shims/font-google.js.map +1 -0
- package/dist/shims/font-local.d.ts +26 -0
- package/dist/shims/font-local.d.ts.map +1 -0
- package/dist/shims/font-local.js +20 -0
- package/dist/shims/font-local.js.map +1 -0
- package/dist/shims/headers.d.ts +2 -1
- package/dist/shims/headers.d.ts.map +1 -1
- package/dist/shims/navigation-client.d.ts +1 -1
- package/dist/shims/navigation-client.d.ts.map +1 -1
- package/dist/shims/navigation.d.ts +3 -2
- package/dist/shims/navigation.d.ts.map +1 -1
- package/dist/utils/directive-parser.d.ts +5 -2
- package/dist/utils/directive-parser.d.ts.map +1 -1
- package/dist/utils/state-machine.d.ts +80 -0
- package/dist/utils/state-machine.d.ts.map +1 -0
- package/package.json +56 -22
- package/src/adapters/cloudflare-dev.ts +177 -0
- package/src/adapters/cloudflare-kv-cache.ts +142 -0
- package/src/adapters/cloudflare.ts +342 -28
- package/src/adapters/compress-module.ts +24 -4
- package/src/adapters/nitro.ts +52 -8
- package/src/adapters/wrangler.d.ts +7 -0
- package/src/cache/cache-api.ts +38 -0
- package/src/cache/handler-store.ts +68 -0
- package/src/cache/index.ts +81 -18
- package/src/cache/singleflight.ts +62 -4
- package/src/cache/sizeof.ts +31 -0
- package/src/cache/timber-cache.ts +24 -20
- package/src/cli.ts +16 -6
- package/src/client/browser-dev.ts +128 -1
- package/src/client/browser-entry/action-dispatch.ts +116 -0
- package/src/client/browser-entry/hmr.ts +81 -0
- package/src/client/browser-entry/hydrate.ts +145 -0
- package/src/client/browser-entry/index.ts +138 -0
- package/src/client/browser-entry/post-hydration.ts +119 -0
- package/src/client/browser-entry/router-init.ts +193 -0
- package/src/client/browser-entry/rsc-stream.ts +157 -0
- package/src/client/browser-entry/scroll.ts +27 -0
- package/src/client/error-boundary.tsx +48 -16
- package/src/client/error-reconstituter.tsx +65 -0
- package/src/client/form.tsx +9 -7
- package/src/client/history.ts +26 -4
- package/src/client/index.ts +19 -38
- package/src/client/internal.ts +57 -0
- package/src/client/link-pending-store.ts +111 -0
- package/src/client/link.tsx +329 -97
- package/src/client/nav-link-store.ts +47 -0
- package/src/client/navigation-api-types.ts +112 -0
- package/src/client/navigation-api.ts +332 -0
- package/src/client/navigation-context.ts +31 -6
- package/src/client/navigation-root.tsx +342 -0
- package/src/client/nuqs-adapter.tsx +16 -3
- package/src/client/router-ref.ts +1 -1
- package/src/client/router.ts +299 -72
- package/src/client/rsc-fetch.ts +97 -8
- package/src/client/segment-cache.ts +1 -1
- package/src/client/segment-outlet.tsx +86 -0
- package/src/client/ssr-data.ts +13 -5
- package/src/client/stale-reload.ts +72 -3
- package/src/client/top-loader.tsx +16 -8
- package/src/client/use-link-status.ts +7 -7
- package/src/client/use-params.ts +7 -5
- package/src/client/{use-navigation-pending.ts → use-pending-navigation.ts} +6 -6
- package/src/client/use-query-states.ts +3 -3
- package/src/client/use-router.ts +1 -1
- package/src/codec.ts +49 -0
- package/src/config-types.ts +225 -0
- package/src/content/index.ts +5 -13
- package/src/cookies/define-cookie.ts +78 -25
- package/src/cookies/index.ts +8 -0
- package/src/fonts/css.ts +2 -1
- package/src/index.ts +295 -354
- package/src/plugin-context.ts +240 -0
- package/src/plugins/adapter-build.ts +9 -3
- package/src/plugins/build-manifest.ts +13 -2
- package/src/plugins/build-report.ts +3 -3
- package/src/plugins/client-chunks.ts +65 -0
- package/src/plugins/content.ts +1 -1
- package/src/plugins/dev-browser-logs.ts +288 -0
- package/src/plugins/dev-error-overlay.ts +70 -1
- package/src/plugins/dev-logs.ts +1 -1
- package/src/plugins/dev-server.ts +70 -9
- package/src/plugins/entries.ts +71 -10
- package/src/plugins/fonts.ts +168 -61
- package/src/plugins/mdx.ts +1 -1
- package/src/plugins/routing.ts +57 -17
- package/src/plugins/server-action-exports.ts +1 -1
- package/src/plugins/server-bundle.ts +32 -1
- package/src/plugins/shims.ts +135 -35
- package/src/plugins/static-build.ts +17 -11
- package/src/routing/codegen.ts +165 -105
- package/src/routing/index.ts +2 -0
- package/src/routing/scanner.ts +93 -23
- package/src/routing/segment-classify.ts +89 -0
- package/src/routing/status-file-lint.ts +3 -2
- package/src/routing/types.ts +17 -4
- package/src/rsc-runtime/rsc.ts +2 -0
- package/src/rsc-runtime/ssr.ts +50 -0
- package/src/rsc-runtime/vendor-types.d.ts +7 -0
- package/src/{search-params/codecs.ts → schema-bridge.ts} +57 -20
- package/src/search-params/define.ts +482 -0
- package/src/search-params/index.ts +14 -20
- package/src/search-params/registry.ts +2 -2
- package/src/search-params/wrappers.ts +85 -0
- package/src/segment-params/define.ts +279 -0
- package/src/segment-params/index.ts +9 -0
- package/src/server/access-gate.tsx +70 -29
- package/src/server/action-client.ts +46 -11
- package/src/server/action-encryption.ts +144 -0
- package/src/server/action-handler.ts +21 -4
- package/src/server/actions.ts +10 -9
- package/src/server/als-registry.ts +34 -6
- package/src/server/build-manifest.ts +10 -4
- package/src/server/compress.ts +25 -7
- package/src/server/debug.ts +1 -1
- package/src/server/default-logger.ts +99 -0
- package/src/server/deny-page-resolver.ts +154 -0
- package/src/server/deny-renderer.ts +24 -38
- package/src/server/dev-holding-server.ts +185 -0
- package/src/server/dev-warnings.ts +4 -49
- package/src/server/early-hints.ts +36 -15
- package/src/server/error-boundary-wrapper.ts +74 -22
- package/src/server/fallback-error.ts +31 -15
- package/src/server/flight-injection-state.ts +113 -0
- package/src/server/flight-scripts.ts +62 -0
- package/src/server/flush.ts +2 -1
- package/src/server/form-data.ts +76 -0
- package/src/server/html-injectors.ts +280 -120
- package/src/server/index.ts +25 -177
- package/src/server/internal.ts +169 -0
- package/src/server/logger.ts +44 -36
- package/src/server/middleware-runner.ts +31 -4
- package/src/server/node-stream-transforms.ts +509 -0
- package/src/server/page-deny-boundary.tsx +56 -0
- package/src/server/pipeline-interception.ts +17 -16
- package/src/server/pipeline-metadata.ts +13 -0
- package/src/server/pipeline.ts +227 -62
- package/src/server/primitives.ts +111 -28
- package/src/server/render-timeout.ts +108 -0
- package/src/server/request-context.ts +293 -132
- package/src/server/route-element-builder.ts +283 -191
- package/src/server/route-handler.ts +24 -4
- package/src/server/route-matcher.ts +24 -20
- package/src/server/rsc-entry/api-handler.ts +15 -16
- package/src/server/rsc-entry/error-renderer.ts +300 -89
- package/src/server/rsc-entry/helpers.ts +134 -5
- package/src/server/rsc-entry/index.ts +200 -112
- package/src/server/rsc-entry/rsc-payload.ts +65 -18
- package/src/server/rsc-entry/rsc-stream.ts +65 -13
- package/src/server/rsc-entry/ssr-bridge.ts +14 -5
- package/src/server/rsc-entry/ssr-renderer.ts +168 -38
- package/src/server/safe-load.ts +60 -0
- package/src/server/sitemap-generator.ts +338 -0
- package/src/server/sitemap-handler.ts +126 -0
- package/src/server/slot-resolver.ts +244 -229
- package/src/server/ssr-entry.ts +211 -32
- package/src/server/ssr-render.ts +289 -67
- package/src/server/ssr-wrappers.tsx +139 -0
- package/src/server/status-code-resolver.ts +1 -1
- package/src/server/stream-utils.ts +213 -0
- package/src/server/tracing.ts +20 -9
- package/src/server/tree-builder.ts +92 -58
- package/src/server/types.ts +3 -6
- package/src/server/version-skew.ts +104 -0
- package/src/shared/merge-search-params.ts +55 -0
- package/src/shims/font-google.ts +1 -1
- package/src/shims/font-local.ts +34 -0
- package/src/shims/headers.ts +5 -1
- package/src/shims/navigation-client.ts +1 -1
- package/src/shims/navigation.ts +7 -2
- package/src/utils/directive-parser.ts +5 -2
- package/src/utils/state-machine.ts +111 -0
- package/dist/_chunks/als-registry-B7DbZ2hS.js.map +0 -1
- package/dist/_chunks/debug-gwlJkDuf.js.map +0 -1
- package/dist/_chunks/format-DviM89f0.js.map +0 -1
- package/dist/_chunks/interception-BOoWmLUA.js.map +0 -1
- package/dist/_chunks/request-context-DIkVh_jG.js +0 -330
- package/dist/_chunks/request-context-DIkVh_jG.js.map +0 -1
- package/dist/_chunks/tracing-CemImE6h.js.map +0 -1
- package/dist/_chunks/use-cookie-DX-l1_5E.js +0 -91
- package/dist/_chunks/use-cookie-DX-l1_5E.js.map +0 -1
- package/dist/_chunks/use-query-states-D5KaffOK.js.map +0 -1
- package/dist/cache/register-cached-function.d.ts +0 -17
- package/dist/cache/register-cached-function.d.ts.map +0 -1
- package/dist/client/browser-entry.d.ts +0 -21
- package/dist/client/browser-entry.d.ts.map +0 -1
- package/dist/client/link-status-provider.d.ts +0 -11
- package/dist/client/link-status-provider.d.ts.map +0 -1
- package/dist/client/transition-root.d.ts.map +0 -1
- package/dist/client/use-navigation-pending.d.ts.map +0 -1
- package/dist/cookies/index.js.map +0 -1
- package/dist/plugins/cache-transform.d.ts +0 -36
- package/dist/plugins/cache-transform.d.ts.map +0 -1
- package/dist/plugins/dynamic-transform.d.ts +0 -72
- package/dist/plugins/dynamic-transform.d.ts.map +0 -1
- package/dist/search-params/analyze.d.ts +0 -54
- package/dist/search-params/analyze.d.ts.map +0 -1
- package/dist/search-params/builtin-codecs.d.ts +0 -105
- package/dist/search-params/builtin-codecs.d.ts.map +0 -1
- package/dist/search-params/codecs.d.ts +0 -53
- package/dist/search-params/codecs.d.ts.map +0 -1
- package/dist/search-params/create.d.ts +0 -106
- package/dist/search-params/create.d.ts.map +0 -1
- package/dist/server/prerender.d.ts +0 -77
- package/dist/server/prerender.d.ts.map +0 -1
- package/dist/server/response-cache.d.ts +0 -54
- package/dist/server/response-cache.d.ts.map +0 -1
- package/src/cache/register-cached-function.ts +0 -103
- package/src/client/browser-entry.ts +0 -678
- package/src/client/link-status-provider.tsx +0 -30
- package/src/client/transition-root.tsx +0 -166
- package/src/plugins/cache-transform.ts +0 -199
- package/src/plugins/dynamic-transform.ts +0 -161
- package/src/search-params/analyze.ts +0 -192
- package/src/search-params/builtin-codecs.ts +0 -228
- package/src/search-params/create.ts +0 -321
- package/src/server/prerender.ts +0 -139
- package/src/server/response-cache.ts +0 -410
|
@@ -0,0 +1,279 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* defineSegmentParams — factory for typed route param coercion.
|
|
3
|
+
*
|
|
4
|
+
* Creates a ParamsDefinition that coerces raw string params from the
|
|
5
|
+
* URL into typed values. Used by exporting from params.ts (segment-level)
|
|
6
|
+
* convention file.
|
|
7
|
+
*
|
|
8
|
+
* Reuses the shared Codec<T> protocol with Standard Schema auto-detection,
|
|
9
|
+
* same pattern as defineSearchParams. Runtime constraints are stricter:
|
|
10
|
+
* - serialize must return string (not null — path segments can't be omitted)
|
|
11
|
+
* - parse throwing → 404 (invalid param value)
|
|
12
|
+
*
|
|
13
|
+
* Design doc: design/07a-route-params-triage.md
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import type { Codec } from '../codec.js';
|
|
17
|
+
import {
|
|
18
|
+
type StandardSchemaV1,
|
|
19
|
+
validateSync,
|
|
20
|
+
isStandardSchema,
|
|
21
|
+
isCodec,
|
|
22
|
+
} from '../schema-bridge.js';
|
|
23
|
+
|
|
24
|
+
// ---------------------------------------------------------------------------
|
|
25
|
+
// Server-only ALS reference for .get()
|
|
26
|
+
// ---------------------------------------------------------------------------
|
|
27
|
+
|
|
28
|
+
// Same pattern as search-params: eagerly registered at server startup
|
|
29
|
+
// to avoid dynamic imports that lose ALS context. See TIM-523.
|
|
30
|
+
let _getSegmentParamsFn: (() => Promise<Record<string, string | string[]>>) | undefined;
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Register the getSegmentParams function. Called once at module load time
|
|
34
|
+
* from request-context.ts to avoid dynamic import at call time.
|
|
35
|
+
* @internal
|
|
36
|
+
*/
|
|
37
|
+
export function _setGetSegmentParamsFn(fn: () => Promise<Record<string, string | string[]>>): void {
|
|
38
|
+
_getSegmentParamsFn = fn;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function getSegmentParamsFromAls(): Promise<Record<string, string | string[]>> {
|
|
42
|
+
if (!_getSegmentParamsFn) {
|
|
43
|
+
throw new Error(
|
|
44
|
+
'[timber] segmentParams.get() is only available on the server. ' +
|
|
45
|
+
'Use useSegmentParams() on the client.'
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
return _getSegmentParamsFn();
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ---------------------------------------------------------------------------
|
|
52
|
+
// Types
|
|
53
|
+
// ---------------------------------------------------------------------------
|
|
54
|
+
|
|
55
|
+
/** Infer the output type from a Codec or StandardSchemaV1. */
|
|
56
|
+
export type InferParamField<V> =
|
|
57
|
+
V extends Codec<infer T> ? T : V extends StandardSchemaV1<infer T> ? T : never;
|
|
58
|
+
|
|
59
|
+
/** Acceptable field value for defineParams: a Codec or a Standard Schema. */
|
|
60
|
+
export type ParamField<T = unknown> = Codec<T> | StandardSchemaV1<T>;
|
|
61
|
+
|
|
62
|
+
export type { StandardSchemaV1 };
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* A typed route params definition.
|
|
66
|
+
*
|
|
67
|
+
* Returned by defineParams(). Provides parse (string → typed) and
|
|
68
|
+
* serialize (typed → string) for each declared param.
|
|
69
|
+
*/
|
|
70
|
+
export interface ParamsDefinition<T extends Record<string, unknown>> {
|
|
71
|
+
/** Parse raw string params into typed values. Throws on invalid values. */
|
|
72
|
+
parse(raw: Record<string, string | string[]>): T;
|
|
73
|
+
|
|
74
|
+
/** Serialize typed values back to strings for URL construction. */
|
|
75
|
+
serialize(values: T): Record<string, string>;
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Get typed segment params from the current request context (ALS).
|
|
79
|
+
*
|
|
80
|
+
* Server-only. Reads getSegmentParams() from ALS and coerces through
|
|
81
|
+
* this definition's codecs, returning fully typed params.
|
|
82
|
+
*
|
|
83
|
+
* ```ts
|
|
84
|
+
* // app/products/[id]/params.ts
|
|
85
|
+
* export const segmentParams = defineSegmentParams({ id: z.coerce.number() })
|
|
86
|
+
*
|
|
87
|
+
* // app/products/[id]/page.tsx
|
|
88
|
+
* import { segmentParams } from './params'
|
|
89
|
+
* export default async function Page() {
|
|
90
|
+
* const { id } = await segmentParams.get() // id: number
|
|
91
|
+
* }
|
|
92
|
+
* ```
|
|
93
|
+
*/
|
|
94
|
+
get(): Promise<T>;
|
|
95
|
+
|
|
96
|
+
/** Read-only codec map. */
|
|
97
|
+
codecs: { [K in keyof T]: Codec<T[K]> };
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// ---------------------------------------------------------------------------
|
|
101
|
+
// Internal helpers
|
|
102
|
+
// ---------------------------------------------------------------------------
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* Wrap a Standard Schema into a Codec for route params.
|
|
106
|
+
*
|
|
107
|
+
* Unlike fromSchema for search params:
|
|
108
|
+
* - Parse throws on failure (no fallback to default)
|
|
109
|
+
* - Serialize returns string (not null)
|
|
110
|
+
*/
|
|
111
|
+
function fromParamSchema<T>(fieldName: string, schema: StandardSchemaV1<T>): Codec<T> {
|
|
112
|
+
return {
|
|
113
|
+
parse(value: string | string[] | undefined): T {
|
|
114
|
+
// Route params are always strings (single segment) or string[] (catch-all)
|
|
115
|
+
const input = Array.isArray(value) ? value : value;
|
|
116
|
+
|
|
117
|
+
const result = validateSync(schema, input);
|
|
118
|
+
if (!result.issues) {
|
|
119
|
+
return result.value;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// For route params, parse failure means the param is invalid → throw
|
|
123
|
+
const messages = result.issues.map((i) => i.message).join(', ');
|
|
124
|
+
throw new Error(`[timber] Param '${fieldName}' coercion failed: ${messages}`);
|
|
125
|
+
},
|
|
126
|
+
|
|
127
|
+
serialize(value: T): string | null {
|
|
128
|
+
if (value === null || value === undefined) {
|
|
129
|
+
return null;
|
|
130
|
+
}
|
|
131
|
+
// Catch-all segments produce arrays — join with '/' for path reconstruction
|
|
132
|
+
if (Array.isArray(value)) {
|
|
133
|
+
return value.join('/');
|
|
134
|
+
}
|
|
135
|
+
return String(value);
|
|
136
|
+
},
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* Resolve a field value to a Codec. Auto-detects Standard Schema objects.
|
|
142
|
+
*/
|
|
143
|
+
function resolveField(fieldName: string, value: ParamField): Codec<unknown> {
|
|
144
|
+
if (isCodec(value)) {
|
|
145
|
+
return value;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
if (isStandardSchema(value)) {
|
|
149
|
+
return fromParamSchema(fieldName, value);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
throw new Error(
|
|
153
|
+
`[timber] defineSegmentParams: field '${fieldName}' is not a valid codec or Standard Schema. ` +
|
|
154
|
+
`Expected an object with { parse, serialize } methods, or a Standard Schema object ` +
|
|
155
|
+
`(Zod, Valibot, ArkType).`
|
|
156
|
+
);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* Validate that no codec's serialize returns null.
|
|
161
|
+
* Route params are structural — they must produce a valid path segment.
|
|
162
|
+
*/
|
|
163
|
+
function validateSerialize(codecMap: Record<string, Codec<unknown>>): void {
|
|
164
|
+
for (const [key, codec] of Object.entries(codecMap)) {
|
|
165
|
+
// Test serialize with a sample parsed value to check for null
|
|
166
|
+
// We can't exhaustively test, but we can check that serialize(parse("test"))
|
|
167
|
+
// doesn't return null for a basic input.
|
|
168
|
+
try {
|
|
169
|
+
const testValue = codec.parse('test');
|
|
170
|
+
const serialized = codec.serialize(testValue);
|
|
171
|
+
if (serialized === null) {
|
|
172
|
+
throw new Error(
|
|
173
|
+
`[timber] defineSegmentParams: field '${key}' codec.serialize() returned null.\n` +
|
|
174
|
+
` Route params are path segments — they cannot be omitted.\n` +
|
|
175
|
+
` Ensure serialize() always returns a string.`
|
|
176
|
+
);
|
|
177
|
+
}
|
|
178
|
+
} catch (e) {
|
|
179
|
+
// parse('test') may throw for strict codecs (e.g., number-only).
|
|
180
|
+
// That's fine — it means the codec validates. We only care about
|
|
181
|
+
// serialize returning null, which we can't test without a valid value.
|
|
182
|
+
if (e instanceof Error && e.message.includes('returned null')) {
|
|
183
|
+
throw e;
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
// ---------------------------------------------------------------------------
|
|
190
|
+
// Factory
|
|
191
|
+
// ---------------------------------------------------------------------------
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* Create a ParamsDefinition from a map of codecs and/or Standard Schema objects.
|
|
195
|
+
*
|
|
196
|
+
* ```ts
|
|
197
|
+
* // app/products/[id]/layout.tsx
|
|
198
|
+
* import { defineSegmentParams } from '@timber-js/app/segment-params'
|
|
199
|
+
* import { z } from 'zod/v4'
|
|
200
|
+
*
|
|
201
|
+
* export const segmentParams = defineSegmentParams({
|
|
202
|
+
* id: z.coerce.number().int().positive(),
|
|
203
|
+
* })
|
|
204
|
+
* ```
|
|
205
|
+
*/
|
|
206
|
+
export function defineSegmentParams<C extends Record<string, ParamField>>(
|
|
207
|
+
codecs: C
|
|
208
|
+
): ParamsDefinition<{ [K in keyof C]: InferParamField<C[K]> }> {
|
|
209
|
+
type T = { [K in keyof C]: InferParamField<C[K]> };
|
|
210
|
+
|
|
211
|
+
const resolvedCodecs: Record<string, Codec<unknown>> = {};
|
|
212
|
+
|
|
213
|
+
for (const [key, value] of Object.entries(codecs)) {
|
|
214
|
+
resolvedCodecs[key] = resolveField(key, value as ParamField);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
// Validate that serialize doesn't return null
|
|
218
|
+
validateSerialize(resolvedCodecs);
|
|
219
|
+
|
|
220
|
+
// ---- parse ----
|
|
221
|
+
function parse(raw: Record<string, string | string[]>): T {
|
|
222
|
+
const result: Record<string, unknown> = {};
|
|
223
|
+
|
|
224
|
+
for (const [key, codec] of Object.entries(resolvedCodecs)) {
|
|
225
|
+
const rawValue = raw[key];
|
|
226
|
+
// Route params are always present (the route matched)
|
|
227
|
+
result[key] = codec.parse(rawValue);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return result as T;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
// ---- serialize ----
|
|
234
|
+
function serialize(values: T): Record<string, string> {
|
|
235
|
+
const result: Record<string, string> = {};
|
|
236
|
+
|
|
237
|
+
for (const [key, codec] of Object.entries(resolvedCodecs)) {
|
|
238
|
+
const serialized = codec.serialize(values[key as keyof T] as unknown);
|
|
239
|
+
if (serialized === null) {
|
|
240
|
+
throw new Error(
|
|
241
|
+
`[timber] params.serialize: field '${key}' serialized to null. ` +
|
|
242
|
+
`Route params must produce a valid path segment.`
|
|
243
|
+
);
|
|
244
|
+
}
|
|
245
|
+
result[key] = serialized;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
return result;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// ---- get ----
|
|
252
|
+
// ALS-backed: reads segment params from the current request context.
|
|
253
|
+
// Server-only — throws on client.
|
|
254
|
+
//
|
|
255
|
+
// The pipeline already coerces params via coerceSegmentParams() which
|
|
256
|
+
// calls parse() and stores typed values in ALS via setSegmentParams().
|
|
257
|
+
// We return those directly instead of re-parsing, because codecs may
|
|
258
|
+
// not be idempotent (e.g., a codec that only accepts raw strings would
|
|
259
|
+
// throw if given an already-parsed value). See TIM-574.
|
|
260
|
+
async function get(): Promise<T> {
|
|
261
|
+
if (typeof window !== 'undefined') {
|
|
262
|
+
throw new Error(
|
|
263
|
+
'[timber] segmentParams.get() is server-only. ' + 'Use useSegmentParams() on the client.'
|
|
264
|
+
);
|
|
265
|
+
}
|
|
266
|
+
const params = await getSegmentParamsFromAls();
|
|
267
|
+
// params are already coerced by the pipeline — return as-is.
|
|
268
|
+
return params as unknown as T;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
const definition: ParamsDefinition<T> = {
|
|
272
|
+
parse,
|
|
273
|
+
serialize,
|
|
274
|
+
get,
|
|
275
|
+
codecs: resolvedCodecs as { [K in keyof T]: Codec<T[K]> },
|
|
276
|
+
};
|
|
277
|
+
|
|
278
|
+
return definition;
|
|
279
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
// @timber-js/app/segment-params — Typed route param coercion
|
|
2
|
+
//
|
|
3
|
+
// This is the import path for defineSegmentParams and related types.
|
|
4
|
+
// For search params, import from @timber-js/app/search-params instead.
|
|
5
|
+
//
|
|
6
|
+
// See design/07-routing.md §"params.ts Convention File"
|
|
7
|
+
|
|
8
|
+
export type { ParamsDefinition, InferParamField, ParamField } from './define.js';
|
|
9
|
+
export { defineSegmentParams } from './define.js';
|
|
@@ -17,6 +17,8 @@ import { DenySignal, RedirectSignal } from './primitives.js';
|
|
|
17
17
|
import type { AccessGateProps, SlotAccessGateProps, ReactElement } from './tree-builder.js';
|
|
18
18
|
import { withSpan, setSpanAttribute } from './tracing.js';
|
|
19
19
|
import { isDebug } from './debug.js';
|
|
20
|
+
import type { DenyPageEntry } from './deny-page-resolver.js';
|
|
21
|
+
import { renderMatchingDenyPage, setDenyStatus } from './deny-page-resolver.js';
|
|
20
22
|
|
|
21
23
|
// ─── AccessGate ─────────────────────────────────────────────────────────────
|
|
22
24
|
|
|
@@ -35,24 +37,20 @@ import { isDebug } from './debug.js';
|
|
|
35
37
|
* gets the same data by calling the same cached functions (React.cache dedup).
|
|
36
38
|
*/
|
|
37
39
|
export function AccessGate(props: AccessGateProps): ReactElement | Promise<ReactElement> {
|
|
38
|
-
const { accessFn,
|
|
40
|
+
const { accessFn, segmentName, verdict, denyPages, children } = props;
|
|
39
41
|
|
|
40
42
|
// Fast path: replay pre-computed verdict from the pre-render pass.
|
|
41
|
-
// This is synchronous — Suspense boundaries cannot interfere with the
|
|
42
|
-
// status code because the signal throws before any async work.
|
|
43
43
|
if (verdict !== undefined) {
|
|
44
44
|
if (verdict === 'pass') {
|
|
45
45
|
return children;
|
|
46
46
|
}
|
|
47
|
-
// Throw the stored DenySignal or RedirectSignal synchronously.
|
|
48
|
-
// React catches this as a render-phase throw — the flush controller
|
|
49
|
-
// produces the correct HTTP status code.
|
|
50
47
|
throw verdict;
|
|
51
48
|
}
|
|
52
49
|
|
|
53
|
-
//
|
|
54
|
-
//
|
|
55
|
-
|
|
50
|
+
// Primary path: call accessFn directly during render.
|
|
51
|
+
// If denyPages is provided, catch DenySignal and render the deny page
|
|
52
|
+
// in-tree — no throw reaches React Flight, no second render pass.
|
|
53
|
+
return accessGateFallback(accessFn, segmentName, denyPages, children);
|
|
56
54
|
}
|
|
57
55
|
|
|
58
56
|
/**
|
|
@@ -61,28 +59,41 @@ export function AccessGate(props: AccessGateProps): ReactElement | Promise<React
|
|
|
61
59
|
*/
|
|
62
60
|
async function accessGateFallback(
|
|
63
61
|
accessFn: AccessGateProps['accessFn'],
|
|
64
|
-
params: AccessGateProps['params'],
|
|
65
|
-
searchParams: AccessGateProps['searchParams'],
|
|
66
62
|
segmentName: AccessGateProps['segmentName'],
|
|
63
|
+
denyPages: DenyPageEntry[] | undefined,
|
|
67
64
|
children: ReactElement
|
|
68
65
|
): Promise<ReactElement> {
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
66
|
+
try {
|
|
67
|
+
await withSpan('timber.access', { 'timber.segment': segmentName ?? 'unknown' }, async () => {
|
|
68
|
+
try {
|
|
69
|
+
await accessFn();
|
|
70
|
+
await setSpanAttribute('timber.result', 'pass');
|
|
71
|
+
} catch (error: unknown) {
|
|
72
|
+
if (error instanceof DenySignal) {
|
|
73
|
+
await setSpanAttribute('timber.result', 'deny');
|
|
74
|
+
await setSpanAttribute('timber.deny_status', error.status);
|
|
75
|
+
if (error.sourceFile) {
|
|
76
|
+
await setSpanAttribute('timber.deny_file', error.sourceFile);
|
|
77
|
+
}
|
|
78
|
+
} else if (error instanceof RedirectSignal) {
|
|
79
|
+
await setSpanAttribute('timber.result', 'redirect');
|
|
79
80
|
}
|
|
80
|
-
|
|
81
|
-
|
|
81
|
+
throw error;
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
} catch (error: unknown) {
|
|
85
|
+
// Catch DenySignal and render the deny page in-tree.
|
|
86
|
+
// No throw reaches React Flight — clean stream, single render pass.
|
|
87
|
+
// RedirectSignal and other errors propagate normally.
|
|
88
|
+
if (error instanceof DenySignal && denyPages) {
|
|
89
|
+
const denyElement = renderMatchingDenyPage(denyPages, error.status, error.data);
|
|
90
|
+
if (denyElement) {
|
|
91
|
+
setDenyStatus(error.status);
|
|
92
|
+
return denyElement;
|
|
82
93
|
}
|
|
83
|
-
throw error;
|
|
84
94
|
}
|
|
85
|
-
|
|
95
|
+
throw error;
|
|
96
|
+
}
|
|
86
97
|
|
|
87
98
|
return children;
|
|
88
99
|
}
|
|
@@ -96,18 +107,27 @@ async function accessGateFallback(
|
|
|
96
107
|
* The HTTP status code is unaffected — slot denial is a UI concern, not
|
|
97
108
|
* a protocol concern. The parent layout and sibling slots still render.
|
|
98
109
|
*
|
|
110
|
+
* DeniedComponent is passed instead of a pre-built element so that
|
|
111
|
+
* DenySignal.data can be forwarded as the dangerouslyPassData prop
|
|
112
|
+
* and the slot name can be passed as the slot prop. See TIM-488.
|
|
113
|
+
*
|
|
99
114
|
* redirect() in slot access.ts is a dev-mode error — redirecting from a
|
|
100
115
|
* slot doesn't make architectural sense.
|
|
101
116
|
*/
|
|
102
117
|
export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactElement> {
|
|
103
|
-
const { accessFn,
|
|
118
|
+
const { accessFn, DeniedComponent, slotName, createElement, defaultFallback, children } = props;
|
|
104
119
|
|
|
105
120
|
try {
|
|
106
|
-
await accessFn(
|
|
121
|
+
await accessFn();
|
|
107
122
|
} catch (error: unknown) {
|
|
108
123
|
// DenySignal → graceful degradation (denied.tsx → default.tsx → null)
|
|
124
|
+
// Build the denied element dynamically so DenySignal.data is forwarded.
|
|
109
125
|
if (error instanceof DenySignal) {
|
|
110
|
-
return
|
|
126
|
+
return (
|
|
127
|
+
buildDeniedFallback(DeniedComponent, slotName, error.data, createElement) ??
|
|
128
|
+
defaultFallback ??
|
|
129
|
+
null
|
|
130
|
+
);
|
|
111
131
|
}
|
|
112
132
|
|
|
113
133
|
// RedirectSignal in slot access → dev-mode error.
|
|
@@ -123,7 +143,11 @@ export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactE
|
|
|
123
143
|
);
|
|
124
144
|
}
|
|
125
145
|
// In production, treat as a deny — render fallback rather than crash.
|
|
126
|
-
return
|
|
146
|
+
return (
|
|
147
|
+
buildDeniedFallback(DeniedComponent, slotName, undefined, createElement) ??
|
|
148
|
+
defaultFallback ??
|
|
149
|
+
null
|
|
150
|
+
);
|
|
127
151
|
}
|
|
128
152
|
|
|
129
153
|
// Unhandled error — re-throw so error boundaries can catch it.
|
|
@@ -141,3 +165,20 @@ export async function SlotAccessGate(props: SlotAccessGateProps): Promise<ReactE
|
|
|
141
165
|
// Access passed — render slot content.
|
|
142
166
|
return children;
|
|
143
167
|
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Build the denied fallback element dynamically with DenySignal data.
|
|
171
|
+
* Returns null if no DeniedComponent is available.
|
|
172
|
+
*/
|
|
173
|
+
function buildDeniedFallback(
|
|
174
|
+
DeniedComponent: SlotAccessGateProps['DeniedComponent'],
|
|
175
|
+
slotName: string,
|
|
176
|
+
data: unknown,
|
|
177
|
+
createElement: SlotAccessGateProps['createElement']
|
|
178
|
+
): ReactElement | null {
|
|
179
|
+
if (!DeniedComponent) return null;
|
|
180
|
+
return createElement(DeniedComponent, {
|
|
181
|
+
slot: slotName,
|
|
182
|
+
dangerouslyPassData: data,
|
|
183
|
+
});
|
|
184
|
+
}
|
|
@@ -138,22 +138,45 @@ export interface ActionBuilder<TCtx> {
|
|
|
138
138
|
/** Declare the input schema. Validation errors are returned typed. */
|
|
139
139
|
schema<TInput>(schema: ActionSchema<TInput>): ActionBuilderWithSchema<TCtx, TInput>;
|
|
140
140
|
/** Define the action body without input validation. */
|
|
141
|
-
action<TData>(
|
|
141
|
+
action<TData>(
|
|
142
|
+
fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>
|
|
143
|
+
): ActionFn<undefined, TData>;
|
|
142
144
|
}
|
|
143
145
|
|
|
144
146
|
/** Builder after .schema() has been called. */
|
|
145
147
|
export interface ActionBuilderWithSchema<TCtx, TInput> {
|
|
146
148
|
/** Define the action body with validated input. */
|
|
147
|
-
action<TData>(fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>): ActionFn<TData>;
|
|
149
|
+
action<TData>(fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>): ActionFn<TInput, TData>;
|
|
148
150
|
}
|
|
149
151
|
|
|
150
152
|
/**
|
|
151
|
-
* The final action function. Callable
|
|
153
|
+
* The final action function. Callable three ways:
|
|
152
154
|
* - Direct: action(input) → Promise<ActionResult<TData>>
|
|
153
155
|
* - React useActionState: action(prevState, formData) → Promise<ActionResult<TData>>
|
|
156
|
+
* - React <form action={fn}>: action(formData) → void (return value ignored by React)
|
|
157
|
+
*
|
|
158
|
+
* The third overload exists purely for type compatibility with React's
|
|
159
|
+
* `<form action>` prop, which expects `(formData: FormData) => void`.
|
|
160
|
+
* At runtime the function still returns Promise<ActionResult>, but React
|
|
161
|
+
* discards it. This lets validated actions be passed directly to forms
|
|
162
|
+
* without casts.
|
|
163
|
+
*/
|
|
164
|
+
/**
|
|
165
|
+
* Map schema output keys to `string | undefined` for form-facing APIs.
|
|
166
|
+
* HTML form values are always strings, and fields can be absent.
|
|
167
|
+
* Gives autocomplete for field names without lying about value types.
|
|
154
168
|
*/
|
|
155
|
-
export type
|
|
156
|
-
|
|
169
|
+
export type InputHint<T> =
|
|
170
|
+
T extends Record<string, unknown> ? { [K in keyof T]: string | undefined } : T;
|
|
171
|
+
|
|
172
|
+
export type ActionFn<TInput = unknown, TData = unknown> = {
|
|
173
|
+
/** <form action={fn}> compatibility — React discards the return value. */
|
|
174
|
+
(formData: FormData): void;
|
|
175
|
+
/** Direct call: action(input) — optional when TInput is undefined (no-schema actions). */
|
|
176
|
+
(
|
|
177
|
+
...args: undefined extends TInput ? [input?: TInput] : [input: TInput]
|
|
178
|
+
): Promise<ActionResult<TData>>;
|
|
179
|
+
/** React useActionState: action(prevState, formData) */
|
|
157
180
|
(prevState: ActionResult<TData> | null, formData: FormData): Promise<ActionResult<TData>>;
|
|
158
181
|
};
|
|
159
182
|
|
|
@@ -183,8 +206,9 @@ async function runActionMiddleware<TCtx>(
|
|
|
183
206
|
|
|
184
207
|
// Re-export parseFormData for use throughout the framework
|
|
185
208
|
import { parseFormData } from './form-data.js';
|
|
186
|
-
import { formatSize } from '
|
|
209
|
+
import { formatSize } from '../utils/format.js';
|
|
187
210
|
import { isDebug, isDevMode } from './debug.js';
|
|
211
|
+
import { RedirectSignal, DenySignal } from './primitives.js';
|
|
188
212
|
|
|
189
213
|
/**
|
|
190
214
|
* Extract validation errors from a schema error.
|
|
@@ -286,7 +310,7 @@ export function createActionClient<TCtx = Record<string, never>>(
|
|
|
286
310
|
function buildAction<TInput, TData>(
|
|
287
311
|
schema: ActionSchema<TInput> | undefined,
|
|
288
312
|
fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>
|
|
289
|
-
): ActionFn<TData> {
|
|
313
|
+
): ActionFn<TInput, TData> {
|
|
290
314
|
async function actionHandler(...args: unknown[]): Promise<ActionResult<TData>> {
|
|
291
315
|
try {
|
|
292
316
|
// Run middleware
|
|
@@ -366,22 +390,33 @@ export function createActionClient<TCtx = Record<string, never>>(
|
|
|
366
390
|
const data = await fn({ ctx, input });
|
|
367
391
|
return { data };
|
|
368
392
|
} catch (error) {
|
|
393
|
+
// Re-throw redirect/deny signals — these are control flow, not errors.
|
|
394
|
+
// They must propagate to executeAction() which converts them to proper
|
|
395
|
+
// HTTP responses (302 redirect, 4xx deny). Catching them here would
|
|
396
|
+
// wrap them as INTERNAL_ERROR and break redirect()/redirectExternal()/deny().
|
|
397
|
+
if (error instanceof RedirectSignal || error instanceof DenySignal) {
|
|
398
|
+
throw error;
|
|
399
|
+
}
|
|
369
400
|
return handleActionError(error);
|
|
370
401
|
}
|
|
371
402
|
}
|
|
372
403
|
|
|
373
|
-
return actionHandler as ActionFn<TData>;
|
|
404
|
+
return actionHandler as ActionFn<TInput, TData>;
|
|
374
405
|
}
|
|
375
406
|
|
|
376
407
|
return {
|
|
377
408
|
schema<TInput>(schema: ActionSchema<TInput>) {
|
|
378
409
|
return {
|
|
379
|
-
action<TData>(
|
|
410
|
+
action<TData>(
|
|
411
|
+
fn: (ctx: ActionContext<TCtx, TInput>) => Promise<TData>
|
|
412
|
+
): ActionFn<TInput, TData> {
|
|
380
413
|
return buildAction(schema, fn);
|
|
381
414
|
},
|
|
382
415
|
};
|
|
383
416
|
},
|
|
384
|
-
action<TData>(
|
|
417
|
+
action<TData>(
|
|
418
|
+
fn: (ctx: ActionContext<TCtx, undefined>) => Promise<TData>
|
|
419
|
+
): ActionFn<undefined, TData> {
|
|
385
420
|
return buildAction(undefined, fn as (ctx: ActionContext<TCtx, unknown>) => Promise<TData>);
|
|
386
421
|
},
|
|
387
422
|
};
|
|
@@ -410,7 +445,7 @@ export function createActionClient<TCtx = Record<string, never>>(
|
|
|
410
445
|
export function validated<TInput, TData>(
|
|
411
446
|
schema: ActionSchema<TInput>,
|
|
412
447
|
handler: (input: TInput) => Promise<TData>
|
|
413
|
-
): ActionFn<TData> {
|
|
448
|
+
): ActionFn<TInput, TData> {
|
|
414
449
|
return createActionClient()
|
|
415
450
|
.schema(schema)
|
|
416
451
|
.action(async ({ input }) => handler(input));
|