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