@rangojs/router 0.0.0-experimental.13 → 0.0.0-experimental.131
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/AGENTS.md +9 -0
- package/README.md +1040 -5
- package/dist/__internal.d.ts +83 -0
- package/dist/__internal.d.ts.map +1 -0
- package/dist/__internal.js +19 -0
- package/dist/__internal.js.map +1 -0
- package/dist/__mocks__/version.d.ts +7 -0
- package/dist/__mocks__/version.d.ts.map +1 -0
- package/dist/__mocks__/version.js +7 -0
- package/dist/__mocks__/version.js.map +1 -0
- package/dist/__tests__/client-href.test.d.ts +2 -0
- package/dist/__tests__/client-href.test.d.ts.map +1 -0
- package/dist/__tests__/client-href.test.js +74 -0
- package/dist/__tests__/client-href.test.js.map +1 -0
- package/dist/__tests__/component-utils.test.d.ts +2 -0
- package/dist/__tests__/component-utils.test.d.ts.map +1 -0
- package/dist/__tests__/component-utils.test.js +51 -0
- package/dist/__tests__/component-utils.test.js.map +1 -0
- package/dist/__tests__/event-controller.test.d.ts +2 -0
- package/dist/__tests__/event-controller.test.d.ts.map +1 -0
- package/dist/__tests__/event-controller.test.js +538 -0
- package/dist/__tests__/event-controller.test.js.map +1 -0
- package/dist/__tests__/helpers/route-tree.d.ts +118 -0
- package/dist/__tests__/helpers/route-tree.d.ts.map +1 -0
- package/dist/__tests__/helpers/route-tree.js +374 -0
- package/dist/__tests__/helpers/route-tree.js.map +1 -0
- package/dist/__tests__/match-result.test.d.ts +2 -0
- package/dist/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/__tests__/match-result.test.js +154 -0
- package/dist/__tests__/match-result.test.js.map +1 -0
- package/dist/__tests__/navigation-store.test.d.ts +2 -0
- package/dist/__tests__/navigation-store.test.d.ts.map +1 -0
- package/dist/__tests__/navigation-store.test.js +440 -0
- package/dist/__tests__/navigation-store.test.js.map +1 -0
- package/dist/__tests__/partial-update.test.d.ts +2 -0
- package/dist/__tests__/partial-update.test.d.ts.map +1 -0
- package/dist/__tests__/partial-update.test.js +1009 -0
- package/dist/__tests__/partial-update.test.js.map +1 -0
- package/dist/__tests__/reverse-types.test.d.ts +8 -0
- package/dist/__tests__/reverse-types.test.d.ts.map +1 -0
- package/dist/__tests__/reverse-types.test.js +656 -0
- package/dist/__tests__/reverse-types.test.js.map +1 -0
- package/dist/__tests__/route-definition.test.d.ts +2 -0
- package/dist/__tests__/route-definition.test.d.ts.map +1 -0
- package/dist/__tests__/route-definition.test.js +55 -0
- package/dist/__tests__/route-definition.test.js.map +1 -0
- package/dist/__tests__/router-helpers.test.d.ts +2 -0
- package/dist/__tests__/router-helpers.test.d.ts.map +1 -0
- package/dist/__tests__/router-helpers.test.js +377 -0
- package/dist/__tests__/router-helpers.test.js.map +1 -0
- package/dist/__tests__/router-integration-2.test.d.ts +2 -0
- package/dist/__tests__/router-integration-2.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration-2.test.js +426 -0
- package/dist/__tests__/router-integration-2.test.js.map +1 -0
- package/dist/__tests__/router-integration.test.d.ts +2 -0
- package/dist/__tests__/router-integration.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration.test.js +1051 -0
- package/dist/__tests__/router-integration.test.js.map +1 -0
- package/dist/__tests__/search-params.test.d.ts +5 -0
- package/dist/__tests__/search-params.test.d.ts.map +1 -0
- package/dist/__tests__/search-params.test.js +306 -0
- package/dist/__tests__/search-params.test.js.map +1 -0
- package/dist/__tests__/segment-system.test.d.ts +2 -0
- package/dist/__tests__/segment-system.test.d.ts.map +1 -0
- package/dist/__tests__/segment-system.test.js +627 -0
- package/dist/__tests__/segment-system.test.js.map +1 -0
- package/dist/__tests__/static-handler-types.test.d.ts +8 -0
- package/dist/__tests__/static-handler-types.test.d.ts.map +1 -0
- package/dist/__tests__/static-handler-types.test.js +63 -0
- package/dist/__tests__/static-handler-types.test.js.map +1 -0
- package/dist/__tests__/urls.test.d.ts +2 -0
- package/dist/__tests__/urls.test.d.ts.map +1 -0
- package/dist/__tests__/urls.test.js +421 -0
- package/dist/__tests__/urls.test.js.map +1 -0
- package/dist/__tests__/use-mount.test.d.ts +2 -0
- package/dist/__tests__/use-mount.test.d.ts.map +1 -0
- package/dist/__tests__/use-mount.test.js +35 -0
- package/dist/__tests__/use-mount.test.js.map +1 -0
- package/dist/bin/rango.d.ts +2 -0
- package/dist/bin/rango.d.ts.map +1 -0
- package/dist/bin/rango.js +1712 -212
- package/dist/bin/rango.js.map +1 -0
- package/dist/browser/event-controller.d.ts +191 -0
- package/dist/browser/event-controller.d.ts.map +1 -0
- package/dist/browser/event-controller.js +559 -0
- package/dist/browser/event-controller.js.map +1 -0
- package/dist/browser/index.d.ts +2 -0
- package/dist/browser/index.d.ts.map +1 -0
- package/dist/browser/index.js +14 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/browser/link-interceptor.d.ts +38 -0
- package/dist/browser/link-interceptor.d.ts.map +1 -0
- package/dist/browser/link-interceptor.js +99 -0
- package/dist/browser/link-interceptor.js.map +1 -0
- package/dist/browser/logging.d.ts +10 -0
- package/dist/browser/logging.d.ts.map +1 -0
- package/dist/browser/logging.js +29 -0
- package/dist/browser/logging.js.map +1 -0
- package/dist/browser/lru-cache.d.ts +17 -0
- package/dist/browser/lru-cache.d.ts.map +1 -0
- package/dist/browser/lru-cache.js +50 -0
- package/dist/browser/lru-cache.js.map +1 -0
- package/dist/browser/merge-segment-loaders.d.ts +39 -0
- package/dist/browser/merge-segment-loaders.d.ts.map +1 -0
- package/dist/browser/merge-segment-loaders.js +102 -0
- package/dist/browser/merge-segment-loaders.js.map +1 -0
- package/dist/browser/navigation-bridge.d.ts +102 -0
- package/dist/browser/navigation-bridge.d.ts.map +1 -0
- package/dist/browser/navigation-bridge.js +708 -0
- package/dist/browser/navigation-bridge.js.map +1 -0
- package/dist/browser/navigation-client.d.ts +25 -0
- package/dist/browser/navigation-client.d.ts.map +1 -0
- package/dist/browser/navigation-client.js +157 -0
- package/dist/browser/navigation-client.js.map +1 -0
- package/dist/browser/navigation-store.d.ts +101 -0
- package/dist/browser/navigation-store.d.ts.map +1 -0
- package/dist/browser/navigation-store.js +625 -0
- package/dist/browser/navigation-store.js.map +1 -0
- package/dist/browser/partial-update.d.ts +75 -0
- package/dist/browser/partial-update.d.ts.map +1 -0
- package/dist/browser/partial-update.js +426 -0
- package/dist/browser/partial-update.js.map +1 -0
- package/dist/browser/react/Link.d.ts +86 -0
- package/dist/browser/react/Link.d.ts.map +1 -0
- package/dist/browser/react/Link.js +128 -0
- package/dist/browser/react/Link.js.map +1 -0
- package/dist/browser/react/NavigationProvider.d.ts +63 -0
- package/dist/browser/react/NavigationProvider.d.ts.map +1 -0
- package/dist/browser/react/NavigationProvider.js +216 -0
- package/dist/browser/react/NavigationProvider.js.map +1 -0
- package/dist/browser/react/ScrollRestoration.d.ts +75 -0
- package/dist/browser/react/ScrollRestoration.d.ts.map +1 -0
- package/dist/browser/react/ScrollRestoration.js +57 -0
- package/dist/browser/react/ScrollRestoration.js.map +1 -0
- package/dist/browser/react/context.d.ts +46 -0
- package/dist/browser/react/context.d.ts.map +1 -0
- package/dist/browser/react/context.js +10 -0
- package/dist/browser/react/context.js.map +1 -0
- package/dist/browser/react/index.d.ts +11 -0
- package/dist/browser/react/index.d.ts.map +1 -0
- package/dist/browser/react/index.js +22 -0
- package/dist/browser/react/index.js.map +1 -0
- package/dist/browser/react/location-state-shared.d.ts +63 -0
- package/dist/browser/react/location-state-shared.d.ts.map +1 -0
- package/dist/browser/react/location-state-shared.js +81 -0
- package/dist/browser/react/location-state-shared.js.map +1 -0
- package/dist/browser/react/location-state.d.ts +23 -0
- package/dist/browser/react/location-state.d.ts.map +1 -0
- package/dist/browser/react/location-state.js +29 -0
- package/dist/browser/react/location-state.js.map +1 -0
- package/dist/browser/react/mount-context.d.ts +24 -0
- package/dist/browser/react/mount-context.d.ts.map +1 -0
- package/dist/browser/react/mount-context.js +24 -0
- package/dist/browser/react/mount-context.js.map +1 -0
- package/dist/browser/react/use-action.d.ts +64 -0
- package/dist/browser/react/use-action.d.ts.map +1 -0
- package/dist/browser/react/use-action.js +134 -0
- package/dist/browser/react/use-action.js.map +1 -0
- package/dist/browser/react/use-client-cache.d.ts +41 -0
- package/dist/browser/react/use-client-cache.d.ts.map +1 -0
- package/{src/browser/react/use-client-cache.ts → dist/browser/react/use-client-cache.js} +9 -26
- package/dist/browser/react/use-client-cache.js.map +1 -0
- package/dist/browser/react/use-handle.d.ts +31 -0
- package/dist/browser/react/use-handle.d.ts.map +1 -0
- package/dist/browser/react/use-handle.js +144 -0
- package/dist/browser/react/use-handle.js.map +1 -0
- package/dist/browser/react/use-href.d.ts +33 -0
- package/dist/browser/react/use-href.d.ts.map +1 -0
- package/dist/browser/react/use-href.js +39 -0
- package/dist/browser/react/use-href.js.map +1 -0
- package/dist/browser/react/use-link-status.d.ts +37 -0
- package/dist/browser/react/use-link-status.d.ts.map +1 -0
- package/dist/browser/react/use-link-status.js +99 -0
- package/dist/browser/react/use-link-status.js.map +1 -0
- package/dist/browser/react/use-mount.d.ts +25 -0
- package/dist/browser/react/use-mount.d.ts.map +1 -0
- package/dist/browser/react/use-mount.js +30 -0
- package/dist/browser/react/use-mount.js.map +1 -0
- package/dist/browser/react/use-navigation.d.ts +27 -0
- package/dist/browser/react/use-navigation.d.ts.map +1 -0
- package/dist/browser/react/use-navigation.js +87 -0
- package/dist/browser/react/use-navigation.js.map +1 -0
- package/dist/browser/react/use-segments.d.ts +38 -0
- package/dist/browser/react/use-segments.d.ts.map +1 -0
- package/dist/browser/react/use-segments.js +130 -0
- package/dist/browser/react/use-segments.js.map +1 -0
- package/dist/browser/request-controller.d.ts +26 -0
- package/dist/browser/request-controller.d.ts.map +1 -0
- package/dist/browser/request-controller.js +147 -0
- package/dist/browser/request-controller.js.map +1 -0
- package/dist/browser/rsc-router.d.ts +129 -0
- package/dist/browser/rsc-router.d.ts.map +1 -0
- package/dist/browser/rsc-router.js +195 -0
- package/dist/browser/rsc-router.js.map +1 -0
- package/dist/browser/scroll-restoration.d.ts +93 -0
- package/dist/browser/scroll-restoration.d.ts.map +1 -0
- package/dist/browser/scroll-restoration.js +321 -0
- package/dist/browser/scroll-restoration.js.map +1 -0
- package/dist/browser/segment-structure-assert.d.ts +17 -0
- package/dist/browser/segment-structure-assert.d.ts.map +1 -0
- package/dist/browser/segment-structure-assert.js +59 -0
- package/dist/browser/segment-structure-assert.js.map +1 -0
- package/dist/browser/server-action-bridge.d.ts +26 -0
- package/dist/browser/server-action-bridge.d.ts.map +1 -0
- package/dist/browser/server-action-bridge.js +668 -0
- package/dist/browser/server-action-bridge.js.map +1 -0
- package/dist/browser/shallow.d.ts +12 -0
- package/dist/browser/shallow.d.ts.map +1 -0
- package/dist/browser/shallow.js +34 -0
- package/dist/browser/shallow.js.map +1 -0
- package/dist/browser/types.d.ts +369 -0
- package/dist/browser/types.d.ts.map +1 -0
- package/dist/browser/types.js +2 -0
- package/dist/browser/types.js.map +1 -0
- package/dist/build/__tests__/generate-cli.test.d.ts +2 -0
- package/dist/build/__tests__/generate-cli.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-cli.test.js +237 -0
- package/dist/build/__tests__/generate-cli.test.js.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.js +119 -0
- package/dist/build/__tests__/generate-manifest.test.js.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts +2 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.js +620 -0
- package/dist/build/__tests__/generate-route-types.test.js.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.js +308 -0
- package/dist/build/__tests__/per-router-manifest.test.js.map +1 -0
- package/dist/build/generate-manifest.d.ts +81 -0
- package/dist/build/generate-manifest.d.ts.map +1 -0
- package/dist/build/generate-manifest.js +276 -0
- package/dist/build/generate-manifest.js.map +1 -0
- package/dist/build/generate-route-types.d.ts +115 -0
- package/dist/build/generate-route-types.d.ts.map +1 -0
- package/dist/build/generate-route-types.js +740 -0
- package/dist/build/generate-route-types.js.map +1 -0
- package/dist/build/index.d.ts +21 -0
- package/dist/build/index.d.ts.map +1 -0
- package/dist/build/index.js +21 -0
- package/dist/build/index.js.map +1 -0
- package/dist/build/route-trie.d.ts +71 -0
- package/dist/build/route-trie.d.ts.map +1 -0
- package/dist/build/route-trie.js +175 -0
- package/dist/build/route-trie.js.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts +2 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.js +208 -0
- package/dist/cache/__tests__/cache-scope.test.js.map +1 -0
- package/dist/cache/__tests__/document-cache.test.d.ts +2 -0
- package/dist/cache/__tests__/document-cache.test.d.ts.map +1 -0
- package/dist/cache/__tests__/document-cache.test.js +345 -0
- package/dist/cache/__tests__/document-cache.test.js.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.js +425 -0
- package/dist/cache/__tests__/memory-segment-store.test.js.map +1 -0
- package/dist/cache/__tests__/memory-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-store.test.js +367 -0
- package/dist/cache/__tests__/memory-store.test.js.map +1 -0
- package/dist/cache/cache-scope.d.ts +102 -0
- package/dist/cache/cache-scope.d.ts.map +1 -0
- package/dist/cache/cache-scope.js +440 -0
- package/dist/cache/cache-scope.js.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts +2 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js +330 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js.map +1 -0
- package/dist/cache/cf/cf-cache-store.d.ts +165 -0
- package/dist/cache/cf/cf-cache-store.d.ts.map +1 -0
- package/dist/cache/cf/cf-cache-store.js +242 -0
- package/dist/cache/cf/cf-cache-store.js.map +1 -0
- package/dist/cache/cf/index.d.ts +14 -0
- package/dist/cache/cf/index.d.ts.map +1 -0
- package/dist/cache/cf/index.js +17 -0
- package/dist/cache/cf/index.js.map +1 -0
- package/dist/cache/document-cache.d.ts +64 -0
- package/dist/cache/document-cache.d.ts.map +1 -0
- package/dist/cache/document-cache.js +228 -0
- package/dist/cache/document-cache.js.map +1 -0
- package/dist/cache/index.d.ts +19 -0
- package/dist/cache/index.d.ts.map +1 -0
- package/dist/cache/index.js +21 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/cache/memory-segment-store.d.ts +110 -0
- package/dist/cache/memory-segment-store.d.ts.map +1 -0
- package/dist/cache/memory-segment-store.js +117 -0
- package/dist/cache/memory-segment-store.js.map +1 -0
- package/dist/cache/memory-store.d.ts +41 -0
- package/dist/cache/memory-store.d.ts.map +1 -0
- package/dist/cache/memory-store.js +191 -0
- package/dist/cache/memory-store.js.map +1 -0
- package/dist/cache/types.d.ts +317 -0
- package/dist/cache/types.d.ts.map +1 -0
- package/dist/cache/types.js +12 -0
- package/dist/cache/types.js.map +1 -0
- package/dist/client.d.ts +248 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +367 -0
- package/dist/client.js.map +1 -0
- package/dist/client.rsc.d.ts +26 -0
- package/dist/client.rsc.d.ts.map +1 -0
- package/dist/client.rsc.js +46 -0
- package/dist/client.rsc.js.map +1 -0
- package/dist/component-utils.d.ts +36 -0
- package/dist/component-utils.d.ts.map +1 -0
- package/dist/component-utils.js +61 -0
- package/dist/component-utils.js.map +1 -0
- package/dist/components/DefaultDocument.d.ts +13 -0
- package/dist/components/DefaultDocument.d.ts.map +1 -0
- package/dist/components/DefaultDocument.js +15 -0
- package/dist/components/DefaultDocument.js.map +1 -0
- package/dist/debug.d.ts +58 -0
- package/dist/debug.d.ts.map +1 -0
- package/dist/debug.js +157 -0
- package/dist/debug.js.map +1 -0
- package/dist/default-error-boundary.d.ts +11 -0
- package/dist/default-error-boundary.d.ts.map +1 -0
- package/dist/default-error-boundary.js +45 -0
- package/dist/default-error-boundary.js.map +1 -0
- package/dist/deps/browser.d.ts +2 -0
- package/dist/deps/browser.d.ts.map +1 -0
- package/dist/deps/browser.js +3 -0
- package/dist/deps/browser.js.map +1 -0
- package/dist/deps/html-stream-client.d.ts +2 -0
- package/dist/deps/html-stream-client.d.ts.map +1 -0
- package/dist/deps/html-stream-client.js +3 -0
- package/dist/deps/html-stream-client.js.map +1 -0
- package/dist/deps/html-stream-server.d.ts +2 -0
- package/dist/deps/html-stream-server.d.ts.map +1 -0
- package/dist/deps/html-stream-server.js +3 -0
- package/dist/deps/html-stream-server.js.map +1 -0
- package/dist/deps/rsc.d.ts +2 -0
- package/dist/deps/rsc.d.ts.map +1 -0
- package/dist/deps/rsc.js +4 -0
- package/dist/deps/rsc.js.map +1 -0
- package/dist/deps/ssr.d.ts +2 -0
- package/dist/deps/ssr.d.ts.map +1 -0
- package/dist/deps/ssr.js +3 -0
- package/dist/deps/ssr.js.map +1 -0
- package/dist/errors.d.ts +174 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +241 -0
- package/dist/errors.js.map +1 -0
- package/dist/handle.d.ts +78 -0
- package/dist/handle.d.ts.map +1 -0
- package/dist/handle.js +82 -0
- package/dist/handle.js.map +1 -0
- package/dist/handles/MetaTags.d.ts +14 -0
- package/dist/handles/MetaTags.d.ts.map +1 -0
- package/dist/handles/MetaTags.js +136 -0
- package/dist/handles/MetaTags.js.map +1 -0
- package/dist/handles/index.d.ts +6 -0
- package/dist/handles/index.d.ts.map +1 -0
- package/{src/handles/index.ts → dist/handles/index.js} +1 -1
- package/dist/handles/index.js.map +1 -0
- package/dist/handles/meta.d.ts +39 -0
- package/dist/handles/meta.d.ts.map +1 -0
- package/dist/handles/meta.js +202 -0
- package/dist/handles/meta.js.map +1 -0
- package/dist/host/__tests__/errors.test.d.ts +2 -0
- package/dist/host/__tests__/errors.test.d.ts.map +1 -0
- package/dist/host/__tests__/errors.test.js +76 -0
- package/dist/host/__tests__/errors.test.js.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js +732 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.js +251 -0
- package/dist/host/__tests__/pattern-matcher.test.js.map +1 -0
- package/dist/host/__tests__/router.test.d.ts +2 -0
- package/dist/host/__tests__/router.test.d.ts.map +1 -0
- package/dist/host/__tests__/router.test.js +241 -0
- package/dist/host/__tests__/router.test.js.map +1 -0
- package/dist/host/__tests__/testing.test.d.ts +2 -0
- package/dist/host/__tests__/testing.test.d.ts.map +1 -0
- package/dist/host/__tests__/testing.test.js +64 -0
- package/dist/host/__tests__/testing.test.js.map +1 -0
- package/dist/host/__tests__/utils.test.d.ts +2 -0
- package/dist/host/__tests__/utils.test.d.ts.map +1 -0
- package/dist/host/__tests__/utils.test.js +29 -0
- package/dist/host/__tests__/utils.test.js.map +1 -0
- package/dist/host/cookie-handler.d.ts +34 -0
- package/dist/host/cookie-handler.d.ts.map +1 -0
- package/dist/host/cookie-handler.js +124 -0
- package/dist/host/cookie-handler.js.map +1 -0
- package/dist/host/errors.d.ts +56 -0
- package/dist/host/errors.d.ts.map +1 -0
- package/dist/host/errors.js +79 -0
- package/dist/host/errors.js.map +1 -0
- package/dist/host/index.d.ts +29 -0
- package/dist/host/index.d.ts.map +1 -0
- package/dist/host/index.js +32 -0
- package/dist/host/index.js.map +1 -0
- package/dist/host/pattern-matcher.d.ts +36 -0
- package/dist/host/pattern-matcher.d.ts.map +1 -0
- package/dist/host/pattern-matcher.js +172 -0
- package/dist/host/pattern-matcher.js.map +1 -0
- package/dist/host/router.d.ts +26 -0
- package/dist/host/router.d.ts.map +1 -0
- package/dist/host/router.js +218 -0
- package/dist/host/router.js.map +1 -0
- package/dist/host/testing.d.ts +36 -0
- package/dist/host/testing.d.ts.map +1 -0
- package/dist/host/testing.js +55 -0
- package/dist/host/testing.js.map +1 -0
- package/dist/host/types.d.ts +115 -0
- package/dist/host/types.d.ts.map +1 -0
- package/dist/host/types.js +7 -0
- package/dist/host/types.js.map +1 -0
- package/dist/host/utils.d.ts +21 -0
- package/dist/host/utils.d.ts.map +1 -0
- package/dist/host/utils.js +23 -0
- package/dist/host/utils.js.map +1 -0
- package/dist/href-client.d.ts +131 -0
- package/dist/href-client.d.ts.map +1 -0
- package/dist/href-client.js +64 -0
- package/dist/href-client.js.map +1 -0
- package/{src/href-context.ts → dist/href-context.d.ts} +7 -11
- package/dist/href-context.d.ts.map +1 -0
- package/dist/href-context.js +21 -0
- package/dist/href-context.js.map +1 -0
- package/dist/index.d.ts +73 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +91 -0
- package/dist/index.js.map +1 -0
- package/dist/index.rsc.d.ts +32 -0
- package/dist/index.rsc.d.ts.map +1 -0
- package/dist/index.rsc.js +40 -0
- package/dist/index.rsc.js.map +1 -0
- package/dist/internal-debug.d.ts +2 -0
- package/dist/internal-debug.d.ts.map +1 -0
- package/dist/internal-debug.js +5 -0
- package/dist/internal-debug.js.map +1 -0
- package/dist/loader.d.ts +14 -0
- package/dist/loader.d.ts.map +1 -0
- package/dist/loader.js +20 -0
- package/dist/loader.js.map +1 -0
- package/dist/loader.rsc.d.ts +19 -0
- package/dist/loader.rsc.d.ts.map +1 -0
- package/dist/loader.rsc.js +99 -0
- package/dist/loader.rsc.js.map +1 -0
- package/dist/network-error-thrower.d.ts +17 -0
- package/dist/network-error-thrower.d.ts.map +1 -0
- package/dist/network-error-thrower.js +14 -0
- package/dist/network-error-thrower.js.map +1 -0
- package/dist/outlet-context.d.ts +13 -0
- package/dist/outlet-context.d.ts.map +1 -0
- package/dist/outlet-context.js +3 -0
- package/dist/outlet-context.js.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts +2 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.js +148 -0
- package/dist/prerender/__tests__/param-hash.test.js.map +1 -0
- package/dist/prerender/param-hash.d.ts +16 -0
- package/dist/prerender/param-hash.d.ts.map +1 -0
- package/dist/prerender/param-hash.js +36 -0
- package/dist/prerender/param-hash.js.map +1 -0
- package/dist/prerender/store.d.ts +38 -0
- package/dist/prerender/store.d.ts.map +1 -0
- package/dist/prerender/store.js +61 -0
- package/dist/prerender/store.js.map +1 -0
- package/dist/prerender.d.ts +66 -0
- package/dist/prerender.d.ts.map +1 -0
- package/dist/prerender.js +57 -0
- package/dist/prerender.js.map +1 -0
- package/dist/reverse.d.ts +196 -0
- package/dist/reverse.d.ts.map +1 -0
- package/dist/reverse.js +78 -0
- package/dist/reverse.js.map +1 -0
- package/dist/root-error-boundary.d.ts +33 -0
- package/dist/root-error-boundary.d.ts.map +1 -0
- package/dist/root-error-boundary.js +165 -0
- package/dist/root-error-boundary.js.map +1 -0
- package/dist/route-content-wrapper.d.ts +46 -0
- package/dist/route-content-wrapper.d.ts.map +1 -0
- package/dist/route-content-wrapper.js +77 -0
- package/dist/route-content-wrapper.js.map +1 -0
- package/dist/route-definition.d.ts +421 -0
- package/dist/route-definition.d.ts.map +1 -0
- package/dist/route-definition.js +868 -0
- package/dist/route-definition.js.map +1 -0
- package/dist/route-map-builder.d.ts +155 -0
- package/dist/route-map-builder.d.ts.map +1 -0
- package/dist/route-map-builder.js +237 -0
- package/dist/route-map-builder.js.map +1 -0
- package/dist/route-types.d.ts +165 -0
- package/dist/route-types.d.ts.map +1 -0
- package/dist/route-types.js +7 -0
- package/dist/route-types.js.map +1 -0
- package/dist/router/__tests__/handler-context.test.d.ts +2 -0
- package/dist/router/__tests__/handler-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/handler-context.test.js +65 -0
- package/dist/router/__tests__/handler-context.test.js.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts +2 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js +221 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js.map +1 -0
- package/dist/router/__tests__/match-context.test.d.ts +2 -0
- package/dist/router/__tests__/match-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-context.test.js +92 -0
- package/dist/router/__tests__/match-context.test.js.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts +2 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.js +417 -0
- package/dist/router/__tests__/match-pipelines.test.js.map +1 -0
- package/dist/router/__tests__/match-result.test.d.ts +2 -0
- package/dist/router/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-result.test.js +457 -0
- package/dist/router/__tests__/match-result.test.js.map +1 -0
- package/dist/router/__tests__/on-error.test.d.ts +2 -0
- package/dist/router/__tests__/on-error.test.d.ts.map +1 -0
- package/dist/router/__tests__/on-error.test.js +678 -0
- package/dist/router/__tests__/on-error.test.js.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts +2 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.js +629 -0
- package/dist/router/__tests__/pattern-matching.test.js.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts +2 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js +155 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js.map +1 -0
- package/dist/router/error-handling.d.ts +77 -0
- package/dist/router/error-handling.d.ts.map +1 -0
- package/dist/router/error-handling.js +202 -0
- package/dist/router/error-handling.js.map +1 -0
- package/dist/router/handler-context.d.ts +20 -0
- package/dist/router/handler-context.d.ts.map +1 -0
- package/dist/router/handler-context.js +198 -0
- package/dist/router/handler-context.js.map +1 -0
- package/dist/router/intercept-resolution.d.ts +66 -0
- package/dist/router/intercept-resolution.d.ts.map +1 -0
- package/dist/router/intercept-resolution.js +246 -0
- package/dist/router/intercept-resolution.js.map +1 -0
- package/dist/router/loader-resolution.d.ts +64 -0
- package/dist/router/loader-resolution.d.ts.map +1 -0
- package/dist/router/loader-resolution.js +284 -0
- package/dist/router/loader-resolution.js.map +1 -0
- package/dist/router/logging.d.ts +15 -0
- package/dist/router/logging.d.ts.map +1 -0
- package/dist/router/logging.js +99 -0
- package/dist/router/logging.js.map +1 -0
- package/dist/router/manifest.d.ts +22 -0
- package/dist/router/manifest.d.ts.map +1 -0
- package/dist/router/manifest.js +181 -0
- package/dist/router/manifest.js.map +1 -0
- package/dist/router/match-api.d.ts +35 -0
- package/dist/router/match-api.d.ts.map +1 -0
- package/dist/router/match-api.js +406 -0
- package/dist/router/match-api.js.map +1 -0
- package/dist/router/match-context.d.ts +206 -0
- package/dist/router/match-context.d.ts.map +1 -0
- package/dist/router/match-context.js +17 -0
- package/dist/router/match-context.js.map +1 -0
- package/dist/router/match-middleware/background-revalidation.d.ts +127 -0
- package/dist/router/match-middleware/background-revalidation.d.ts.map +1 -0
- package/dist/router/match-middleware/background-revalidation.js +75 -0
- package/dist/router/match-middleware/background-revalidation.js.map +1 -0
- package/dist/router/match-middleware/cache-lookup.d.ts +112 -0
- package/dist/router/match-middleware/cache-lookup.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-lookup.js +257 -0
- package/dist/router/match-middleware/cache-lookup.js.map +1 -0
- package/dist/router/match-middleware/cache-store.d.ts +113 -0
- package/dist/router/match-middleware/cache-store.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-store.js +108 -0
- package/dist/router/match-middleware/cache-store.js.map +1 -0
- package/dist/router/match-middleware/index.d.ts +81 -0
- package/dist/router/match-middleware/index.d.ts.map +1 -0
- package/dist/router/match-middleware/index.js +80 -0
- package/dist/router/match-middleware/index.js.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts +117 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.js +134 -0
- package/dist/router/match-middleware/intercept-resolution.js.map +1 -0
- package/dist/router/match-middleware/segment-resolution.d.ts +99 -0
- package/dist/router/match-middleware/segment-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/segment-resolution.js +53 -0
- package/dist/router/match-middleware/segment-resolution.js.map +1 -0
- package/dist/router/match-pipelines.d.ts +147 -0
- package/dist/router/match-pipelines.d.ts.map +1 -0
- package/dist/router/match-pipelines.js +82 -0
- package/dist/router/match-pipelines.js.map +1 -0
- package/dist/router/match-result.d.ts +126 -0
- package/dist/router/match-result.d.ts.map +1 -0
- package/dist/router/match-result.js +93 -0
- package/dist/router/match-result.js.map +1 -0
- package/dist/router/metrics.d.ts +20 -0
- package/dist/router/metrics.d.ts.map +1 -0
- package/dist/router/metrics.js +47 -0
- package/dist/router/metrics.js.map +1 -0
- package/dist/router/middleware.d.ts +249 -0
- package/dist/router/middleware.d.ts.map +1 -0
- package/dist/router/middleware.js +434 -0
- package/dist/router/middleware.js.map +1 -0
- package/dist/router/middleware.test.d.ts +2 -0
- package/dist/router/middleware.test.d.ts.map +1 -0
- package/dist/router/middleware.test.js +816 -0
- package/dist/router/middleware.test.js.map +1 -0
- package/dist/router/pattern-matching.d.ts +149 -0
- package/dist/router/pattern-matching.d.ts.map +1 -0
- package/dist/router/pattern-matching.js +349 -0
- package/dist/router/pattern-matching.js.map +1 -0
- package/dist/router/revalidation.d.ts +44 -0
- package/dist/router/revalidation.d.ts.map +1 -0
- package/dist/router/revalidation.js +147 -0
- package/dist/router/revalidation.js.map +1 -0
- package/dist/router/router-context.d.ts +135 -0
- package/dist/router/router-context.d.ts.map +1 -0
- package/dist/router/router-context.js +36 -0
- package/dist/router/router-context.js.map +1 -0
- package/dist/router/segment-resolution.d.ts +127 -0
- package/dist/router/segment-resolution.d.ts.map +1 -0
- package/dist/router/segment-resolution.js +919 -0
- package/dist/router/segment-resolution.js.map +1 -0
- package/dist/router/trie-matching.d.ts +40 -0
- package/dist/router/trie-matching.d.ts.map +1 -0
- package/dist/router/trie-matching.js +127 -0
- package/dist/router/trie-matching.js.map +1 -0
- package/dist/router/types.d.ts +136 -0
- package/dist/router/types.d.ts.map +1 -0
- package/dist/router/types.js +7 -0
- package/dist/router/types.js.map +1 -0
- package/dist/router.d.ts +753 -0
- package/dist/router.d.ts.map +1 -0
- package/dist/router.gen.d.ts +6 -0
- package/dist/router.gen.d.ts.map +1 -0
- package/dist/router.gen.js +6 -0
- package/dist/router.gen.js.map +1 -0
- package/dist/router.js +1304 -0
- package/dist/router.js.map +1 -0
- package/dist/rsc/__tests__/helpers.test.d.ts +2 -0
- package/dist/rsc/__tests__/helpers.test.d.ts.map +1 -0
- package/dist/rsc/__tests__/helpers.test.js +140 -0
- package/dist/rsc/__tests__/helpers.test.js.map +1 -0
- package/dist/rsc/handler.d.ts +45 -0
- package/dist/rsc/handler.d.ts.map +1 -0
- package/dist/rsc/handler.js +1172 -0
- package/dist/rsc/handler.js.map +1 -0
- package/dist/rsc/helpers.d.ts +16 -0
- package/dist/rsc/helpers.d.ts.map +1 -0
- package/dist/rsc/helpers.js +55 -0
- package/dist/rsc/helpers.js.map +1 -0
- package/dist/rsc/index.d.ts +22 -0
- package/dist/rsc/index.d.ts.map +1 -0
- package/dist/rsc/index.js +23 -0
- package/dist/rsc/index.js.map +1 -0
- package/dist/rsc/nonce.d.ts +9 -0
- package/dist/rsc/nonce.d.ts.map +1 -0
- package/dist/rsc/nonce.js +18 -0
- package/dist/rsc/nonce.js.map +1 -0
- package/dist/rsc/types.d.ts +206 -0
- package/dist/rsc/types.d.ts.map +1 -0
- package/dist/rsc/types.js +8 -0
- package/dist/rsc/types.js.map +1 -0
- package/dist/search-params.d.ts +103 -0
- package/dist/search-params.d.ts.map +1 -0
- package/dist/search-params.js +74 -0
- package/dist/search-params.js.map +1 -0
- package/dist/segment-system.d.ts +75 -0
- package/dist/segment-system.d.ts.map +1 -0
- package/dist/segment-system.js +336 -0
- package/dist/segment-system.js.map +1 -0
- package/dist/server/context.d.ts +245 -0
- package/dist/server/context.d.ts.map +1 -0
- package/dist/server/context.js +197 -0
- package/dist/server/context.js.map +1 -0
- package/dist/server/fetchable-loader-store.d.ts +18 -0
- package/dist/server/fetchable-loader-store.d.ts.map +1 -0
- package/dist/server/fetchable-loader-store.js +18 -0
- package/dist/server/fetchable-loader-store.js.map +1 -0
- package/dist/server/handle-store.d.ts +85 -0
- package/dist/server/handle-store.d.ts.map +1 -0
- package/dist/server/handle-store.js +142 -0
- package/dist/server/handle-store.js.map +1 -0
- package/dist/server/loader-registry.d.ts +55 -0
- package/dist/server/loader-registry.d.ts.map +1 -0
- package/dist/server/loader-registry.js +132 -0
- package/dist/server/loader-registry.js.map +1 -0
- package/dist/server/request-context.d.ts +226 -0
- package/dist/server/request-context.d.ts.map +1 -0
- package/dist/server/request-context.js +290 -0
- package/dist/server/request-context.js.map +1 -0
- package/dist/server/root-layout.d.ts +4 -0
- package/dist/server/root-layout.d.ts.map +1 -0
- package/dist/server/root-layout.js +5 -0
- package/dist/server/root-layout.js.map +1 -0
- package/dist/server.d.ts +15 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +20 -0
- package/dist/server.js.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts +2 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.js +132 -0
- package/dist/ssr/__tests__/ssr-handler.test.js.map +1 -0
- package/dist/ssr/index.d.ts +98 -0
- package/dist/ssr/index.d.ts.map +1 -0
- package/dist/ssr/index.js +158 -0
- package/dist/ssr/index.js.map +1 -0
- package/dist/static-handler.d.ts +50 -0
- package/dist/static-handler.d.ts.map +1 -0
- package/dist/static-handler.gen.d.ts +5 -0
- package/dist/static-handler.gen.d.ts.map +1 -0
- package/dist/static-handler.gen.js +5 -0
- package/dist/static-handler.gen.js.map +1 -0
- package/dist/static-handler.js +29 -0
- package/dist/static-handler.js.map +1 -0
- package/dist/testing/vitest.js +82 -0
- package/dist/theme/ThemeProvider.d.ts +20 -0
- package/dist/theme/ThemeProvider.d.ts.map +1 -0
- package/dist/theme/ThemeProvider.js +240 -0
- package/dist/theme/ThemeProvider.js.map +1 -0
- package/dist/theme/ThemeScript.d.ts +48 -0
- package/dist/theme/ThemeScript.d.ts.map +1 -0
- package/dist/theme/ThemeScript.js +13 -0
- package/dist/theme/ThemeScript.js.map +1 -0
- package/dist/theme/__tests__/theme.test.d.ts +2 -0
- package/dist/theme/__tests__/theme.test.d.ts.map +1 -0
- package/dist/theme/__tests__/theme.test.js +103 -0
- package/dist/theme/__tests__/theme.test.js.map +1 -0
- package/dist/theme/constants.d.ts +29 -0
- package/dist/theme/constants.d.ts.map +1 -0
- package/dist/theme/constants.js +48 -0
- package/dist/theme/constants.js.map +1 -0
- package/dist/theme/index.d.ts +31 -0
- package/dist/theme/index.d.ts.map +1 -0
- package/dist/theme/index.js +36 -0
- package/dist/theme/index.js.map +1 -0
- package/dist/theme/theme-context.d.ts +40 -0
- package/dist/theme/theme-context.d.ts.map +1 -0
- package/dist/theme/theme-context.js +60 -0
- package/dist/theme/theme-context.js.map +1 -0
- package/dist/theme/theme-script.d.ts +27 -0
- package/dist/theme/theme-script.d.ts.map +1 -0
- package/dist/theme/theme-script.js +147 -0
- package/dist/theme/theme-script.js.map +1 -0
- package/dist/theme/types.d.ts +163 -0
- package/dist/theme/types.d.ts.map +1 -0
- package/dist/theme/types.js +11 -0
- package/dist/theme/types.js.map +1 -0
- package/dist/theme/use-theme.d.ts +12 -0
- package/dist/theme/use-theme.d.ts.map +1 -0
- package/dist/theme/use-theme.js +40 -0
- package/dist/theme/use-theme.js.map +1 -0
- package/dist/types.d.ts +1479 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +10 -0
- package/dist/types.js.map +1 -0
- package/dist/urls.d.ts +441 -0
- package/dist/urls.d.ts.map +1 -0
- package/dist/urls.gen.d.ts +8 -0
- package/dist/urls.gen.d.ts.map +1 -0
- package/dist/urls.gen.js +8 -0
- package/dist/urls.gen.js.map +1 -0
- package/dist/urls.js +443 -0
- package/dist/urls.js.map +1 -0
- package/dist/use-loader.d.ts +127 -0
- package/dist/use-loader.d.ts.map +1 -0
- package/dist/use-loader.js +237 -0
- package/dist/use-loader.js.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts +2 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js +294 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.js +224 -0
- package/dist/vite/__tests__/expose-id-utils.test.js.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js +647 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.js +39 -0
- package/dist/vite/__tests__/expose-router-id.test.js.map +1 -0
- package/dist/vite/ast-handler-extract.d.ts +49 -0
- package/dist/vite/ast-handler-extract.d.ts.map +1 -0
- package/dist/vite/ast-handler-extract.js +249 -0
- package/dist/vite/ast-handler-extract.js.map +1 -0
- package/dist/vite/expose-action-id.d.ts +19 -0
- package/dist/vite/expose-action-id.d.ts.map +1 -0
- package/dist/vite/expose-action-id.js +250 -0
- package/dist/vite/expose-action-id.js.map +1 -0
- package/dist/vite/expose-id-utils.d.ts +69 -0
- package/dist/vite/expose-id-utils.d.ts.map +1 -0
- package/dist/vite/expose-id-utils.js +289 -0
- package/dist/vite/expose-id-utils.js.map +1 -0
- package/dist/vite/expose-internal-ids.d.ts +22 -0
- package/dist/vite/expose-internal-ids.d.ts.map +1 -0
- package/dist/vite/expose-internal-ids.js +886 -0
- package/dist/vite/expose-internal-ids.js.map +1 -0
- package/dist/vite/index.d.ts +149 -0
- package/dist/vite/index.d.ts.map +1 -0
- package/dist/vite/index.js +6263 -2733
- package/dist/vite/index.js.bak +5448 -0
- package/dist/vite/index.js.map +1 -0
- package/dist/vite/package-resolution.d.ts +43 -0
- package/dist/vite/package-resolution.d.ts.map +1 -0
- package/{src/vite/package-resolution.ts → dist/vite/package-resolution.js} +53 -66
- package/dist/vite/package-resolution.js.map +1 -0
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/dist/vite/virtual-entries.d.ts +25 -0
- package/dist/vite/virtual-entries.d.ts.map +1 -0
- package/{src/vite/virtual-entries.ts → dist/vite/virtual-entries.js} +12 -16
- package/dist/vite/virtual-entries.js.map +1 -0
- package/package.json +123 -68
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +312 -0
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +486 -0
- package/skills/caching/SKILL.md +349 -24
- package/skills/composability/SKILL.md +197 -0
- package/skills/css/SKILL.md +76 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +87 -62
- package/skills/fonts/SKILL.md +6 -4
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +557 -79
- package/skills/host-router/SKILL.md +278 -0
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +175 -8
- package/skills/layout/SKILL.md +128 -5
- package/skills/links/SKILL.md +304 -25
- package/skills/loader/SKILL.md +604 -54
- package/skills/middleware/SKILL.md +213 -37
- package/skills/migrate-nextjs/SKILL.md +584 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +41 -10
- package/skills/observability/SKILL.md +172 -0
- package/skills/parallel/SKILL.md +276 -3
- package/skills/prerender/SKILL.md +432 -52
- package/skills/rango/SKILL.md +313 -21
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +248 -120
- package/skills/route/SKILL.md +287 -21
- package/skills/router-setup/SKILL.md +231 -33
- package/skills/server-actions/SKILL.md +775 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/testing/SKILL.md +129 -0
- package/skills/testing/bindings.md +89 -0
- package/skills/testing/cache-prerender.md +124 -0
- package/skills/testing/client-components.md +122 -0
- package/skills/testing/e2e-parity.md +125 -0
- package/skills/testing/flight.md +92 -0
- package/skills/testing/handles.md +129 -0
- package/skills/testing/loader.md +128 -0
- package/skills/testing/middleware.md +99 -0
- package/skills/testing/render-handler.md +121 -0
- package/skills/testing/response-routes.md +95 -0
- package/skills/testing/reverse-and-types.md +84 -0
- package/skills/testing/server-actions.md +107 -0
- package/skills/testing/server-tree.md +128 -0
- package/skills/testing/setup.md +120 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +547 -107
- package/skills/use-cache/SKILL.md +355 -0
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +116 -0
- package/src/__internal.ts +77 -44
- package/src/bin/rango.ts +312 -15
- package/src/browser/action-coordinator.ts +114 -0
- package/src/browser/action-fence.ts +47 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/cookie-name.ts +140 -0
- package/src/browser/event-controller.ts +162 -200
- package/src/browser/history-state.ts +101 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/invalidate-client-cache.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +11 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +323 -563
- package/src/browser/navigation-client.ts +219 -75
- package/src/browser/navigation-store-handle.ts +38 -0
- package/src/browser/navigation-store.ts +104 -112
- package/src/browser/navigation-transaction.ts +247 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +328 -348
- package/src/browser/prefetch/cache.ts +324 -0
- package/src/browser/prefetch/fetch.ts +357 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +194 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +194 -0
- package/src/browser/react/Link.tsx +253 -71
- package/src/browser/react/NavigationProvider.tsx +155 -34
- package/src/browser/react/ScrollRestoration.tsx +10 -6
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +53 -0
- package/src/browser/react/index.ts +0 -48
- package/src/browser/react/location-state-shared.ts +260 -60
- package/src/browser/react/location-state.ts +90 -20
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +35 -66
- package/src/browser/react/use-handle.ts +39 -126
- package/src/browser/react/use-link-status.ts +6 -9
- package/src/browser/react/use-navigation.ts +44 -68
- package/src/browser/react/use-params.ts +75 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +98 -0
- package/src/browser/react/use-search-params.ts +51 -0
- package/src/browser/react/use-segments.ts +72 -99
- package/src/browser/response-adapter.ts +124 -0
- package/src/browser/rsc-router.tsx +290 -72
- package/src/browser/scroll-restoration.ts +132 -49
- package/src/browser/segment-reconciler.ts +243 -0
- package/src/browser/segment-structure-assert.ts +17 -1
- package/src/browser/server-action-bridge.ts +621 -613
- package/src/browser/types.ts +175 -50
- package/src/browser/validate-redirect-origin.ts +56 -0
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +123 -56
- package/src/build/generate-route-types.ts +41 -1038
- package/src/build/index.ts +9 -6
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +165 -34
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +113 -0
- package/src/build/route-types/include-resolution.ts +418 -0
- package/src/build/route-types/param-extraction.ts +51 -0
- package/src/build/route-types/per-module-writer.ts +131 -0
- package/src/build/route-types/router-processing.ts +651 -0
- package/src/build/route-types/scan-filter.ts +85 -0
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +220 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-error.ts +104 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +165 -0
- package/src/cache/cache-runtime.ts +444 -0
- package/src/cache/cache-scope.ts +231 -325
- package/src/cache/cache-tag.ts +98 -0
- package/src/cache/cf/cf-cache-store.ts +2644 -75
- package/src/cache/cf/index.ts +17 -17
- package/src/cache/document-cache.ts +172 -92
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +104 -0
- package/src/cache/index.ts +11 -35
- package/src/cache/memory-segment-store.ts +307 -30
- package/src/cache/profile-registry.ts +49 -0
- package/src/cache/read-through-swr.ts +164 -0
- package/src/cache/segment-codec.ts +240 -0
- package/src/cache/tag-invalidation.ts +230 -0
- package/src/cache/taint.ts +153 -0
- package/src/cache/types.ts +94 -211
- package/src/client.rsc.tsx +8 -21
- package/src/client.tsx +123 -347
- package/src/cloudflare/index.ts +11 -0
- package/src/cloudflare/tracing.ts +109 -0
- package/src/component-utils.ts +23 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +168 -0
- package/src/debug.ts +19 -9
- package/src/decode-loader-results.ts +36 -0
- package/src/defer.ts +196 -0
- package/src/deps/ssr.ts +0 -1
- package/src/errors.ts +106 -10
- package/src/handle.ts +76 -23
- package/src/handles/MetaTags.tsx +73 -34
- package/src/handles/breadcrumbs.ts +77 -0
- package/src/handles/meta.ts +30 -52
- package/src/host/cookie-handler.ts +21 -51
- package/src/host/errors.ts +8 -32
- package/src/host/index.ts +12 -9
- package/src/host/pattern-matcher.ts +34 -77
- package/src/host/router.ts +151 -121
- package/src/host/testing.ts +45 -32
- package/src/host/types.ts +52 -11
- package/src/host/utils.ts +2 -2
- package/src/href-client.ts +192 -57
- package/src/index.rsc.ts +174 -34
- package/src/index.ts +241 -73
- package/src/internal-debug.ts +8 -4
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +31 -99
- package/src/loader.ts +30 -12
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +4 -7
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +41 -0
- package/src/prerender/param-hash.ts +14 -13
- package/src/prerender/store.ts +121 -21
- package/src/prerender.ts +445 -24
- package/src/redirect-origin.ts +100 -0
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +198 -128
- package/src/root-error-boundary.tsx +42 -48
- package/src/route-content-wrapper.tsx +10 -72
- package/src/route-definition/dsl-helpers.ts +1116 -0
- package/src/route-definition/helper-factories.ts +90 -0
- package/src/route-definition/helpers-types.ts +506 -0
- package/src/route-definition/index.ts +55 -0
- package/src/route-definition/redirect.ts +135 -0
- package/src/route-definition/resolve-handler-use.ts +161 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-definition.ts +1 -1450
- package/src/route-map-builder.ts +82 -144
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +72 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +215 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +22 -26
- package/src/router/find-match.ts +181 -0
- package/src/router/handler-context.ts +372 -125
- package/src/router/instrument.ts +346 -0
- package/src/router/intercept-resolution.ts +34 -27
- package/src/router/lazy-includes.ts +200 -0
- package/src/router/loader-resolution.ts +381 -147
- package/src/router/logging.ts +106 -6
- package/src/router/manifest.ts +88 -51
- package/src/router/match-api.ts +162 -245
- package/src/router/match-context.ts +4 -24
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +90 -89
- package/src/router/match-middleware/cache-lookup.ts +297 -150
- package/src/router/match-middleware/cache-store.ts +75 -33
- package/src/router/match-middleware/intercept-resolution.ts +44 -43
- package/src/router/match-middleware/segment-resolution.ts +64 -22
- package/src/router/match-pipelines.ts +11 -87
- package/src/router/match-result.ts +121 -50
- package/src/router/metrics.ts +219 -28
- package/src/router/middleware-types.ts +93 -0
- package/src/router/middleware.ts +421 -413
- package/src/router/navigation-snapshot.ts +131 -0
- package/src/router/params-util.ts +23 -0
- package/src/router/pattern-matching.ts +263 -79
- package/src/router/prerender-match.ts +541 -0
- package/src/router/preview-match.ts +100 -0
- package/src/router/request-classification.ts +276 -0
- package/src/router/revalidation.ts +143 -44
- package/src/router/route-snapshot.ts +244 -0
- package/src/router/router-context.ts +41 -47
- package/src/router/router-interfaces.ts +525 -0
- package/src/router/router-options.ts +726 -0
- package/src/router/router-registry.ts +21 -0
- package/src/router/segment-resolution/fresh.ts +747 -0
- package/src/router/segment-resolution/helpers.ts +314 -0
- package/src/router/segment-resolution/loader-cache.ts +207 -0
- package/src/router/segment-resolution/revalidation.ts +1322 -0
- package/src/router/segment-resolution/static-store.ts +81 -0
- package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +24 -1354
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/state-cookie-name.ts +33 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +261 -0
- package/src/router/telemetry.ts +377 -0
- package/src/router/timeout.ts +128 -0
- package/src/router/tracing.ts +206 -0
- package/src/router/trie-matching.ts +172 -60
- package/src/router/types.ts +23 -70
- package/src/router/url-params.ts +44 -0
- package/src/router.ts +748 -2376
- package/src/rsc/handler-context.ts +46 -0
- package/src/rsc/handler.ts +861 -1141
- package/src/rsc/helpers.ts +269 -19
- package/src/rsc/index.ts +1 -21
- package/src/rsc/json-route-result.ts +38 -0
- package/src/rsc/loader-fetch.ts +235 -0
- package/src/rsc/manifest-init.ts +77 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +155 -0
- package/src/rsc/progressive-enhancement.ts +413 -0
- package/src/rsc/redirect-guard.ts +99 -0
- package/src/rsc/response-error.ts +104 -0
- package/src/rsc/response-route-handler.ts +374 -0
- package/src/rsc/rsc-rendering.ts +261 -0
- package/src/rsc/runtime-warnings.ts +55 -0
- package/src/rsc/server-action.ts +376 -0
- package/src/rsc/ssr-setup.ts +144 -0
- package/src/rsc/types.ts +58 -12
- package/src/runtime-env.ts +18 -0
- package/src/search-params.ts +70 -74
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +134 -0
- package/src/segment-system.tsx +292 -134
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +439 -85
- package/src/server/cookie-store.ts +265 -0
- package/src/server/fetchable-loader-store.ts +11 -6
- package/src/server/handle-store.ts +112 -31
- package/src/server/loader-registry.ts +23 -82
- package/src/server/request-context.ts +724 -143
- package/src/server.ts +26 -164
- package/src/ssr/index.tsx +113 -36
- package/src/static-handler.ts +45 -18
- package/src/testing/cache-status.ts +162 -0
- package/src/testing/collect-handle.ts +40 -0
- package/src/testing/dispatch.ts +618 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +188 -0
- package/src/testing/e2e/index.ts +128 -0
- package/src/testing/e2e/matchers.ts +35 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +387 -0
- package/src/testing/e2e/server.ts +195 -0
- package/src/testing/flight-matchers.ts +97 -0
- package/src/testing/flight-normalize.ts +11 -0
- package/src/testing/flight-runtime.d.ts +57 -0
- package/src/testing/flight-tree.ts +682 -0
- package/src/testing/flight.entry.ts +52 -0
- package/src/testing/flight.ts +232 -0
- package/src/testing/generated-routes.ts +183 -0
- package/src/testing/index.ts +99 -0
- package/src/testing/internal/context.ts +348 -0
- package/src/testing/internal/flight-client-globals.ts +30 -0
- package/src/testing/internal/seed-vars.ts +54 -0
- package/src/testing/render-handler.ts +330 -0
- package/src/testing/render-route.tsx +566 -0
- package/src/testing/run-loader.ts +378 -0
- package/src/testing/run-middleware.ts +205 -0
- package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
- package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
- package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
- package/src/testing/vitest-stubs/version.ts +5 -0
- package/src/testing/vitest.ts +305 -0
- package/src/theme/ThemeProvider.tsx +21 -67
- package/src/theme/ThemeScript.tsx +5 -11
- package/src/theme/constants.ts +5 -14
- package/src/theme/index.ts +3 -20
- package/src/theme/theme-context.ts +5 -35
- package/src/theme/theme-script.ts +21 -32
- package/src/theme/use-theme.ts +0 -3
- package/src/types/boundaries.ts +123 -0
- package/src/types/cache-types.ts +207 -0
- package/src/types/error-types.ts +132 -0
- package/src/types/global-namespace.ts +113 -0
- package/src/types/handler-context.ts +839 -0
- package/src/types/index.ts +79 -0
- package/src/types/loader-types.ts +212 -0
- package/src/types/request-scope.ts +107 -0
- package/src/types/route-config.ts +126 -0
- package/src/types/route-entry.ts +114 -0
- package/src/types/segments.ts +171 -0
- package/src/types.ts +1 -1795
- package/src/urls/include-helper.ts +160 -0
- package/src/urls/index.ts +43 -0
- package/src/urls/path-helper-types.ts +386 -0
- package/src/urls/path-helper.ts +275 -0
- package/src/urls/pattern-types.ts +124 -0
- package/src/urls/response-types.ts +109 -0
- package/src/urls/type-extraction.ts +291 -0
- package/src/urls/urls-function.ts +81 -0
- package/src/urls.ts +1 -1323
- package/src/use-loader.tsx +406 -141
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +182 -0
- package/src/vite/discovery/discover-routers.ts +389 -0
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +467 -0
- package/src/vite/discovery/route-types-writer.ts +214 -0
- package/src/vite/discovery/self-gen-tracking.ts +73 -0
- package/src/vite/discovery/state.ts +161 -0
- package/src/vite/discovery/virtual-module-codegen.ts +183 -0
- package/src/vite/index.ts +17 -2259
- package/src/vite/plugin-types.ts +221 -0
- package/src/vite/plugins/cjs-to-esm.ts +83 -0
- package/src/vite/plugins/client-ref-dedup.ts +120 -0
- package/src/vite/plugins/client-ref-hashing.ts +118 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +88 -110
- package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +13 -87
- package/src/vite/plugins/expose-ids/export-analysis.ts +338 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +141 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +57 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +124 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +806 -0
- package/src/vite/plugins/performance-tracks.ts +89 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +313 -0
- package/src/vite/plugins/version-injector.ts +79 -0
- package/src/vite/plugins/version-plugin.ts +275 -0
- package/src/vite/plugins/virtual-entries.ts +108 -0
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +444 -0
- package/src/vite/router-discovery.ts +1581 -0
- package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +193 -37
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +132 -0
- package/src/vite/utils/client-chunks.ts +184 -0
- package/src/vite/utils/forward-user-plugins.ts +171 -0
- package/src/vite/utils/manifest-utils.ts +15 -0
- package/src/vite/utils/package-resolution.ts +89 -0
- package/src/vite/utils/prerender-utils.ts +223 -0
- package/src/vite/utils/shared-utils.ts +219 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/browser/shallow.ts +0 -35
- package/src/cache/memory-store.ts +0 -253
- package/src/router.gen.ts +0 -6
- package/src/static-handler.gen.ts +0 -5
- package/src/urls.gen.ts +0 -8
- package/src/vite/expose-internal-ids.ts +0 -1167
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
package/src/rsc/handler.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/// <reference types="@vitejs/plugin-rsc/types" />
|
|
2
|
-
/// <reference path="../vite/version.d.ts" />
|
|
2
|
+
/// <reference path="../vite/plugins/version.d.ts" />
|
|
3
3
|
/**
|
|
4
4
|
* RSC Request Handler
|
|
5
5
|
*
|
|
@@ -8,72 +8,93 @@
|
|
|
8
8
|
*/
|
|
9
9
|
|
|
10
10
|
import { createElement } from "react";
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
13
|
-
import type { ResponseError } from "../urls.js";
|
|
14
|
-
import { getLoaderLazy } from "../server/loader-registry.js";
|
|
15
|
-
import {
|
|
16
|
-
matchMiddleware,
|
|
17
|
-
executeMiddleware,
|
|
18
|
-
executeLoaderMiddleware,
|
|
19
|
-
} from "../router/middleware.js";
|
|
11
|
+
import { isRouteNotFoundError } from "../errors.js";
|
|
12
|
+
import { matchMiddleware, executeMiddleware } from "../router/middleware.js";
|
|
20
13
|
import {
|
|
21
14
|
runWithRequestContext,
|
|
22
15
|
setRequestContextParams,
|
|
23
16
|
requireRequestContext,
|
|
17
|
+
getRequestContext,
|
|
18
|
+
_getRequestContext,
|
|
24
19
|
createRequestContext,
|
|
25
|
-
type ExecutionContext,
|
|
26
20
|
} from "../server/request-context.js";
|
|
27
21
|
import * as rscDeps from "@vitejs/plugin-rsc/rsc";
|
|
28
|
-
|
|
29
22
|
import type {
|
|
30
23
|
RscPayload,
|
|
31
|
-
ReactFormState,
|
|
32
24
|
CreateRSCHandlerOptions,
|
|
25
|
+
LoadSSRModule,
|
|
26
|
+
SSRModule,
|
|
33
27
|
} from "./types.js";
|
|
34
|
-
import {
|
|
35
|
-
|
|
28
|
+
import {
|
|
29
|
+
createResponseWithMergedHeaders,
|
|
30
|
+
finalizeResponse,
|
|
31
|
+
interceptRedirectForPartial,
|
|
32
|
+
buildRouteMiddlewareEntries,
|
|
33
|
+
} from "./helpers.js";
|
|
34
|
+
import { guardOutgoingRedirect } from "./redirect-guard.js";
|
|
35
|
+
import { isWebSocketUpgradeResponse } from "../response-utils.js";
|
|
36
|
+
import {
|
|
37
|
+
handleResponseRoute,
|
|
38
|
+
type ResponseRouteMatch,
|
|
39
|
+
} from "./response-route-handler.js";
|
|
40
|
+
import { generateNonce, nonce as nonceToken } from "./nonce.js";
|
|
36
41
|
import { VERSION } from "@rangojs/router:version";
|
|
37
42
|
import type { ErrorPhase } from "../types.js";
|
|
43
|
+
import type { RouterRequestInput } from "../router/router-interfaces.js";
|
|
38
44
|
import { invokeOnError } from "../router/error-handling.js";
|
|
39
45
|
import {
|
|
40
|
-
|
|
46
|
+
createReverseFunction,
|
|
47
|
+
stripInternalParams,
|
|
48
|
+
} from "../router/handler-context.js";
|
|
49
|
+
import { contextSet } from "../context-var.js";
|
|
50
|
+
import {
|
|
41
51
|
hasCachedManifest,
|
|
42
|
-
setCachedManifest,
|
|
43
52
|
getRouteTrie,
|
|
44
|
-
setRouteTrie,
|
|
45
53
|
getPrecomputedEntries,
|
|
46
54
|
waitForManifestReady,
|
|
47
55
|
getRouterManifest,
|
|
48
56
|
getRouterTrie,
|
|
49
|
-
setRouterManifest,
|
|
50
|
-
setRouterTrie,
|
|
51
57
|
} from "../route-map-builder.js";
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
58
|
+
import type { HandlerContext } from "./handler-context.js";
|
|
59
|
+
import type { CacheErrorCategory } from "../cache/cache-error.js";
|
|
60
|
+
import type { SegmentCacheStore } from "../cache/types.js";
|
|
61
|
+
import { buildRouterTrieFromUrlpatterns } from "./manifest-init.js";
|
|
62
|
+
import { handleProgressiveEnhancement } from "./progressive-enhancement.js";
|
|
63
|
+
import {
|
|
64
|
+
executeServerAction,
|
|
65
|
+
revalidateAfterAction,
|
|
66
|
+
type ActionContinuation,
|
|
67
|
+
} from "./server-action.js";
|
|
68
|
+
import { handleLoaderFetch } from "./loader-fetch.js";
|
|
69
|
+
import {
|
|
70
|
+
checkRequestOrigin,
|
|
71
|
+
ORIGIN_CHECK_PHASE_BY_MODE,
|
|
72
|
+
} from "./origin-guard.js";
|
|
73
|
+
import { handleRscRendering } from "./rsc-rendering.js";
|
|
74
|
+
import {
|
|
75
|
+
withTimeout,
|
|
76
|
+
RouterTimeoutError,
|
|
77
|
+
createDefaultTimeoutResponse,
|
|
78
|
+
type TimeoutPhase,
|
|
79
|
+
} from "../router/timeout.js";
|
|
80
|
+
import {
|
|
81
|
+
createMetricsStore,
|
|
82
|
+
appendMetric,
|
|
83
|
+
buildMetricsTiming,
|
|
84
|
+
} from "../router/metrics.js";
|
|
85
|
+
import { observePhase, observeEvent, PHASES } from "../router/instrument.js";
|
|
86
|
+
import {
|
|
87
|
+
startSSRSetup,
|
|
88
|
+
getSSRSetup,
|
|
89
|
+
mayNeedSSR,
|
|
90
|
+
isRscRequest,
|
|
91
|
+
SSR_SETUP_VAR,
|
|
92
|
+
} from "./ssr-setup.js";
|
|
93
|
+
import {
|
|
94
|
+
classifyRequest,
|
|
95
|
+
type RequestPlan,
|
|
96
|
+
type ExecutableRequestPlan,
|
|
97
|
+
} from "../router/request-classification.js";
|
|
77
98
|
|
|
78
99
|
/**
|
|
79
100
|
* Create an RSC request handler.
|
|
@@ -108,12 +129,35 @@ function createResponseErrorPayload(error: unknown, isDev: boolean): ResponseErr
|
|
|
108
129
|
* });
|
|
109
130
|
* ```
|
|
110
131
|
*/
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Response that tells the client to do a full document navigation. Shared by
|
|
135
|
+
* the terminal reload plans (version-mismatch and app-switch): an empty 200
|
|
136
|
+
* carrying X-RSC-Reload, which the client turns into window.location.href.
|
|
137
|
+
*/
|
|
138
|
+
function createReloadResponse(reloadUrl: string) {
|
|
139
|
+
return createResponseWithMergedHeaders(null, {
|
|
140
|
+
status: 200,
|
|
141
|
+
headers: {
|
|
142
|
+
"X-RSC-Reload": reloadUrl,
|
|
143
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
144
|
+
},
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
|
|
111
148
|
export function createRSCHandler<
|
|
112
149
|
TEnv = unknown,
|
|
113
150
|
TRoutes extends Record<string, string> = Record<string, string>,
|
|
114
151
|
>(options: CreateRSCHandlerOptions<TEnv, TRoutes>) {
|
|
115
152
|
const { router, version = VERSION, nonce: nonceProvider } = options;
|
|
116
153
|
|
|
154
|
+
// Handler-owned registry of explicit per-scope stores from cache({ store }).
|
|
155
|
+
// Lives in the closure so it is scoped per handler (multi-router deployments
|
|
156
|
+
// get separate registries) and accumulates every explicit store this handler
|
|
157
|
+
// resolves across requests. updateTag()/revalidateTag() iterate it to reach
|
|
158
|
+
// stores not covered by the app-level ctx._cacheStore.
|
|
159
|
+
const explicitTaggedStores = new Set<SegmentCacheStore>();
|
|
160
|
+
|
|
117
161
|
// Use provided deps or default to @vitejs/plugin-rsc/rsc exports
|
|
118
162
|
const deps = options.deps ?? rscDeps;
|
|
119
163
|
const {
|
|
@@ -125,33 +169,175 @@ export function createRSCHandler<
|
|
|
125
169
|
decodeFormState,
|
|
126
170
|
} = deps;
|
|
127
171
|
|
|
128
|
-
// Use provided loadSSRModule or default to vite RSC module loader
|
|
129
|
-
|
|
172
|
+
// Use provided loadSSRModule or default to vite RSC module loader.
|
|
173
|
+
// In production the SSR module is stable across requests, so memoize
|
|
174
|
+
// the dynamic import to avoid repeated module resolution overhead.
|
|
175
|
+
// In dev mode Vite may hot-reload the module, so skip memoization.
|
|
176
|
+
const rawLoadSSRModule: LoadSSRModule =
|
|
130
177
|
options.loadSSRModule ??
|
|
131
178
|
(() => import.meta.viteRsc.loadModule("ssr", "index"));
|
|
179
|
+
let _ssrModulePromise: Promise<SSRModule> | undefined;
|
|
180
|
+
const loadSSRModule: LoadSSRModule =
|
|
181
|
+
process.env.NODE_ENV === "production"
|
|
182
|
+
? () =>
|
|
183
|
+
(_ssrModulePromise ??= rawLoadSSRModule().catch((err) => {
|
|
184
|
+
_ssrModulePromise = undefined;
|
|
185
|
+
throw err;
|
|
186
|
+
}))
|
|
187
|
+
: rawLoadSSRModule;
|
|
132
188
|
|
|
133
189
|
/**
|
|
134
|
-
*
|
|
135
|
-
*
|
|
190
|
+
* Per-request error reporter that deduplicates via the ALS request context.
|
|
191
|
+
*
|
|
192
|
+
* Uses the same _reportedErrors WeakSet as the router layer so errors
|
|
193
|
+
* that propagate across layers are only reported once per request.
|
|
136
194
|
*/
|
|
137
195
|
function callOnError(
|
|
138
196
|
error: unknown,
|
|
139
197
|
phase: ErrorPhase,
|
|
140
198
|
context: Parameters<typeof invokeOnError<TEnv>>[3],
|
|
141
199
|
): void {
|
|
200
|
+
// Guard: abort signal handlers fire asynchronously outside the ALS
|
|
201
|
+
// request scope, so the context may be gone. Skip dedup in that
|
|
202
|
+
// case — the error is from a cancelled stream, not a real failure.
|
|
203
|
+
const reqCtx = _getRequestContext();
|
|
204
|
+
if (error != null && typeof error === "object" && reqCtx) {
|
|
205
|
+
if (reqCtx._reportedErrors.has(error)) return;
|
|
206
|
+
reqCtx._reportedErrors.add(error);
|
|
207
|
+
}
|
|
142
208
|
invokeOnError(router.onError, error, phase, context, "RSC");
|
|
143
209
|
}
|
|
144
210
|
|
|
145
|
-
|
|
211
|
+
function getRequiredRouteMap(): Record<string, string> {
|
|
212
|
+
const routeMap = getRouterManifest(router.id);
|
|
213
|
+
if (!routeMap) {
|
|
214
|
+
throw new Error(
|
|
215
|
+
`Route manifest for router "${router.id}" is not available.`,
|
|
216
|
+
);
|
|
217
|
+
}
|
|
218
|
+
return routeMap;
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Handle a timeout by reporting the error, emitting telemetry,
|
|
223
|
+
* and returning either the custom onTimeout response or a default 504.
|
|
224
|
+
*/
|
|
225
|
+
async function handleTimeoutResponse(
|
|
146
226
|
request: Request,
|
|
147
|
-
env: TEnv
|
|
148
|
-
|
|
227
|
+
env: TEnv,
|
|
228
|
+
url: URL,
|
|
229
|
+
phase: TimeoutPhase,
|
|
230
|
+
durationMs: number,
|
|
231
|
+
routeKey?: string,
|
|
232
|
+
actionId?: string,
|
|
233
|
+
): Promise<Response> {
|
|
234
|
+
const timeoutError = new RouterTimeoutError(phase, durationMs);
|
|
235
|
+
|
|
236
|
+
callOnError(timeoutError, phase === "action" ? "action" : "handler", {
|
|
237
|
+
request,
|
|
238
|
+
url,
|
|
239
|
+
env,
|
|
240
|
+
routeKey,
|
|
241
|
+
actionId,
|
|
242
|
+
handledByBoundary: false,
|
|
243
|
+
metadata: { timeout: true, phase, durationMs },
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
observeEvent({
|
|
247
|
+
type: "request.timeout",
|
|
248
|
+
timestamp: performance.now(),
|
|
249
|
+
phase,
|
|
250
|
+
pathname: url.pathname,
|
|
251
|
+
routeKey,
|
|
252
|
+
actionId,
|
|
253
|
+
durationMs,
|
|
254
|
+
customHandler: !!router.onTimeout,
|
|
255
|
+
});
|
|
256
|
+
|
|
257
|
+
if (router.onTimeout) {
|
|
258
|
+
try {
|
|
259
|
+
return await router.onTimeout({
|
|
260
|
+
phase,
|
|
261
|
+
request,
|
|
262
|
+
url,
|
|
263
|
+
env,
|
|
264
|
+
routeKey,
|
|
265
|
+
actionId,
|
|
266
|
+
durationMs,
|
|
267
|
+
});
|
|
268
|
+
} catch (e) {
|
|
269
|
+
if (process.env.NODE_ENV !== "production") {
|
|
270
|
+
console.error("[RSC] onTimeout callback error:", e);
|
|
271
|
+
}
|
|
272
|
+
return createDefaultTimeoutResponse(phase);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
return createDefaultTimeoutResponse(phase);
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
/**
|
|
280
|
+
* Build a 200 Flight response that carries a redirect URL and optional state.
|
|
281
|
+
* Used when a partial/action request results in a redirect -- fetch
|
|
282
|
+
* auto-follows 3xx so we send the redirect as payload metadata instead.
|
|
283
|
+
*/
|
|
284
|
+
function createRedirectFlightResponse(
|
|
285
|
+
redirectUrl: string,
|
|
286
|
+
locationState?: Record<string, unknown>,
|
|
287
|
+
external?: boolean,
|
|
288
|
+
): Response {
|
|
289
|
+
const redirectPayload: RscPayload = {
|
|
290
|
+
metadata: {
|
|
291
|
+
pathname: redirectUrl,
|
|
292
|
+
segments: [],
|
|
293
|
+
redirect: { url: redirectUrl, ...(external && { external: true }) },
|
|
294
|
+
...(locationState && { locationState }),
|
|
295
|
+
},
|
|
296
|
+
};
|
|
297
|
+
const rscStream = renderToReadableStream<RscPayload>(redirectPayload);
|
|
298
|
+
return createResponseWithMergedHeaders(rscStream, {
|
|
299
|
+
status: 200,
|
|
300
|
+
headers: { "content-type": "text/x-component;charset=utf-8" },
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
// Bundle shared dependencies for extracted handler functions.
|
|
305
|
+
// callOnError reads from ALS so it's inherently per-request scoped.
|
|
306
|
+
const handlerCtx: HandlerContext<TEnv> = {
|
|
307
|
+
router,
|
|
308
|
+
version,
|
|
309
|
+
renderToReadableStream,
|
|
310
|
+
decodeReply,
|
|
311
|
+
createTemporaryReferenceSet,
|
|
312
|
+
loadServerAction,
|
|
313
|
+
decodeAction,
|
|
314
|
+
decodeFormState,
|
|
315
|
+
loadSSRModule,
|
|
316
|
+
callOnError,
|
|
317
|
+
getRequiredRouteMap,
|
|
318
|
+
createRedirectFlightResponse,
|
|
319
|
+
resolveStreamMode: async (request, env, url) => {
|
|
320
|
+
const resolver = router.ssr?.resolveStreaming;
|
|
321
|
+
if (!resolver) return "stream";
|
|
322
|
+
return resolver({ request, env, url });
|
|
149
323
|
},
|
|
324
|
+
};
|
|
325
|
+
|
|
326
|
+
return async function handler(
|
|
327
|
+
request: Request,
|
|
328
|
+
input: RouterRequestInput<TEnv> = {},
|
|
150
329
|
): Promise<Response> {
|
|
151
330
|
const handlerStart = performance.now();
|
|
331
|
+
// Create the metrics store at handler start so handler:total has startTime=0
|
|
332
|
+
// and all metrics are relative to the request entry point.
|
|
333
|
+
const earlyMetricsStore = router.debugPerformance
|
|
334
|
+
? createMetricsStore(true, handlerStart)
|
|
335
|
+
: undefined;
|
|
336
|
+
|
|
337
|
+
const { env = {} as TEnv, vars: initialVars, ctx: executionCtx } = input;
|
|
152
338
|
|
|
153
339
|
// Connection warmup: return 204 immediately before any processing
|
|
154
|
-
if (router
|
|
340
|
+
if (router?.warmupEnabled && request.method === "HEAD") {
|
|
155
341
|
const warmupUrl = new URL(request.url);
|
|
156
342
|
if (warmupUrl.searchParams.has("_rsc_warmup")) {
|
|
157
343
|
return new Response(null, { status: 204 });
|
|
@@ -175,24 +361,27 @@ export function createRSCHandler<
|
|
|
175
361
|
const mwMatchDur = performance.now() - mwMatchStart;
|
|
176
362
|
|
|
177
363
|
// Shared variables between middleware and route handlers
|
|
178
|
-
// Initialize from
|
|
179
|
-
const variables: Record<string, any> =
|
|
180
|
-
|
|
181
|
-
|
|
364
|
+
// Initialize from input.vars if provided (allows pre-seeding from worker entry)
|
|
365
|
+
const variables: Record<string, any> = initialVars
|
|
366
|
+
? { ...initialVars }
|
|
367
|
+
: {};
|
|
182
368
|
|
|
183
|
-
// Store nonce
|
|
369
|
+
// Store nonce via ContextVar token and string key for backward compat
|
|
184
370
|
if (nonce) {
|
|
371
|
+
contextSet(variables, nonceToken, nonce);
|
|
185
372
|
variables.nonce = nonce;
|
|
186
373
|
}
|
|
187
374
|
|
|
188
375
|
// Resolve cache store configuration
|
|
189
376
|
// Priority: options.cache (handler override) > router.cache (router default)
|
|
190
377
|
// Store is enabled only if: config provided, enabled, and no ?__no_cache query param
|
|
191
|
-
let cacheStore
|
|
378
|
+
let cacheStore: SegmentCacheStore | undefined;
|
|
192
379
|
const cacheOption = options.cache ?? router.cache;
|
|
193
380
|
if (cacheOption && !url.searchParams.has("__no_cache")) {
|
|
194
381
|
const cacheConfig =
|
|
195
|
-
typeof cacheOption === "function"
|
|
382
|
+
typeof cacheOption === "function"
|
|
383
|
+
? cacheOption(env, executionCtx)
|
|
384
|
+
: cacheOption;
|
|
196
385
|
|
|
197
386
|
if (cacheConfig.enabled !== false) {
|
|
198
387
|
cacheStore = cacheConfig.store;
|
|
@@ -223,56 +412,7 @@ export function createRSCHandler<
|
|
|
223
412
|
// Cloudflare dev: generate manifest inline for this router.
|
|
224
413
|
// Each router generates its own manifest independently so
|
|
225
414
|
// multi-router setups (host routing) work correctly.
|
|
226
|
-
|
|
227
|
-
await import("../build/generate-manifest.js");
|
|
228
|
-
const generated = generateManifest(router.urlpatterns);
|
|
229
|
-
if (
|
|
230
|
-
generated._routeAncestry &&
|
|
231
|
-
Object.keys(generated._routeAncestry).length > 0
|
|
232
|
-
) {
|
|
233
|
-
const { buildRouteTrie } = await import("../build/route-trie.js");
|
|
234
|
-
// Map each route to its include() staticPrefix so the trie
|
|
235
|
-
// returns the correct sp for lazy entry lookup in findMatch.
|
|
236
|
-
const routeToStaticPrefix: Record<string, string> = {};
|
|
237
|
-
for (const name of Object.keys(generated.routeManifest)) {
|
|
238
|
-
routeToStaticPrefix[name] = "";
|
|
239
|
-
}
|
|
240
|
-
// Override with prefix from include() entries so the trie
|
|
241
|
-
// returns the correct sp for lazy entry lookup in findMatch.
|
|
242
|
-
// Walk recursively to include routes in nested includes.
|
|
243
|
-
if (generated.prefixTree) {
|
|
244
|
-
const visitPrefixNode = (node: any): void => {
|
|
245
|
-
const sp = node.staticPrefix || "";
|
|
246
|
-
for (const route of (node.routes || [])) {
|
|
247
|
-
routeToStaticPrefix[route] = sp;
|
|
248
|
-
}
|
|
249
|
-
for (const child of Object.values(node.children || {})) {
|
|
250
|
-
visitPrefixNode(child);
|
|
251
|
-
}
|
|
252
|
-
};
|
|
253
|
-
for (const node of Object.values(generated.prefixTree)) {
|
|
254
|
-
visitPrefixNode(node);
|
|
255
|
-
}
|
|
256
|
-
}
|
|
257
|
-
const trie = buildRouteTrie(
|
|
258
|
-
generated.routeManifest,
|
|
259
|
-
generated._routeAncestry,
|
|
260
|
-
routeToStaticPrefix,
|
|
261
|
-
generated.routeTrailingSlash,
|
|
262
|
-
generated.prerenderRoutes ? new Set(generated.prerenderRoutes) : undefined,
|
|
263
|
-
generated.passthroughRoutes ? new Set(generated.passthroughRoutes) : undefined,
|
|
264
|
-
generated.responseTypeRoutes,
|
|
265
|
-
);
|
|
266
|
-
setRouterTrie(router.id, trie);
|
|
267
|
-
// Set global trie only if not already set by another router
|
|
268
|
-
if (!getRouteTrie()) {
|
|
269
|
-
setRouteTrie(trie);
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
setRouterManifest(router.id, generated.routeManifest);
|
|
273
|
-
// Merge into global manifest (needed for reverse/href across routers)
|
|
274
|
-
const existing = hasCachedManifest() ? getGlobalRouteMap() : {};
|
|
275
|
-
setCachedManifest({ ...existing, ...generated.routeManifest });
|
|
415
|
+
await buildRouterTrieFromUrlpatterns(router);
|
|
276
416
|
}
|
|
277
417
|
if (!getRouterManifest(router.id) && !hasCachedManifest()) {
|
|
278
418
|
throw new Error(
|
|
@@ -280,10 +420,17 @@ export function createRSCHandler<
|
|
|
280
420
|
);
|
|
281
421
|
}
|
|
282
422
|
}
|
|
283
|
-
const manifestCacheDur = performance.now() - manifestCacheStart;
|
|
284
423
|
|
|
285
|
-
//
|
|
286
|
-
// This
|
|
424
|
+
// Rebuild the trie when the manifest exists but the per-router trie is
|
|
425
|
+
// missing. This happens in dev mode after HMR: the virtual module sets
|
|
426
|
+
// the manifest (from fresh gen files) but the trie is intentionally not
|
|
427
|
+
// injected to avoid stale discovery-time data. Without the trie, route
|
|
428
|
+
// matching falls back to regex iteration which does not handle wildcard
|
|
429
|
+
// priority correctly (catch-all patterns match before specific routes).
|
|
430
|
+
if (!getRouterTrie(router.id) && router.urlpatterns) {
|
|
431
|
+
await buildRouterTrieFromUrlpatterns(router);
|
|
432
|
+
}
|
|
433
|
+
const manifestCacheDur = performance.now() - manifestCacheStart;
|
|
287
434
|
|
|
288
435
|
// Create unified request context with all methods
|
|
289
436
|
// Includes: stub response, handle store, loader memoization, use(), cookies, headers, cache store
|
|
@@ -295,9 +442,30 @@ export function createRSCHandler<
|
|
|
295
442
|
url,
|
|
296
443
|
variables,
|
|
297
444
|
cacheStore,
|
|
298
|
-
|
|
445
|
+
explicitTaggedStores,
|
|
446
|
+
cacheProfiles: router.cacheProfiles,
|
|
447
|
+
executionContext: executionCtx,
|
|
299
448
|
themeConfig: router.themeConfig,
|
|
449
|
+
stateCookieName: router.resolvedStateCookieName,
|
|
450
|
+
version,
|
|
300
451
|
});
|
|
452
|
+
if (earlyMetricsStore) {
|
|
453
|
+
requestContext._debugPerformance = true;
|
|
454
|
+
requestContext._metricsStore = earlyMetricsStore;
|
|
455
|
+
}
|
|
456
|
+
// Wire background error reporting so "use cache" and other subsystems
|
|
457
|
+
// can surface non-fatal errors through the router's onError callback.
|
|
458
|
+
requestContext._reportBackgroundError = (
|
|
459
|
+
error: unknown,
|
|
460
|
+
category: CacheErrorCategory,
|
|
461
|
+
) => {
|
|
462
|
+
callOnError(error, "cache", {
|
|
463
|
+
request,
|
|
464
|
+
url,
|
|
465
|
+
metadata: { category },
|
|
466
|
+
});
|
|
467
|
+
};
|
|
468
|
+
|
|
301
469
|
const ctxCreateDur = performance.now() - ctxCreateStart;
|
|
302
470
|
|
|
303
471
|
// Accumulate handler-level timing for Server-Timing header
|
|
@@ -319,296 +487,132 @@ export function createRSCHandler<
|
|
|
319
487
|
// - Server components during rendering
|
|
320
488
|
// - Error boundaries
|
|
321
489
|
// - Streaming
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
url: URL,
|
|
348
|
-
variables: Record<string, any>,
|
|
349
|
-
nonce: string | undefined,
|
|
350
|
-
): Promise<Response> {
|
|
351
|
-
// First, check for route-level middleware
|
|
352
|
-
const previewStart = performance.now();
|
|
353
|
-
const preview = await router.previewMatch(request, env);
|
|
354
|
-
const previewDur = performance.now() - previewStart;
|
|
355
|
-
const handlerTiming: string[] = variables.__handlerTiming || [];
|
|
356
|
-
handlerTiming.push(`handler-preview-match;dur=${previewDur.toFixed(2)}`);
|
|
357
|
-
// Response route short-circuit: skip entire RSC pipeline
|
|
358
|
-
if (preview?.responseType && preview.handler) {
|
|
359
|
-
const isPartial = url.searchParams.has("_rsc_partial");
|
|
360
|
-
|
|
361
|
-
// Partial requests (client-side navigation) to response routes
|
|
362
|
-
// get X-RSC-Reload to trigger hard navigation in the browser
|
|
363
|
-
if (isPartial) {
|
|
364
|
-
const cleanUrl = new URL(url);
|
|
365
|
-
cleanUrl.searchParams.delete("_rsc_partial");
|
|
366
|
-
cleanUrl.searchParams.delete("_rsc_segments");
|
|
367
|
-
cleanUrl.searchParams.delete("_rsc_v");
|
|
368
|
-
cleanUrl.searchParams.delete("_rsc_stale");
|
|
369
|
-
cleanUrl.searchParams.delete("_rsc_action");
|
|
370
|
-
cleanUrl.searchParams.delete("_rsc_prev");
|
|
371
|
-
|
|
372
|
-
return createResponseWithMergedHeaders(null, {
|
|
373
|
-
status: 200,
|
|
374
|
-
headers: {
|
|
375
|
-
"X-RSC-Reload": cleanUrl.toString(),
|
|
376
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
377
|
-
},
|
|
378
|
-
});
|
|
379
|
-
}
|
|
490
|
+
// Store basename on request context (scoped per-request via existing ALS)
|
|
491
|
+
requestContext._basename = router.basename;
|
|
492
|
+
|
|
493
|
+
// Resolved span tracing for this request (read at each traced phase).
|
|
494
|
+
requestContext._tracing = router.tracing;
|
|
495
|
+
|
|
496
|
+
// The "rango.request" span is opened inside the request context so the
|
|
497
|
+
// Cloudflare runner can read executionContext.tracing, and so every nested
|
|
498
|
+
// phase span (and the platform's automatic KV/D1/fetch spans) nests under
|
|
499
|
+
// it. Construction-bound: the span ends when the Response is built, never
|
|
500
|
+
// wrapping the streamed body. metric:false — handler:total is metered
|
|
501
|
+
// directly below (a grand total incl. the pre-context bootstrap timings).
|
|
502
|
+
// When tracing is off this is a direct pass-through.
|
|
503
|
+
return runWithRequestContext(requestContext, () =>
|
|
504
|
+
observePhase(PHASES.request, async (span) => {
|
|
505
|
+
span.setAttribute("http.method", request.method);
|
|
506
|
+
// The matched route template is not known until match() runs later, so
|
|
507
|
+
// emit the concrete path as url.path (low-level), NOT http.route — the
|
|
508
|
+
// latter is reserved for the low-cardinality template (OTel convention).
|
|
509
|
+
span.setAttribute("url.path", url.pathname);
|
|
510
|
+
|
|
511
|
+
// Core handler logic (wrapped by middleware)
|
|
512
|
+
const coreHandler = async (): Promise<Response> => {
|
|
513
|
+
return coreRequestHandler(request, env, url, variables, nonce);
|
|
514
|
+
};
|
|
380
515
|
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
if (name.startsWith("/")) {
|
|
393
|
-
if (!hrefParams) return name;
|
|
394
|
-
return name.replace(/:([^/]+)/g, (_, key) => {
|
|
395
|
-
const value = hrefParams[key];
|
|
396
|
-
if (value === undefined) throw new Error(`Missing param "${key}" for path "${name}"`);
|
|
397
|
-
return encodeURIComponent(value);
|
|
398
|
-
});
|
|
399
|
-
}
|
|
400
|
-
return name;
|
|
401
|
-
},
|
|
402
|
-
get: (key: string) => variables[key],
|
|
403
|
-
header: (name: string, value: string) => reqCtx.header(name, value),
|
|
404
|
-
setCookie: (name: string, value: string, options?: any) => reqCtx.setCookie(name, value, options),
|
|
405
|
-
};
|
|
516
|
+
// Execute middleware chain if any, otherwise call core handler directly
|
|
517
|
+
let response: Response;
|
|
518
|
+
if (matchedMiddleware.length > 0) {
|
|
519
|
+
const mwResponse = await executeMiddleware(
|
|
520
|
+
matchedMiddleware,
|
|
521
|
+
request,
|
|
522
|
+
env,
|
|
523
|
+
variables,
|
|
524
|
+
coreHandler,
|
|
525
|
+
createReverseFunction(getRequiredRouteMap()),
|
|
526
|
+
);
|
|
406
527
|
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
if (result instanceof Response) {
|
|
415
|
-
const mergedHeaders: Record<string, string> = {};
|
|
416
|
-
result.headers.forEach((value, key) => {
|
|
417
|
-
mergedHeaders[key] = value;
|
|
418
|
-
});
|
|
419
|
-
return createResponseWithMergedHeaders(result.body, {
|
|
420
|
-
status: result.status,
|
|
421
|
-
headers: mergedHeaders,
|
|
422
|
-
});
|
|
423
|
-
}
|
|
424
|
-
return createResponseWithMergedHeaders(
|
|
425
|
-
JSON.stringify({ data: result }),
|
|
426
|
-
{ status: 200, headers: { "content-type": "application/json;charset=utf-8" } },
|
|
427
|
-
);
|
|
428
|
-
} catch (error) {
|
|
429
|
-
callOnError(error, "handler", errorCtx);
|
|
430
|
-
const isDev = process.env.NODE_ENV !== "production";
|
|
431
|
-
const status = error instanceof RouterError ? error.status : 500;
|
|
432
|
-
return createResponseWithMergedHeaders(
|
|
433
|
-
JSON.stringify({ error: createResponseErrorPayload(error, isDev) }),
|
|
434
|
-
{ status, headers: { "content-type": "application/json;charset=utf-8" } },
|
|
528
|
+
if (
|
|
529
|
+
url.searchParams.has("_rsc_partial") ||
|
|
530
|
+
url.searchParams.has("_rsc_action")
|
|
531
|
+
) {
|
|
532
|
+
const intercepted = interceptRedirectForPartial(
|
|
533
|
+
mwResponse,
|
|
534
|
+
createRedirectFlightResponse,
|
|
435
535
|
);
|
|
536
|
+
response = intercepted ?? finalizeResponse(mwResponse);
|
|
537
|
+
} else {
|
|
538
|
+
response = finalizeResponse(mwResponse);
|
|
436
539
|
}
|
|
540
|
+
} else {
|
|
541
|
+
response = await coreHandler();
|
|
437
542
|
}
|
|
438
543
|
|
|
439
|
-
//
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
{ status: 200, headers: { "content-type": "application/xml;charset=utf-8" } },
|
|
472
|
-
);
|
|
473
|
-
case "md":
|
|
474
|
-
return createResponseWithMergedHeaders(
|
|
475
|
-
String(result),
|
|
476
|
-
{ status: 200, headers: { "content-type": "text/markdown;charset=utf-8" } },
|
|
477
|
-
);
|
|
478
|
-
default:
|
|
479
|
-
// image, stream, any -- must return Response
|
|
480
|
-
throw new Error(
|
|
481
|
-
`Response route handler for "${preview.responseType}" must return a Response object, got ${typeof result}`
|
|
482
|
-
);
|
|
483
|
-
}
|
|
484
|
-
} catch (error) {
|
|
485
|
-
callOnError(error, "handler", errorCtx);
|
|
486
|
-
const isDev = process.env.NODE_ENV !== "production";
|
|
487
|
-
const status = error instanceof RouterError ? error.status : 500;
|
|
488
|
-
const message = error instanceof RouterError
|
|
489
|
-
? error.message
|
|
490
|
-
: isDev && error instanceof Error
|
|
491
|
-
? error.message
|
|
492
|
-
: "Internal Server Error";
|
|
493
|
-
return createResponseWithMergedHeaders(message, {
|
|
494
|
-
status,
|
|
495
|
-
headers: { "content-type": "text/plain;charset=utf-8" },
|
|
496
|
-
});
|
|
544
|
+
// Finalize metrics after all middleware (including post-next work)
|
|
545
|
+
// has completed so :post spans are captured in the timeline.
|
|
546
|
+
// Handler timing parts are always emitted (even without debug metrics)
|
|
547
|
+
// so non-debug requests still get bootstrap Server-Timing entries.
|
|
548
|
+
const handlerTimingArr: string[] = variables.__handlerTiming || [];
|
|
549
|
+
// Preserve any existing Server-Timing set by response routes or middleware
|
|
550
|
+
const existingTiming = response.headers.get("Server-Timing");
|
|
551
|
+
const timingParts = existingTiming
|
|
552
|
+
? [existingTiming, ...handlerTimingArr]
|
|
553
|
+
: [...handlerTimingArr];
|
|
554
|
+
|
|
555
|
+
const metricsStore = requestContext._metricsStore;
|
|
556
|
+
if (metricsStore) {
|
|
557
|
+
// When the store was created at handler start (earlyMetricsStore),
|
|
558
|
+
// handler:total covers the full request. When ctx.debugPerformance()
|
|
559
|
+
// created the store mid-request, use its requestStart to avoid a
|
|
560
|
+
// negative startTime offset.
|
|
561
|
+
const totalStart = earlyMetricsStore
|
|
562
|
+
? handlerStart
|
|
563
|
+
: metricsStore.requestStart;
|
|
564
|
+
appendMetric(
|
|
565
|
+
metricsStore,
|
|
566
|
+
"handler:total",
|
|
567
|
+
totalStart,
|
|
568
|
+
performance.now() - totalStart,
|
|
569
|
+
);
|
|
570
|
+
const metricsTiming = buildMetricsTiming(
|
|
571
|
+
request.method,
|
|
572
|
+
url.pathname,
|
|
573
|
+
metricsStore,
|
|
574
|
+
);
|
|
575
|
+
if (metricsTiming) timingParts.push(metricsTiming);
|
|
497
576
|
}
|
|
498
|
-
};
|
|
499
577
|
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
if (preview.negotiated) {
|
|
504
|
-
response.headers.append("Vary", "Accept");
|
|
578
|
+
const fullTiming = timingParts.join(", ");
|
|
579
|
+
if (fullTiming && !isWebSocketUpgradeResponse(response)) {
|
|
580
|
+
response.headers.set("Server-Timing", fullTiming);
|
|
505
581
|
}
|
|
506
|
-
return response;
|
|
507
|
-
};
|
|
508
|
-
|
|
509
|
-
if (preview.routeMiddleware && preview.routeMiddleware.length > 0) {
|
|
510
|
-
const middlewareEntries = preview.routeMiddleware.map((mw) => ({
|
|
511
|
-
entry: {
|
|
512
|
-
pattern: null,
|
|
513
|
-
regex: null,
|
|
514
|
-
paramNames: [],
|
|
515
|
-
handler: mw.handler,
|
|
516
|
-
mountPrefix: null,
|
|
517
|
-
},
|
|
518
|
-
params: mw.params,
|
|
519
|
-
}));
|
|
520
|
-
return executeMiddleware(middlewareEntries, request, env, variables, callHandlerWithVary);
|
|
521
|
-
}
|
|
522
|
-
|
|
523
|
-
return callHandlerWithVary();
|
|
524
|
-
}
|
|
525
|
-
|
|
526
|
-
// Wrap RSC handler to append Vary: Accept on content-negotiated routes
|
|
527
|
-
const rscHandler = async () => {
|
|
528
|
-
const response = await coreRequestHandlerInner(request, env, url, variables, nonce);
|
|
529
|
-
if (preview?.negotiated) {
|
|
530
|
-
response.headers.append("Vary", "Accept");
|
|
531
|
-
}
|
|
532
|
-
return response;
|
|
533
|
-
};
|
|
534
582
|
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
},
|
|
545
|
-
params: mw.params,
|
|
546
|
-
}));
|
|
547
|
-
|
|
548
|
-
// Execute route middleware wrapping the actual request handling
|
|
549
|
-
return executeMiddleware(middlewareEntries, request, env, variables, rscHandler);
|
|
550
|
-
}
|
|
551
|
-
|
|
552
|
-
// No route middleware, proceed directly
|
|
553
|
-
return rscHandler();
|
|
554
|
-
}
|
|
583
|
+
// Single open-redirect chokepoint: every response (PE, full-page,
|
|
584
|
+
// middleware short-circuit, response-route) funnels through here, so
|
|
585
|
+
// guarding browser-followed (3xx) redirects once covers them all and any
|
|
586
|
+
// future redirect exit. Soft SPA/Flight redirects are 200/204 and pass
|
|
587
|
+
// through untouched (validated client-side instead).
|
|
588
|
+
return guardOutgoingRedirect(response, url.origin, router.basename);
|
|
589
|
+
}),
|
|
590
|
+
);
|
|
591
|
+
};
|
|
555
592
|
|
|
556
|
-
//
|
|
557
|
-
|
|
593
|
+
// Core request handling logic (separated for middleware wrapping).
|
|
594
|
+
// Uses the classify → execute model: classifyRequest produces a RequestPlan,
|
|
595
|
+
// then execution dispatches on the plan mode.
|
|
596
|
+
async function coreRequestHandler(
|
|
558
597
|
request: Request,
|
|
559
598
|
env: TEnv,
|
|
560
599
|
url: URL,
|
|
561
600
|
variables: Record<string, any>,
|
|
562
601
|
nonce: string | undefined,
|
|
563
602
|
): Promise<Response> {
|
|
564
|
-
const
|
|
565
|
-
const isAction =
|
|
566
|
-
request.headers.has("rsc-action") || url.searchParams.has("_rsc_action");
|
|
567
|
-
const actionId =
|
|
568
|
-
request.headers.get("rsc-action") || url.searchParams.get("_rsc_action");
|
|
569
|
-
|
|
570
|
-
// Version mismatch detection - client may have stale code after HMR/deployment
|
|
571
|
-
// If versions don't match, tell the client to reload
|
|
572
|
-
const clientVersion = url.searchParams.get("_rsc_v");
|
|
573
|
-
if (version && clientVersion && clientVersion !== version) {
|
|
574
|
-
console.log(
|
|
575
|
-
`[RSC] Version mismatch: client=${clientVersion}, server=${version}. Forcing reload.`,
|
|
576
|
-
);
|
|
603
|
+
const handlerTiming: string[] = variables.__handlerTiming || [];
|
|
577
604
|
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
cleanUrl.searchParams.delete("_rsc_partial");
|
|
581
|
-
cleanUrl.searchParams.delete("_rsc_segments");
|
|
582
|
-
cleanUrl.searchParams.delete("_rsc_v");
|
|
583
|
-
cleanUrl.searchParams.delete("_rsc_stale");
|
|
584
|
-
cleanUrl.searchParams.delete("_rsc_action");
|
|
585
|
-
cleanUrl.searchParams.delete("_rsc_prev");
|
|
586
|
-
|
|
587
|
-
// For actions, reload current page (referer)
|
|
588
|
-
// For navigation, load the target URL
|
|
589
|
-
const reloadUrl = isAction
|
|
590
|
-
? request.headers.get("referer") || cleanUrl.toString()
|
|
591
|
-
: cleanUrl.toString();
|
|
592
|
-
|
|
593
|
-
// Return special response that tells client to reload
|
|
594
|
-
return createResponseWithMergedHeaders(null, {
|
|
595
|
-
status: 200,
|
|
596
|
-
headers: {
|
|
597
|
-
"X-RSC-Reload": reloadUrl,
|
|
598
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
599
|
-
},
|
|
600
|
-
});
|
|
601
|
-
}
|
|
602
|
-
// Debug manifest endpoint: ?__debug_manifest on any route.
|
|
603
|
-
// Always available in dev, requires allowDebugManifest option in production.
|
|
605
|
+
// Debug manifest endpoint: handled before classification since it
|
|
606
|
+
// doesn't need a route match and needs trie access from the closure.
|
|
604
607
|
const isDev = process.env.NODE_ENV !== "production";
|
|
605
608
|
if (
|
|
606
609
|
url.searchParams.has("__debug_manifest") &&
|
|
607
610
|
(isDev || router.allowDebugManifest)
|
|
608
611
|
) {
|
|
609
612
|
const trie = getRouterTrie(router.id) ?? getRouteTrie();
|
|
610
|
-
const routeManifest =
|
|
611
|
-
const { extractAncestryFromTrie } =
|
|
613
|
+
const routeManifest = getRequiredRouteMap();
|
|
614
|
+
const { extractAncestryFromTrie } =
|
|
615
|
+
await import("../build/route-trie.js");
|
|
612
616
|
return new Response(
|
|
613
617
|
JSON.stringify(
|
|
614
618
|
{
|
|
@@ -627,851 +631,567 @@ export function createRSCHandler<
|
|
|
627
631
|
);
|
|
628
632
|
}
|
|
629
633
|
|
|
630
|
-
//
|
|
631
|
-
|
|
632
|
-
|
|
634
|
+
// ---- 1. Classify ----
|
|
635
|
+
// classifyRequest may throw RouteNotFoundError for unknown routes.
|
|
636
|
+
// In that case, fall through to a full-render plan so the pipeline
|
|
637
|
+
// can render the 404 page via the existing error handling path.
|
|
638
|
+
const classifyStart = performance.now();
|
|
639
|
+
let plan: RequestPlan<TEnv>;
|
|
633
640
|
try {
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
641
|
+
plan = await classifyRequest<TEnv>(request, url, {
|
|
642
|
+
findMatch: router.findMatch,
|
|
643
|
+
routerVersion: version,
|
|
644
|
+
routerId: router.id,
|
|
645
|
+
});
|
|
646
|
+
} catch (error) {
|
|
647
|
+
if (isRouteNotFoundError(error)) {
|
|
648
|
+
// Let the render path handle 404 — match()/matchPartial() will
|
|
649
|
+
// re-throw RouteNotFoundError and the catch block in
|
|
650
|
+
// executeRenderWithMiddleware renders the not-found page.
|
|
651
|
+
plan = {
|
|
652
|
+
mode: "full-render",
|
|
653
|
+
route: {
|
|
654
|
+
matched: null as any,
|
|
655
|
+
manifestEntry: null as any,
|
|
656
|
+
entries: [],
|
|
657
|
+
routeKey: "",
|
|
658
|
+
localRouteName: "",
|
|
659
|
+
params: {},
|
|
660
|
+
routeMiddleware: [],
|
|
661
|
+
cacheScope: null,
|
|
662
|
+
isPassthrough: false,
|
|
663
|
+
},
|
|
664
|
+
negotiated: false,
|
|
665
|
+
};
|
|
666
|
+
} else {
|
|
667
|
+
throw error;
|
|
647
668
|
}
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
669
|
+
}
|
|
670
|
+
const classifyDur = performance.now() - classifyStart;
|
|
671
|
+
handlerTiming.push(`handler-classify;dur=${classifyDur.toFixed(2)}`);
|
|
672
|
+
|
|
673
|
+
// ---- 2. Terminal plans (no execution needed) ----
|
|
674
|
+
if (plan.mode === "redirect") {
|
|
675
|
+
// Redirects are handled by the pipeline (match/matchPartial),
|
|
676
|
+
// but for partial requests we short-circuit with a Flight redirect.
|
|
677
|
+
if (url.searchParams.has("_rsc_partial")) {
|
|
678
|
+
return createRedirectFlightResponse(plan.redirectUrl);
|
|
654
679
|
}
|
|
680
|
+
// Full requests: let the pipeline handle the redirect via match()
|
|
681
|
+
// which returns { redirect: url }. Fall through to full-render.
|
|
682
|
+
}
|
|
655
683
|
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
684
|
+
if (plan.mode === "version-mismatch") {
|
|
685
|
+
console.log(
|
|
686
|
+
`[RSC] Version mismatch: client=${url.searchParams.get("_rsc_v")}, server=${version}. Forcing reload.`,
|
|
687
|
+
);
|
|
688
|
+
return createReloadResponse(plan.reloadUrl);
|
|
689
|
+
}
|
|
690
|
+
|
|
691
|
+
if (plan.mode === "app-switch") {
|
|
692
|
+
// Cross-app SPA navigation crossed a host-router app boundary. Force a
|
|
693
|
+
// real document navigation so the target app's document is re-established
|
|
694
|
+
// (stylesheets, theme, warmup, prefetch-TTL). See request-classification.
|
|
695
|
+
return createReloadResponse(plan.reloadUrl);
|
|
696
|
+
}
|
|
663
697
|
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
return await handleRscRendering(
|
|
698
|
+
// ---- 3. Origin guard (gate for action/loader/PE modes) ----
|
|
699
|
+
const originPhase = ORIGIN_CHECK_PHASE_BY_MODE[plan.mode];
|
|
700
|
+
if (originPhase) {
|
|
701
|
+
const originResult = await checkRequestOrigin(
|
|
669
702
|
request,
|
|
670
|
-
env,
|
|
671
703
|
url,
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
704
|
+
router.originCheck,
|
|
705
|
+
env,
|
|
706
|
+
router.id,
|
|
707
|
+
originPhase,
|
|
675
708
|
);
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
// Signal the browser to hard-navigate so it renders the raw response.
|
|
682
|
-
// Only for 200 — redirects (3xx) work already because the browser follows
|
|
683
|
-
// them automatically to a URL that serves Flight data.
|
|
684
|
-
if (isPartial && error.status === 200) {
|
|
685
|
-
console.warn(
|
|
686
|
-
`[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
|
|
687
|
-
`Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
|
|
688
|
-
);
|
|
689
|
-
const cleanUrl = new URL(url);
|
|
690
|
-
cleanUrl.searchParams.delete("_rsc_partial");
|
|
691
|
-
cleanUrl.searchParams.delete("_rsc_segments");
|
|
692
|
-
cleanUrl.searchParams.delete("_rsc_v");
|
|
693
|
-
cleanUrl.searchParams.delete("_rsc_stale");
|
|
694
|
-
cleanUrl.searchParams.delete("_rsc_action");
|
|
695
|
-
cleanUrl.searchParams.delete("_rsc_prev");
|
|
696
|
-
return createResponseWithMergedHeaders(null, {
|
|
697
|
-
status: 200,
|
|
698
|
-
headers: {
|
|
699
|
-
"X-RSC-Reload": cleanUrl.toString(),
|
|
700
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
701
|
-
},
|
|
702
|
-
});
|
|
703
|
-
}
|
|
704
|
-
return error;
|
|
705
|
-
}
|
|
709
|
+
if (originResult) {
|
|
710
|
+
const originError = new Error(
|
|
711
|
+
`Origin check rejected: ${request.headers.get("origin") ?? "none"} vs ${request.headers.get("host") ?? "none"}`,
|
|
712
|
+
);
|
|
713
|
+
originError.name = "OriginCheckError";
|
|
706
714
|
|
|
707
|
-
|
|
708
|
-
// Check both instanceof and error.name for cross-bundle compatibility
|
|
709
|
-
const isRouteNotFound =
|
|
710
|
-
error instanceof RouteNotFoundError ||
|
|
711
|
-
(error instanceof Error && error.name === "RouteNotFoundError");
|
|
712
|
-
if (isRouteNotFound) {
|
|
713
|
-
callOnError(error, "routing", {
|
|
715
|
+
callOnError(originError, "origin", {
|
|
714
716
|
request,
|
|
715
717
|
url,
|
|
716
718
|
env,
|
|
717
|
-
handledByBoundary:
|
|
718
|
-
});
|
|
719
|
-
|
|
720
|
-
// Get notFound component from router options or use default
|
|
721
|
-
const notFoundOption = router.notFound;
|
|
722
|
-
const notFoundComponent =
|
|
723
|
-
typeof notFoundOption === "function"
|
|
724
|
-
? notFoundOption({ pathname: url.pathname })
|
|
725
|
-
: (notFoundOption ?? createElement("h1", null, "Not Found"));
|
|
726
|
-
|
|
727
|
-
// Create a simple segment for the 404 page
|
|
728
|
-
const notFoundSegment = {
|
|
729
|
-
id: "notFound",
|
|
730
|
-
namespace: "notFound",
|
|
731
|
-
type: "route" as const,
|
|
732
|
-
index: 0,
|
|
733
|
-
component: notFoundComponent,
|
|
734
|
-
params: {},
|
|
735
|
-
};
|
|
736
|
-
|
|
737
|
-
// Render with rootLayout to maintain app shell
|
|
738
|
-
const root = await renderSegments([notFoundSegment], {
|
|
739
|
-
rootLayout: router.rootLayout,
|
|
740
|
-
// No routeName for not-found routes
|
|
741
|
-
});
|
|
742
|
-
|
|
743
|
-
const payload: RscPayload = {
|
|
744
|
-
root,
|
|
719
|
+
handledByBoundary: false,
|
|
745
720
|
metadata: {
|
|
746
|
-
|
|
747
|
-
|
|
748
|
-
|
|
749
|
-
diff: [],
|
|
750
|
-
isPartial: false,
|
|
751
|
-
handles: handleStore.stream(),
|
|
752
|
-
version,
|
|
753
|
-
themeConfig: router.themeConfig,
|
|
754
|
-
warmupEnabled: router.warmupEnabled,
|
|
755
|
-
initialTheme: requireRequestContext().theme,
|
|
756
|
-
// No routeName for not-found routes
|
|
721
|
+
phase: originPhase,
|
|
722
|
+
origin: request.headers.get("origin"),
|
|
723
|
+
host: request.headers.get("host"),
|
|
757
724
|
},
|
|
758
|
-
};
|
|
759
|
-
|
|
760
|
-
const rscStream = renderToReadableStream(payload);
|
|
761
|
-
|
|
762
|
-
// Determine if this is an RSC request or HTML request
|
|
763
|
-
const isRscRequest =
|
|
764
|
-
(!request.headers.get("accept")?.includes("text/html") &&
|
|
765
|
-
!url.searchParams.has("__html")) ||
|
|
766
|
-
url.searchParams.has("__rsc");
|
|
767
|
-
|
|
768
|
-
if (isRscRequest) {
|
|
769
|
-
return createResponseWithMergedHeaders(rscStream, {
|
|
770
|
-
status: 404,
|
|
771
|
-
headers: { "content-type": "text/x-component;charset=utf-8" },
|
|
772
|
-
});
|
|
773
|
-
}
|
|
774
|
-
|
|
775
|
-
// Delegate to SSR for HTML response
|
|
776
|
-
const ssrModule = await loadSSRModule();
|
|
777
|
-
const htmlStream = await ssrModule.renderHTML(rscStream, { nonce });
|
|
725
|
+
});
|
|
778
726
|
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
727
|
+
observeEvent({
|
|
728
|
+
type: "request.origin-rejected",
|
|
729
|
+
timestamp: performance.now(),
|
|
730
|
+
method: request.method,
|
|
731
|
+
pathname: url.pathname,
|
|
732
|
+
phase: originPhase,
|
|
733
|
+
origin: request.headers.get("origin"),
|
|
734
|
+
host: request.headers.get("host"),
|
|
782
735
|
});
|
|
783
|
-
}
|
|
784
736
|
|
|
785
|
-
|
|
786
|
-
|
|
787
|
-
request,
|
|
788
|
-
url,
|
|
789
|
-
env,
|
|
790
|
-
handledByBoundary: false,
|
|
791
|
-
});
|
|
792
|
-
console.error(`[RSC] Error:`, error);
|
|
793
|
-
throw error;
|
|
737
|
+
return originResult;
|
|
738
|
+
}
|
|
794
739
|
}
|
|
740
|
+
|
|
741
|
+
// ---- 4. Execute ----
|
|
742
|
+
return executeRequest(
|
|
743
|
+
plan as ExecutableRequestPlan<TEnv>,
|
|
744
|
+
request,
|
|
745
|
+
env,
|
|
746
|
+
url,
|
|
747
|
+
variables,
|
|
748
|
+
nonce,
|
|
749
|
+
);
|
|
795
750
|
}
|
|
796
751
|
|
|
797
|
-
//
|
|
798
|
-
//
|
|
799
|
-
//
|
|
800
|
-
//
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
async function handleProgressiveEnhancement(
|
|
752
|
+
// Execute a classified request plan. Dispatches to the appropriate handler
|
|
753
|
+
// based on plan.mode. Lives in the createRSCHandler closure for access to
|
|
754
|
+
// handlerCtx, router, callOnError, etc.
|
|
755
|
+
// Only receives executable plans (version-mismatch is handled above).
|
|
756
|
+
async function executeRequest(
|
|
757
|
+
plan: ExecutableRequestPlan<TEnv>,
|
|
804
758
|
request: Request,
|
|
805
759
|
env: TEnv,
|
|
806
760
|
url: URL,
|
|
807
|
-
|
|
808
|
-
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
761
|
+
variables: Record<string, any>,
|
|
809
762
|
nonce: string | undefined,
|
|
810
|
-
): Promise<Response
|
|
811
|
-
|
|
812
|
-
const
|
|
813
|
-
contentType.includes("multipart/form-data") ||
|
|
814
|
-
contentType.includes("application/x-www-form-urlencoded");
|
|
815
|
-
|
|
816
|
-
if (request.method !== "POST" || isAction || !isFormSubmission) {
|
|
817
|
-
return null;
|
|
818
|
-
}
|
|
819
|
-
|
|
820
|
-
// Clone the request to read FormData without consuming it
|
|
821
|
-
const formData = await request.clone().formData();
|
|
822
|
-
|
|
823
|
-
// Look for React's progressive enhancement hidden fields
|
|
824
|
-
let isDirectAction = false;
|
|
825
|
-
let isUseActionState = false;
|
|
826
|
-
let directActionId: string | null = null;
|
|
763
|
+
): Promise<Response> {
|
|
764
|
+
// Common setup
|
|
765
|
+
const handleStore = requireRequestContext()._handleStore;
|
|
827
766
|
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
|
|
835
|
-
|
|
767
|
+
// Wire up error reporting for late streaming-handle failures
|
|
768
|
+
handleStore.onError = (error: Error) => {
|
|
769
|
+
const reqCtx = requireRequestContext();
|
|
770
|
+
callOnError(error, "handler", {
|
|
771
|
+
request,
|
|
772
|
+
url,
|
|
773
|
+
routeKey: reqCtx._routeName,
|
|
774
|
+
params: reqCtx.params as Record<string, string>,
|
|
775
|
+
handledByBoundary: true,
|
|
776
|
+
});
|
|
777
|
+
observeEvent({
|
|
778
|
+
type: "handler.error",
|
|
779
|
+
timestamp: performance.now(),
|
|
780
|
+
error,
|
|
781
|
+
handledByBoundary: true,
|
|
782
|
+
pathname: url.pathname,
|
|
783
|
+
routeKey: reqCtx._routeName,
|
|
784
|
+
params: reqCtx.params as Record<string, string>,
|
|
785
|
+
});
|
|
786
|
+
};
|
|
836
787
|
|
|
837
|
-
|
|
838
|
-
|
|
788
|
+
// Set route params early so all execution paths can access ctx.params.
|
|
789
|
+
// Also store the classified snapshot so match/matchPartial can reuse it
|
|
790
|
+
// instead of calling resolveRoute again.
|
|
791
|
+
if (plan.mode !== "redirect") {
|
|
792
|
+
setRequestContextParams(plan.route.params, plan.route.routeKey);
|
|
793
|
+
requireRequestContext()._classifiedRoute = plan.route;
|
|
839
794
|
}
|
|
840
795
|
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
|
|
845
|
-
|
|
846
|
-
|
|
847
|
-
|
|
848
|
-
|
|
849
|
-
|
|
850
|
-
|
|
796
|
+
const routeReverse = createReverseFunction(getRequiredRouteMap());
|
|
797
|
+
|
|
798
|
+
// ---- Response route: skip entire RSC pipeline ----
|
|
799
|
+
if (plan.mode === "response") {
|
|
800
|
+
// Build ResponseRouteMatch from plan fields. handleResponseRoute
|
|
801
|
+
// expects a flat object with params at the top level.
|
|
802
|
+
const responseMatch: ResponseRouteMatch = {
|
|
803
|
+
responseType: plan.responseType,
|
|
804
|
+
handler: plan.handler,
|
|
805
|
+
params: plan.route.params,
|
|
806
|
+
negotiated: plan.negotiated,
|
|
807
|
+
manifestEntry: plan.manifestEntry,
|
|
808
|
+
routeMiddleware: plan.routeMiddleware,
|
|
809
|
+
};
|
|
810
|
+
const responseOutcome = await withTimeout(
|
|
811
|
+
handleResponseRoute(
|
|
812
|
+
handlerCtx,
|
|
813
|
+
responseMatch,
|
|
851
814
|
request,
|
|
852
|
-
url,
|
|
853
815
|
env,
|
|
854
|
-
handledByBoundary: false,
|
|
855
|
-
});
|
|
856
|
-
console.error("[RSC] Progressive enhancement action error:", error);
|
|
857
|
-
}
|
|
858
|
-
} else if (isDirectAction && directActionId) {
|
|
859
|
-
const temporaryReferences = createTemporaryReferenceSet();
|
|
860
|
-
|
|
861
|
-
let args: unknown[] = [];
|
|
862
|
-
try {
|
|
863
|
-
args = await decodeReply(formData, { temporaryReferences });
|
|
864
|
-
} catch {
|
|
865
|
-
args = [formData];
|
|
866
|
-
}
|
|
867
|
-
|
|
868
|
-
try {
|
|
869
|
-
const loadedAction = await loadServerAction(directActionId);
|
|
870
|
-
actionResult = await loadedAction.apply(null, args);
|
|
871
|
-
} catch (error) {
|
|
872
|
-
callOnError(error, "action", {
|
|
873
|
-
request,
|
|
874
816
|
url,
|
|
817
|
+
variables,
|
|
818
|
+
),
|
|
819
|
+
router.timeouts.renderStartMs,
|
|
820
|
+
"render-start",
|
|
821
|
+
);
|
|
822
|
+
if (responseOutcome.timedOut) {
|
|
823
|
+
return handleTimeoutResponse(
|
|
824
|
+
request,
|
|
875
825
|
env,
|
|
876
|
-
|
|
877
|
-
|
|
878
|
-
|
|
879
|
-
|
|
826
|
+
url,
|
|
827
|
+
"render-start",
|
|
828
|
+
responseOutcome.durationMs,
|
|
829
|
+
plan.route.routeKey,
|
|
830
|
+
);
|
|
880
831
|
}
|
|
832
|
+
const response = responseOutcome.result;
|
|
833
|
+
if (plan.negotiated && !isWebSocketUpgradeResponse(response)) {
|
|
834
|
+
response.headers.append("Vary", "Accept");
|
|
835
|
+
}
|
|
836
|
+
return response;
|
|
881
837
|
}
|
|
882
838
|
|
|
883
|
-
//
|
|
884
|
-
|
|
885
|
-
|
|
886
|
-
|
|
887
|
-
|
|
839
|
+
// SSR setup: kick off in parallel for modes that need HTML rendering.
|
|
840
|
+
// Placed after response-route short-circuit so response/mime routes
|
|
841
|
+
// never pay for SSR work.
|
|
842
|
+
if (plan.mode !== "loader" && mayNeedSSR(request, url)) {
|
|
843
|
+
variables[SSR_SETUP_VAR] = startSSRSetup(
|
|
844
|
+
handlerCtx,
|
|
888
845
|
request,
|
|
889
|
-
url,
|
|
890
846
|
env,
|
|
891
|
-
|
|
892
|
-
|
|
893
|
-
|
|
894
|
-
|
|
895
|
-
|
|
896
|
-
// Re-render the page and return HTML
|
|
897
|
-
const renderRequest = new Request(url.toString(), {
|
|
898
|
-
method: "GET",
|
|
899
|
-
headers: new Headers({ accept: "text/html" }),
|
|
900
|
-
});
|
|
901
|
-
|
|
902
|
-
const match = await router.match(renderRequest, env);
|
|
903
|
-
|
|
904
|
-
if (match.redirect) {
|
|
905
|
-
return new Response(null, {
|
|
906
|
-
status: 308,
|
|
907
|
-
headers: { Location: match.redirect },
|
|
908
|
-
});
|
|
847
|
+
url,
|
|
848
|
+
router.debugPerformance
|
|
849
|
+
? () => requireRequestContext()._metricsStore
|
|
850
|
+
: undefined,
|
|
851
|
+
);
|
|
909
852
|
}
|
|
910
853
|
|
|
911
|
-
|
|
912
|
-
|
|
913
|
-
|
|
914
|
-
|
|
915
|
-
const payload: RscPayload = {
|
|
916
|
-
root,
|
|
917
|
-
metadata: {
|
|
918
|
-
pathname: url.pathname,
|
|
919
|
-
segments: match.segments,
|
|
920
|
-
matched: match.matched,
|
|
921
|
-
diff: match.diff,
|
|
922
|
-
isPartial: false,
|
|
923
|
-
rootLayout: router.rootLayout,
|
|
924
|
-
handles: handleStore.stream(),
|
|
925
|
-
version,
|
|
926
|
-
themeConfig: router.themeConfig,
|
|
927
|
-
warmupEnabled: router.warmupEnabled,
|
|
928
|
-
initialTheme: requireRequestContext().theme,
|
|
929
|
-
},
|
|
930
|
-
formState: actionResult,
|
|
931
|
-
};
|
|
932
|
-
|
|
933
|
-
const rscStream = renderToReadableStream<RscPayload>(payload);
|
|
934
|
-
const ssrModule = await loadSSRModule();
|
|
935
|
-
const htmlStream = await ssrModule.renderHTML(rscStream, {
|
|
936
|
-
formState: reactFormState,
|
|
937
|
-
nonce,
|
|
938
|
-
});
|
|
939
|
-
|
|
940
|
-
return new Response(htmlStream, {
|
|
941
|
-
headers: { "content-type": "text/html;charset=utf-8" },
|
|
942
|
-
});
|
|
943
|
-
}
|
|
944
|
-
|
|
945
|
-
// ============================================================================
|
|
946
|
-
// SERVER ACTION HANDLER
|
|
947
|
-
// ============================================================================
|
|
948
|
-
async function handleServerAction(
|
|
949
|
-
request: Request,
|
|
950
|
-
env: TEnv,
|
|
951
|
-
url: URL,
|
|
952
|
-
actionId: string,
|
|
953
|
-
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
954
|
-
): Promise<Response> {
|
|
955
|
-
const temporaryReferences = createTemporaryReferenceSet();
|
|
956
|
-
|
|
957
|
-
// Decode action arguments from request body
|
|
958
|
-
const contentType = request.headers.get("content-type") || "";
|
|
959
|
-
let args: unknown[] = [];
|
|
960
|
-
let actionFormData: FormData | undefined;
|
|
961
|
-
|
|
962
|
-
try {
|
|
963
|
-
const body = contentType.includes("multipart/form-data")
|
|
964
|
-
? await request.formData()
|
|
965
|
-
: await request.text();
|
|
966
|
-
|
|
967
|
-
if (body instanceof FormData) {
|
|
968
|
-
actionFormData = body;
|
|
969
|
-
}
|
|
970
|
-
|
|
971
|
-
if (hasBodyContent(body)) {
|
|
972
|
-
args = await decodeReply(body, { temporaryReferences });
|
|
973
|
-
}
|
|
974
|
-
} catch (error) {
|
|
975
|
-
callOnError(error, "action", {
|
|
854
|
+
// ---- Loader fetch ----
|
|
855
|
+
if (plan.mode === "loader") {
|
|
856
|
+
return handleLoaderFetch(
|
|
857
|
+
handlerCtx,
|
|
976
858
|
request,
|
|
977
|
-
url,
|
|
978
859
|
env,
|
|
979
|
-
|
|
980
|
-
|
|
981
|
-
|
|
982
|
-
|
|
983
|
-
cause: error,
|
|
984
|
-
});
|
|
860
|
+
url,
|
|
861
|
+
variables,
|
|
862
|
+
plan.route.params,
|
|
863
|
+
);
|
|
985
864
|
}
|
|
986
865
|
|
|
987
|
-
//
|
|
988
|
-
|
|
989
|
-
|
|
990
|
-
|
|
991
|
-
|
|
992
|
-
try {
|
|
993
|
-
loadedAction = await loadServerAction(actionId);
|
|
994
|
-
const data = await loadedAction!.apply(null, args);
|
|
995
|
-
returnValue = { ok: true, data };
|
|
996
|
-
} catch (error) {
|
|
997
|
-
returnValue = { ok: false, data: error };
|
|
998
|
-
actionStatus = 500;
|
|
999
|
-
|
|
1000
|
-
// Try to render error boundary
|
|
1001
|
-
const errorResult = await router.matchError(request, env, error, "route");
|
|
1002
|
-
|
|
1003
|
-
// Report the action error (handledByBoundary indicates if error boundary will render)
|
|
1004
|
-
callOnError(error, "action", {
|
|
866
|
+
// ---- Progressive enhancement ----
|
|
867
|
+
if (plan.mode === "pe-render") {
|
|
868
|
+
const peResult = await handleProgressiveEnhancement(
|
|
869
|
+
handlerCtx,
|
|
1005
870
|
request,
|
|
1006
|
-
url,
|
|
1007
871
|
env,
|
|
1008
|
-
|
|
1009
|
-
|
|
1010
|
-
|
|
1011
|
-
|
|
1012
|
-
|
|
1013
|
-
|
|
1014
|
-
|
|
1015
|
-
|
|
1016
|
-
|
|
1017
|
-
|
|
1018
|
-
|
|
1019
|
-
|
|
1020
|
-
isPartial: true,
|
|
1021
|
-
matched: errorResult.matched,
|
|
1022
|
-
diff: errorResult.diff,
|
|
1023
|
-
isError: true,
|
|
1024
|
-
handles: handleStore.stream(),
|
|
1025
|
-
version,
|
|
1026
|
-
},
|
|
1027
|
-
returnValue,
|
|
1028
|
-
};
|
|
1029
|
-
|
|
1030
|
-
const rscStream = renderToReadableStream<RscPayload>(payload, {
|
|
1031
|
-
temporaryReferences,
|
|
1032
|
-
});
|
|
1033
|
-
|
|
1034
|
-
return createResponseWithMergedHeaders(rscStream, {
|
|
1035
|
-
status: actionStatus,
|
|
1036
|
-
headers: { "content-type": "text/x-component;charset=utf-8" },
|
|
1037
|
-
});
|
|
1038
|
-
}
|
|
872
|
+
url,
|
|
873
|
+
false, // isAction = false for PE
|
|
874
|
+
handleStore,
|
|
875
|
+
nonce,
|
|
876
|
+
{
|
|
877
|
+
routeMiddleware: plan.route.routeMiddleware,
|
|
878
|
+
variables,
|
|
879
|
+
routeReverse,
|
|
880
|
+
},
|
|
881
|
+
);
|
|
882
|
+
if (peResult) return peResult;
|
|
883
|
+
// PE handler returned null (not a PE form) — fall through to render
|
|
1039
884
|
}
|
|
1040
885
|
|
|
1041
|
-
//
|
|
1042
|
-
|
|
1043
|
-
|
|
1044
|
-
|
|
1045
|
-
|
|
1046
|
-
|
|
1047
|
-
|
|
1048
|
-
|
|
1049
|
-
|
|
1050
|
-
|
|
1051
|
-
|
|
1052
|
-
|
|
1053
|
-
|
|
1054
|
-
|
|
1055
|
-
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
1060
|
-
|
|
1061
|
-
|
|
1062
|
-
|
|
1063
|
-
|
|
886
|
+
// ---- Action: execute action, then revalidate wrapped in route middleware ----
|
|
887
|
+
if (plan.mode === "action") {
|
|
888
|
+
let actionContinuation: ActionContinuation | undefined;
|
|
889
|
+
try {
|
|
890
|
+
// Instrument the action execution as its own phase (action:<actionId> +
|
|
891
|
+
// rango.action), so a POST shows the mutation time AND which action ran,
|
|
892
|
+
// not just the downstream revalidation render. The action's own
|
|
893
|
+
// loaders/fetches nest under rango.action.
|
|
894
|
+
const actionOutcome = await withTimeout(
|
|
895
|
+
observePhase(PHASES.action(plan.actionId), () =>
|
|
896
|
+
executeServerAction(
|
|
897
|
+
handlerCtx,
|
|
898
|
+
request,
|
|
899
|
+
env,
|
|
900
|
+
url,
|
|
901
|
+
plan.actionId,
|
|
902
|
+
handleStore,
|
|
903
|
+
),
|
|
904
|
+
),
|
|
905
|
+
router.timeouts.actionMs,
|
|
906
|
+
"action",
|
|
907
|
+
);
|
|
908
|
+
if (actionOutcome.timedOut) {
|
|
909
|
+
return handleTimeoutResponse(
|
|
910
|
+
request,
|
|
911
|
+
env,
|
|
912
|
+
url,
|
|
913
|
+
"action",
|
|
914
|
+
actionOutcome.durationMs,
|
|
915
|
+
plan.route.routeKey,
|
|
916
|
+
plan.actionId,
|
|
917
|
+
);
|
|
918
|
+
}
|
|
919
|
+
const result = actionOutcome.result;
|
|
920
|
+
// Response means redirect or error boundary — done.
|
|
921
|
+
if (result instanceof Response) return result;
|
|
922
|
+
actionContinuation = result;
|
|
923
|
+
} catch (error) {
|
|
924
|
+
callOnError(error, "action", {
|
|
925
|
+
request,
|
|
926
|
+
url,
|
|
927
|
+
env,
|
|
928
|
+
actionId: plan.actionId,
|
|
929
|
+
handledByBoundary: false,
|
|
1064
930
|
});
|
|
931
|
+
console.error(`[RSC] Action error:`, error);
|
|
932
|
+
throw error;
|
|
1065
933
|
}
|
|
1066
934
|
|
|
1067
|
-
|
|
1068
|
-
|
|
1069
|
-
|
|
1070
|
-
|
|
1071
|
-
|
|
1072
|
-
|
|
1073
|
-
|
|
1074
|
-
|
|
1075
|
-
|
|
1076
|
-
|
|
1077
|
-
|
|
1078
|
-
|
|
1079
|
-
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
|
|
1083
|
-
|
|
1084
|
-
|
|
1085
|
-
|
|
1086
|
-
},
|
|
1087
|
-
returnValue,
|
|
1088
|
-
};
|
|
1089
|
-
|
|
1090
|
-
const rscStream = renderToReadableStream<RscPayload>(payload, {
|
|
1091
|
-
temporaryReferences,
|
|
1092
|
-
});
|
|
1093
|
-
|
|
1094
|
-
const headers: Record<string, string> = {
|
|
1095
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
1096
|
-
};
|
|
1097
|
-
if (serverTiming) {
|
|
1098
|
-
headers["Server-Timing"] = serverTiming;
|
|
1099
|
-
}
|
|
1100
|
-
|
|
1101
|
-
return createResponseWithMergedHeaders(rscStream, {
|
|
1102
|
-
status: actionStatus,
|
|
1103
|
-
headers,
|
|
1104
|
-
});
|
|
1105
|
-
}
|
|
1106
|
-
|
|
1107
|
-
// Return updated segments
|
|
1108
|
-
setRequestContextParams(matchResult.params);
|
|
1109
|
-
|
|
1110
|
-
const renderStart = performance.now();
|
|
1111
|
-
|
|
1112
|
-
const renderDuration = performance.now() - renderStart;
|
|
1113
|
-
const serverTiming = matchResult.serverTiming
|
|
1114
|
-
? `${matchResult.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
|
|
1115
|
-
: `rendering;dur=${renderDuration.toFixed(2)}`;
|
|
1116
|
-
|
|
1117
|
-
const payload: RscPayload = {
|
|
1118
|
-
root: null,
|
|
1119
|
-
metadata: {
|
|
1120
|
-
pathname: url.pathname,
|
|
1121
|
-
segments: matchResult.segments,
|
|
1122
|
-
isPartial: true,
|
|
1123
|
-
matched: matchResult.matched,
|
|
1124
|
-
diff: matchResult.diff,
|
|
1125
|
-
slots: matchResult.slots,
|
|
1126
|
-
handles: handleStore.stream(),
|
|
1127
|
-
version,
|
|
1128
|
-
},
|
|
1129
|
-
returnValue,
|
|
1130
|
-
};
|
|
1131
|
-
|
|
1132
|
-
const rscStream = renderToReadableStream<RscPayload>(payload, {
|
|
1133
|
-
temporaryReferences,
|
|
1134
|
-
});
|
|
1135
|
-
|
|
1136
|
-
const actionHeaders: Record<string, string> = {
|
|
1137
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
1138
|
-
};
|
|
1139
|
-
if (serverTiming) {
|
|
1140
|
-
actionHeaders["Server-Timing"] = serverTiming;
|
|
935
|
+
// Revalidation render wrapped in route middleware.
|
|
936
|
+
// Actions from client-side navigation include _rsc_partial — preserve
|
|
937
|
+
// the partial flag so the revalidation returns a Flight stream, not HTML.
|
|
938
|
+
// App-switch is already excluded by classifyRequest (would be full-render).
|
|
939
|
+
const isPartialAction = url.searchParams.has("_rsc_partial");
|
|
940
|
+
return executeRenderWithMiddleware(
|
|
941
|
+
plan.route.routeMiddleware,
|
|
942
|
+
plan.negotiated,
|
|
943
|
+
plan.route.routeKey,
|
|
944
|
+
routeReverse,
|
|
945
|
+
request,
|
|
946
|
+
env,
|
|
947
|
+
url,
|
|
948
|
+
variables,
|
|
949
|
+
nonce,
|
|
950
|
+
handleStore,
|
|
951
|
+
isPartialAction,
|
|
952
|
+
actionContinuation,
|
|
953
|
+
);
|
|
1141
954
|
}
|
|
1142
955
|
|
|
1143
|
-
|
|
1144
|
-
|
|
1145
|
-
|
|
1146
|
-
|
|
956
|
+
// Full render, partial render, fallen-through PE, and full-page redirect all
|
|
957
|
+
// render through the same middleware-wrapped path. Only full/partial-render
|
|
958
|
+
// carry negotiation + the partial flag; pe/redirect render plainly.
|
|
959
|
+
const isPartial = plan.mode === "partial-render";
|
|
960
|
+
const negotiated =
|
|
961
|
+
plan.mode === "full-render" || plan.mode === "partial-render"
|
|
962
|
+
? plan.negotiated
|
|
963
|
+
: false;
|
|
964
|
+
return executeRenderWithMiddleware(
|
|
965
|
+
plan.route.routeMiddleware,
|
|
966
|
+
negotiated,
|
|
967
|
+
plan.route.routeKey,
|
|
968
|
+
routeReverse,
|
|
969
|
+
request,
|
|
970
|
+
env,
|
|
971
|
+
url,
|
|
972
|
+
variables,
|
|
973
|
+
nonce,
|
|
974
|
+
handleStore,
|
|
975
|
+
isPartial,
|
|
976
|
+
);
|
|
1147
977
|
}
|
|
1148
978
|
|
|
1149
|
-
//
|
|
1150
|
-
//
|
|
1151
|
-
//
|
|
1152
|
-
|
|
1153
|
-
|
|
979
|
+
// Shared render execution: wraps handleRscRendering (or revalidateAfterAction)
|
|
980
|
+
// in route middleware and timeout handling. Consolidates the pattern used by
|
|
981
|
+
// action-revalidate, full-render, and partial-render modes.
|
|
982
|
+
async function executeRenderWithMiddleware(
|
|
983
|
+
routeMiddleware: import("../router/middleware-types.js").CollectedMiddleware[],
|
|
984
|
+
negotiated: boolean,
|
|
985
|
+
routeKey: string,
|
|
986
|
+
routeReverse: ReturnType<typeof createReverseFunction>,
|
|
1154
987
|
request: Request,
|
|
1155
988
|
env: TEnv,
|
|
1156
989
|
url: URL,
|
|
1157
990
|
variables: Record<string, any>,
|
|
991
|
+
nonce: string | undefined,
|
|
992
|
+
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
993
|
+
isPartial: boolean,
|
|
994
|
+
actionContinuation?: ActionContinuation,
|
|
1158
995
|
): Promise<Response> {
|
|
1159
|
-
const
|
|
996
|
+
const renderHandler = async (): Promise<Response> => {
|
|
997
|
+
try {
|
|
998
|
+
let response: Response;
|
|
999
|
+
if (actionContinuation) {
|
|
1000
|
+
response = await revalidateAfterAction(
|
|
1001
|
+
handlerCtx,
|
|
1002
|
+
request,
|
|
1003
|
+
env,
|
|
1004
|
+
url,
|
|
1005
|
+
handleStore,
|
|
1006
|
+
actionContinuation,
|
|
1007
|
+
);
|
|
1008
|
+
} else {
|
|
1009
|
+
response = await handleRscRendering(
|
|
1010
|
+
handlerCtx,
|
|
1011
|
+
request,
|
|
1012
|
+
env,
|
|
1013
|
+
url,
|
|
1014
|
+
isPartial,
|
|
1015
|
+
handleStore,
|
|
1016
|
+
nonce,
|
|
1017
|
+
);
|
|
1018
|
+
}
|
|
1019
|
+
if (negotiated && !isWebSocketUpgradeResponse(response)) {
|
|
1020
|
+
response.headers.append("Vary", "Accept");
|
|
1021
|
+
}
|
|
1022
|
+
return response;
|
|
1023
|
+
} catch (error) {
|
|
1024
|
+
// Check if middleware/handler returned Response
|
|
1025
|
+
if (error instanceof Response) {
|
|
1026
|
+
// An action revalidation render is delivered to the client over the
|
|
1027
|
+
// same Flight-parsing path as a partial navigation, so a Response
|
|
1028
|
+
// thrown during it must be converted exactly like a partial one
|
|
1029
|
+
// (raw 200 -> hard-nav hint, 3xx -> Flight redirect). Without this,
|
|
1030
|
+
// the no-middleware path returns the raw Response (the with-middleware
|
|
1031
|
+
// path is already covered by the isPartial || actionContinuation
|
|
1032
|
+
// guard below).
|
|
1033
|
+
const treatAsPartial = isPartial || actionContinuation != null;
|
|
1034
|
+
|
|
1035
|
+
// During partial (client-side navigation), a 200 Response from a handler
|
|
1036
|
+
// means the route serves raw content (JSON, text, etc.), not JSX.
|
|
1037
|
+
// Signal the browser to hard-navigate so it renders the raw response.
|
|
1038
|
+
if (treatAsPartial && error.status === 200) {
|
|
1039
|
+
console.warn(
|
|
1040
|
+
`[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
|
|
1041
|
+
`Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
|
|
1042
|
+
);
|
|
1043
|
+
return createResponseWithMergedHeaders(null, {
|
|
1044
|
+
status: 200,
|
|
1045
|
+
headers: {
|
|
1046
|
+
"X-RSC-Reload": stripInternalParams(url).toString(),
|
|
1047
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
1048
|
+
},
|
|
1049
|
+
});
|
|
1050
|
+
}
|
|
1160
1051
|
|
|
1161
|
-
|
|
1162
|
-
|
|
1163
|
-
|
|
1164
|
-
|
|
1165
|
-
|
|
1052
|
+
if (treatAsPartial) {
|
|
1053
|
+
const intercepted = interceptRedirectForPartial(
|
|
1054
|
+
error,
|
|
1055
|
+
createRedirectFlightResponse,
|
|
1056
|
+
);
|
|
1057
|
+
if (intercepted) return intercepted;
|
|
1058
|
+
}
|
|
1166
1059
|
|
|
1167
|
-
|
|
1168
|
-
|
|
1169
|
-
if (!registeredLoader) {
|
|
1170
|
-
return createResponseWithMergedHeaders(
|
|
1171
|
-
`Loader "${loaderId}" not found in registry`,
|
|
1172
|
-
{ status: 404 },
|
|
1173
|
-
);
|
|
1174
|
-
}
|
|
1060
|
+
return error;
|
|
1061
|
+
}
|
|
1175
1062
|
|
|
1176
|
-
|
|
1177
|
-
|
|
1178
|
-
|
|
1179
|
-
|
|
1063
|
+
// Render 404 page for unmatched routes
|
|
1064
|
+
if (isRouteNotFoundError(error)) {
|
|
1065
|
+
callOnError(error, "routing", {
|
|
1066
|
+
request,
|
|
1067
|
+
url,
|
|
1068
|
+
env,
|
|
1069
|
+
handledByBoundary: true,
|
|
1070
|
+
});
|
|
1180
1071
|
|
|
1181
|
-
|
|
1182
|
-
|
|
1183
|
-
|
|
1184
|
-
|
|
1185
|
-
|
|
1186
|
-
|
|
1187
|
-
|
|
1072
|
+
const notFoundOption = router.notFound;
|
|
1073
|
+
const notFoundComponent =
|
|
1074
|
+
typeof notFoundOption === "function"
|
|
1075
|
+
? notFoundOption({ pathname: url.pathname })
|
|
1076
|
+
: (notFoundOption ?? createElement("h1", null, "Not Found"));
|
|
1077
|
+
|
|
1078
|
+
const notFoundSegment = {
|
|
1079
|
+
id: "notFound",
|
|
1080
|
+
namespace: "notFound",
|
|
1081
|
+
type: "route" as const,
|
|
1082
|
+
index: 0,
|
|
1083
|
+
component: notFoundComponent,
|
|
1084
|
+
params: {},
|
|
1188
1085
|
};
|
|
1189
|
-
loaderParams = jsonBody.params ?? {};
|
|
1190
|
-
loaderBody = jsonBody.body;
|
|
1191
|
-
}
|
|
1192
|
-
} catch {
|
|
1193
|
-
return createResponseWithMergedHeaders("Invalid JSON body", {
|
|
1194
|
-
status: 400,
|
|
1195
|
-
});
|
|
1196
|
-
}
|
|
1197
|
-
} else {
|
|
1198
|
-
const loaderParamsJson = url.searchParams.get("_rsc_loader_params");
|
|
1199
|
-
if (loaderParamsJson) {
|
|
1200
|
-
try {
|
|
1201
|
-
loaderParams = JSON.parse(loaderParamsJson);
|
|
1202
|
-
} catch {
|
|
1203
|
-
return createResponseWithMergedHeaders(
|
|
1204
|
-
"Invalid _rsc_loader_params JSON",
|
|
1205
|
-
{ status: 400 },
|
|
1206
|
-
);
|
|
1207
|
-
}
|
|
1208
|
-
}
|
|
1209
|
-
}
|
|
1210
|
-
|
|
1211
|
-
// Execute the loader with middleware
|
|
1212
|
-
try {
|
|
1213
|
-
const { fn, middleware } = registeredLoader;
|
|
1214
1086
|
|
|
1215
|
-
|
|
1216
|
-
|
|
1217
|
-
|
|
1218
|
-
|
|
1219
|
-
|
|
1220
|
-
|
|
1221
|
-
|
|
1222
|
-
|
|
1223
|
-
|
|
1224
|
-
|
|
1225
|
-
|
|
1226
|
-
|
|
1087
|
+
const payload: RscPayload = {
|
|
1088
|
+
metadata: {
|
|
1089
|
+
pathname: url.pathname,
|
|
1090
|
+
routerId: router.id,
|
|
1091
|
+
basename: router.basename,
|
|
1092
|
+
segments: [notFoundSegment],
|
|
1093
|
+
matched: [],
|
|
1094
|
+
diff: [],
|
|
1095
|
+
isPartial: false,
|
|
1096
|
+
rootLayout: router.rootLayout,
|
|
1097
|
+
handles: handleStore.stream(),
|
|
1098
|
+
version,
|
|
1099
|
+
stateCookieName: router.resolvedStateCookieName,
|
|
1100
|
+
themeConfig: router.themeConfig,
|
|
1101
|
+
warmupEnabled: router.warmupEnabled,
|
|
1102
|
+
initialTheme: requireRequestContext().theme,
|
|
1103
|
+
},
|
|
1227
1104
|
};
|
|
1228
1105
|
|
|
1229
|
-
const
|
|
1106
|
+
const rscStream = renderToReadableStream(payload, {
|
|
1107
|
+
onError: (error: unknown) => {
|
|
1108
|
+
callOnError(error, "rendering", { request, url, env });
|
|
1109
|
+
},
|
|
1110
|
+
});
|
|
1230
1111
|
|
|
1231
|
-
|
|
1232
|
-
|
|
1112
|
+
if (isRscRequest(request, url, isPartial)) {
|
|
1113
|
+
return createResponseWithMergedHeaders(rscStream, {
|
|
1114
|
+
status: 404,
|
|
1115
|
+
headers: {
|
|
1116
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
1117
|
+
// Router identity for the client's pre-decode integrity check; a
|
|
1118
|
+
// same-app 404 matches and applies in place. See response-adapter.
|
|
1119
|
+
"X-RSC-Router-Id": router.id,
|
|
1120
|
+
},
|
|
1121
|
+
});
|
|
1233
1122
|
}
|
|
1234
|
-
const loaderPayload: LoaderPayload = { loaderResult: result };
|
|
1235
|
-
const rscStream =
|
|
1236
|
-
renderToReadableStream<LoaderPayload>(loaderPayload);
|
|
1237
1123
|
|
|
1238
|
-
|
|
1239
|
-
|
|
1124
|
+
const [ssrModule, streamMode] = await getSSRSetup(
|
|
1125
|
+
handlerCtx,
|
|
1126
|
+
request,
|
|
1127
|
+
env,
|
|
1128
|
+
url,
|
|
1129
|
+
requireRequestContext()._metricsStore,
|
|
1130
|
+
);
|
|
1131
|
+
const htmlStream = await ssrModule.renderHTML(rscStream, {
|
|
1132
|
+
nonce,
|
|
1133
|
+
streamMode,
|
|
1240
1134
|
});
|
|
1241
|
-
},
|
|
1242
|
-
);
|
|
1243
|
-
} catch (error) {
|
|
1244
|
-
const err = error instanceof Error ? error : new Error(String(error));
|
|
1245
|
-
const isDev = process.env.NODE_ENV !== "production";
|
|
1246
|
-
|
|
1247
|
-
console.error("[RSC] Loader error:", error);
|
|
1248
|
-
|
|
1249
|
-
callOnError(error, "loader", {
|
|
1250
|
-
request,
|
|
1251
|
-
url,
|
|
1252
|
-
env,
|
|
1253
|
-
loaderName: loaderId,
|
|
1254
|
-
handledByBoundary: false,
|
|
1255
|
-
});
|
|
1256
|
-
|
|
1257
|
-
const errorPayload = {
|
|
1258
|
-
loaderResult: null,
|
|
1259
|
-
loaderError: {
|
|
1260
|
-
message: isDev ? err.message : "An error occurred",
|
|
1261
|
-
name: err.name,
|
|
1262
|
-
},
|
|
1263
|
-
};
|
|
1264
|
-
const rscStream = renderToReadableStream(errorPayload);
|
|
1265
|
-
|
|
1266
|
-
return createResponseWithMergedHeaders(rscStream, {
|
|
1267
|
-
status: 500,
|
|
1268
|
-
headers: { "content-type": "text/x-component;charset=utf-8" },
|
|
1269
|
-
});
|
|
1270
|
-
}
|
|
1271
|
-
}
|
|
1272
1135
|
|
|
1273
|
-
|
|
1274
|
-
|
|
1275
|
-
|
|
1276
|
-
async function handleRscRendering(
|
|
1277
|
-
request: Request,
|
|
1278
|
-
env: TEnv,
|
|
1279
|
-
url: URL,
|
|
1280
|
-
isPartial: boolean,
|
|
1281
|
-
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
1282
|
-
nonce: string | undefined,
|
|
1283
|
-
): Promise<Response> {
|
|
1284
|
-
// Retrieve handler-level timing from variables
|
|
1285
|
-
const reqCtx = requireRequestContext();
|
|
1286
|
-
const handlerTimingArr: string[] = reqCtx.var.__handlerTiming || [];
|
|
1287
|
-
const handlerStart: number = reqCtx.var.__handlerStart || 0;
|
|
1288
|
-
|
|
1289
|
-
let payload: RscPayload;
|
|
1290
|
-
let serverTiming: string | undefined;
|
|
1291
|
-
|
|
1292
|
-
if (isPartial) {
|
|
1293
|
-
// Partial render (navigation)
|
|
1294
|
-
const result = await router.matchPartial(request, env);
|
|
1295
|
-
|
|
1296
|
-
if (!result) {
|
|
1297
|
-
// Fall back to full render
|
|
1298
|
-
const match = await router.match(request, env);
|
|
1299
|
-
setRequestContextParams(match.params);
|
|
1300
|
-
|
|
1301
|
-
if (match.redirect) {
|
|
1302
|
-
return createResponseWithMergedHeaders(null, {
|
|
1303
|
-
status: 308,
|
|
1304
|
-
headers: { Location: match.redirect },
|
|
1136
|
+
return createResponseWithMergedHeaders(htmlStream, {
|
|
1137
|
+
status: 404,
|
|
1138
|
+
headers: { "content-type": "text/html;charset=utf-8" },
|
|
1305
1139
|
});
|
|
1306
1140
|
}
|
|
1307
1141
|
|
|
1308
|
-
|
|
1309
|
-
|
|
1310
|
-
|
|
1311
|
-
|
|
1312
|
-
|
|
1313
|
-
|
|
1314
|
-
? `${match.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
|
|
1315
|
-
: `rendering;dur=${renderDuration.toFixed(2)}`;
|
|
1316
|
-
|
|
1317
|
-
payload = {
|
|
1318
|
-
root,
|
|
1319
|
-
metadata: {
|
|
1320
|
-
pathname: url.pathname,
|
|
1321
|
-
segments: match.segments,
|
|
1322
|
-
matched: match.matched,
|
|
1323
|
-
diff: match.diff,
|
|
1324
|
-
isPartial: false,
|
|
1325
|
-
handles: handleStore.stream(),
|
|
1326
|
-
version,
|
|
1327
|
-
themeConfig: router.themeConfig,
|
|
1328
|
-
initialTheme: reqCtx.theme,
|
|
1329
|
-
},
|
|
1330
|
-
};
|
|
1331
|
-
} else {
|
|
1332
|
-
setRequestContextParams(result.params);
|
|
1333
|
-
serverTiming = result.serverTiming;
|
|
1334
|
-
|
|
1335
|
-
payload = {
|
|
1336
|
-
root: null,
|
|
1337
|
-
metadata: {
|
|
1338
|
-
pathname: url.pathname,
|
|
1339
|
-
segments: result.segments,
|
|
1340
|
-
matched: result.matched,
|
|
1341
|
-
diff: result.diff,
|
|
1342
|
-
isPartial: true,
|
|
1343
|
-
slots: result.slots,
|
|
1344
|
-
handles: handleStore.stream(),
|
|
1345
|
-
version,
|
|
1346
|
-
},
|
|
1347
|
-
};
|
|
1348
|
-
}
|
|
1349
|
-
} else {
|
|
1350
|
-
// Full render (initial page load)
|
|
1351
|
-
const match = await router.match(request, env);
|
|
1352
|
-
setRequestContextParams(match.params);
|
|
1353
|
-
|
|
1354
|
-
if (match.redirect) {
|
|
1355
|
-
return createResponseWithMergedHeaders(null, {
|
|
1356
|
-
status: 308,
|
|
1357
|
-
headers: { Location: match.redirect },
|
|
1142
|
+
// Report unhandled errors
|
|
1143
|
+
callOnError(error, "routing", {
|
|
1144
|
+
request,
|
|
1145
|
+
url,
|
|
1146
|
+
env,
|
|
1147
|
+
handledByBoundary: false,
|
|
1358
1148
|
});
|
|
1149
|
+
console.error(`[RSC] Error:`, error);
|
|
1150
|
+
throw error;
|
|
1359
1151
|
}
|
|
1152
|
+
};
|
|
1360
1153
|
|
|
1361
|
-
|
|
1362
|
-
|
|
1363
|
-
|
|
1364
|
-
|
|
1365
|
-
|
|
1366
|
-
|
|
1367
|
-
|
|
1368
|
-
|
|
1369
|
-
|
|
1370
|
-
|
|
1371
|
-
const serializedSegments = await serializeSegments(nonLoaderSegments);
|
|
1372
|
-
const handles: Record<string, Record<string, unknown[]>> = {};
|
|
1373
|
-
for (const seg of nonLoaderSegments) {
|
|
1374
|
-
const segHandles = handleStore.getDataForSegment(seg.id);
|
|
1375
|
-
if (Object.keys(segHandles).length > 0) {
|
|
1376
|
-
handles[seg.id] = segHandles;
|
|
1377
|
-
}
|
|
1378
|
-
}
|
|
1379
|
-
return new Response(
|
|
1380
|
-
JSON.stringify({
|
|
1381
|
-
segments: serializedSegments,
|
|
1382
|
-
handles,
|
|
1383
|
-
routeName: match.routeName,
|
|
1384
|
-
params: match.params,
|
|
1385
|
-
}),
|
|
1386
|
-
{ headers: { "Content-Type": "application/json" } },
|
|
1154
|
+
// Wrap the render path in a renderStartMs timeout
|
|
1155
|
+
const executeRender = async (): Promise<Response> => {
|
|
1156
|
+
if (routeMiddleware.length > 0) {
|
|
1157
|
+
const mwResponse = await executeMiddleware(
|
|
1158
|
+
buildRouteMiddlewareEntries<TEnv>(routeMiddleware),
|
|
1159
|
+
request,
|
|
1160
|
+
env,
|
|
1161
|
+
variables,
|
|
1162
|
+
renderHandler,
|
|
1163
|
+
routeReverse,
|
|
1387
1164
|
);
|
|
1388
|
-
} else {
|
|
1389
|
-
const renderStart = performance.now();
|
|
1390
|
-
const root = renderSegments(match.segments, {
|
|
1391
|
-
rootLayout: router.rootLayout,
|
|
1392
|
-
});
|
|
1393
|
-
const renderDuration = performance.now() - renderStart;
|
|
1394
|
-
serverTiming = match.serverTiming
|
|
1395
|
-
? `${match.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
|
|
1396
|
-
: `rendering;dur=${renderDuration.toFixed(2)}`;
|
|
1397
|
-
|
|
1398
|
-
payload = {
|
|
1399
|
-
root,
|
|
1400
|
-
metadata: {
|
|
1401
|
-
pathname: url.pathname,
|
|
1402
|
-
segments: match.segments,
|
|
1403
|
-
matched: match.matched,
|
|
1404
|
-
diff: match.diff,
|
|
1405
|
-
isPartial: false,
|
|
1406
|
-
rootLayout: router.rootLayout,
|
|
1407
|
-
handles: handleStore.stream(),
|
|
1408
|
-
version,
|
|
1409
|
-
themeConfig: router.themeConfig,
|
|
1410
|
-
initialTheme: reqCtx.theme,
|
|
1411
|
-
},
|
|
1412
|
-
};
|
|
1413
|
-
}
|
|
1414
|
-
}
|
|
1415
1165
|
|
|
1416
|
-
|
|
1417
|
-
|
|
1418
|
-
|
|
1419
|
-
|
|
1420
|
-
|
|
1421
|
-
|
|
1422
|
-
|
|
1423
|
-
(!request.headers.get("accept")?.includes("text/html") &&
|
|
1424
|
-
!url.searchParams.has("__html")) ||
|
|
1425
|
-
url.searchParams.has("__rsc");
|
|
1426
|
-
|
|
1427
|
-
// Build complete Server-Timing: handler phases + match/manifest + rendering + RSC serialize
|
|
1428
|
-
const timingParts: string[] = [...handlerTimingArr];
|
|
1429
|
-
if (serverTiming) {
|
|
1430
|
-
timingParts.push(serverTiming);
|
|
1431
|
-
}
|
|
1432
|
-
timingParts.push(`rsc-serialize;dur=${rscSerializeDur.toFixed(2)}`);
|
|
1166
|
+
if (isPartial || actionContinuation) {
|
|
1167
|
+
const intercepted = interceptRedirectForPartial(
|
|
1168
|
+
mwResponse,
|
|
1169
|
+
createRedirectFlightResponse,
|
|
1170
|
+
);
|
|
1171
|
+
if (intercepted) return intercepted;
|
|
1172
|
+
}
|
|
1433
1173
|
|
|
1434
|
-
|
|
1435
|
-
const fullTiming = timingParts.join(", ");
|
|
1436
|
-
const rscHeaders: Record<string, string> = {
|
|
1437
|
-
"content-type": "text/x-component;charset=utf-8",
|
|
1438
|
-
vary: "accept",
|
|
1439
|
-
};
|
|
1440
|
-
if (fullTiming) {
|
|
1441
|
-
rscHeaders["Server-Timing"] = fullTiming;
|
|
1174
|
+
return finalizeResponse(mwResponse);
|
|
1442
1175
|
}
|
|
1443
|
-
return createResponseWithMergedHeaders(rscStream, {
|
|
1444
|
-
headers: rscHeaders,
|
|
1445
|
-
});
|
|
1446
|
-
}
|
|
1447
|
-
|
|
1448
|
-
// Delegate to SSR for HTML response
|
|
1449
|
-
const ssrModuleStart = performance.now();
|
|
1450
|
-
const ssrModule = await loadSSRModule();
|
|
1451
|
-
const ssrModuleDur = performance.now() - ssrModuleStart;
|
|
1452
|
-
timingParts.push(`ssr-module-load;dur=${ssrModuleDur.toFixed(2)}`);
|
|
1453
|
-
|
|
1454
|
-
const ssrRenderStart = performance.now();
|
|
1455
|
-
const htmlStream = await ssrModule.renderHTML(rscStream, { nonce });
|
|
1456
|
-
const ssrRenderDur = performance.now() - ssrRenderStart;
|
|
1457
|
-
timingParts.push(`ssr-render-html;dur=${ssrRenderDur.toFixed(2)}`);
|
|
1458
|
-
|
|
1459
|
-
// Add total handler duration
|
|
1460
|
-
if (handlerStart) {
|
|
1461
|
-
const totalHandler = performance.now() - handlerStart;
|
|
1462
|
-
timingParts.push(`handler-total;dur=${totalHandler.toFixed(2)}`);
|
|
1463
|
-
}
|
|
1464
1176
|
|
|
1465
|
-
|
|
1466
|
-
const htmlHeaders: Record<string, string> = {
|
|
1467
|
-
"content-type": "text/html;charset=utf-8",
|
|
1177
|
+
return renderHandler();
|
|
1468
1178
|
};
|
|
1469
|
-
if (fullTiming) {
|
|
1470
|
-
htmlHeaders["Server-Timing"] = fullTiming;
|
|
1471
|
-
}
|
|
1472
1179
|
|
|
1473
|
-
|
|
1474
|
-
|
|
1475
|
-
|
|
1180
|
+
const renderOutcome = await withTimeout(
|
|
1181
|
+
executeRender(),
|
|
1182
|
+
router.timeouts.renderStartMs,
|
|
1183
|
+
"render-start",
|
|
1184
|
+
);
|
|
1185
|
+
if (renderOutcome.timedOut) {
|
|
1186
|
+
return handleTimeoutResponse(
|
|
1187
|
+
request,
|
|
1188
|
+
env,
|
|
1189
|
+
url,
|
|
1190
|
+
"render-start",
|
|
1191
|
+
renderOutcome.durationMs,
|
|
1192
|
+
routeKey,
|
|
1193
|
+
);
|
|
1194
|
+
}
|
|
1195
|
+
return renderOutcome.result;
|
|
1476
1196
|
}
|
|
1477
1197
|
}
|