@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
|
@@ -7,17 +7,18 @@
|
|
|
7
7
|
*
|
|
8
8
|
* This module handles:
|
|
9
9
|
* 1. Loading page/layout components from the segment chain
|
|
10
|
-
* 2.
|
|
10
|
+
* 2. Collecting access.ts checks (executed inside render by AccessPreRunner)
|
|
11
11
|
* 3. Resolving metadata (static object or async function, both exported as `metadata`)
|
|
12
12
|
* 4. Building the React element tree (page → error boundaries → access gates → layouts)
|
|
13
13
|
* 5. Resolving parallel slots
|
|
14
|
+
* 6. Wrapping the tree with AccessPreRunner for React.cache-scoped access checks
|
|
14
15
|
*
|
|
15
16
|
* See design/02-rendering-pipeline.md, design/04-authorization.md
|
|
16
17
|
*/
|
|
17
18
|
|
|
18
19
|
import { createElement } from 'react';
|
|
19
20
|
|
|
20
|
-
import { withSpan
|
|
21
|
+
import { withSpan } from './tracing.js';
|
|
21
22
|
import type { RouteMatch } from './pipeline.js';
|
|
22
23
|
import type { ManifestSegmentNode } from './route-matcher.js';
|
|
23
24
|
import { resolveMetadata, renderMetadataToElements } from './metadata.js';
|
|
@@ -26,13 +27,57 @@ import type { Metadata } from './types.js';
|
|
|
26
27
|
import { METADATA_ROUTE_CONVENTIONS, getMetadataRouteAutoLink } from './metadata-routes.js';
|
|
27
28
|
import { DenySignal, RedirectSignal } from './primitives.js';
|
|
28
29
|
import { AccessGate } from './access-gate.js';
|
|
30
|
+
import { PageDenyBoundary } from './page-deny-boundary.js';
|
|
31
|
+
import { buildDenyPageChain, renderMatchingDenyPage, setDenyStatus } from './deny-page-resolver.js';
|
|
32
|
+
import type { DenyPageEntry } from './deny-page-resolver.js';
|
|
29
33
|
import { resolveSlotElement } from './slot-resolver.js';
|
|
30
|
-
import { SegmentProvider } from '
|
|
31
|
-
|
|
32
|
-
import type { SearchParamsDefinition } from '#/search-params/create.js';
|
|
34
|
+
import { SegmentProvider } from '../client/segment-context.js';
|
|
35
|
+
|
|
33
36
|
import { wrapSegmentWithErrorBoundaries } from './error-boundary-wrapper.js';
|
|
34
37
|
import type { InterceptionContext } from './pipeline.js';
|
|
35
38
|
import { shouldSkipSegment } from './state-tree-diff.js';
|
|
39
|
+
import { loadModule } from './safe-load.js';
|
|
40
|
+
|
|
41
|
+
// ─── Client Reference Detection ──────────────────────────────────────────
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Symbol used by React Flight to mark client references.
|
|
45
|
+
* Client references are proxy objects created by @vitejs/plugin-rsc for
|
|
46
|
+
* 'use client' modules in the RSC environment. They must be passed to
|
|
47
|
+
* createElement() — calling them as functions throws:
|
|
48
|
+
* "Unexpectedly client reference export 'default' is called on server"
|
|
49
|
+
*/
|
|
50
|
+
const CLIENT_REFERENCE_TAG = Symbol.for('react.client.reference');
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Detect whether a component is a React client reference.
|
|
54
|
+
* Client references have $$typeof set to Symbol.for('react.client.reference')
|
|
55
|
+
* by registerClientReference() in the React Flight server runtime.
|
|
56
|
+
*
|
|
57
|
+
* Used to skip OTEL tracing wrappers that would call the component as a
|
|
58
|
+
* function. Client components must go through createElement only — they are
|
|
59
|
+
* serialized as references in the RSC Flight stream, not executed on the server.
|
|
60
|
+
*/
|
|
61
|
+
export function isClientReference(component: unknown): boolean {
|
|
62
|
+
return (
|
|
63
|
+
component != null &&
|
|
64
|
+
typeof component === 'function' &&
|
|
65
|
+
(component as unknown as Record<string, unknown>).$$typeof === CLIENT_REFERENCE_TAG
|
|
66
|
+
);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// ─── Param Coercion Error ─────────────────────────────────────────────────
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Thrown when a defineSegmentParams codec's parse() fails.
|
|
73
|
+
* The pipeline catches this and responds with 404.
|
|
74
|
+
*/
|
|
75
|
+
export class ParamCoercionError extends Error {
|
|
76
|
+
constructor(message: string) {
|
|
77
|
+
super(message);
|
|
78
|
+
this.name = 'ParamCoercionError';
|
|
79
|
+
}
|
|
80
|
+
}
|
|
36
81
|
|
|
37
82
|
// ─── Types ────────────────────────────────────────────────────────────────
|
|
38
83
|
|
|
@@ -73,6 +118,10 @@ export interface RouteElementResult {
|
|
|
73
118
|
/**
|
|
74
119
|
* Wraps a DenySignal or RedirectSignal with the layout components loaded
|
|
75
120
|
* so far, enabling the caller to render deny pages inside the layout shell.
|
|
121
|
+
*
|
|
122
|
+
* @deprecated No longer thrown by buildRouteElement since TIM-662. Access
|
|
123
|
+
* checks now run inside AccessPreRunner during renderToReadableStream, and
|
|
124
|
+
* signals are caught by onError. Kept for backward compat with external code.
|
|
76
125
|
*/
|
|
77
126
|
export class RouteSignalWithContext extends Error {
|
|
78
127
|
constructor(
|
|
@@ -84,6 +133,64 @@ export class RouteSignalWithContext extends Error {
|
|
|
84
133
|
}
|
|
85
134
|
}
|
|
86
135
|
|
|
136
|
+
// ─── Module Processing Helpers ─────────────────────────────────────────────
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Reject the legacy `generateMetadata` export with a helpful migration message.
|
|
140
|
+
* Throws if the module exports `generateMetadata` instead of `metadata`.
|
|
141
|
+
*/
|
|
142
|
+
function rejectLegacyGenerateMetadata(mod: Record<string, unknown>, filePath: string): void {
|
|
143
|
+
if ('generateMetadata' in mod) {
|
|
144
|
+
throw new Error(
|
|
145
|
+
`${filePath}: "generateMetadata" is not a valid export. ` +
|
|
146
|
+
`Export an async function named "metadata" instead.\n\n` +
|
|
147
|
+
` // Before\n` +
|
|
148
|
+
` export async function generateMetadata({ params }) { ... }\n\n` +
|
|
149
|
+
` // After\n` +
|
|
150
|
+
` export async function metadata() { ... }`
|
|
151
|
+
);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Extract and resolve metadata from a module (layout or page).
|
|
157
|
+
* Handles both static metadata objects and async metadata functions.
|
|
158
|
+
* Returns the resolved Metadata, or null if none exported.
|
|
159
|
+
*
|
|
160
|
+
* Metadata functions no longer receive { params } — they access params
|
|
161
|
+
* via getSegmentParams() from ALS, same as page/layout components.
|
|
162
|
+
*/
|
|
163
|
+
async function extractMetadata(
|
|
164
|
+
mod: Record<string, unknown>,
|
|
165
|
+
segment: ManifestSegmentNode
|
|
166
|
+
): Promise<Metadata | null> {
|
|
167
|
+
if (typeof mod.metadata === 'function') {
|
|
168
|
+
type MetadataFn = () => Promise<Metadata>;
|
|
169
|
+
return (
|
|
170
|
+
(await withSpan(
|
|
171
|
+
'timber.metadata',
|
|
172
|
+
{ 'timber.segment': segment.segmentName ?? segment.urlPath },
|
|
173
|
+
() => (mod.metadata as MetadataFn)()
|
|
174
|
+
)) ?? null
|
|
175
|
+
);
|
|
176
|
+
}
|
|
177
|
+
if (mod.metadata) {
|
|
178
|
+
return mod.metadata as Metadata;
|
|
179
|
+
}
|
|
180
|
+
return null;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* Extract `deferSuspenseFor` from a module and return the maximum
|
|
185
|
+
* of the current value and the module's value.
|
|
186
|
+
*/
|
|
187
|
+
function extractDeferSuspenseFor(mod: Record<string, unknown>, current: number): number {
|
|
188
|
+
if (typeof mod.deferSuspenseFor === 'number' && mod.deferSuspenseFor > current) {
|
|
189
|
+
return mod.deferSuspenseFor;
|
|
190
|
+
}
|
|
191
|
+
return current;
|
|
192
|
+
}
|
|
193
|
+
|
|
87
194
|
// ─── Builder ──────────────────────────────────────────────────────────────
|
|
88
195
|
|
|
89
196
|
/**
|
|
@@ -95,6 +202,10 @@ export class RouteSignalWithContext extends Error {
|
|
|
95
202
|
*
|
|
96
203
|
* Does NOT serialize to RSC Flight — the caller decides whether to render
|
|
97
204
|
* to a stream or use the element directly (e.g., for action revalidation).
|
|
205
|
+
*
|
|
206
|
+
* Access checks are collected but NOT executed here. They run inside
|
|
207
|
+
* AccessPreRunner during renderToReadableStream so that access.ts and
|
|
208
|
+
* render components share the same React.cache scope (TIM-662).
|
|
98
209
|
*/
|
|
99
210
|
export async function buildRouteElement(
|
|
100
211
|
req: Request,
|
|
@@ -104,9 +215,6 @@ export async function buildRouteElement(
|
|
|
104
215
|
): Promise<RouteElementResult> {
|
|
105
216
|
const segments = match.segments as unknown as ManifestSegmentNode[];
|
|
106
217
|
|
|
107
|
-
// Params are passed as a Promise to match Next.js 15+ convention.
|
|
108
|
-
const paramsPromise = Promise.resolve(match.params);
|
|
109
|
-
|
|
110
218
|
// Load all modules along the segment chain
|
|
111
219
|
const metadataEntries: Array<{ metadata: Metadata; isPage: boolean }> = [];
|
|
112
220
|
const layoutComponents: LayoutComponentEntry[] = [];
|
|
@@ -119,94 +227,41 @@ export async function buildRouteElement(
|
|
|
119
227
|
|
|
120
228
|
// Load layout
|
|
121
229
|
if (segment.layout) {
|
|
122
|
-
const mod =
|
|
230
|
+
const mod = await loadModule(segment.layout);
|
|
123
231
|
if (mod.default) {
|
|
124
232
|
layoutComponents.push({
|
|
125
233
|
component: mod.default as (...args: unknown[]) => unknown,
|
|
126
234
|
segment,
|
|
127
235
|
});
|
|
128
236
|
}
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
` // After\n` +
|
|
138
|
-
` export async function metadata({ params }) { ... }`
|
|
139
|
-
);
|
|
140
|
-
}
|
|
141
|
-
// Unified metadata export: static object or async function
|
|
142
|
-
if (typeof mod.metadata === 'function') {
|
|
143
|
-
type MetadataFn = (props: Record<string, unknown>) => Promise<Metadata>;
|
|
144
|
-
const generated = await withSpan(
|
|
145
|
-
'timber.metadata',
|
|
146
|
-
{ 'timber.segment': segment.segmentName ?? segment.urlPath },
|
|
147
|
-
() => (mod.metadata as MetadataFn)({ params: paramsPromise })
|
|
148
|
-
);
|
|
149
|
-
if (generated) {
|
|
150
|
-
metadataEntries.push({ metadata: generated, isPage: false });
|
|
151
|
-
}
|
|
152
|
-
} else if (mod.metadata) {
|
|
153
|
-
metadataEntries.push({ metadata: mod.metadata as Metadata, isPage: false });
|
|
154
|
-
}
|
|
155
|
-
// deferSuspenseFor hold window — max across all segments
|
|
156
|
-
if (typeof mod.deferSuspenseFor === 'number' && mod.deferSuspenseFor > deferSuspenseFor) {
|
|
157
|
-
deferSuspenseFor = mod.deferSuspenseFor;
|
|
237
|
+
|
|
238
|
+
// Param coercion is handled in the pipeline (Stage 2c) before
|
|
239
|
+
// middleware and rendering. See coerceSegmentParams() in pipeline.ts.
|
|
240
|
+
|
|
241
|
+
rejectLegacyGenerateMetadata(mod, segment.layout.filePath ?? segment.urlPath);
|
|
242
|
+
const layoutMetadata = await extractMetadata(mod, segment);
|
|
243
|
+
if (layoutMetadata) {
|
|
244
|
+
metadataEntries.push({ metadata: layoutMetadata, isPage: false });
|
|
158
245
|
}
|
|
246
|
+
deferSuspenseFor = extractDeferSuspenseFor(mod, deferSuspenseFor);
|
|
159
247
|
}
|
|
160
248
|
|
|
161
249
|
// Load page (leaf segment only)
|
|
162
250
|
if (isLeaf && segment.page) {
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
default?: SearchParamsDefinition<Record<string, unknown>>;
|
|
168
|
-
};
|
|
169
|
-
if (spMod.default) {
|
|
170
|
-
const rawSearchParams = new URL(req.url).searchParams;
|
|
171
|
-
const parsed = spMod.default.parse(rawSearchParams);
|
|
172
|
-
setParsedSearchParams(parsed);
|
|
173
|
-
}
|
|
174
|
-
}
|
|
251
|
+
const mod = await loadModule(segment.page);
|
|
252
|
+
|
|
253
|
+
// Param coercion is handled in the pipeline (Stage 2c) before
|
|
254
|
+
// middleware and rendering. See coerceSegmentParams() in pipeline.ts.
|
|
175
255
|
|
|
176
|
-
const mod = (await segment.page.load()) as Record<string, unknown>;
|
|
177
256
|
if (mod.default) {
|
|
178
257
|
PageComponent = mod.default as (...args: unknown[]) => unknown;
|
|
179
258
|
}
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
`${filePath}: "generateMetadata" is not a valid export. ` +
|
|
185
|
-
`Export an async function named "metadata" instead.\n\n` +
|
|
186
|
-
` // Before\n` +
|
|
187
|
-
` export async function generateMetadata({ params }) { ... }\n\n` +
|
|
188
|
-
` // After\n` +
|
|
189
|
-
` export async function metadata({ params }) { ... }`
|
|
190
|
-
);
|
|
191
|
-
}
|
|
192
|
-
// Unified metadata export: static object or async function
|
|
193
|
-
if (typeof mod.metadata === 'function') {
|
|
194
|
-
type MetadataFn = (props: Record<string, unknown>) => Promise<Metadata>;
|
|
195
|
-
const generated = await withSpan(
|
|
196
|
-
'timber.metadata',
|
|
197
|
-
{ 'timber.segment': segment.segmentName ?? segment.urlPath },
|
|
198
|
-
() => (mod.metadata as MetadataFn)({ params: paramsPromise })
|
|
199
|
-
);
|
|
200
|
-
if (generated) {
|
|
201
|
-
metadataEntries.push({ metadata: generated, isPage: true });
|
|
202
|
-
}
|
|
203
|
-
} else if (mod.metadata) {
|
|
204
|
-
metadataEntries.push({ metadata: mod.metadata as Metadata, isPage: true });
|
|
205
|
-
}
|
|
206
|
-
// deferSuspenseFor hold window — max across all segments
|
|
207
|
-
if (typeof mod.deferSuspenseFor === 'number' && mod.deferSuspenseFor > deferSuspenseFor) {
|
|
208
|
-
deferSuspenseFor = mod.deferSuspenseFor;
|
|
259
|
+
rejectLegacyGenerateMetadata(mod, segment.page.filePath ?? segment.urlPath);
|
|
260
|
+
const pageMetadata = await extractMetadata(mod, segment);
|
|
261
|
+
if (pageMetadata) {
|
|
262
|
+
metadataEntries.push({ metadata: pageMetadata, isPage: true });
|
|
209
263
|
}
|
|
264
|
+
deferSuspenseFor = extractDeferSuspenseFor(mod, deferSuspenseFor);
|
|
210
265
|
}
|
|
211
266
|
}
|
|
212
267
|
|
|
@@ -214,54 +269,27 @@ export async function buildRouteElement(
|
|
|
214
269
|
throw new Error(`No page component found for route: ${new URL(req.url).pathname}`);
|
|
215
270
|
}
|
|
216
271
|
|
|
217
|
-
//
|
|
218
|
-
//
|
|
219
|
-
//
|
|
220
|
-
//
|
|
221
|
-
//
|
|
222
|
-
//
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
try {
|
|
239
|
-
await accessFn({ params: match.params, searchParams: {} });
|
|
240
|
-
await setSpanAttribute('timber.result', 'pass');
|
|
241
|
-
accessVerdicts.set(si, 'pass');
|
|
242
|
-
} catch (error) {
|
|
243
|
-
if (error instanceof DenySignal) {
|
|
244
|
-
await setSpanAttribute('timber.result', 'deny');
|
|
245
|
-
await setSpanAttribute('timber.deny_status', error.status);
|
|
246
|
-
if (error.sourceFile) {
|
|
247
|
-
await setSpanAttribute('timber.deny_file', error.sourceFile);
|
|
248
|
-
}
|
|
249
|
-
accessVerdicts.set(si, error);
|
|
250
|
-
} else if (error instanceof RedirectSignal) {
|
|
251
|
-
await setSpanAttribute('timber.result', 'redirect');
|
|
252
|
-
accessVerdicts.set(si, error);
|
|
253
|
-
}
|
|
254
|
-
throw error;
|
|
255
|
-
}
|
|
256
|
-
}
|
|
257
|
-
);
|
|
258
|
-
} catch (error) {
|
|
259
|
-
if (error instanceof DenySignal || error instanceof RedirectSignal) {
|
|
260
|
-
throw new RouteSignalWithContext(error, layoutComponents, segments);
|
|
261
|
-
}
|
|
262
|
-
throw error;
|
|
263
|
-
}
|
|
264
|
-
}
|
|
272
|
+
// Access checks are NOT run here. AccessGate components in the element
|
|
273
|
+
// tree call accessFn directly during renderToReadableStream, sharing the
|
|
274
|
+
// same React.cache scope as layout/page components. A requireUser() call
|
|
275
|
+
// in access.ts populates React.cache; the same call in a layout is a hit.
|
|
276
|
+
//
|
|
277
|
+
// Previously (before TIM-662), access checks ran eagerly in a pre-render
|
|
278
|
+
// loop OUTSIDE renderToReadableStream, breaking React.cache dedup.
|
|
279
|
+
//
|
|
280
|
+
// See design/04-authorization.md §"Pre-Render Pass and Verdict Replay"
|
|
281
|
+
|
|
282
|
+
// Build deny page fallback chains for each segment position.
|
|
283
|
+
// When AccessGate or PageDenyBoundary catches a DenySignal, they render
|
|
284
|
+
// the matching deny page in-tree instead of throwing into React Flight.
|
|
285
|
+
// The chain walks from the current segment outward to root, collecting
|
|
286
|
+
// status-code files (403.tsx → 4xx.tsx → error.tsx) in fallback order.
|
|
287
|
+
// See TIM-666.
|
|
288
|
+
const denyPageChains = new Map<number, DenyPageEntry[]>();
|
|
289
|
+
for (let i = 0; i < segments.length; i++) {
|
|
290
|
+
const chain = await buildDenyPageChain(segments, i);
|
|
291
|
+
if (chain.length > 0) {
|
|
292
|
+
denyPageChains.set(i, chain);
|
|
265
293
|
}
|
|
266
294
|
}
|
|
267
295
|
|
|
@@ -293,19 +321,43 @@ export async function buildRouteElement(
|
|
|
293
321
|
// Build element tree: page wrapped in layouts (innermost to outermost)
|
|
294
322
|
const h = createElement as (...args: unknown[]) => React.ReactElement;
|
|
295
323
|
|
|
296
|
-
//
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
324
|
+
// Build the page element.
|
|
325
|
+
// Client references ('use client' pages) must NOT be called as functions —
|
|
326
|
+
// they are proxy objects that throw when invoked. They must go through
|
|
327
|
+
// createElement only, which serializes them as client references in the
|
|
328
|
+
// RSC Flight stream. OTEL tracing is skipped for client components.
|
|
329
|
+
// See TIM-627 for the original bug.
|
|
330
|
+
// Build the page element.
|
|
331
|
+
// Server component pages are wrapped in PageDenyBoundary which calls
|
|
332
|
+
// them as async functions and catches DenySignal — rendering the deny
|
|
333
|
+
// page in-tree instead of throwing into React Flight. This eliminates
|
|
334
|
+
// the second render pass for deny pages. See TIM-666.
|
|
335
|
+
//
|
|
336
|
+
// Client reference pages ('use client') can't call deny() (server-only),
|
|
337
|
+
// so they go through createElement normally — no wrapper needed.
|
|
338
|
+
const leafIndex = segments.length - 1;
|
|
339
|
+
const leafDenyPages = denyPageChains.get(leafIndex);
|
|
340
|
+
let element: React.ReactElement;
|
|
341
|
+
if (isClientReference(PageComponent)) {
|
|
342
|
+
element = h(PageComponent, {});
|
|
343
|
+
} else if (leafDenyPages && leafDenyPages.length > 0) {
|
|
344
|
+
// Server component page WITH deny page chain — wrap in PageDenyBoundary
|
|
345
|
+
element = h(PageDenyBoundary, {
|
|
346
|
+
Page: PageComponent,
|
|
347
|
+
route: match.segments[leafIndex]?.urlPath ?? '/',
|
|
348
|
+
denyPages: leafDenyPages,
|
|
349
|
+
});
|
|
350
|
+
} else {
|
|
351
|
+
// Server component page WITHOUT deny page chain — trace only
|
|
352
|
+
const TracedPage = async (props: Record<string, unknown>) => {
|
|
353
|
+
return withSpan(
|
|
354
|
+
'timber.page',
|
|
355
|
+
{ 'timber.route': match.segments[leafIndex]?.urlPath ?? '/' },
|
|
356
|
+
() => (PageComponent as (props: Record<string, unknown>) => unknown)(props)
|
|
357
|
+
);
|
|
358
|
+
};
|
|
359
|
+
element = h(TracedPage, {});
|
|
360
|
+
}
|
|
309
361
|
|
|
310
362
|
// Build a lookup of layout components by segment for O(1) access.
|
|
311
363
|
const layoutBySegment = new Map(
|
|
@@ -357,11 +409,28 @@ export async function buildRouteElement(
|
|
|
357
409
|
segment.segmentType !== 'group';
|
|
358
410
|
|
|
359
411
|
if (skip) {
|
|
360
|
-
// Skip this segment
|
|
361
|
-
// Access.ts already ran in the pre-render loop (security guarantee).
|
|
412
|
+
// Skip this segment's layout/error boundaries — the client uses its cached version.
|
|
362
413
|
// Metadata was already resolved above (head elements are correct).
|
|
363
414
|
// Record for X-Timber-Skipped-Segments header (outermost first, so prepend).
|
|
364
415
|
skippedSegments.unshift(segment.urlPath);
|
|
416
|
+
|
|
417
|
+
// SECURITY: Even though the layout is skipped, AccessGate MUST still
|
|
418
|
+
// wrap the element tree. access.ts runs on every navigation regardless
|
|
419
|
+
// of cached layouts or state tree content.
|
|
420
|
+
// See design/13-security.md §"Auth always runs" (test #11).
|
|
421
|
+
if (segment.access) {
|
|
422
|
+
const accessMod = await loadModule(segment.access);
|
|
423
|
+
const accessFn = accessMod.default as (() => unknown) | undefined;
|
|
424
|
+
if (accessFn) {
|
|
425
|
+
element = h(AccessGate, {
|
|
426
|
+
accessFn,
|
|
427
|
+
segmentName: segment.segmentName,
|
|
428
|
+
denyPages: denyPageChains.get(i),
|
|
429
|
+
children: element,
|
|
430
|
+
});
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
|
|
365
434
|
continue;
|
|
366
435
|
}
|
|
367
436
|
|
|
@@ -372,29 +441,17 @@ export async function buildRouteElement(
|
|
|
372
441
|
}
|
|
373
442
|
|
|
374
443
|
// Wrap with error boundaries from this segment (inside layout).
|
|
444
|
+
// Keep ALL error boundaries (including 4xx) — they're the safety net for
|
|
445
|
+
// DenySignal from nested server components that escape AccessGate/PageDenyBoundary
|
|
446
|
+
// try/catch. Status-code files must be 'use client' TSX or MDX to serialize
|
|
447
|
+
// as error boundary fallbacks. See TIM-666.
|
|
375
448
|
element = await wrapSegmentWithErrorBoundaries(segment, element, h);
|
|
376
449
|
|
|
377
|
-
// Wrap
|
|
378
|
-
//
|
|
379
|
-
//
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
const accessFn = accessMod.default as
|
|
383
|
-
| ((ctx: { params: Record<string, string | string[]>; searchParams: unknown }) => unknown)
|
|
384
|
-
| undefined;
|
|
385
|
-
if (accessFn) {
|
|
386
|
-
element = h(AccessGate, {
|
|
387
|
-
accessFn,
|
|
388
|
-
params: match.params,
|
|
389
|
-
searchParams: {},
|
|
390
|
-
segmentName: segment.segmentName,
|
|
391
|
-
verdict: accessVerdicts.get(i),
|
|
392
|
-
children: element,
|
|
393
|
-
});
|
|
394
|
-
}
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
// Wrap with layout if this segment has one — traced with OTEL span
|
|
450
|
+
// Wrap with layout BEFORE AccessGate — AccessGate is OUTSIDE the layout.
|
|
451
|
+
// When AccessGate denies, the layout never renders. The deny page appears
|
|
452
|
+
// at the AccessGate level, wrapped by PARENT layouts only.
|
|
453
|
+
// This prevents leaking layout UI (sidebars, nav) on denied pages.
|
|
454
|
+
// See design/04-authorization.md §"Access Failure".
|
|
398
455
|
if (layoutComponent) {
|
|
399
456
|
// Resolve parallel slots for this layout
|
|
400
457
|
const slotProps: Record<string, unknown> = {};
|
|
@@ -403,7 +460,6 @@ export async function buildRouteElement(
|
|
|
403
460
|
slotProps[slotName] = await resolveSlotElement(
|
|
404
461
|
slotNode as ManifestSegmentNode,
|
|
405
462
|
match,
|
|
406
|
-
paramsPromise,
|
|
407
463
|
h,
|
|
408
464
|
interception
|
|
409
465
|
);
|
|
@@ -412,43 +468,79 @@ export async function buildRouteElement(
|
|
|
412
468
|
const segmentPath = segment.urlPath.split('/');
|
|
413
469
|
const parallelRouteKeys = Object.keys(segment.slots ?? {});
|
|
414
470
|
|
|
415
|
-
//
|
|
416
|
-
//
|
|
417
|
-
//
|
|
418
|
-
// from the root "layout /".
|
|
419
|
-
const segmentForSpan = segment;
|
|
420
|
-
const layoutComponentForSpan = layoutComponent;
|
|
421
|
-
const segmentLabel =
|
|
422
|
-
segmentForSpan.segmentType === 'group'
|
|
423
|
-
? `${segmentForSpan.urlPath === '/' ? '' : segmentForSpan.urlPath}/${segmentForSpan.segmentName}`
|
|
424
|
-
: segmentForSpan.urlPath;
|
|
425
|
-
const TracedLayout = async (props: Record<string, unknown>) => {
|
|
426
|
-
return withSpan('timber.layout', { 'timber.segment': segmentLabel }, () =>
|
|
427
|
-
(layoutComponentForSpan as (props: Record<string, unknown>) => unknown)(props)
|
|
428
|
-
);
|
|
429
|
-
};
|
|
430
|
-
|
|
431
|
-
// segmentId uniquely identifies this segment for client-side element
|
|
432
|
-
// caching. For route groups, urlPath is shared with the parent (both "/"),
|
|
433
|
-
// so we include the group name to distinguish them. Without this, the
|
|
434
|
-
// segment merger's element cache would conflate root and group elements.
|
|
471
|
+
// For route groups, urlPath is shared with the parent (both "/"),
|
|
472
|
+
// so include the group name to distinguish them. Used for both OTEL
|
|
473
|
+
// span labels and client-side element caching (segmentId).
|
|
435
474
|
const segmentId =
|
|
436
475
|
segment.segmentType === 'group'
|
|
437
476
|
? `${segment.urlPath === '/' ? '' : segment.urlPath}/${segment.segmentName}`
|
|
438
477
|
: segment.urlPath;
|
|
439
478
|
|
|
479
|
+
// Build the layout element.
|
|
480
|
+
// Same client reference guard as pages — client layouts must not be
|
|
481
|
+
// called as functions. OTEL tracing is skipped for client components.
|
|
482
|
+
let layoutElement: React.ReactElement;
|
|
483
|
+
if (isClientReference(layoutComponent)) {
|
|
484
|
+
layoutElement = h(layoutComponent, {
|
|
485
|
+
...slotProps,
|
|
486
|
+
children: element,
|
|
487
|
+
});
|
|
488
|
+
} else {
|
|
489
|
+
// Server component layout — wrap with OTEL tracing AND DenySignal
|
|
490
|
+
// catching. If the layout calls deny(), the signal is caught here
|
|
491
|
+
// and the matching deny page renders in-tree (same pattern as
|
|
492
|
+
// AccessGate and PageDenyBoundary). Without this, DenySignal
|
|
493
|
+
// escapes to React Flight onError and triggers the re-render
|
|
494
|
+
// fallback path. See TIM-668, design/04-authorization.md.
|
|
495
|
+
const layoutComponentRef = layoutComponent;
|
|
496
|
+
const layoutDenyPages = denyPageChains.get(i);
|
|
497
|
+
const TracedLayout = async (props: Record<string, unknown>) => {
|
|
498
|
+
try {
|
|
499
|
+
return await withSpan('timber.layout', { 'timber.segment': segmentId }, () =>
|
|
500
|
+
(layoutComponentRef as (props: Record<string, unknown>) => unknown)(props)
|
|
501
|
+
);
|
|
502
|
+
} catch (error: unknown) {
|
|
503
|
+
if (error instanceof DenySignal && layoutDenyPages) {
|
|
504
|
+
const denyElement = renderMatchingDenyPage(layoutDenyPages, error.status, error.data);
|
|
505
|
+
if (denyElement) {
|
|
506
|
+
setDenyStatus(error.status);
|
|
507
|
+
return denyElement;
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
// Non-deny errors (RedirectSignal, runtime errors) propagate normally.
|
|
511
|
+
throw error;
|
|
512
|
+
}
|
|
513
|
+
};
|
|
514
|
+
layoutElement = h(TracedLayout, {
|
|
515
|
+
...slotProps,
|
|
516
|
+
children: element,
|
|
517
|
+
});
|
|
518
|
+
}
|
|
519
|
+
|
|
440
520
|
element = h(SegmentProvider, {
|
|
441
521
|
segments: segmentPath,
|
|
442
522
|
segmentId,
|
|
443
523
|
parallelRouteKeys,
|
|
444
|
-
children:
|
|
445
|
-
...slotProps,
|
|
446
|
-
params: paramsPromise,
|
|
447
|
-
searchParams: {},
|
|
448
|
-
children: element,
|
|
449
|
-
}),
|
|
524
|
+
children: layoutElement,
|
|
450
525
|
});
|
|
451
526
|
}
|
|
527
|
+
|
|
528
|
+
// Wrap in AccessGate OUTSIDE the layout.
|
|
529
|
+
// If access denies, the deny page renders here — the layout above
|
|
530
|
+
// never executes. Parent layouts (from outer iterations) form the shell.
|
|
531
|
+
// See TIM-662, TIM-666, design/04-authorization.md §"Access Failure".
|
|
532
|
+
if (segment.access) {
|
|
533
|
+
const accessMod = await loadModule(segment.access);
|
|
534
|
+
const accessFn = accessMod.default as (() => unknown) | undefined;
|
|
535
|
+
if (accessFn) {
|
|
536
|
+
element = h(AccessGate, {
|
|
537
|
+
accessFn,
|
|
538
|
+
segmentName: segment.segmentName,
|
|
539
|
+
denyPages: denyPageChains.get(i),
|
|
540
|
+
children: element,
|
|
541
|
+
});
|
|
542
|
+
}
|
|
543
|
+
}
|
|
452
544
|
}
|
|
453
545
|
|
|
454
546
|
return {
|
|
@@ -9,6 +9,7 @@
|
|
|
9
9
|
*/
|
|
10
10
|
|
|
11
11
|
import type { RouteContext } from './types.js';
|
|
12
|
+
import { logRouteError } from './logger.js';
|
|
12
13
|
|
|
13
14
|
// ─── Types ───────────────────────────────────────────────────────────────
|
|
14
15
|
|
|
@@ -122,7 +123,7 @@ async function runHandler(handler: RouteHandler, ctx: RouteContext): Promise<Res
|
|
|
122
123
|
const res = await handler(ctx);
|
|
123
124
|
return mergeResponseHeaders(res, ctx.headers);
|
|
124
125
|
} catch (error) {
|
|
125
|
-
|
|
126
|
+
logRouteError({ method: ctx.req.method, path: new URL(ctx.req.url).pathname, error });
|
|
126
127
|
return new Response(null, { status: 500 });
|
|
127
128
|
}
|
|
128
129
|
}
|
|
@@ -140,10 +141,29 @@ function mergeResponseHeaders(res: Response, ctxHeaders: Headers): Response {
|
|
|
140
141
|
});
|
|
141
142
|
if (!hasCtxHeaders) return res;
|
|
142
143
|
|
|
143
|
-
// Merge: ctx.headers first, then handler response headers override
|
|
144
|
+
// Merge: ctx.headers first, then handler response headers override.
|
|
145
|
+
// Set-Cookie needs special handling: Headers.set() replaces all values
|
|
146
|
+
// for a key, but each Set-Cookie must be its own header per RFC 6265 §4.1.
|
|
147
|
+
// Use append for Set-Cookie, set for everything else.
|
|
144
148
|
const merged = new Headers();
|
|
145
|
-
ctxHeaders.forEach((value, key) =>
|
|
146
|
-
|
|
149
|
+
ctxHeaders.forEach((value, key) => {
|
|
150
|
+
if (key.toLowerCase() === 'set-cookie') {
|
|
151
|
+
merged.append(key, value);
|
|
152
|
+
} else {
|
|
153
|
+
merged.set(key, value);
|
|
154
|
+
}
|
|
155
|
+
});
|
|
156
|
+
// Response Set-Cookie headers: use getSetCookie() to preserve individual
|
|
157
|
+
// cookies (forEach joins them with ", " into one entry).
|
|
158
|
+
const resCookies = res.headers.getSetCookie();
|
|
159
|
+
for (const cookie of resCookies) {
|
|
160
|
+
merged.append('Set-Cookie', cookie);
|
|
161
|
+
}
|
|
162
|
+
res.headers.forEach((value, key) => {
|
|
163
|
+
if (key.toLowerCase() !== 'set-cookie') {
|
|
164
|
+
merged.set(key, value);
|
|
165
|
+
}
|
|
166
|
+
});
|
|
147
167
|
|
|
148
168
|
return new Response(res.body, {
|
|
149
169
|
status: res.status,
|