@rangojs/router 0.0.0-experimental.18 → 0.0.0-experimental.1878aa49
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/{CLAUDE.md → AGENTS.md} +4 -0
- package/README.md +91 -19
- 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 +348 -80
- 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 +883 -444
- 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/dist/vite/package-resolution.js +112 -0
- package/dist/vite/package-resolution.js.map +1 -0
- package/dist/vite/virtual-entries.d.ts +25 -0
- package/dist/vite/virtual-entries.d.ts.map +1 -0
- package/dist/vite/virtual-entries.js +110 -0
- package/dist/vite/virtual-entries.js.map +1 -0
- package/package.json +17 -16
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +32 -0
- package/skills/caching/SKILL.md +49 -8
- package/skills/document-cache/SKILL.md +2 -2
- package/skills/hooks/SKILL.md +34 -32
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +79 -0
- package/skills/layout/SKILL.md +62 -2
- package/skills/links/SKILL.md +3 -1
- package/skills/loader/SKILL.md +166 -23
- package/skills/middleware/SKILL.md +83 -0
- package/skills/parallel/SKILL.md +183 -2
- package/skills/prerender/SKILL.md +189 -19
- package/skills/rango/SKILL.md +0 -1
- package/skills/route/SKILL.md +76 -5
- package/skills/router-setup/SKILL.md +172 -5
- package/skills/typesafety/SKILL.md +35 -23
- package/src/__internal.ts +93 -1
- package/src/bin/rango.ts +56 -19
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +30 -27
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +1 -1
- package/src/browser/link-interceptor.ts +4 -3
- package/src/browser/merge-segment-loaders.ts +9 -2
- package/src/browser/navigation-bridge.ts +160 -31
- package/src/browser/navigation-client.ts +141 -88
- package/src/browser/navigation-store.ts +44 -39
- package/src/browser/navigation-transaction.ts +55 -214
- package/src/browser/partial-update.ts +181 -164
- package/src/browser/prefetch/cache.ts +206 -0
- package/src/browser/prefetch/fetch.ts +150 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +160 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/react/Link.tsx +97 -36
- package/src/browser/react/NavigationProvider.tsx +49 -5
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/index.ts +2 -6
- package/src/browser/react/location-state-shared.ts +1 -1
- package/src/browser/react/location-state.ts +2 -0
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/use-action.ts +9 -1
- package/src/browser/react/use-handle.ts +3 -25
- package/src/browser/react/use-params.ts +2 -4
- package/src/browser/react/use-pathname.ts +2 -3
- package/src/browser/react/use-router.ts +22 -9
- package/src/browser/react/use-search-params.ts +2 -1
- package/src/browser/react/use-segments.ts +7 -60
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +154 -64
- package/src/browser/scroll-restoration.ts +37 -37
- package/src/browser/segment-reconciler.ts +6 -1
- package/src/browser/server-action-bridge.ts +463 -426
- package/src/browser/types.ts +61 -36
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +8 -0
- package/src/build/generate-route-types.ts +7 -0
- package/src/build/route-trie.ts +19 -3
- package/src/build/route-types/codegen.ts +13 -4
- package/src/build/route-types/include-resolution.ts +21 -1
- package/src/build/route-types/per-module-writer.ts +15 -3
- package/src/build/route-types/router-processing.ts +391 -90
- package/src/build/runtime-discovery.ts +13 -1
- 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 +136 -96
- package/src/cache/cache-scope.ts +119 -80
- package/src/cache/cf/cf-cache-store.ts +462 -15
- package/src/cache/cf/index.ts +5 -1
- package/src/cache/document-cache.ts +88 -53
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/index.ts +1 -0
- package/src/cache/memory-segment-store.ts +18 -7
- package/src/cache/profile-registry.ts +43 -8
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +101 -112
- package/src/cache/taint.ts +81 -0
- package/src/client.rsc.tsx +2 -1
- package/src/client.tsx +54 -76
- package/src/context-var.ts +72 -2
- package/src/debug.ts +2 -2
- package/src/errors.ts +6 -1
- package/src/handle.ts +1 -1
- package/src/handles/MetaTags.tsx +5 -2
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/host/cookie-handler.ts +8 -3
- package/src/host/index.ts +0 -3
- package/src/host/router.ts +14 -1
- package/src/href-client.ts +3 -1
- package/src/index.rsc.ts +24 -23
- package/src/index.ts +45 -52
- package/src/loader.rsc.ts +12 -4
- package/src/loader.ts +8 -0
- package/src/prerender/store.ts +60 -18
- package/src/prerender.ts +76 -18
- package/src/reverse.ts +11 -7
- package/src/root-error-boundary.tsx +30 -26
- package/src/route-definition/dsl-helpers.ts +41 -13
- package/src/route-definition/helpers-types.ts +6 -5
- package/src/route-definition/index.ts +0 -3
- package/src/route-definition/redirect.ts +24 -4
- package/src/route-map-builder.ts +45 -3
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +7 -0
- package/src/router/content-negotiation.ts +1 -1
- package/src/router/debug-manifest.ts +16 -3
- package/src/router/find-match.ts +4 -2
- package/src/router/handler-context.ts +159 -34
- package/src/router/intercept-resolution.ts +17 -8
- package/src/router/lazy-includes.ts +8 -1
- package/src/router/loader-resolution.ts +11 -3
- package/src/router/logging.ts +104 -4
- package/src/router/manifest.ts +45 -7
- package/src/router/match-api.ts +62 -8
- package/src/router/match-context.ts +3 -0
- package/src/router/match-handlers.ts +185 -11
- package/src/router/match-middleware/background-revalidation.ts +94 -86
- package/src/router/match-middleware/cache-lookup.ts +138 -13
- package/src/router/match-middleware/cache-store.ts +55 -10
- package/src/router/match-middleware/intercept-resolution.ts +9 -7
- package/src/router/match-middleware/segment-resolution.ts +8 -5
- package/src/router/match-pipelines.ts +8 -43
- package/src/router/match-result.ts +22 -15
- package/src/router/metrics.ts +238 -13
- package/src/router/middleware-types.ts +60 -12
- package/src/router/middleware.ts +214 -42
- package/src/router/pattern-matching.ts +61 -10
- package/src/router/prerender-match.ts +40 -8
- package/src/router/preview-match.ts +7 -1
- package/src/router/revalidation.ts +87 -8
- package/src/router/router-context.ts +20 -0
- package/src/router/router-interfaces.ts +73 -5
- package/src/router/router-options.ts +250 -19
- package/src/router/segment-resolution/fresh.ts +277 -45
- package/src/router/segment-resolution/helpers.ts +49 -25
- package/src/router/segment-resolution/loader-cache.ts +38 -146
- package/src/router/segment-resolution/revalidation.ts +619 -275
- package/src/router/segment-wrappers.ts +5 -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 +20 -2
- package/src/router/types.ts +8 -1
- package/src/router.ts +230 -25
- package/src/rsc/handler-context.ts +11 -0
- package/src/rsc/handler.ts +492 -96
- package/src/rsc/helpers.ts +25 -16
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +84 -42
- package/src/rsc/manifest-init.ts +5 -1
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +249 -19
- package/src/rsc/response-route-handler.ts +37 -26
- package/src/rsc/rsc-rendering.ts +40 -41
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +150 -68
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +18 -1
- package/src/search-params.ts +38 -23
- package/src/segment-system.tsx +140 -4
- package/src/server/context.ts +187 -20
- package/src/server/fetchable-loader-store.ts +11 -6
- package/src/server/handle-store.ts +66 -9
- package/src/server/loader-registry.ts +20 -54
- package/src/server/request-context.ts +188 -41
- package/src/server.ts +6 -0
- package/src/ssr/index.tsx +66 -26
- package/src/static-handler.ts +7 -0
- package/src/theme/ThemeProvider.tsx +6 -1
- package/src/theme/index.ts +4 -18
- package/src/theme/theme-context.ts +1 -28
- package/src/theme/theme-script.ts +2 -1
- package/src/types/cache-types.ts +9 -4
- package/src/types/error-types.ts +3 -0
- package/src/types/global-namespace.ts +9 -0
- package/src/types/handler-context.ts +167 -55
- package/src/types/loader-types.ts +11 -5
- package/src/types/route-config.ts +17 -8
- package/src/types/route-entry.ts +35 -0
- package/src/types/segments.ts +2 -5
- package/src/urls/include-helper.ts +49 -8
- package/src/urls/index.ts +1 -0
- package/src/urls/path-helper-types.ts +30 -12
- package/src/urls/path-helper.ts +18 -3
- package/src/urls/pattern-types.ts +33 -1
- package/src/urls/response-types.ts +27 -2
- package/src/urls/type-extraction.ts +23 -15
- package/src/use-loader.tsx +12 -4
- package/src/vite/discovery/bundle-postprocess.ts +32 -52
- package/src/vite/discovery/discover-routers.ts +53 -23
- package/src/vite/discovery/prerender-collection.ts +58 -41
- package/src/vite/discovery/route-types-writer.ts +7 -7
- package/src/vite/discovery/state.ts +4 -9
- package/src/vite/discovery/virtual-module-codegen.ts +5 -2
- package/src/vite/index.ts +4 -3
- package/src/vite/plugin-types.ts +0 -83
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +3 -3
- package/src/vite/plugins/expose-action-id.ts +1 -3
- package/src/vite/plugins/performance-tracks.ts +88 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +91 -3
- package/src/vite/plugins/version-plugin.ts +13 -1
- package/src/vite/rango.ts +172 -210
- package/src/vite/router-discovery.ts +115 -44
- package/src/vite/utils/banner.ts +3 -3
- package/src/vite/utils/prerender-utils.ts +81 -0
- package/src/vite/utils/shared-utils.ts +6 -3
- package/skills/testing/SKILL.md +0 -226
- package/src/browser/prefetch-cache.ts +0 -48
- package/src/browser/prefetch-fetch.ts +0 -109
- package/src/browser/prefetch-queue.ts +0 -81
- package/src/browser/request-controller.ts +0 -164
- package/src/route-definition/route-function.ts +0 -119
- package/src/router.gen.ts +0 -6
- package/src/static-handler.gen.ts +0 -5
- package/src/urls.gen.ts +0 -8
- /package/src/browser/{prefetch-observer.ts → prefetch/observer.ts} +0 -0
|
@@ -9,7 +9,6 @@ import {
|
|
|
9
9
|
reconcileSegments,
|
|
10
10
|
reconcileErrorSegments,
|
|
11
11
|
} from "./segment-reconciler.js";
|
|
12
|
-
import { classifyActionResponse } from "./action-response-classifier.js";
|
|
13
12
|
import { startTransition } from "react";
|
|
14
13
|
import type { EventController } from "./event-controller.js";
|
|
15
14
|
import {
|
|
@@ -22,6 +21,15 @@ import {
|
|
|
22
21
|
isBrowserDebugEnabled,
|
|
23
22
|
startBrowserTransaction,
|
|
24
23
|
} from "./logging.js";
|
|
24
|
+
import { validateRedirectOrigin } from "./validate-redirect-origin.js";
|
|
25
|
+
import {
|
|
26
|
+
extractRscHeaderUrl,
|
|
27
|
+
emptyResponse,
|
|
28
|
+
teeWithCompletion,
|
|
29
|
+
} from "./response-adapter.js";
|
|
30
|
+
import { mergeLocationState } from "./history-state.js";
|
|
31
|
+
import { classifyActionOutcome } from "./action-coordinator.js";
|
|
32
|
+
import { getAppVersion } from "./app-version.js";
|
|
25
33
|
|
|
26
34
|
// Polyfill Symbol.dispose/asyncDispose for Safari and older browsers
|
|
27
35
|
if (typeof Symbol.dispose === "undefined") {
|
|
@@ -36,8 +44,6 @@ if (typeof Symbol.asyncDispose === "undefined") {
|
|
|
36
44
|
*/
|
|
37
45
|
export interface ServerActionBridgeConfigWithController extends ServerActionBridgeConfig {
|
|
38
46
|
eventController: EventController;
|
|
39
|
-
/** RSC version from initial payload metadata */
|
|
40
|
-
version?: string;
|
|
41
47
|
/** Callback to trigger SPA navigation (for action redirects) */
|
|
42
48
|
onNavigate?: (
|
|
43
49
|
url: string,
|
|
@@ -68,7 +74,6 @@ export function createServerActionBridge(
|
|
|
68
74
|
deps,
|
|
69
75
|
onUpdate,
|
|
70
76
|
renderSegments,
|
|
71
|
-
version,
|
|
72
77
|
onNavigate,
|
|
73
78
|
} = config;
|
|
74
79
|
|
|
@@ -79,7 +84,7 @@ export function createServerActionBridge(
|
|
|
79
84
|
client,
|
|
80
85
|
onUpdate,
|
|
81
86
|
renderSegments,
|
|
82
|
-
|
|
87
|
+
getVersion: getAppVersion,
|
|
83
88
|
});
|
|
84
89
|
|
|
85
90
|
/**
|
|
@@ -92,27 +97,31 @@ export function createServerActionBridge(
|
|
|
92
97
|
interceptSourceUrl?: string | null;
|
|
93
98
|
}): Promise<void> {
|
|
94
99
|
const src = opts?.interceptSourceUrl ?? null;
|
|
95
|
-
|
|
100
|
+
const navTx = createNavigationTransaction(
|
|
96
101
|
store,
|
|
97
102
|
eventController,
|
|
98
103
|
window.location.href,
|
|
99
104
|
{ replace: true, skipLoadingState: true },
|
|
100
105
|
);
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
106
|
+
try {
|
|
107
|
+
await fetchPartialUpdate(
|
|
108
|
+
window.location.href,
|
|
109
|
+
opts?.segments ?? [],
|
|
110
|
+
false,
|
|
111
|
+
navTx.handle.signal,
|
|
112
|
+
navTx.with({
|
|
113
|
+
url: window.location.href,
|
|
114
|
+
storeOnly: true,
|
|
115
|
+
...(src ? { intercept: true, interceptSourceUrl: src } : {}),
|
|
116
|
+
}),
|
|
117
|
+
{
|
|
118
|
+
type: "action" as const,
|
|
119
|
+
...(src ? { interceptSourceUrl: src } : {}),
|
|
120
|
+
},
|
|
121
|
+
);
|
|
122
|
+
} finally {
|
|
123
|
+
navTx[Symbol.dispose]();
|
|
124
|
+
}
|
|
116
125
|
}
|
|
117
126
|
|
|
118
127
|
/**
|
|
@@ -130,406 +139,499 @@ export function createServerActionBridge(
|
|
|
130
139
|
log("action start", { id, argsCount: args.length });
|
|
131
140
|
|
|
132
141
|
// Start action in event controller - handles lifecycle tracking
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
142
|
+
const handle = eventController.startAction(id, args);
|
|
143
|
+
try {
|
|
144
|
+
const segmentState = store.getSegmentState();
|
|
145
|
+
|
|
146
|
+
// Mark cache as stale immediately when action starts
|
|
147
|
+
// This ensures SWR pattern kicks in if user navigates away during action
|
|
148
|
+
store.markCacheAsStaleAndBroadcast();
|
|
149
|
+
|
|
150
|
+
// Create temporary references for serialization
|
|
151
|
+
const temporaryReferences = deps.createTemporaryReferenceSet();
|
|
152
|
+
|
|
153
|
+
// Capture URL pathname at action start to detect navigation during action
|
|
154
|
+
// Must use window.location (not store.path) because intercepts change URL
|
|
155
|
+
// without changing store.path (e.g., /kanban -> /kanban/card/1)
|
|
156
|
+
const actionStartPathname = window.location.pathname;
|
|
157
|
+
|
|
158
|
+
// Build action request URL with current segments
|
|
159
|
+
const url = new URL(window.location.href);
|
|
160
|
+
url.searchParams.set("_rsc_action", id);
|
|
161
|
+
url.searchParams.set(
|
|
162
|
+
"_rsc_segments",
|
|
163
|
+
segmentState.currentSegmentIds.join(","),
|
|
164
|
+
);
|
|
165
|
+
// Add version param for version mismatch detection
|
|
166
|
+
const version = getAppVersion();
|
|
167
|
+
if (version) {
|
|
168
|
+
url.searchParams.set("_rsc_v", version);
|
|
169
|
+
}
|
|
170
|
+
// Add router ID for app switch detection
|
|
171
|
+
const rid = store.getRouterId?.();
|
|
172
|
+
if (rid) {
|
|
173
|
+
url.searchParams.set("_rsc_rid", rid);
|
|
174
|
+
}
|
|
136
175
|
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
store.markCacheAsStaleAndBroadcast();
|
|
176
|
+
// Encode arguments
|
|
177
|
+
const encodedBody = await deps.encodeReply(args, { temporaryReferences });
|
|
140
178
|
|
|
141
|
-
|
|
142
|
-
|
|
179
|
+
log("sending action request", {
|
|
180
|
+
url: url.href,
|
|
181
|
+
bodyType: typeof encodedBody,
|
|
182
|
+
isFormData: encodedBody instanceof FormData,
|
|
183
|
+
segmentCount: segmentState.currentSegmentIds.length,
|
|
184
|
+
});
|
|
143
185
|
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
186
|
+
// Track when the stream completes
|
|
187
|
+
let resolveStreamComplete: () => void;
|
|
188
|
+
const streamComplete = new Promise<void>((resolve) => {
|
|
189
|
+
resolveStreamComplete = resolve;
|
|
190
|
+
});
|
|
148
191
|
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
192
|
+
// Get intercept source URL if in intercept context
|
|
193
|
+
const interceptSourceUrl = store.getInterceptSourceUrl();
|
|
194
|
+
|
|
195
|
+
// Track streaming token - will be set when response arrives
|
|
196
|
+
let streamingToken: { end(): void } | null = null;
|
|
197
|
+
|
|
198
|
+
// Use a dedicated abort controller for the fetch so we can cancel network
|
|
199
|
+
// I/O without disrupting the Flight stream once the response has arrived.
|
|
200
|
+
// Aborting a response mid-stream causes React's Flight decoder to throw
|
|
201
|
+
// asynchronous unhandled errors (BodyStreamBuffer was aborted).
|
|
202
|
+
const fetchAbort = new AbortController();
|
|
203
|
+
const onHandleAbort = () => fetchAbort.abort();
|
|
204
|
+
handle.signal.addEventListener("abort", onHandleAbort, { once: true });
|
|
205
|
+
|
|
206
|
+
// Send action request with stream tracking
|
|
207
|
+
const responsePromise = fetch(url, {
|
|
208
|
+
method: "POST",
|
|
209
|
+
headers: {
|
|
210
|
+
"rsc-action": id,
|
|
211
|
+
"X-RSC-Router-Client-Path": segmentState.currentUrl,
|
|
212
|
+
...(tx && { "X-RSC-Router-Request-Id": tx.requestId }),
|
|
213
|
+
...(interceptSourceUrl && {
|
|
214
|
+
"X-RSC-Router-Intercept-Source": interceptSourceUrl,
|
|
215
|
+
}),
|
|
216
|
+
},
|
|
217
|
+
body: encodedBody,
|
|
218
|
+
signal: fetchAbort.signal,
|
|
219
|
+
}).then(async (response) => {
|
|
220
|
+
// Response arrived — disconnect fetch abort from handle abort so
|
|
221
|
+
// abortAllActions() doesn't disrupt the in-progress Flight stream.
|
|
222
|
+
handle.signal.removeEventListener("abort", onHandleAbort);
|
|
223
|
+
|
|
224
|
+
// Check for version mismatch - server wants us to reload
|
|
225
|
+
const reload = extractRscHeaderUrl(response, "X-RSC-Reload");
|
|
226
|
+
if (reload === "blocked") {
|
|
227
|
+
resolveStreamComplete();
|
|
228
|
+
return emptyResponse();
|
|
229
|
+
}
|
|
230
|
+
if (reload) {
|
|
231
|
+
log("version mismatch on action, reloading", {
|
|
232
|
+
reloadUrl: reload.url,
|
|
233
|
+
});
|
|
234
|
+
window.location.href = reload.url;
|
|
235
|
+
return new Promise<Response>(() => {});
|
|
236
|
+
}
|
|
160
237
|
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
// Get intercept source URL if in intercept context
|
|
178
|
-
const interceptSourceUrl = store.getInterceptSourceUrl();
|
|
179
|
-
|
|
180
|
-
// Track streaming token - will be set when response arrives
|
|
181
|
-
let streamingToken: { end(): void } | null = null;
|
|
182
|
-
|
|
183
|
-
// Send action request with stream tracking
|
|
184
|
-
const responsePromise = fetch(url, {
|
|
185
|
-
method: "POST",
|
|
186
|
-
headers: {
|
|
187
|
-
"rsc-action": id,
|
|
188
|
-
"X-RSC-Router-Client-Path": segmentState.currentUrl,
|
|
189
|
-
...(tx && { "X-RSC-Router-Request-Id": tx.requestId }),
|
|
190
|
-
// Send intercept source URL so server can maintain intercept context
|
|
191
|
-
...(interceptSourceUrl && {
|
|
192
|
-
"X-RSC-Router-Intercept-Source": interceptSourceUrl,
|
|
193
|
-
}),
|
|
194
|
-
},
|
|
195
|
-
body: encodedBody,
|
|
196
|
-
}).then(async (response) => {
|
|
197
|
-
// Check for version mismatch - server wants us to reload
|
|
198
|
-
const reloadUrl = response.headers.get("X-RSC-Reload");
|
|
199
|
-
if (reloadUrl) {
|
|
200
|
-
// Validate origin to prevent open redirect via crafted headers
|
|
201
|
-
try {
|
|
202
|
-
const target = new URL(reloadUrl, window.location.origin);
|
|
203
|
-
if (target.origin !== window.location.origin) {
|
|
204
|
-
throw new Error(
|
|
205
|
-
`X-RSC-Reload blocked: origin mismatch (${target.origin})`,
|
|
206
|
-
);
|
|
238
|
+
// Simple redirect from action (no state, no RSC payload).
|
|
239
|
+
// Short-circuits before createFromFetch — no Flight deserialization needed.
|
|
240
|
+
// Check handle.signal.aborted to avoid redirecting from a stale action
|
|
241
|
+
// when the user has already navigated away.
|
|
242
|
+
const redirect = extractRscHeaderUrl(response, "X-RSC-Redirect");
|
|
243
|
+
if (redirect && redirect !== "blocked" && !handle.signal.aborted) {
|
|
244
|
+
log("action simple redirect", { url: redirect.url });
|
|
245
|
+
handle.complete(undefined);
|
|
246
|
+
if (onNavigate) {
|
|
247
|
+
await onNavigate(redirect.url, {
|
|
248
|
+
replace: true,
|
|
249
|
+
_skipCache: true,
|
|
250
|
+
});
|
|
251
|
+
} else {
|
|
252
|
+
window.location.href = redirect.url;
|
|
207
253
|
}
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
254
|
+
return new Promise<Response>(() => {});
|
|
255
|
+
}
|
|
256
|
+
if (redirect === "blocked") {
|
|
257
|
+
resolveStreamComplete();
|
|
258
|
+
return emptyResponse();
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
// Start streaming immediately when response arrives
|
|
262
|
+
if (!handle.signal.aborted) {
|
|
263
|
+
streamingToken = handle.startStreaming();
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
return teeWithCompletion(response, () => {
|
|
267
|
+
log("stream complete");
|
|
268
|
+
streamingToken?.end();
|
|
269
|
+
resolveStreamComplete();
|
|
270
|
+
});
|
|
271
|
+
});
|
|
272
|
+
|
|
273
|
+
// Deserialize response (MUST use same temporaryReferences)
|
|
274
|
+
let payload: RscPayload;
|
|
275
|
+
try {
|
|
276
|
+
payload = await deps.createFromFetch<RscPayload>(responsePromise, {
|
|
277
|
+
temporaryReferences,
|
|
278
|
+
});
|
|
279
|
+
} catch (error) {
|
|
280
|
+
// Clean up streaming token on error (may be null if fetch failed before .then() ran)
|
|
281
|
+
// The token is assigned in .then() callback which runs before this catch block,
|
|
282
|
+
// but TypeScript doesn't track cross-async assignments, so use type assertion
|
|
283
|
+
(streamingToken as { end(): void } | null)?.end();
|
|
284
|
+
// resolveStreamComplete is assigned in the Promise constructor so it's safe to call
|
|
285
|
+
resolveStreamComplete!();
|
|
286
|
+
|
|
287
|
+
// Silently swallow abort errors — the action was intentionally cancelled
|
|
288
|
+
// (e.g., user navigated away or abortAllActions was called).
|
|
289
|
+
// Return undefined instead of throwing to avoid surfacing as a page error.
|
|
290
|
+
// Check both DOMException AbortError and stream-level abort messages
|
|
291
|
+
// (BodyStreamBuffer was aborted) that propagate from the aborted fetch.
|
|
292
|
+
if (handle.signal.aborted) {
|
|
293
|
+
return undefined;
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
// Convert network-level errors to NetworkError for proper handling
|
|
297
|
+
const networkError = toNetworkError(error, {
|
|
298
|
+
url: url.toString(),
|
|
299
|
+
operation: "action",
|
|
300
|
+
});
|
|
301
|
+
if (networkError) {
|
|
302
|
+
handle.fail(networkError);
|
|
303
|
+
emitNetworkError(onUpdate, networkError, segmentState.currentUrl);
|
|
304
|
+
throw networkError;
|
|
211
305
|
}
|
|
212
|
-
|
|
213
|
-
window.location.href = reloadUrl;
|
|
214
|
-
// Return a never-resolving promise to prevent further processing
|
|
215
|
-
return new Promise<Response>(() => {});
|
|
306
|
+
throw error;
|
|
216
307
|
}
|
|
217
308
|
|
|
218
|
-
|
|
219
|
-
|
|
309
|
+
log("action response received", {
|
|
310
|
+
isPartial: payload.metadata?.isPartial,
|
|
311
|
+
isError: payload.metadata?.isError,
|
|
312
|
+
matchedCount: payload.metadata?.matched?.length ?? 0,
|
|
313
|
+
diffCount: payload.metadata?.diff?.length ?? 0,
|
|
314
|
+
});
|
|
315
|
+
// Guard: if the action was aborted while streaming (e.g., user navigated
|
|
316
|
+
// away or abortAllActions fired), bail out before any reconcile/render/cache
|
|
317
|
+
// writes to avoid overwriting the current UI with stale action results.
|
|
318
|
+
if (handle.signal.aborted) {
|
|
319
|
+
log("action aborted after response, skipping reconciliation");
|
|
320
|
+
return undefined;
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
// Process response
|
|
324
|
+
const { metadata, returnValue } = payload;
|
|
325
|
+
|
|
326
|
+
// Handle action redirect: server converted the redirect to a Flight payload
|
|
327
|
+
// so we can perform SPA navigation instead of a full page reload.
|
|
220
328
|
// Check handle.signal.aborted to avoid redirecting from a stale action
|
|
221
329
|
// when the user has already navigated away.
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
330
|
+
if (metadata?.redirect && !handle.signal.aborted) {
|
|
331
|
+
const redirectUrl = validateRedirectOrigin(
|
|
332
|
+
metadata.redirect.url,
|
|
333
|
+
window.location.origin,
|
|
334
|
+
);
|
|
335
|
+
if (!redirectUrl) {
|
|
336
|
+
log("blocked action redirect payload", {
|
|
337
|
+
url: metadata.redirect.url,
|
|
227
338
|
});
|
|
339
|
+
handle.complete(returnValue?.data);
|
|
340
|
+
return returnValue?.data;
|
|
228
341
|
}
|
|
229
|
-
|
|
342
|
+
const redirectState = metadata.locationState;
|
|
343
|
+
log("action redirect", { url: redirectUrl });
|
|
344
|
+
handle.complete(returnValue?.data);
|
|
230
345
|
if (onNavigate) {
|
|
231
|
-
await onNavigate(
|
|
346
|
+
await onNavigate(redirectUrl, {
|
|
347
|
+
state: redirectState,
|
|
232
348
|
replace: true,
|
|
233
349
|
_skipCache: true,
|
|
234
350
|
});
|
|
235
351
|
} else {
|
|
236
|
-
window.location.href =
|
|
352
|
+
window.location.href = redirectUrl;
|
|
237
353
|
}
|
|
238
|
-
return
|
|
354
|
+
return returnValue?.data;
|
|
239
355
|
}
|
|
240
356
|
|
|
241
|
-
//
|
|
242
|
-
|
|
243
|
-
|
|
357
|
+
// Bail out if the action was aborted after deserialization (e.g. user
|
|
358
|
+
// navigated away or abortAllActions was called while the Flight stream
|
|
359
|
+
// was being consumed). Without this check the code below would mutate
|
|
360
|
+
// the store / UI for a stale action.
|
|
361
|
+
if (handle.signal.aborted) {
|
|
362
|
+
log("action aborted after deserialization, skipping mutations");
|
|
363
|
+
return returnValue?.data;
|
|
244
364
|
}
|
|
245
365
|
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
366
|
+
const { matched, diff, segments, isPartial, isError } = metadata || {};
|
|
367
|
+
|
|
368
|
+
// Log action result
|
|
369
|
+
if (returnValue && !returnValue.ok) {
|
|
370
|
+
console.error(`[Browser] Action failed:`, returnValue.data);
|
|
251
371
|
}
|
|
252
372
|
|
|
253
|
-
//
|
|
254
|
-
|
|
373
|
+
// Handle error responses with error boundary UI
|
|
374
|
+
if (isError && isPartial && segments && diff) {
|
|
375
|
+
log("processing error boundary response");
|
|
255
376
|
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
while (true) {
|
|
261
|
-
const { done } = await reader.read();
|
|
262
|
-
if (done) break;
|
|
263
|
-
}
|
|
264
|
-
} finally {
|
|
265
|
-
reader.releaseLock();
|
|
266
|
-
log("stream complete");
|
|
267
|
-
streamingToken?.end();
|
|
268
|
-
resolveStreamComplete();
|
|
377
|
+
// Fail current handle BEFORE aborting all actions so the event controller
|
|
378
|
+
// records the error state (abortAllActions clears inflight entries)
|
|
379
|
+
if (returnValue && !returnValue.ok) {
|
|
380
|
+
handle.fail(returnValue.data);
|
|
269
381
|
}
|
|
270
|
-
})().catch((error) => {
|
|
271
|
-
console.error("[STREAMING] Error reading tracking stream:", error);
|
|
272
|
-
streamingToken?.end();
|
|
273
|
-
});
|
|
274
382
|
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
status: response.status,
|
|
279
|
-
statusText: response.statusText,
|
|
280
|
-
});
|
|
281
|
-
});
|
|
383
|
+
// Abort all other pending action requests - error takes precedence
|
|
384
|
+
// This prevents other actions from completing and overwriting the error UI
|
|
385
|
+
eventController.abortAllActions();
|
|
282
386
|
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
(streamingToken as { end(): void } | null)?.end();
|
|
294
|
-
// resolveStreamComplete is assigned in the Promise constructor so it's safe to call
|
|
295
|
-
resolveStreamComplete!();
|
|
296
|
-
|
|
297
|
-
// Convert network-level errors to NetworkError for proper handling
|
|
298
|
-
const networkError = toNetworkError(error, {
|
|
299
|
-
url: url.toString(),
|
|
300
|
-
operation: "action",
|
|
301
|
-
});
|
|
302
|
-
if (networkError) {
|
|
303
|
-
handle.fail(networkError);
|
|
304
|
-
emitNetworkError(onUpdate, networkError, segmentState.currentUrl);
|
|
305
|
-
throw networkError;
|
|
306
|
-
}
|
|
307
|
-
throw error;
|
|
308
|
-
}
|
|
387
|
+
// Clear concurrent action tracking - no consolidation needed when showing error
|
|
388
|
+
handle.clearConsolidation();
|
|
389
|
+
|
|
390
|
+
// Get current page's cached segments
|
|
391
|
+
const currentKey = store.getHistoryKey();
|
|
392
|
+
const cached = store.getCachedSegments(currentKey);
|
|
393
|
+
const cachedSegments = cached?.segments || [];
|
|
394
|
+
|
|
395
|
+
// Reconcile error segments with cached tree
|
|
396
|
+
const errorResult = reconcileErrorSegments(cachedSegments, segments);
|
|
309
397
|
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
// Process response
|
|
318
|
-
const { metadata, returnValue } = payload;
|
|
319
|
-
|
|
320
|
-
// Handle action redirect: server converted the redirect to a Flight payload
|
|
321
|
-
// so we can perform SPA navigation instead of a full page reload.
|
|
322
|
-
// Check handle.signal.aborted to avoid redirecting from a stale action
|
|
323
|
-
// when the user has already navigated away.
|
|
324
|
-
if (metadata?.redirect && !handle.signal.aborted) {
|
|
325
|
-
const { url: redirectUrl } = metadata.redirect;
|
|
326
|
-
const redirectState = metadata.locationState;
|
|
327
|
-
console.log(`[Browser] Action redirect to ${redirectUrl}`);
|
|
328
|
-
handle.complete(returnValue?.data);
|
|
329
|
-
if (onNavigate) {
|
|
330
|
-
await onNavigate(redirectUrl, {
|
|
331
|
-
state: redirectState,
|
|
332
|
-
replace: true,
|
|
333
|
-
_skipCache: true,
|
|
398
|
+
// Render the full tree with error segment merged with parent layouts
|
|
399
|
+
const errorTree = await renderSegments(errorResult.mainSegments, {
|
|
400
|
+
isAction: true,
|
|
401
|
+
interceptSegments:
|
|
402
|
+
errorResult.interceptSegments.length > 0
|
|
403
|
+
? errorResult.interceptSegments
|
|
404
|
+
: undefined,
|
|
334
405
|
});
|
|
335
|
-
} else {
|
|
336
|
-
window.location.href = redirectUrl;
|
|
337
|
-
}
|
|
338
|
-
return returnValue?.data;
|
|
339
|
-
}
|
|
340
406
|
|
|
341
|
-
|
|
407
|
+
// Re-check route stability after async renderSegments — user may have
|
|
408
|
+
// navigated away while the error tree was being prepared.
|
|
409
|
+
if (window.location.pathname !== actionStartPathname) {
|
|
410
|
+
log("user navigated during error render, skipping");
|
|
411
|
+
if (returnValue && !returnValue.ok) {
|
|
412
|
+
throw returnValue.data;
|
|
413
|
+
}
|
|
414
|
+
handle.complete(undefined);
|
|
415
|
+
return undefined;
|
|
416
|
+
}
|
|
417
|
+
const currentKeyNow = store.getHistoryKey();
|
|
418
|
+
if (currentKeyNow !== currentKey) {
|
|
419
|
+
log("history key changed during error render, skipping cache update");
|
|
420
|
+
if (returnValue && !returnValue.ok) {
|
|
421
|
+
throw returnValue.data;
|
|
422
|
+
}
|
|
423
|
+
handle.complete(undefined);
|
|
424
|
+
return undefined;
|
|
425
|
+
}
|
|
342
426
|
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
427
|
+
// Update UI with error boundary
|
|
428
|
+
startTransition(() => {
|
|
429
|
+
onUpdate({ root: errorTree, metadata: metadata! });
|
|
430
|
+
});
|
|
431
|
+
|
|
432
|
+
// Update segment tracking to exclude error segment IDs
|
|
433
|
+
const errorSegmentIds = new Set(diff);
|
|
434
|
+
const segmentIdsAfterError = segmentState.currentSegmentIds.filter(
|
|
435
|
+
(id) => !errorSegmentIds.has(id),
|
|
436
|
+
);
|
|
437
|
+
|
|
438
|
+
// Update store state
|
|
439
|
+
store.setSegmentIds(segmentIdsAfterError);
|
|
440
|
+
const currentHandleData = eventController.getHandleState().data;
|
|
441
|
+
store.cacheSegmentsForHistory(
|
|
442
|
+
currentKey,
|
|
443
|
+
errorResult.segments,
|
|
444
|
+
currentHandleData,
|
|
445
|
+
);
|
|
446
|
+
|
|
447
|
+
// Throw the error so the action promise rejects
|
|
448
|
+
if (returnValue && !returnValue.ok) {
|
|
449
|
+
throw returnValue.data;
|
|
450
|
+
}
|
|
451
|
+
|
|
452
|
+
// No error in returnValue (shouldn't happen with isError: true)
|
|
453
|
+
handle.complete(undefined);
|
|
454
|
+
return undefined;
|
|
455
|
+
}
|
|
347
456
|
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
457
|
+
if (!isPartial) {
|
|
458
|
+
// Protocol invariant: action revalidation responses MUST be partial.
|
|
459
|
+
// The server always sends isPartial: true for successful revalidation
|
|
460
|
+
// and isPartial: true + isError: true for error boundary responses.
|
|
461
|
+
// A non-partial payload here indicates a server-side bug.
|
|
462
|
+
throw new Error(
|
|
463
|
+
`[Browser] Action response missing isPartial — the server must ` +
|
|
464
|
+
`always send partial payloads for action revalidation.`,
|
|
465
|
+
);
|
|
466
|
+
}
|
|
351
467
|
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
468
|
+
log("processing partial update", {
|
|
469
|
+
serverSegments: segments?.length ?? 0,
|
|
470
|
+
diff: diff?.join(", ") ?? "",
|
|
471
|
+
matched: matched?.join(", ") ?? "",
|
|
472
|
+
});
|
|
355
473
|
|
|
356
|
-
//
|
|
357
|
-
|
|
474
|
+
// Record revalidated segments for concurrent action tracking
|
|
475
|
+
if (diff) {
|
|
476
|
+
handle.recordRevalidatedSegments(diff);
|
|
477
|
+
}
|
|
358
478
|
|
|
359
|
-
// Get current page's cached segments
|
|
479
|
+
// Get current page's cached segments for merging
|
|
360
480
|
const currentKey = store.getHistoryKey();
|
|
361
481
|
const cached = store.getCachedSegments(currentKey);
|
|
362
482
|
const cachedSegments = cached?.segments || [];
|
|
363
483
|
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
// Render the full tree with error segment merged with parent layouts
|
|
368
|
-
const errorTree = await renderSegments(errorResult.mainSegments, {
|
|
369
|
-
isAction: true,
|
|
370
|
-
interceptSegments:
|
|
371
|
-
errorResult.interceptSegments.length > 0
|
|
372
|
-
? errorResult.interceptSegments
|
|
373
|
-
: undefined,
|
|
374
|
-
});
|
|
484
|
+
if (!matched) {
|
|
485
|
+
throw new Error("No matched segments in response");
|
|
486
|
+
}
|
|
375
487
|
|
|
376
|
-
//
|
|
377
|
-
|
|
378
|
-
|
|
488
|
+
// Reconcile server segments with cached segments (single source of truth)
|
|
489
|
+
const reconciled = reconcileSegments({
|
|
490
|
+
actor: "action",
|
|
491
|
+
matched,
|
|
492
|
+
diff: diff || [],
|
|
493
|
+
serverSegments: segments || [],
|
|
494
|
+
cachedSegments,
|
|
379
495
|
});
|
|
496
|
+
const fullSegments = reconciled.segments;
|
|
380
497
|
|
|
381
|
-
|
|
382
|
-
const errorSegmentIds = new Set(diff);
|
|
383
|
-
const segmentIdsAfterError = segmentState.currentSegmentIds.filter(
|
|
384
|
-
(id) => !errorSegmentIds.has(id),
|
|
385
|
-
);
|
|
386
|
-
|
|
387
|
-
// Update store state
|
|
388
|
-
store.setSegmentIds(segmentIdsAfterError);
|
|
389
|
-
const currentHandleData = eventController.getHandleState().data;
|
|
390
|
-
store.cacheSegmentsForHistory(
|
|
391
|
-
currentKey,
|
|
392
|
-
errorResult.segments,
|
|
393
|
-
currentHandleData,
|
|
394
|
-
);
|
|
498
|
+
const returnData = returnValue?.data;
|
|
395
499
|
|
|
396
|
-
// Throw the error so the action promise rejects
|
|
397
500
|
if (returnValue && !returnValue.ok) {
|
|
398
501
|
handle.fail(returnValue.data);
|
|
399
502
|
throw returnValue.data;
|
|
400
503
|
}
|
|
401
504
|
|
|
402
|
-
//
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
505
|
+
// Classify the post-reconciliation scenario
|
|
506
|
+
const scenario = classifyActionOutcome({
|
|
507
|
+
handleId: handle.id,
|
|
508
|
+
inflightActions: eventController.getInflightActions(),
|
|
509
|
+
hadAnyConcurrentActions: eventController.hadAnyConcurrentActions(),
|
|
510
|
+
revalidatedSegments: handle.getRevalidatedSegments(),
|
|
511
|
+
actionStartPathname,
|
|
512
|
+
currentPathname: window.location.pathname,
|
|
513
|
+
actionStartLocationKey: locationKey,
|
|
514
|
+
currentLocationKey: window.history.state?.key,
|
|
515
|
+
reconciledSegmentCount: fullSegments.length,
|
|
516
|
+
matchedCount: matched.length,
|
|
517
|
+
currentInterceptSource: store.getInterceptSourceUrl(),
|
|
518
|
+
});
|
|
413
519
|
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
520
|
+
switch (scenario.type) {
|
|
521
|
+
case "navigated-away": {
|
|
522
|
+
log("user navigated away during action", {
|
|
523
|
+
from: actionStartPathname,
|
|
524
|
+
to: window.location.pathname,
|
|
525
|
+
historyKeyChanged: scenario.historyKeyChanged,
|
|
526
|
+
});
|
|
527
|
+
// Clear concurrent action tracking - don't consolidate for old route's segments
|
|
528
|
+
handle.clearConsolidation();
|
|
529
|
+
|
|
530
|
+
if (scenario.historyKeyChanged) {
|
|
531
|
+
if (!scenario.onInterceptRoute) {
|
|
532
|
+
store.markCacheAsStaleAndBroadcast();
|
|
533
|
+
refetchRoute().catch((error) => {
|
|
534
|
+
if (isBackgroundSuppressible(error)) return;
|
|
535
|
+
console.error(
|
|
536
|
+
"[Browser] Background revalidation failed:",
|
|
537
|
+
error,
|
|
538
|
+
);
|
|
539
|
+
});
|
|
540
|
+
}
|
|
541
|
+
break;
|
|
542
|
+
}
|
|
419
543
|
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
544
|
+
// Same history key but different pathname - safe to refetch current route
|
|
545
|
+
store.markCacheAsStaleAndBroadcast();
|
|
546
|
+
await refetchRoute({
|
|
547
|
+
interceptSourceUrl: store.getInterceptSourceUrl(),
|
|
548
|
+
});
|
|
549
|
+
break;
|
|
550
|
+
}
|
|
424
551
|
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
552
|
+
case "hmr-missing": {
|
|
553
|
+
console.warn(
|
|
554
|
+
`[Browser] Missing segments after action (HMR detected), refetching...`,
|
|
555
|
+
);
|
|
556
|
+
await refetchRoute({ interceptSourceUrl });
|
|
557
|
+
store.broadcastCacheInvalidation();
|
|
558
|
+
break;
|
|
559
|
+
}
|
|
429
560
|
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
561
|
+
case "consolidation-needed": {
|
|
562
|
+
log("consolidation fetch needed", {
|
|
563
|
+
segmentIds: scenario.segmentIds,
|
|
564
|
+
});
|
|
565
|
+
// Calculate segments to send (exclude the ones we want fresh)
|
|
566
|
+
const currentSegmentIds = store.getSegmentState().currentSegmentIds;
|
|
567
|
+
const segmentsToSend = currentSegmentIds.filter(
|
|
568
|
+
(sid) => !scenario.segmentIds.includes(sid),
|
|
569
|
+
);
|
|
433
570
|
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
actor: "action",
|
|
437
|
-
matched,
|
|
438
|
-
diff: diff || [],
|
|
439
|
-
serverSegments: segments || [],
|
|
440
|
-
cachedSegments,
|
|
441
|
-
});
|
|
442
|
-
const fullSegments = reconciled.segments;
|
|
443
|
-
|
|
444
|
-
const returnData = returnValue?.data;
|
|
445
|
-
|
|
446
|
-
if (returnValue && !returnValue.ok) {
|
|
447
|
-
handle.fail(returnValue.data);
|
|
448
|
-
throw returnValue.data;
|
|
449
|
-
}
|
|
571
|
+
// Clear consolidation tracking before fetch
|
|
572
|
+
handle.clearConsolidation();
|
|
450
573
|
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
actionStartPathname,
|
|
459
|
-
currentPathname: window.location.pathname,
|
|
460
|
-
actionStartLocationKey: locationKey,
|
|
461
|
-
currentLocationKey: window.history.state?.key,
|
|
462
|
-
reconciledSegmentCount: fullSegments.length,
|
|
463
|
-
matchedCount: matched.length,
|
|
464
|
-
consolidationSegments: consolidationSegments || null,
|
|
465
|
-
otherFetchingActionCount: otherFetchingActions.length,
|
|
466
|
-
currentInterceptSource: store.getInterceptSourceUrl(),
|
|
467
|
-
});
|
|
468
|
-
|
|
469
|
-
switch (scenario.type) {
|
|
470
|
-
case "navigated-away": {
|
|
471
|
-
log("user navigated away during action", {
|
|
472
|
-
from: actionStartPathname,
|
|
473
|
-
to: window.location.pathname,
|
|
474
|
-
historyKeyChanged: scenario.historyKeyChanged,
|
|
475
|
-
});
|
|
476
|
-
// Clear concurrent action tracking - don't consolidate for old route's segments
|
|
477
|
-
handle.clearConsolidation();
|
|
574
|
+
await refetchRoute({
|
|
575
|
+
segments: segmentsToSend,
|
|
576
|
+
interceptSourceUrl,
|
|
577
|
+
});
|
|
578
|
+
store.broadcastCacheInvalidation();
|
|
579
|
+
break;
|
|
580
|
+
}
|
|
478
581
|
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
582
|
+
case "concurrent-skip": {
|
|
583
|
+
log("skipping UI update, other actions fetching", {
|
|
584
|
+
otherCount: scenario.otherFetchingCount,
|
|
585
|
+
});
|
|
586
|
+
// Only update store if history key hasn't changed (user didn't navigate away)
|
|
587
|
+
const currentKeyNow = store.getHistoryKey();
|
|
588
|
+
if (currentKeyNow === currentKey) {
|
|
589
|
+
store.setSegmentIds(matched);
|
|
590
|
+
const currentHandleData = eventController.getHandleState().data;
|
|
591
|
+
store.cacheSegmentsForHistory(
|
|
592
|
+
currentKey,
|
|
593
|
+
fullSegments,
|
|
594
|
+
currentHandleData,
|
|
595
|
+
);
|
|
486
596
|
}
|
|
487
597
|
break;
|
|
488
598
|
}
|
|
489
599
|
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
600
|
+
case "normal": {
|
|
601
|
+
// Prepare new tree (await loader data resolution)
|
|
602
|
+
const newTree = await renderSegments(reconciled.mainSegments, {
|
|
603
|
+
isAction: true,
|
|
604
|
+
interceptSegments:
|
|
605
|
+
reconciled.interceptSegments.length > 0
|
|
606
|
+
? reconciled.interceptSegments
|
|
607
|
+
: undefined,
|
|
608
|
+
});
|
|
497
609
|
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
store.broadcastCacheInvalidation();
|
|
504
|
-
break;
|
|
505
|
-
}
|
|
610
|
+
// Re-check if user navigated away (could happen during async renderSegments)
|
|
611
|
+
if (window.location.pathname !== actionStartPathname) {
|
|
612
|
+
log("user navigated during render, skipping");
|
|
613
|
+
break;
|
|
614
|
+
}
|
|
506
615
|
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
616
|
+
// Verify the store's current key still matches what we captured at action start
|
|
617
|
+
// If they differ, user navigated away and we should NOT cache under the old key
|
|
618
|
+
const currentKeyNow = store.getHistoryKey();
|
|
619
|
+
if (currentKeyNow !== currentKey) {
|
|
620
|
+
log("history key changed during action, skipping cache update");
|
|
621
|
+
break;
|
|
622
|
+
}
|
|
514
623
|
|
|
515
|
-
|
|
516
|
-
|
|
624
|
+
startTransition(() => {
|
|
625
|
+
onUpdate({ root: newTree, metadata: metadata! });
|
|
626
|
+
});
|
|
517
627
|
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
break;
|
|
524
|
-
}
|
|
628
|
+
// Apply server-set location state to history.state (non-redirect flow)
|
|
629
|
+
const actionLocationState = metadata?.locationState;
|
|
630
|
+
if (actionLocationState) {
|
|
631
|
+
mergeLocationState(actionLocationState);
|
|
632
|
+
}
|
|
525
633
|
|
|
526
|
-
|
|
527
|
-
log("skipping UI update, other actions fetching", {
|
|
528
|
-
otherCount: scenario.otherFetchingCount,
|
|
529
|
-
});
|
|
530
|
-
// Only update store if history key hasn't changed (user didn't navigate away)
|
|
531
|
-
const currentKeyNow = store.getHistoryKey();
|
|
532
|
-
if (currentKeyNow === currentKey) {
|
|
634
|
+
// Update store state
|
|
533
635
|
store.setSegmentIds(matched);
|
|
534
636
|
const currentHandleData = eventController.getHandleState().data;
|
|
535
637
|
store.cacheSegmentsForHistory(
|
|
@@ -537,71 +639,16 @@ export function createServerActionBridge(
|
|
|
537
639
|
fullSegments,
|
|
538
640
|
currentHandleData,
|
|
539
641
|
);
|
|
540
|
-
|
|
541
|
-
break;
|
|
542
|
-
}
|
|
543
|
-
|
|
544
|
-
case "normal": {
|
|
545
|
-
// Prepare new tree (await loader data resolution)
|
|
546
|
-
const newTree = await renderSegments(reconciled.mainSegments, {
|
|
547
|
-
isAction: true,
|
|
548
|
-
interceptSegments:
|
|
549
|
-
reconciled.interceptSegments.length > 0
|
|
550
|
-
? reconciled.interceptSegments
|
|
551
|
-
: undefined,
|
|
552
|
-
});
|
|
553
|
-
|
|
554
|
-
// Re-check if user navigated away (could happen during async renderSegments)
|
|
555
|
-
if (window.location.pathname !== actionStartPathname) {
|
|
556
|
-
log("user navigated during render, skipping");
|
|
557
|
-
break;
|
|
558
|
-
}
|
|
559
|
-
|
|
560
|
-
// Verify the store's current key still matches what we captured at action start
|
|
561
|
-
// If they differ, user navigated away and we should NOT cache under the old key
|
|
562
|
-
const currentKeyNow = store.getHistoryKey();
|
|
563
|
-
if (currentKeyNow !== currentKey) {
|
|
564
|
-
log("history key changed during action, skipping cache update");
|
|
642
|
+
store.markCacheAsStaleAndBroadcast();
|
|
565
643
|
break;
|
|
566
644
|
}
|
|
567
|
-
|
|
568
|
-
startTransition(() => {
|
|
569
|
-
onUpdate({ root: newTree, metadata: metadata! });
|
|
570
|
-
});
|
|
571
|
-
|
|
572
|
-
// Apply server-set location state to history.state (non-redirect flow)
|
|
573
|
-
const actionLocationState = metadata?.locationState;
|
|
574
|
-
if (actionLocationState) {
|
|
575
|
-
const merged = {
|
|
576
|
-
...window.history.state,
|
|
577
|
-
...actionLocationState,
|
|
578
|
-
};
|
|
579
|
-
window.history.replaceState(merged, "", window.location.href);
|
|
580
|
-
// Notify useLocationState hooks so they re-read from history.state
|
|
581
|
-
if (
|
|
582
|
-
Object.keys(actionLocationState).some((k) =>
|
|
583
|
-
k.startsWith("__rsc_ls_"),
|
|
584
|
-
)
|
|
585
|
-
) {
|
|
586
|
-
window.dispatchEvent(new Event("__rsc_locationstate"));
|
|
587
|
-
}
|
|
588
|
-
}
|
|
589
|
-
|
|
590
|
-
// Update store state
|
|
591
|
-
store.setSegmentIds(matched);
|
|
592
|
-
const currentHandleData = eventController.getHandleState().data;
|
|
593
|
-
store.cacheSegmentsForHistory(
|
|
594
|
-
currentKey,
|
|
595
|
-
fullSegments,
|
|
596
|
-
currentHandleData,
|
|
597
|
-
);
|
|
598
|
-
store.markCacheAsStaleAndBroadcast();
|
|
599
|
-
break;
|
|
600
645
|
}
|
|
601
|
-
}
|
|
602
646
|
|
|
603
|
-
|
|
604
|
-
|
|
647
|
+
handle.complete(returnData);
|
|
648
|
+
return returnData;
|
|
649
|
+
} finally {
|
|
650
|
+
handle[Symbol.dispose]();
|
|
651
|
+
}
|
|
605
652
|
}
|
|
606
653
|
|
|
607
654
|
return {
|
|
@@ -616,16 +663,6 @@ export function createServerActionBridge(
|
|
|
616
663
|
deps.setServerCallback(handleServerAction);
|
|
617
664
|
isRegistered = true;
|
|
618
665
|
},
|
|
619
|
-
|
|
620
|
-
/**
|
|
621
|
-
* Unregister the server action callback
|
|
622
|
-
*/
|
|
623
|
-
unregister(): void {
|
|
624
|
-
if (!isRegistered) {
|
|
625
|
-
return;
|
|
626
|
-
}
|
|
627
|
-
isRegistered = false;
|
|
628
|
-
},
|
|
629
666
|
};
|
|
630
667
|
}
|
|
631
668
|
|