@rangojs/router 0.0.0-experimental.12 → 0.0.0-experimental.120
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 +1037 -4
- 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 +1704 -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/dist/browser/react/use-client-cache.js +39 -0
- 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/dist/handles/index.js +6 -0
- 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 +5994 -2763
- package/dist/vite/index.js.bak +5448 -0
- package/dist/vite/index.js.map +1 -0
- package/dist/vite/index.named-routes.gen.ts +103 -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 +64 -54
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +252 -0
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +484 -0
- package/skills/caching/SKILL.md +202 -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 +537 -70
- package/skills/host-router/SKILL.md +264 -0
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +173 -8
- package/skills/layout/SKILL.md +123 -5
- package/skills/links/SKILL.md +304 -25
- package/skills/loader/SKILL.md +600 -54
- package/skills/middleware/SKILL.md +211 -37
- package/skills/migrate-nextjs/SKILL.md +562 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +41 -10
- package/skills/observability/SKILL.md +137 -0
- package/skills/parallel/SKILL.md +271 -3
- package/skills/prerender/SKILL.md +413 -52
- package/skills/rango/SKILL.md +301 -21
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +248 -120
- package/skills/route/SKILL.md +279 -21
- package/skills/router-setup/SKILL.md +211 -33
- package/skills/server-actions/SKILL.md +751 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +547 -107
- package/skills/use-cache/SKILL.md +353 -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 +102 -4
- package/src/bin/rango.ts +312 -15
- package/src/browser/action-coordinator.ts +114 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +172 -128
- 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/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 +309 -562
- package/src/browser/navigation-client.ts +206 -75
- package/src/browser/navigation-store.ts +80 -63
- package/src/browser/navigation-transaction.ts +279 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +355 -314
- package/src/browser/prefetch/cache.ts +363 -0
- package/src/browser/prefetch/fetch.ts +350 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +191 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +136 -0
- package/src/browser/react/Link.tsx +255 -71
- package/src/browser/react/NavigationProvider.tsx +153 -33
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +55 -0
- package/src/browser/react/index.ts +15 -12
- package/src/browser/react/location-state-shared.ts +269 -56
- package/src/browser/react/location-state.ts +90 -19
- 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 +29 -51
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +41 -123
- package/src/browser/react/use-link-status.ts +6 -5
- package/src/browser/react/use-navigation.ts +44 -65
- package/src/browser/react/use-params.ts +77 -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 +97 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +85 -99
- package/src/browser/response-adapter.ts +124 -0
- package/src/browser/rsc-router.tsx +275 -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 +508 -610
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +149 -48
- package/src/browser/validate-redirect-origin.ts +29 -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 -1029
- package/src/build/index.ts +4 -0
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +127 -31
- 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 +102 -0
- package/src/build/route-types/include-resolution.ts +418 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +131 -0
- package/src/build/route-types/router-processing.ts +659 -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-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +354 -0
- package/src/cache/cache-scope.ts +170 -308
- package/src/cache/cf/cf-cache-store.ts +696 -30
- package/src/cache/cf/index.ts +13 -3
- package/src/cache/document-cache.ts +116 -77
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +144 -0
- package/src/cache/index.ts +1 -15
- package/src/cache/memory-segment-store.ts +192 -13
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +153 -0
- package/src/cache/types.ts +78 -124
- package/src/client.rsc.tsx +6 -1
- package/src/client.tsx +117 -322
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +156 -0
- package/src/debug.ts +19 -9
- package/src/decode-loader-results.ts +36 -0
- package/src/errors.ts +106 -10
- package/src/handle.ts +54 -11
- package/src/handles/MetaTags.tsx +73 -20
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/handles/meta.ts +30 -13
- package/src/host/cookie-handler.ts +21 -15
- package/src/host/errors.ts +8 -8
- package/src/host/index.ts +6 -9
- package/src/host/pattern-matcher.ts +27 -27
- package/src/host/router.ts +176 -82
- package/src/host/testing.ts +8 -8
- package/src/host/types.ts +46 -9
- package/src/host/utils.ts +2 -2
- package/src/href-client.ts +196 -57
- package/src/index.rsc.ts +112 -42
- package/src/index.ts +178 -74
- package/src/internal-debug.ts +9 -3
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +15 -93
- package/src/loader.ts +20 -10
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +4 -2
- package/src/prerender/store.ts +125 -18
- package/src/prerender.ts +389 -23
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +198 -128
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +13 -32
- package/src/route-definition/dsl-helpers.ts +1109 -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 +101 -0
- package/src/route-definition/resolve-handler-use.ts +155 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-definition.ts +1 -1450
- package/src/route-map-builder.ts +87 -133
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +72 -41
- package/src/router/content-negotiation.ts +228 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +10 -10
- package/src/router/find-match.ts +208 -0
- package/src/router/handler-context.ts +372 -125
- package/src/router/intercept-resolution.ts +24 -26
- package/src/router/lazy-includes.ts +256 -0
- package/src/router/loader-resolution.ts +367 -140
- package/src/router/logging.ts +112 -6
- package/src/router/manifest.ts +89 -28
- package/src/router/match-api.ts +175 -238
- package/src/router/match-context.ts +4 -2
- package/src/router/match-handlers.ts +441 -0
- package/src/router/match-middleware/background-revalidation.ts +97 -89
- package/src/router/match-middleware/cache-lookup.ts +289 -54
- package/src/router/match-middleware/cache-store.ts +78 -17
- package/src/router/match-middleware/intercept-resolution.ts +45 -22
- package/src/router/match-middleware/segment-resolution.ts +73 -9
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +146 -24
- package/src/router/metrics.ts +241 -16
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +209 -0
- package/src/router/middleware.ts +354 -384
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +279 -44
- package/src/router/prerender-match.ts +527 -0
- package/src/router/preview-match.ts +100 -0
- package/src/router/request-classification.ts +313 -0
- package/src/router/revalidation.ts +163 -16
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +41 -21
- package/src/router/router-interfaces.ts +500 -0
- package/src/router/router-options.ts +642 -0
- package/src/router/router-registry.ts +21 -0
- package/src/router/segment-resolution/fresh.ts +769 -0
- package/src/router/segment-resolution/helpers.ts +268 -0
- package/src/router/segment-resolution/loader-cache.ts +199 -0
- package/src/router/segment-resolution/revalidation.ts +1420 -0
- package/src/router/segment-resolution/static-store.ts +81 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +21 -1354
- package/src/router/segment-wrappers.ts +291 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +133 -42
- package/src/router/types.ts +24 -9
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +710 -2373
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +821 -1128
- package/src/rsc/helpers.ts +182 -19
- package/src/rsc/index.ts +1 -21
- package/src/rsc/loader-fetch.ts +229 -0
- package/src/rsc/manifest-init.ts +77 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +161 -0
- package/src/rsc/progressive-enhancement.ts +395 -0
- package/src/rsc/response-error.ts +104 -0
- package/src/rsc/response-route-handler.ts +362 -0
- package/src/rsc/rsc-rendering.ts +240 -0
- package/src/rsc/runtime-warnings.ts +41 -0
- package/src/rsc/server-action.ts +350 -0
- package/src/rsc/ssr-setup.ts +144 -0
- package/src/rsc/types.ts +52 -11
- package/src/search-params.ts +70 -58
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +237 -73
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +439 -85
- package/src/server/cookie-store.ts +214 -0
- package/src/server/fetchable-loader-store.ts +11 -6
- package/src/server/handle-store.ts +113 -15
- package/src/server/loader-registry.ts +24 -64
- package/src/server/request-context.ts +640 -107
- package/src/server.ts +26 -164
- package/src/ssr/index.tsx +107 -30
- package/src/static-handler.ts +39 -9
- package/src/theme/ThemeProvider.tsx +21 -15
- package/src/theme/ThemeScript.tsx +5 -5
- package/src/theme/constants.ts +5 -2
- package/src/theme/index.ts +4 -14
- package/src/theme/theme-context.ts +4 -30
- package/src/theme/theme-script.ts +21 -18
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +113 -0
- package/src/types/handler-context.ts +809 -0
- package/src/types/index.ts +89 -0
- package/src/types/loader-types.ts +212 -0
- package/src/types/request-scope.ts +126 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +120 -0
- package/src/types/segments.ts +184 -0
- package/src/types.ts +1 -1795
- package/src/urls/include-helper.ts +164 -0
- package/src/urls/index.ts +49 -0
- package/src/urls/path-helper-types.ts +386 -0
- package/src/urls/path-helper.ts +329 -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 +94 -0
- package/src/urls.ts +1 -1323
- package/src/use-loader.tsx +559 -108
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +181 -0
- package/src/vite/discovery/discover-routers.ts +379 -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 +480 -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 +150 -0
- package/src/vite/discovery/virtual-module-codegen.ts +193 -0
- package/src/vite/index.ts +17 -2261
- package/src/vite/plugin-types.ts +170 -0
- package/src/vite/plugins/cjs-to-esm.ts +94 -0
- package/src/vite/plugins/client-ref-dedup.ts +131 -0
- package/src/vite/plugins/client-ref-hashing.ts +128 -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 +214 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +109 -60
- package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +32 -51
- package/src/vite/plugins/expose-ids/export-analysis.ts +376 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +156 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +72 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +127 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +796 -0
- package/src/vite/plugins/performance-tracks.ts +92 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +338 -0
- package/src/vite/plugins/version-injector.ts +99 -0
- package/src/vite/plugins/version-plugin.ts +323 -0
- package/src/vite/plugins/virtual-entries.ts +123 -0
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +549 -0
- package/src/vite/router-discovery.ts +1567 -0
- package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +194 -22
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +139 -0
- package/src/vite/utils/client-chunks.ts +190 -0
- package/src/vite/utils/forward-user-plugins.ts +193 -0
- package/src/vite/utils/manifest-utils.ts +19 -0
- package/src/vite/utils/package-resolution.ts +161 -0
- package/src/vite/utils/prerender-utils.ts +222 -0
- package/src/vite/utils/shared-utils.ts +251 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- 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,92 @@
|
|
|
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 { isWebSocketUpgradeResponse } from "../response-utils.js";
|
|
35
|
+
import {
|
|
36
|
+
handleResponseRoute,
|
|
37
|
+
type ResponseRouteMatch,
|
|
38
|
+
} from "./response-route-handler.js";
|
|
39
|
+
import { generateNonce, nonce as nonceToken } from "./nonce.js";
|
|
36
40
|
import { VERSION } from "@rangojs/router:version";
|
|
37
41
|
import type { ErrorPhase } from "../types.js";
|
|
42
|
+
import type { RouterRequestInput } from "../router/router-interfaces.js";
|
|
38
43
|
import { invokeOnError } from "../router/error-handling.js";
|
|
39
44
|
import {
|
|
40
|
-
|
|
45
|
+
createReverseFunction,
|
|
46
|
+
stripInternalParams,
|
|
47
|
+
} from "../router/handler-context.js";
|
|
48
|
+
import { getRouterContext } from "../router/router-context.js";
|
|
49
|
+
import { resolveSink, safeEmit } from "../router/telemetry.js";
|
|
50
|
+
import { contextSet } from "../context-var.js";
|
|
51
|
+
import {
|
|
41
52
|
hasCachedManifest,
|
|
42
|
-
setCachedManifest,
|
|
43
53
|
getRouteTrie,
|
|
44
|
-
setRouteTrie,
|
|
45
54
|
getPrecomputedEntries,
|
|
46
55
|
waitForManifestReady,
|
|
47
56
|
getRouterManifest,
|
|
48
57
|
getRouterTrie,
|
|
49
|
-
setRouterManifest,
|
|
50
|
-
setRouterTrie,
|
|
51
58
|
} 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
|
-
|
|
59
|
+
import type { HandlerContext } from "./handler-context.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 {
|
|
86
|
+
startSSRSetup,
|
|
87
|
+
getSSRSetup,
|
|
88
|
+
mayNeedSSR,
|
|
89
|
+
isRscRequest,
|
|
90
|
+
SSR_SETUP_VAR,
|
|
91
|
+
} from "./ssr-setup.js";
|
|
92
|
+
import {
|
|
93
|
+
classifyRequest,
|
|
94
|
+
type RequestPlan,
|
|
95
|
+
type ExecutableRequestPlan,
|
|
96
|
+
} from "../router/request-classification.js";
|
|
77
97
|
|
|
78
98
|
/**
|
|
79
99
|
* Create an RSC request handler.
|
|
@@ -108,6 +128,22 @@ function createResponseErrorPayload(error: unknown, isDev: boolean): ResponseErr
|
|
|
108
128
|
* });
|
|
109
129
|
* ```
|
|
110
130
|
*/
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Response that tells the client to do a full document navigation. Shared by
|
|
134
|
+
* the terminal reload plans (version-mismatch and app-switch): an empty 200
|
|
135
|
+
* carrying X-RSC-Reload, which the client turns into window.location.href.
|
|
136
|
+
*/
|
|
137
|
+
function createReloadResponse(reloadUrl: string) {
|
|
138
|
+
return createResponseWithMergedHeaders(null, {
|
|
139
|
+
status: 200,
|
|
140
|
+
headers: {
|
|
141
|
+
"X-RSC-Reload": reloadUrl,
|
|
142
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
143
|
+
},
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
|
|
111
147
|
export function createRSCHandler<
|
|
112
148
|
TEnv = unknown,
|
|
113
149
|
TRoutes extends Record<string, string> = Record<string, string>,
|
|
@@ -125,33 +161,182 @@ export function createRSCHandler<
|
|
|
125
161
|
decodeFormState,
|
|
126
162
|
} = deps;
|
|
127
163
|
|
|
128
|
-
// Use provided loadSSRModule or default to vite RSC module loader
|
|
129
|
-
|
|
164
|
+
// Use provided loadSSRModule or default to vite RSC module loader.
|
|
165
|
+
// In production the SSR module is stable across requests, so memoize
|
|
166
|
+
// the dynamic import to avoid repeated module resolution overhead.
|
|
167
|
+
// In dev mode Vite may hot-reload the module, so skip memoization.
|
|
168
|
+
const rawLoadSSRModule: LoadSSRModule =
|
|
130
169
|
options.loadSSRModule ??
|
|
131
170
|
(() => import.meta.viteRsc.loadModule("ssr", "index"));
|
|
171
|
+
let _ssrModulePromise: Promise<SSRModule> | undefined;
|
|
172
|
+
const loadSSRModule: LoadSSRModule =
|
|
173
|
+
process.env.NODE_ENV === "production"
|
|
174
|
+
? () =>
|
|
175
|
+
(_ssrModulePromise ??= rawLoadSSRModule().catch((err) => {
|
|
176
|
+
_ssrModulePromise = undefined;
|
|
177
|
+
throw err;
|
|
178
|
+
}))
|
|
179
|
+
: rawLoadSSRModule;
|
|
132
180
|
|
|
133
181
|
/**
|
|
134
|
-
*
|
|
135
|
-
*
|
|
182
|
+
* Per-request error reporter that deduplicates via the ALS request context.
|
|
183
|
+
*
|
|
184
|
+
* Uses the same _reportedErrors WeakSet as the router layer so errors
|
|
185
|
+
* that propagate across layers are only reported once per request.
|
|
136
186
|
*/
|
|
137
187
|
function callOnError(
|
|
138
188
|
error: unknown,
|
|
139
189
|
phase: ErrorPhase,
|
|
140
190
|
context: Parameters<typeof invokeOnError<TEnv>>[3],
|
|
141
191
|
): void {
|
|
192
|
+
// Guard: abort signal handlers fire asynchronously outside the ALS
|
|
193
|
+
// request scope, so the context may be gone. Skip dedup in that
|
|
194
|
+
// case — the error is from a cancelled stream, not a real failure.
|
|
195
|
+
const reqCtx = _getRequestContext();
|
|
196
|
+
if (error != null && typeof error === "object" && reqCtx) {
|
|
197
|
+
if (reqCtx._reportedErrors.has(error)) return;
|
|
198
|
+
reqCtx._reportedErrors.add(error);
|
|
199
|
+
}
|
|
142
200
|
invokeOnError(router.onError, error, phase, context, "RSC");
|
|
143
201
|
}
|
|
144
202
|
|
|
145
|
-
|
|
203
|
+
function getRequiredRouteMap(): Record<string, string> {
|
|
204
|
+
const routeMap = getRouterManifest(router.id);
|
|
205
|
+
if (!routeMap) {
|
|
206
|
+
throw new Error(
|
|
207
|
+
`Route manifest for router "${router.id}" is not available.`,
|
|
208
|
+
);
|
|
209
|
+
}
|
|
210
|
+
return routeMap;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
/**
|
|
214
|
+
* Handle a timeout by reporting the error, emitting telemetry,
|
|
215
|
+
* and returning either the custom onTimeout response or a default 504.
|
|
216
|
+
*/
|
|
217
|
+
async function handleTimeoutResponse(
|
|
146
218
|
request: Request,
|
|
147
|
-
env: TEnv
|
|
148
|
-
|
|
219
|
+
env: TEnv,
|
|
220
|
+
url: URL,
|
|
221
|
+
phase: TimeoutPhase,
|
|
222
|
+
durationMs: number,
|
|
223
|
+
routeKey?: string,
|
|
224
|
+
actionId?: string,
|
|
225
|
+
): Promise<Response> {
|
|
226
|
+
const timeoutError = new RouterTimeoutError(phase, durationMs);
|
|
227
|
+
|
|
228
|
+
callOnError(timeoutError, phase === "action" ? "action" : "handler", {
|
|
229
|
+
request,
|
|
230
|
+
url,
|
|
231
|
+
env,
|
|
232
|
+
routeKey,
|
|
233
|
+
actionId,
|
|
234
|
+
handledByBoundary: false,
|
|
235
|
+
metadata: { timeout: true, phase, durationMs },
|
|
236
|
+
});
|
|
237
|
+
|
|
238
|
+
try {
|
|
239
|
+
const routerCtx = getRouterContext();
|
|
240
|
+
if (routerCtx?.telemetry) {
|
|
241
|
+
safeEmit(resolveSink(routerCtx.telemetry), {
|
|
242
|
+
type: "request.timeout" as const,
|
|
243
|
+
timestamp: performance.now(),
|
|
244
|
+
requestId: routerCtx.requestId,
|
|
245
|
+
phase,
|
|
246
|
+
pathname: url.pathname,
|
|
247
|
+
routeKey,
|
|
248
|
+
actionId,
|
|
249
|
+
durationMs,
|
|
250
|
+
customHandler: !!router.onTimeout,
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
} catch {
|
|
254
|
+
// Router context may not be available
|
|
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
|
+
): Response {
|
|
288
|
+
const redirectPayload: RscPayload = {
|
|
289
|
+
metadata: {
|
|
290
|
+
pathname: redirectUrl,
|
|
291
|
+
segments: [],
|
|
292
|
+
redirect: { url: redirectUrl },
|
|
293
|
+
...(locationState && { locationState }),
|
|
294
|
+
},
|
|
295
|
+
};
|
|
296
|
+
const rscStream = renderToReadableStream<RscPayload>(redirectPayload);
|
|
297
|
+
return createResponseWithMergedHeaders(rscStream, {
|
|
298
|
+
status: 200,
|
|
299
|
+
headers: { "content-type": "text/x-component;charset=utf-8" },
|
|
300
|
+
});
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
// Bundle shared dependencies for extracted handler functions.
|
|
304
|
+
// callOnError reads from ALS so it's inherently per-request scoped.
|
|
305
|
+
const handlerCtx: HandlerContext<TEnv> = {
|
|
306
|
+
router,
|
|
307
|
+
version,
|
|
308
|
+
renderToReadableStream,
|
|
309
|
+
decodeReply,
|
|
310
|
+
createTemporaryReferenceSet,
|
|
311
|
+
loadServerAction,
|
|
312
|
+
decodeAction,
|
|
313
|
+
decodeFormState,
|
|
314
|
+
loadSSRModule,
|
|
315
|
+
callOnError,
|
|
316
|
+
getRequiredRouteMap,
|
|
317
|
+
createRedirectFlightResponse,
|
|
318
|
+
resolveStreamMode: async (request, env, url) => {
|
|
319
|
+
const resolver = router.ssr?.resolveStreaming;
|
|
320
|
+
if (!resolver) return "stream";
|
|
321
|
+
return resolver({ request, env, url });
|
|
149
322
|
},
|
|
323
|
+
};
|
|
324
|
+
|
|
325
|
+
return async function handler(
|
|
326
|
+
request: Request,
|
|
327
|
+
input: RouterRequestInput<TEnv> = {},
|
|
150
328
|
): Promise<Response> {
|
|
151
329
|
const handlerStart = performance.now();
|
|
330
|
+
// Create the metrics store at handler start so handler:total has startTime=0
|
|
331
|
+
// and all metrics are relative to the request entry point.
|
|
332
|
+
const earlyMetricsStore = router.debugPerformance
|
|
333
|
+
? createMetricsStore(true, handlerStart)
|
|
334
|
+
: undefined;
|
|
335
|
+
|
|
336
|
+
const { env = {} as TEnv, vars: initialVars, ctx: executionCtx } = input;
|
|
152
337
|
|
|
153
338
|
// Connection warmup: return 204 immediately before any processing
|
|
154
|
-
if (router
|
|
339
|
+
if (router?.warmupEnabled && request.method === "HEAD") {
|
|
155
340
|
const warmupUrl = new URL(request.url);
|
|
156
341
|
if (warmupUrl.searchParams.has("_rsc_warmup")) {
|
|
157
342
|
return new Response(null, { status: 204 });
|
|
@@ -175,24 +360,27 @@ export function createRSCHandler<
|
|
|
175
360
|
const mwMatchDur = performance.now() - mwMatchStart;
|
|
176
361
|
|
|
177
362
|
// Shared variables between middleware and route handlers
|
|
178
|
-
// Initialize from
|
|
179
|
-
const variables: Record<string, any> =
|
|
180
|
-
|
|
181
|
-
|
|
363
|
+
// Initialize from input.vars if provided (allows pre-seeding from worker entry)
|
|
364
|
+
const variables: Record<string, any> = initialVars
|
|
365
|
+
? { ...initialVars }
|
|
366
|
+
: {};
|
|
182
367
|
|
|
183
|
-
// Store nonce
|
|
368
|
+
// Store nonce via ContextVar token and string key for backward compat
|
|
184
369
|
if (nonce) {
|
|
370
|
+
contextSet(variables, nonceToken, nonce);
|
|
185
371
|
variables.nonce = nonce;
|
|
186
372
|
}
|
|
187
373
|
|
|
188
374
|
// Resolve cache store configuration
|
|
189
375
|
// Priority: options.cache (handler override) > router.cache (router default)
|
|
190
376
|
// Store is enabled only if: config provided, enabled, and no ?__no_cache query param
|
|
191
|
-
let cacheStore
|
|
377
|
+
let cacheStore: SegmentCacheStore | undefined;
|
|
192
378
|
const cacheOption = options.cache ?? router.cache;
|
|
193
379
|
if (cacheOption && !url.searchParams.has("__no_cache")) {
|
|
194
380
|
const cacheConfig =
|
|
195
|
-
typeof cacheOption === "function"
|
|
381
|
+
typeof cacheOption === "function"
|
|
382
|
+
? cacheOption(env, executionCtx)
|
|
383
|
+
: cacheOption;
|
|
196
384
|
|
|
197
385
|
if (cacheConfig.enabled !== false) {
|
|
198
386
|
cacheStore = cacheConfig.store;
|
|
@@ -223,56 +411,7 @@ export function createRSCHandler<
|
|
|
223
411
|
// Cloudflare dev: generate manifest inline for this router.
|
|
224
412
|
// Each router generates its own manifest independently so
|
|
225
413
|
// 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 });
|
|
414
|
+
await buildRouterTrieFromUrlpatterns(router);
|
|
276
415
|
}
|
|
277
416
|
if (!getRouterManifest(router.id) && !hasCachedManifest()) {
|
|
278
417
|
throw new Error(
|
|
@@ -280,10 +419,17 @@ export function createRSCHandler<
|
|
|
280
419
|
);
|
|
281
420
|
}
|
|
282
421
|
}
|
|
283
|
-
const manifestCacheDur = performance.now() - manifestCacheStart;
|
|
284
422
|
|
|
285
|
-
//
|
|
286
|
-
// This
|
|
423
|
+
// Rebuild the trie when the manifest exists but the per-router trie is
|
|
424
|
+
// missing. This happens in dev mode after HMR: the virtual module sets
|
|
425
|
+
// the manifest (from fresh gen files) but the trie is intentionally not
|
|
426
|
+
// injected to avoid stale discovery-time data. Without the trie, route
|
|
427
|
+
// matching falls back to regex iteration which does not handle wildcard
|
|
428
|
+
// priority correctly (catch-all patterns match before specific routes).
|
|
429
|
+
if (!getRouterTrie(router.id) && router.urlpatterns) {
|
|
430
|
+
await buildRouterTrieFromUrlpatterns(router);
|
|
431
|
+
}
|
|
432
|
+
const manifestCacheDur = performance.now() - manifestCacheStart;
|
|
287
433
|
|
|
288
434
|
// Create unified request context with all methods
|
|
289
435
|
// Includes: stub response, handle store, loader memoization, use(), cookies, headers, cache store
|
|
@@ -295,9 +441,27 @@ export function createRSCHandler<
|
|
|
295
441
|
url,
|
|
296
442
|
variables,
|
|
297
443
|
cacheStore,
|
|
298
|
-
|
|
444
|
+
cacheProfiles: router.cacheProfiles,
|
|
445
|
+
executionContext: executionCtx,
|
|
299
446
|
themeConfig: router.themeConfig,
|
|
300
447
|
});
|
|
448
|
+
if (earlyMetricsStore) {
|
|
449
|
+
requestContext._debugPerformance = true;
|
|
450
|
+
requestContext._metricsStore = earlyMetricsStore;
|
|
451
|
+
}
|
|
452
|
+
// Wire background error reporting so "use cache" and other subsystems
|
|
453
|
+
// can surface non-fatal errors through the router's onError callback.
|
|
454
|
+
requestContext._reportBackgroundError = (
|
|
455
|
+
error: unknown,
|
|
456
|
+
category: string,
|
|
457
|
+
) => {
|
|
458
|
+
callOnError(error, "cache", {
|
|
459
|
+
request,
|
|
460
|
+
url,
|
|
461
|
+
metadata: { category },
|
|
462
|
+
});
|
|
463
|
+
};
|
|
464
|
+
|
|
301
465
|
const ctxCreateDur = performance.now() - ctxCreateStart;
|
|
302
466
|
|
|
303
467
|
// Accumulate handler-level timing for Server-Timing header
|
|
@@ -319,6 +483,9 @@ export function createRSCHandler<
|
|
|
319
483
|
// - Server components during rendering
|
|
320
484
|
// - Error boundaries
|
|
321
485
|
// - Streaming
|
|
486
|
+
// Store basename on request context (scoped per-request via existing ALS)
|
|
487
|
+
requestContext._basename = router.basename;
|
|
488
|
+
|
|
322
489
|
return runWithRequestContext(requestContext, async () => {
|
|
323
490
|
// Core handler logic (wrapped by middleware)
|
|
324
491
|
const coreHandler = async (): Promise<Response> => {
|
|
@@ -326,289 +493,99 @@ export function createRSCHandler<
|
|
|
326
493
|
};
|
|
327
494
|
|
|
328
495
|
// Execute middleware chain if any, otherwise call core handler directly
|
|
496
|
+
let response: Response;
|
|
329
497
|
if (matchedMiddleware.length > 0) {
|
|
330
|
-
|
|
498
|
+
const mwResponse = await executeMiddleware(
|
|
331
499
|
matchedMiddleware,
|
|
332
500
|
request,
|
|
333
501
|
env,
|
|
334
502
|
variables,
|
|
335
503
|
coreHandler,
|
|
504
|
+
createReverseFunction(getRequiredRouteMap()),
|
|
336
505
|
);
|
|
337
|
-
}
|
|
338
|
-
|
|
339
|
-
return coreHandler();
|
|
340
|
-
});
|
|
341
|
-
};
|
|
342
|
-
|
|
343
|
-
// Core request handling logic (separated for middleware wrapping)
|
|
344
|
-
async function coreRequestHandler(
|
|
345
|
-
request: Request,
|
|
346
|
-
env: TEnv,
|
|
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
|
-
}
|
|
380
|
-
|
|
381
|
-
// Build lightweight context for response handler
|
|
382
|
-
const bindings = (env as any)?.Bindings ?? env;
|
|
383
|
-
const reqCtx = requireRequestContext();
|
|
384
|
-
const responseHandlerCtx = {
|
|
385
|
-
request,
|
|
386
|
-
params: preview.params || {},
|
|
387
|
-
env: bindings,
|
|
388
|
-
searchParams: url.searchParams,
|
|
389
|
-
url,
|
|
390
|
-
pathname: url.pathname,
|
|
391
|
-
href: (name: string, hrefParams?: Record<string, string>) => {
|
|
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
|
-
};
|
|
406
|
-
|
|
407
|
-
// Call handler directly, wrapped by route middleware if present
|
|
408
|
-
const callHandler = async () => {
|
|
409
|
-
// JSON response routes: wrap in { data } / { error } envelope
|
|
410
|
-
if (preview.responseType === "json") {
|
|
411
|
-
const errorCtx = { request, url, env };
|
|
412
|
-
try {
|
|
413
|
-
const result = await (preview.handler as Function)(responseHandlerCtx);
|
|
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" } },
|
|
435
|
-
);
|
|
436
|
-
}
|
|
437
|
-
}
|
|
438
506
|
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
return createResponseWithMergedHeaders(result.body, {
|
|
451
|
-
status: result.status,
|
|
452
|
-
headers: mergedHeaders,
|
|
453
|
-
});
|
|
454
|
-
}
|
|
455
|
-
|
|
456
|
-
// Auto-wrap based on response type tag
|
|
457
|
-
switch (preview.responseType) {
|
|
458
|
-
case "text":
|
|
459
|
-
return createResponseWithMergedHeaders(
|
|
460
|
-
String(result),
|
|
461
|
-
{ status: 200, headers: { "content-type": "text/plain;charset=utf-8" } },
|
|
462
|
-
);
|
|
463
|
-
case "html":
|
|
464
|
-
return createResponseWithMergedHeaders(
|
|
465
|
-
String(result),
|
|
466
|
-
{ status: 200, headers: { "content-type": "text/html;charset=utf-8" } },
|
|
467
|
-
);
|
|
468
|
-
case "xml":
|
|
469
|
-
return createResponseWithMergedHeaders(
|
|
470
|
-
String(result),
|
|
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
|
-
});
|
|
497
|
-
}
|
|
498
|
-
};
|
|
499
|
-
|
|
500
|
-
// Wrap callHandler to append Vary: Accept on content-negotiated responses
|
|
501
|
-
const callHandlerWithVary = async () => {
|
|
502
|
-
const response = await callHandler();
|
|
503
|
-
if (preview.negotiated) {
|
|
504
|
-
response.headers.append("Vary", "Accept");
|
|
507
|
+
if (
|
|
508
|
+
url.searchParams.has("_rsc_partial") ||
|
|
509
|
+
url.searchParams.has("_rsc_action")
|
|
510
|
+
) {
|
|
511
|
+
const intercepted = interceptRedirectForPartial(
|
|
512
|
+
mwResponse,
|
|
513
|
+
createRedirectFlightResponse,
|
|
514
|
+
);
|
|
515
|
+
response = intercepted ?? finalizeResponse(mwResponse);
|
|
516
|
+
} else {
|
|
517
|
+
response = finalizeResponse(mwResponse);
|
|
505
518
|
}
|
|
506
|
-
|
|
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);
|
|
519
|
+
} else {
|
|
520
|
+
response = await coreHandler();
|
|
521
521
|
}
|
|
522
522
|
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
523
|
+
// Finalize metrics after all middleware (including post-next work)
|
|
524
|
+
// has completed so :post spans are captured in the timeline.
|
|
525
|
+
// Handler timing parts are always emitted (even without debug metrics)
|
|
526
|
+
// so non-debug requests still get bootstrap Server-Timing entries.
|
|
527
|
+
const handlerTimingArr: string[] = variables.__handlerTiming || [];
|
|
528
|
+
// Preserve any existing Server-Timing set by response routes or middleware
|
|
529
|
+
const existingTiming = response.headers.get("Server-Timing");
|
|
530
|
+
const timingParts = existingTiming
|
|
531
|
+
? [existingTiming, ...handlerTimingArr]
|
|
532
|
+
: [...handlerTimingArr];
|
|
533
|
+
|
|
534
|
+
const metricsStore = requestContext._metricsStore;
|
|
535
|
+
if (metricsStore) {
|
|
536
|
+
// When the store was created at handler start (earlyMetricsStore),
|
|
537
|
+
// handler:total covers the full request. When ctx.debugPerformance()
|
|
538
|
+
// created the store mid-request, use its requestStart to avoid a
|
|
539
|
+
// negative startTime offset.
|
|
540
|
+
const totalStart = earlyMetricsStore
|
|
541
|
+
? handlerStart
|
|
542
|
+
: metricsStore.requestStart;
|
|
543
|
+
appendMetric(
|
|
544
|
+
metricsStore,
|
|
545
|
+
"handler:total",
|
|
546
|
+
totalStart,
|
|
547
|
+
performance.now() - totalStart,
|
|
548
|
+
);
|
|
549
|
+
const metricsTiming = buildMetricsTiming(
|
|
550
|
+
request.method,
|
|
551
|
+
url.pathname,
|
|
552
|
+
metricsStore,
|
|
553
|
+
);
|
|
554
|
+
if (metricsTiming) timingParts.push(metricsTiming);
|
|
531
555
|
}
|
|
532
|
-
return response;
|
|
533
|
-
};
|
|
534
|
-
|
|
535
|
-
if (preview?.routeMiddleware && preview.routeMiddleware.length > 0) {
|
|
536
|
-
// Convert route middleware to app middleware format for execution
|
|
537
|
-
const middlewareEntries = preview.routeMiddleware.map((mw) => ({
|
|
538
|
-
entry: {
|
|
539
|
-
pattern: null,
|
|
540
|
-
regex: null,
|
|
541
|
-
paramNames: [],
|
|
542
|
-
handler: mw.handler,
|
|
543
|
-
mountPrefix: null,
|
|
544
|
-
},
|
|
545
|
-
params: mw.params,
|
|
546
|
-
}));
|
|
547
556
|
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
557
|
+
const fullTiming = timingParts.join(", ");
|
|
558
|
+
if (fullTiming && !isWebSocketUpgradeResponse(response)) {
|
|
559
|
+
response.headers.set("Server-Timing", fullTiming);
|
|
560
|
+
}
|
|
551
561
|
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
}
|
|
562
|
+
return response;
|
|
563
|
+
});
|
|
564
|
+
};
|
|
555
565
|
|
|
556
|
-
//
|
|
557
|
-
|
|
566
|
+
// Core request handling logic (separated for middleware wrapping).
|
|
567
|
+
// Uses the classify → execute model: classifyRequest produces a RequestPlan,
|
|
568
|
+
// then execution dispatches on the plan mode.
|
|
569
|
+
async function coreRequestHandler(
|
|
558
570
|
request: Request,
|
|
559
571
|
env: TEnv,
|
|
560
572
|
url: URL,
|
|
561
573
|
variables: Record<string, any>,
|
|
562
574
|
nonce: string | undefined,
|
|
563
575
|
): 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
|
-
);
|
|
576
|
+
const handlerTiming: string[] = variables.__handlerTiming || [];
|
|
577
577
|
|
|
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.
|
|
578
|
+
// Debug manifest endpoint: handled before classification since it
|
|
579
|
+
// doesn't need a route match and needs trie access from the closure.
|
|
604
580
|
const isDev = process.env.NODE_ENV !== "production";
|
|
605
581
|
if (
|
|
606
582
|
url.searchParams.has("__debug_manifest") &&
|
|
607
583
|
(isDev || router.allowDebugManifest)
|
|
608
584
|
) {
|
|
609
585
|
const trie = getRouterTrie(router.id) ?? getRouteTrie();
|
|
610
|
-
const routeManifest =
|
|
611
|
-
const { extractAncestryFromTrie } =
|
|
586
|
+
const routeManifest = getRequiredRouteMap();
|
|
587
|
+
const { extractAncestryFromTrie } =
|
|
588
|
+
await import("../build/route-trie.js");
|
|
612
589
|
return new Response(
|
|
613
590
|
JSON.stringify(
|
|
614
591
|
{
|
|
@@ -627,851 +604,567 @@ export function createRSCHandler<
|
|
|
627
604
|
);
|
|
628
605
|
}
|
|
629
606
|
|
|
630
|
-
//
|
|
631
|
-
|
|
632
|
-
|
|
607
|
+
// ---- 1. Classify ----
|
|
608
|
+
// classifyRequest may throw RouteNotFoundError for unknown routes.
|
|
609
|
+
// In that case, fall through to a full-render plan so the pipeline
|
|
610
|
+
// can render the 404 page via the existing error handling path.
|
|
611
|
+
const classifyStart = performance.now();
|
|
612
|
+
let plan: RequestPlan<TEnv>;
|
|
633
613
|
try {
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
614
|
+
plan = await classifyRequest<TEnv>(request, url, {
|
|
615
|
+
findMatch: router.findMatch,
|
|
616
|
+
routerVersion: version,
|
|
617
|
+
routerId: router.id,
|
|
618
|
+
});
|
|
619
|
+
} catch (error) {
|
|
620
|
+
if (isRouteNotFoundError(error)) {
|
|
621
|
+
// Let the render path handle 404 — match()/matchPartial() will
|
|
622
|
+
// re-throw RouteNotFoundError and the catch block in
|
|
623
|
+
// executeRenderWithMiddleware renders the not-found page.
|
|
624
|
+
plan = {
|
|
625
|
+
mode: "full-render",
|
|
626
|
+
route: {
|
|
627
|
+
matched: null as any,
|
|
628
|
+
manifestEntry: null as any,
|
|
629
|
+
entries: [],
|
|
630
|
+
routeKey: "",
|
|
631
|
+
localRouteName: "",
|
|
632
|
+
params: {},
|
|
633
|
+
routeMiddleware: [],
|
|
634
|
+
cacheScope: null,
|
|
635
|
+
isPassthrough: false,
|
|
636
|
+
},
|
|
637
|
+
negotiated: false,
|
|
638
|
+
};
|
|
639
|
+
} else {
|
|
640
|
+
throw error;
|
|
647
641
|
}
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
642
|
+
}
|
|
643
|
+
const classifyDur = performance.now() - classifyStart;
|
|
644
|
+
handlerTiming.push(`handler-classify;dur=${classifyDur.toFixed(2)}`);
|
|
645
|
+
|
|
646
|
+
// ---- 2. Terminal plans (no execution needed) ----
|
|
647
|
+
if (plan.mode === "redirect") {
|
|
648
|
+
// Redirects are handled by the pipeline (match/matchPartial),
|
|
649
|
+
// but for partial requests we short-circuit with a Flight redirect.
|
|
650
|
+
if (url.searchParams.has("_rsc_partial")) {
|
|
651
|
+
return createRedirectFlightResponse(plan.redirectUrl);
|
|
654
652
|
}
|
|
653
|
+
// Full requests: let the pipeline handle the redirect via match()
|
|
654
|
+
// which returns { redirect: url }. Fall through to full-render.
|
|
655
|
+
}
|
|
655
656
|
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
662
|
-
|
|
657
|
+
if (plan.mode === "version-mismatch") {
|
|
658
|
+
console.log(
|
|
659
|
+
`[RSC] Version mismatch: client=${url.searchParams.get("_rsc_v")}, server=${version}. Forcing reload.`,
|
|
660
|
+
);
|
|
661
|
+
return createReloadResponse(plan.reloadUrl);
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
if (plan.mode === "app-switch") {
|
|
665
|
+
// Cross-app SPA navigation crossed a host-router app boundary. Force a
|
|
666
|
+
// real document navigation so the target app's document is re-established
|
|
667
|
+
// (stylesheets, theme, warmup, prefetch-TTL). See request-classification.
|
|
668
|
+
return createReloadResponse(plan.reloadUrl);
|
|
669
|
+
}
|
|
663
670
|
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
return await handleRscRendering(
|
|
671
|
+
// ---- 3. Origin guard (gate for action/loader/PE modes) ----
|
|
672
|
+
const originPhase = ORIGIN_CHECK_PHASE_BY_MODE[plan.mode];
|
|
673
|
+
if (originPhase) {
|
|
674
|
+
const originResult = await checkRequestOrigin(
|
|
669
675
|
request,
|
|
670
|
-
env,
|
|
671
676
|
url,
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
677
|
+
router.originCheck,
|
|
678
|
+
env,
|
|
679
|
+
router.id,
|
|
680
|
+
originPhase,
|
|
675
681
|
);
|
|
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
|
-
}
|
|
682
|
+
if (originResult) {
|
|
683
|
+
const originError = new Error(
|
|
684
|
+
`Origin check rejected: ${request.headers.get("origin") ?? "none"} vs ${request.headers.get("host") ?? "none"}`,
|
|
685
|
+
);
|
|
686
|
+
originError.name = "OriginCheckError";
|
|
706
687
|
|
|
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", {
|
|
688
|
+
callOnError(originError, "origin", {
|
|
714
689
|
request,
|
|
715
690
|
url,
|
|
716
691
|
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,
|
|
692
|
+
handledByBoundary: false,
|
|
745
693
|
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
|
|
694
|
+
phase: originPhase,
|
|
695
|
+
origin: request.headers.get("origin"),
|
|
696
|
+
host: request.headers.get("host"),
|
|
757
697
|
},
|
|
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");
|
|
698
|
+
});
|
|
767
699
|
|
|
768
|
-
|
|
769
|
-
|
|
770
|
-
|
|
771
|
-
|
|
772
|
-
|
|
700
|
+
try {
|
|
701
|
+
const routerCtx = getRouterContext();
|
|
702
|
+
if (routerCtx?.telemetry) {
|
|
703
|
+
safeEmit(resolveSink(routerCtx.telemetry), {
|
|
704
|
+
type: "request.origin-rejected" as const,
|
|
705
|
+
timestamp: performance.now(),
|
|
706
|
+
requestId: routerCtx.requestId,
|
|
707
|
+
method: request.method,
|
|
708
|
+
pathname: url.pathname,
|
|
709
|
+
phase: originPhase,
|
|
710
|
+
origin: request.headers.get("origin"),
|
|
711
|
+
host: request.headers.get("host"),
|
|
712
|
+
});
|
|
713
|
+
}
|
|
714
|
+
} catch {
|
|
715
|
+
// Router context may not be available
|
|
773
716
|
}
|
|
774
717
|
|
|
775
|
-
|
|
776
|
-
const ssrModule = await loadSSRModule();
|
|
777
|
-
const htmlStream = await ssrModule.renderHTML(rscStream, { nonce });
|
|
778
|
-
|
|
779
|
-
return createResponseWithMergedHeaders(htmlStream, {
|
|
780
|
-
status: 404,
|
|
781
|
-
headers: { "content-type": "text/html;charset=utf-8" },
|
|
782
|
-
});
|
|
718
|
+
return originResult;
|
|
783
719
|
}
|
|
784
|
-
|
|
785
|
-
// Report unhandled errors
|
|
786
|
-
callOnError(error, "routing", {
|
|
787
|
-
request,
|
|
788
|
-
url,
|
|
789
|
-
env,
|
|
790
|
-
handledByBoundary: false,
|
|
791
|
-
});
|
|
792
|
-
console.error(`[RSC] Error:`, error);
|
|
793
|
-
throw error;
|
|
794
720
|
}
|
|
721
|
+
|
|
722
|
+
// ---- 4. Execute ----
|
|
723
|
+
return executeRequest(
|
|
724
|
+
plan as ExecutableRequestPlan<TEnv>,
|
|
725
|
+
request,
|
|
726
|
+
env,
|
|
727
|
+
url,
|
|
728
|
+
variables,
|
|
729
|
+
nonce,
|
|
730
|
+
);
|
|
795
731
|
}
|
|
796
732
|
|
|
797
|
-
//
|
|
798
|
-
//
|
|
799
|
-
//
|
|
800
|
-
//
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
async function handleProgressiveEnhancement(
|
|
733
|
+
// Execute a classified request plan. Dispatches to the appropriate handler
|
|
734
|
+
// based on plan.mode. Lives in the createRSCHandler closure for access to
|
|
735
|
+
// handlerCtx, router, callOnError, etc.
|
|
736
|
+
// Only receives executable plans (version-mismatch is handled above).
|
|
737
|
+
async function executeRequest(
|
|
738
|
+
plan: ExecutableRequestPlan<TEnv>,
|
|
804
739
|
request: Request,
|
|
805
740
|
env: TEnv,
|
|
806
741
|
url: URL,
|
|
807
|
-
|
|
808
|
-
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
742
|
+
variables: Record<string, any>,
|
|
809
743
|
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;
|
|
744
|
+
): Promise<Response> {
|
|
745
|
+
// Common setup
|
|
746
|
+
const handleStore = requireRequestContext()._handleStore;
|
|
827
747
|
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
748
|
+
// Wire up error reporting for late streaming-handle failures
|
|
749
|
+
handleStore.onError = (error: Error) => {
|
|
750
|
+
const reqCtx = requireRequestContext();
|
|
751
|
+
callOnError(error, "handler", {
|
|
752
|
+
request,
|
|
753
|
+
url,
|
|
754
|
+
routeKey: reqCtx._routeName,
|
|
755
|
+
params: reqCtx.params as Record<string, string>,
|
|
756
|
+
handledByBoundary: true,
|
|
757
|
+
});
|
|
758
|
+
try {
|
|
759
|
+
const routerCtx = getRouterContext();
|
|
760
|
+
if (routerCtx?.telemetry) {
|
|
761
|
+
safeEmit(resolveSink(routerCtx.telemetry), {
|
|
762
|
+
type: "handler.error" as const,
|
|
763
|
+
timestamp: performance.now(),
|
|
764
|
+
requestId: routerCtx.requestId,
|
|
765
|
+
error,
|
|
766
|
+
handledByBoundary: true,
|
|
767
|
+
pathname: url.pathname,
|
|
768
|
+
routeKey: reqCtx._routeName,
|
|
769
|
+
params: reqCtx.params as Record<string, string>,
|
|
770
|
+
});
|
|
771
|
+
}
|
|
772
|
+
} catch {
|
|
773
|
+
// Router context may not be available (e.g. prerender path)
|
|
834
774
|
}
|
|
835
|
-
}
|
|
775
|
+
};
|
|
836
776
|
|
|
837
|
-
|
|
838
|
-
|
|
777
|
+
// Set route params early so all execution paths can access ctx.params.
|
|
778
|
+
// Also store the classified snapshot so match/matchPartial can reuse it
|
|
779
|
+
// instead of calling resolveRoute again.
|
|
780
|
+
if (plan.mode !== "redirect") {
|
|
781
|
+
setRequestContextParams(plan.route.params, plan.route.routeKey);
|
|
782
|
+
requireRequestContext()._classifiedRoute = plan.route;
|
|
839
783
|
}
|
|
840
784
|
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
|
|
845
|
-
|
|
846
|
-
|
|
847
|
-
|
|
848
|
-
|
|
849
|
-
|
|
850
|
-
|
|
785
|
+
const routeReverse = createReverseFunction(getRequiredRouteMap());
|
|
786
|
+
|
|
787
|
+
// ---- Response route: skip entire RSC pipeline ----
|
|
788
|
+
if (plan.mode === "response") {
|
|
789
|
+
// Build ResponseRouteMatch from plan fields. handleResponseRoute
|
|
790
|
+
// expects a flat object with params at the top level.
|
|
791
|
+
const responseMatch: ResponseRouteMatch = {
|
|
792
|
+
responseType: plan.responseType,
|
|
793
|
+
handler: plan.handler,
|
|
794
|
+
params: plan.route.params,
|
|
795
|
+
negotiated: plan.negotiated,
|
|
796
|
+
manifestEntry: plan.manifestEntry,
|
|
797
|
+
routeMiddleware: plan.routeMiddleware,
|
|
798
|
+
};
|
|
799
|
+
const responseOutcome = await withTimeout(
|
|
800
|
+
handleResponseRoute(
|
|
801
|
+
handlerCtx,
|
|
802
|
+
responseMatch,
|
|
851
803
|
request,
|
|
852
|
-
url,
|
|
853
804
|
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
805
|
url,
|
|
806
|
+
variables,
|
|
807
|
+
),
|
|
808
|
+
router.timeouts.renderStartMs,
|
|
809
|
+
"render-start",
|
|
810
|
+
);
|
|
811
|
+
if (responseOutcome.timedOut) {
|
|
812
|
+
return handleTimeoutResponse(
|
|
813
|
+
request,
|
|
875
814
|
env,
|
|
876
|
-
|
|
877
|
-
|
|
878
|
-
|
|
879
|
-
|
|
815
|
+
url,
|
|
816
|
+
"render-start",
|
|
817
|
+
responseOutcome.durationMs,
|
|
818
|
+
plan.route.routeKey,
|
|
819
|
+
);
|
|
820
|
+
}
|
|
821
|
+
const response = responseOutcome.result;
|
|
822
|
+
if (plan.negotiated && !isWebSocketUpgradeResponse(response)) {
|
|
823
|
+
response.headers.append("Vary", "Accept");
|
|
880
824
|
}
|
|
825
|
+
return response;
|
|
881
826
|
}
|
|
882
827
|
|
|
883
|
-
//
|
|
884
|
-
|
|
885
|
-
|
|
886
|
-
|
|
887
|
-
|
|
828
|
+
// SSR setup: kick off in parallel for modes that need HTML rendering.
|
|
829
|
+
// Placed after response-route short-circuit so response/mime routes
|
|
830
|
+
// never pay for SSR work.
|
|
831
|
+
if (plan.mode !== "loader" && mayNeedSSR(request, url)) {
|
|
832
|
+
variables[SSR_SETUP_VAR] = startSSRSetup(
|
|
833
|
+
handlerCtx,
|
|
888
834
|
request,
|
|
889
|
-
url,
|
|
890
835
|
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
|
-
});
|
|
836
|
+
url,
|
|
837
|
+
router.debugPerformance
|
|
838
|
+
? () => requireRequestContext()._metricsStore
|
|
839
|
+
: undefined,
|
|
840
|
+
);
|
|
909
841
|
}
|
|
910
842
|
|
|
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", {
|
|
843
|
+
// ---- Loader fetch ----
|
|
844
|
+
if (plan.mode === "loader") {
|
|
845
|
+
return handleLoaderFetch(
|
|
846
|
+
handlerCtx,
|
|
976
847
|
request,
|
|
977
|
-
url,
|
|
978
848
|
env,
|
|
979
|
-
|
|
980
|
-
|
|
981
|
-
|
|
982
|
-
|
|
983
|
-
cause: error,
|
|
984
|
-
});
|
|
849
|
+
url,
|
|
850
|
+
variables,
|
|
851
|
+
plan.route.params,
|
|
852
|
+
);
|
|
985
853
|
}
|
|
986
854
|
|
|
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", {
|
|
855
|
+
// ---- Progressive enhancement ----
|
|
856
|
+
if (plan.mode === "pe-render") {
|
|
857
|
+
const peResult = await handleProgressiveEnhancement(
|
|
858
|
+
handlerCtx,
|
|
1005
859
|
request,
|
|
1006
|
-
url,
|
|
1007
860
|
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
|
-
}
|
|
861
|
+
url,
|
|
862
|
+
false, // isAction = false for PE
|
|
863
|
+
handleStore,
|
|
864
|
+
nonce,
|
|
865
|
+
{
|
|
866
|
+
routeMiddleware: plan.route.routeMiddleware,
|
|
867
|
+
variables,
|
|
868
|
+
routeReverse,
|
|
869
|
+
},
|
|
870
|
+
);
|
|
871
|
+
if (peResult) return peResult;
|
|
872
|
+
// PE handler returned null (not a PE form) — fall through to render
|
|
1039
873
|
}
|
|
1040
874
|
|
|
1041
|
-
//
|
|
1042
|
-
|
|
1043
|
-
|
|
1044
|
-
|
|
1045
|
-
|
|
1046
|
-
|
|
1047
|
-
|
|
1048
|
-
|
|
1049
|
-
|
|
1050
|
-
|
|
1051
|
-
|
|
1052
|
-
|
|
1053
|
-
|
|
1054
|
-
|
|
1055
|
-
|
|
1056
|
-
|
|
1057
|
-
|
|
1058
|
-
|
|
1059
|
-
|
|
1060
|
-
|
|
1061
|
-
|
|
1062
|
-
|
|
1063
|
-
|
|
875
|
+
// ---- Action: execute action, then revalidate wrapped in route middleware ----
|
|
876
|
+
if (plan.mode === "action") {
|
|
877
|
+
let actionContinuation: ActionContinuation | undefined;
|
|
878
|
+
try {
|
|
879
|
+
const actionOutcome = await withTimeout(
|
|
880
|
+
executeServerAction(
|
|
881
|
+
handlerCtx,
|
|
882
|
+
request,
|
|
883
|
+
env,
|
|
884
|
+
url,
|
|
885
|
+
plan.actionId,
|
|
886
|
+
handleStore,
|
|
887
|
+
),
|
|
888
|
+
router.timeouts.actionMs,
|
|
889
|
+
"action",
|
|
890
|
+
);
|
|
891
|
+
if (actionOutcome.timedOut) {
|
|
892
|
+
return handleTimeoutResponse(
|
|
893
|
+
request,
|
|
894
|
+
env,
|
|
895
|
+
url,
|
|
896
|
+
"action",
|
|
897
|
+
actionOutcome.durationMs,
|
|
898
|
+
plan.route.routeKey,
|
|
899
|
+
plan.actionId,
|
|
900
|
+
);
|
|
901
|
+
}
|
|
902
|
+
const result = actionOutcome.result;
|
|
903
|
+
// Response means redirect or error boundary — done.
|
|
904
|
+
if (result instanceof Response) return result;
|
|
905
|
+
actionContinuation = result;
|
|
906
|
+
} catch (error) {
|
|
907
|
+
callOnError(error, "action", {
|
|
908
|
+
request,
|
|
909
|
+
url,
|
|
910
|
+
env,
|
|
911
|
+
actionId: plan.actionId,
|
|
912
|
+
handledByBoundary: false,
|
|
1064
913
|
});
|
|
914
|
+
console.error(`[RSC] Action error:`, error);
|
|
915
|
+
throw error;
|
|
1065
916
|
}
|
|
1066
917
|
|
|
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;
|
|
918
|
+
// Revalidation render wrapped in route middleware.
|
|
919
|
+
// Actions from client-side navigation include _rsc_partial — preserve
|
|
920
|
+
// the partial flag so the revalidation returns a Flight stream, not HTML.
|
|
921
|
+
// App-switch is already excluded by classifyRequest (would be full-render).
|
|
922
|
+
const isPartialAction = url.searchParams.has("_rsc_partial");
|
|
923
|
+
return executeRenderWithMiddleware(
|
|
924
|
+
plan.route.routeMiddleware,
|
|
925
|
+
plan.negotiated,
|
|
926
|
+
plan.route.routeKey,
|
|
927
|
+
routeReverse,
|
|
928
|
+
request,
|
|
929
|
+
env,
|
|
930
|
+
url,
|
|
931
|
+
variables,
|
|
932
|
+
nonce,
|
|
933
|
+
handleStore,
|
|
934
|
+
isPartialAction,
|
|
935
|
+
actionContinuation,
|
|
936
|
+
);
|
|
1141
937
|
}
|
|
1142
938
|
|
|
1143
|
-
|
|
1144
|
-
|
|
1145
|
-
|
|
1146
|
-
|
|
939
|
+
// Full render, partial render, fallen-through PE, and full-page redirect all
|
|
940
|
+
// render through the same middleware-wrapped path. Only full/partial-render
|
|
941
|
+
// carry negotiation + the partial flag; pe/redirect render plainly.
|
|
942
|
+
const isPartial = plan.mode === "partial-render";
|
|
943
|
+
const negotiated =
|
|
944
|
+
plan.mode === "full-render" || plan.mode === "partial-render"
|
|
945
|
+
? plan.negotiated
|
|
946
|
+
: false;
|
|
947
|
+
return executeRenderWithMiddleware(
|
|
948
|
+
plan.route.routeMiddleware,
|
|
949
|
+
negotiated,
|
|
950
|
+
plan.route.routeKey,
|
|
951
|
+
routeReverse,
|
|
952
|
+
request,
|
|
953
|
+
env,
|
|
954
|
+
url,
|
|
955
|
+
variables,
|
|
956
|
+
nonce,
|
|
957
|
+
handleStore,
|
|
958
|
+
isPartial,
|
|
959
|
+
);
|
|
1147
960
|
}
|
|
1148
961
|
|
|
1149
|
-
//
|
|
1150
|
-
//
|
|
1151
|
-
//
|
|
1152
|
-
|
|
1153
|
-
|
|
962
|
+
// Shared render execution: wraps handleRscRendering (or revalidateAfterAction)
|
|
963
|
+
// in route middleware and timeout handling. Consolidates the pattern used by
|
|
964
|
+
// action-revalidate, full-render, and partial-render modes.
|
|
965
|
+
async function executeRenderWithMiddleware(
|
|
966
|
+
routeMiddleware: import("../router/middleware-types.js").CollectedMiddleware[],
|
|
967
|
+
negotiated: boolean,
|
|
968
|
+
routeKey: string,
|
|
969
|
+
routeReverse: ReturnType<typeof createReverseFunction>,
|
|
1154
970
|
request: Request,
|
|
1155
971
|
env: TEnv,
|
|
1156
972
|
url: URL,
|
|
1157
973
|
variables: Record<string, any>,
|
|
974
|
+
nonce: string | undefined,
|
|
975
|
+
handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
|
|
976
|
+
isPartial: boolean,
|
|
977
|
+
actionContinuation?: ActionContinuation,
|
|
1158
978
|
): Promise<Response> {
|
|
1159
|
-
const
|
|
979
|
+
const renderHandler = async (): Promise<Response> => {
|
|
980
|
+
try {
|
|
981
|
+
let response: Response;
|
|
982
|
+
if (actionContinuation) {
|
|
983
|
+
response = await revalidateAfterAction(
|
|
984
|
+
handlerCtx,
|
|
985
|
+
request,
|
|
986
|
+
env,
|
|
987
|
+
url,
|
|
988
|
+
handleStore,
|
|
989
|
+
actionContinuation,
|
|
990
|
+
);
|
|
991
|
+
} else {
|
|
992
|
+
response = await handleRscRendering(
|
|
993
|
+
handlerCtx,
|
|
994
|
+
request,
|
|
995
|
+
env,
|
|
996
|
+
url,
|
|
997
|
+
isPartial,
|
|
998
|
+
handleStore,
|
|
999
|
+
nonce,
|
|
1000
|
+
);
|
|
1001
|
+
}
|
|
1002
|
+
if (negotiated && !isWebSocketUpgradeResponse(response)) {
|
|
1003
|
+
response.headers.append("Vary", "Accept");
|
|
1004
|
+
}
|
|
1005
|
+
return response;
|
|
1006
|
+
} catch (error) {
|
|
1007
|
+
// Check if middleware/handler returned Response
|
|
1008
|
+
if (error instanceof Response) {
|
|
1009
|
+
// During partial (client-side navigation), a 200 Response from a handler
|
|
1010
|
+
// means the route serves raw content (JSON, text, etc.), not JSX.
|
|
1011
|
+
// Signal the browser to hard-navigate so it renders the raw response.
|
|
1012
|
+
if (isPartial && error.status === 200) {
|
|
1013
|
+
console.warn(
|
|
1014
|
+
`[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
|
|
1015
|
+
`Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
|
|
1016
|
+
);
|
|
1017
|
+
return createResponseWithMergedHeaders(null, {
|
|
1018
|
+
status: 200,
|
|
1019
|
+
headers: {
|
|
1020
|
+
"X-RSC-Reload": stripInternalParams(url).toString(),
|
|
1021
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
1022
|
+
},
|
|
1023
|
+
});
|
|
1024
|
+
}
|
|
1160
1025
|
|
|
1161
|
-
|
|
1162
|
-
|
|
1163
|
-
|
|
1164
|
-
|
|
1165
|
-
|
|
1026
|
+
if (isPartial) {
|
|
1027
|
+
const intercepted = interceptRedirectForPartial(
|
|
1028
|
+
error,
|
|
1029
|
+
createRedirectFlightResponse,
|
|
1030
|
+
);
|
|
1031
|
+
if (intercepted) return intercepted;
|
|
1032
|
+
}
|
|
1166
1033
|
|
|
1167
|
-
|
|
1168
|
-
|
|
1169
|
-
if (!registeredLoader) {
|
|
1170
|
-
return createResponseWithMergedHeaders(
|
|
1171
|
-
`Loader "${loaderId}" not found in registry`,
|
|
1172
|
-
{ status: 404 },
|
|
1173
|
-
);
|
|
1174
|
-
}
|
|
1034
|
+
return error;
|
|
1035
|
+
}
|
|
1175
1036
|
|
|
1176
|
-
|
|
1177
|
-
|
|
1178
|
-
|
|
1179
|
-
|
|
1037
|
+
// Render 404 page for unmatched routes
|
|
1038
|
+
if (isRouteNotFoundError(error)) {
|
|
1039
|
+
callOnError(error, "routing", {
|
|
1040
|
+
request,
|
|
1041
|
+
url,
|
|
1042
|
+
env,
|
|
1043
|
+
handledByBoundary: true,
|
|
1044
|
+
});
|
|
1180
1045
|
|
|
1181
|
-
|
|
1182
|
-
|
|
1183
|
-
|
|
1184
|
-
|
|
1185
|
-
|
|
1186
|
-
|
|
1187
|
-
|
|
1046
|
+
const notFoundOption = router.notFound;
|
|
1047
|
+
const notFoundComponent =
|
|
1048
|
+
typeof notFoundOption === "function"
|
|
1049
|
+
? notFoundOption({ pathname: url.pathname })
|
|
1050
|
+
: (notFoundOption ?? createElement("h1", null, "Not Found"));
|
|
1051
|
+
|
|
1052
|
+
const notFoundSegment = {
|
|
1053
|
+
id: "notFound",
|
|
1054
|
+
namespace: "notFound",
|
|
1055
|
+
type: "route" as const,
|
|
1056
|
+
index: 0,
|
|
1057
|
+
component: notFoundComponent,
|
|
1058
|
+
params: {},
|
|
1188
1059
|
};
|
|
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
1060
|
|
|
1215
|
-
|
|
1216
|
-
|
|
1217
|
-
|
|
1218
|
-
|
|
1219
|
-
|
|
1220
|
-
|
|
1221
|
-
|
|
1222
|
-
|
|
1223
|
-
|
|
1224
|
-
|
|
1225
|
-
|
|
1226
|
-
|
|
1061
|
+
const payload: RscPayload = {
|
|
1062
|
+
metadata: {
|
|
1063
|
+
pathname: url.pathname,
|
|
1064
|
+
routerId: router.id,
|
|
1065
|
+
basename: router.basename,
|
|
1066
|
+
segments: [notFoundSegment],
|
|
1067
|
+
matched: [],
|
|
1068
|
+
diff: [],
|
|
1069
|
+
isPartial: false,
|
|
1070
|
+
rootLayout: router.rootLayout,
|
|
1071
|
+
handles: handleStore.stream(),
|
|
1072
|
+
version,
|
|
1073
|
+
themeConfig: router.themeConfig,
|
|
1074
|
+
warmupEnabled: router.warmupEnabled,
|
|
1075
|
+
initialTheme: requireRequestContext().theme,
|
|
1076
|
+
},
|
|
1227
1077
|
};
|
|
1228
1078
|
|
|
1229
|
-
const
|
|
1079
|
+
const rscStream = renderToReadableStream(payload, {
|
|
1080
|
+
onError: (error: unknown) => {
|
|
1081
|
+
callOnError(error, "rendering", { request, url, env });
|
|
1082
|
+
},
|
|
1083
|
+
});
|
|
1230
1084
|
|
|
1231
|
-
|
|
1232
|
-
|
|
1085
|
+
if (isRscRequest(request, url, isPartial)) {
|
|
1086
|
+
return createResponseWithMergedHeaders(rscStream, {
|
|
1087
|
+
status: 404,
|
|
1088
|
+
headers: {
|
|
1089
|
+
"content-type": "text/x-component;charset=utf-8",
|
|
1090
|
+
// Router identity for the client's pre-decode integrity check; a
|
|
1091
|
+
// same-app 404 matches and applies in place. See response-adapter.
|
|
1092
|
+
"X-RSC-Router-Id": router.id,
|
|
1093
|
+
},
|
|
1094
|
+
});
|
|
1233
1095
|
}
|
|
1234
|
-
const loaderPayload: LoaderPayload = { loaderResult: result };
|
|
1235
|
-
const rscStream =
|
|
1236
|
-
renderToReadableStream<LoaderPayload>(loaderPayload);
|
|
1237
1096
|
|
|
1238
|
-
|
|
1239
|
-
|
|
1097
|
+
const [ssrModule, streamMode] = await getSSRSetup(
|
|
1098
|
+
handlerCtx,
|
|
1099
|
+
request,
|
|
1100
|
+
env,
|
|
1101
|
+
url,
|
|
1102
|
+
requireRequestContext()._metricsStore,
|
|
1103
|
+
);
|
|
1104
|
+
const htmlStream = await ssrModule.renderHTML(rscStream, {
|
|
1105
|
+
nonce,
|
|
1106
|
+
streamMode,
|
|
1240
1107
|
});
|
|
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
1108
|
|
|
1249
|
-
|
|
1250
|
-
|
|
1251
|
-
|
|
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
|
-
|
|
1273
|
-
// ============================================================================
|
|
1274
|
-
// RSC RENDERING HANDLER (Navigation)
|
|
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 },
|
|
1109
|
+
return createResponseWithMergedHeaders(htmlStream, {
|
|
1110
|
+
status: 404,
|
|
1111
|
+
headers: { "content-type": "text/html;charset=utf-8" },
|
|
1305
1112
|
});
|
|
1306
1113
|
}
|
|
1307
1114
|
|
|
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 },
|
|
1115
|
+
// Report unhandled errors
|
|
1116
|
+
callOnError(error, "routing", {
|
|
1117
|
+
request,
|
|
1118
|
+
url,
|
|
1119
|
+
env,
|
|
1120
|
+
handledByBoundary: false,
|
|
1358
1121
|
});
|
|
1122
|
+
console.error(`[RSC] Error:`, error);
|
|
1123
|
+
throw error;
|
|
1359
1124
|
}
|
|
1125
|
+
};
|
|
1360
1126
|
|
|
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" } },
|
|
1127
|
+
// Wrap the render path in a renderStartMs timeout
|
|
1128
|
+
const executeRender = async (): Promise<Response> => {
|
|
1129
|
+
if (routeMiddleware.length > 0) {
|
|
1130
|
+
const mwResponse = await executeMiddleware(
|
|
1131
|
+
buildRouteMiddlewareEntries<TEnv>(routeMiddleware),
|
|
1132
|
+
request,
|
|
1133
|
+
env,
|
|
1134
|
+
variables,
|
|
1135
|
+
renderHandler,
|
|
1136
|
+
routeReverse,
|
|
1387
1137
|
);
|
|
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
1138
|
|
|
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)}`);
|
|
1139
|
+
if (isPartial || actionContinuation) {
|
|
1140
|
+
const intercepted = interceptRedirectForPartial(
|
|
1141
|
+
mwResponse,
|
|
1142
|
+
createRedirectFlightResponse,
|
|
1143
|
+
);
|
|
1144
|
+
if (intercepted) return intercepted;
|
|
1145
|
+
}
|
|
1433
1146
|
|
|
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;
|
|
1147
|
+
return finalizeResponse(mwResponse);
|
|
1442
1148
|
}
|
|
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
1149
|
|
|
1465
|
-
|
|
1466
|
-
const htmlHeaders: Record<string, string> = {
|
|
1467
|
-
"content-type": "text/html;charset=utf-8",
|
|
1150
|
+
return renderHandler();
|
|
1468
1151
|
};
|
|
1469
|
-
if (fullTiming) {
|
|
1470
|
-
htmlHeaders["Server-Timing"] = fullTiming;
|
|
1471
|
-
}
|
|
1472
1152
|
|
|
1473
|
-
|
|
1474
|
-
|
|
1475
|
-
|
|
1153
|
+
const renderOutcome = await withTimeout(
|
|
1154
|
+
executeRender(),
|
|
1155
|
+
router.timeouts.renderStartMs,
|
|
1156
|
+
"render-start",
|
|
1157
|
+
);
|
|
1158
|
+
if (renderOutcome.timedOut) {
|
|
1159
|
+
return handleTimeoutResponse(
|
|
1160
|
+
request,
|
|
1161
|
+
env,
|
|
1162
|
+
url,
|
|
1163
|
+
"render-start",
|
|
1164
|
+
renderOutcome.durationMs,
|
|
1165
|
+
routeKey,
|
|
1166
|
+
);
|
|
1167
|
+
}
|
|
1168
|
+
return renderOutcome.result;
|
|
1476
1169
|
}
|
|
1477
1170
|
}
|