@rangojs/router 0.0.0-experimental.13 → 0.0.0-experimental.130
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +9 -0
- package/README.md +1040 -5
- package/dist/__internal.d.ts +83 -0
- package/dist/__internal.d.ts.map +1 -0
- package/dist/__internal.js +19 -0
- package/dist/__internal.js.map +1 -0
- package/dist/__mocks__/version.d.ts +7 -0
- package/dist/__mocks__/version.d.ts.map +1 -0
- package/dist/__mocks__/version.js +7 -0
- package/dist/__mocks__/version.js.map +1 -0
- package/dist/__tests__/client-href.test.d.ts +2 -0
- package/dist/__tests__/client-href.test.d.ts.map +1 -0
- package/dist/__tests__/client-href.test.js +74 -0
- package/dist/__tests__/client-href.test.js.map +1 -0
- package/dist/__tests__/component-utils.test.d.ts +2 -0
- package/dist/__tests__/component-utils.test.d.ts.map +1 -0
- package/dist/__tests__/component-utils.test.js +51 -0
- package/dist/__tests__/component-utils.test.js.map +1 -0
- package/dist/__tests__/event-controller.test.d.ts +2 -0
- package/dist/__tests__/event-controller.test.d.ts.map +1 -0
- package/dist/__tests__/event-controller.test.js +538 -0
- package/dist/__tests__/event-controller.test.js.map +1 -0
- package/dist/__tests__/helpers/route-tree.d.ts +118 -0
- package/dist/__tests__/helpers/route-tree.d.ts.map +1 -0
- package/dist/__tests__/helpers/route-tree.js +374 -0
- package/dist/__tests__/helpers/route-tree.js.map +1 -0
- package/dist/__tests__/match-result.test.d.ts +2 -0
- package/dist/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/__tests__/match-result.test.js +154 -0
- package/dist/__tests__/match-result.test.js.map +1 -0
- package/dist/__tests__/navigation-store.test.d.ts +2 -0
- package/dist/__tests__/navigation-store.test.d.ts.map +1 -0
- package/dist/__tests__/navigation-store.test.js +440 -0
- package/dist/__tests__/navigation-store.test.js.map +1 -0
- package/dist/__tests__/partial-update.test.d.ts +2 -0
- package/dist/__tests__/partial-update.test.d.ts.map +1 -0
- package/dist/__tests__/partial-update.test.js +1009 -0
- package/dist/__tests__/partial-update.test.js.map +1 -0
- package/dist/__tests__/reverse-types.test.d.ts +8 -0
- package/dist/__tests__/reverse-types.test.d.ts.map +1 -0
- package/dist/__tests__/reverse-types.test.js +656 -0
- package/dist/__tests__/reverse-types.test.js.map +1 -0
- package/dist/__tests__/route-definition.test.d.ts +2 -0
- package/dist/__tests__/route-definition.test.d.ts.map +1 -0
- package/dist/__tests__/route-definition.test.js +55 -0
- package/dist/__tests__/route-definition.test.js.map +1 -0
- package/dist/__tests__/router-helpers.test.d.ts +2 -0
- package/dist/__tests__/router-helpers.test.d.ts.map +1 -0
- package/dist/__tests__/router-helpers.test.js +377 -0
- package/dist/__tests__/router-helpers.test.js.map +1 -0
- package/dist/__tests__/router-integration-2.test.d.ts +2 -0
- package/dist/__tests__/router-integration-2.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration-2.test.js +426 -0
- package/dist/__tests__/router-integration-2.test.js.map +1 -0
- package/dist/__tests__/router-integration.test.d.ts +2 -0
- package/dist/__tests__/router-integration.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration.test.js +1051 -0
- package/dist/__tests__/router-integration.test.js.map +1 -0
- package/dist/__tests__/search-params.test.d.ts +5 -0
- package/dist/__tests__/search-params.test.d.ts.map +1 -0
- package/dist/__tests__/search-params.test.js +306 -0
- package/dist/__tests__/search-params.test.js.map +1 -0
- package/dist/__tests__/segment-system.test.d.ts +2 -0
- package/dist/__tests__/segment-system.test.d.ts.map +1 -0
- package/dist/__tests__/segment-system.test.js +627 -0
- package/dist/__tests__/segment-system.test.js.map +1 -0
- package/dist/__tests__/static-handler-types.test.d.ts +8 -0
- package/dist/__tests__/static-handler-types.test.d.ts.map +1 -0
- package/dist/__tests__/static-handler-types.test.js +63 -0
- package/dist/__tests__/static-handler-types.test.js.map +1 -0
- package/dist/__tests__/urls.test.d.ts +2 -0
- package/dist/__tests__/urls.test.d.ts.map +1 -0
- package/dist/__tests__/urls.test.js +421 -0
- package/dist/__tests__/urls.test.js.map +1 -0
- package/dist/__tests__/use-mount.test.d.ts +2 -0
- package/dist/__tests__/use-mount.test.d.ts.map +1 -0
- package/dist/__tests__/use-mount.test.js +35 -0
- package/dist/__tests__/use-mount.test.js.map +1 -0
- package/dist/bin/rango.d.ts +2 -0
- package/dist/bin/rango.d.ts.map +1 -0
- package/dist/bin/rango.js +1712 -212
- package/dist/bin/rango.js.map +1 -0
- package/dist/browser/event-controller.d.ts +191 -0
- package/dist/browser/event-controller.d.ts.map +1 -0
- package/dist/browser/event-controller.js +559 -0
- package/dist/browser/event-controller.js.map +1 -0
- package/dist/browser/index.d.ts +2 -0
- package/dist/browser/index.d.ts.map +1 -0
- package/dist/browser/index.js +14 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/browser/link-interceptor.d.ts +38 -0
- package/dist/browser/link-interceptor.d.ts.map +1 -0
- package/dist/browser/link-interceptor.js +99 -0
- package/dist/browser/link-interceptor.js.map +1 -0
- package/dist/browser/logging.d.ts +10 -0
- package/dist/browser/logging.d.ts.map +1 -0
- package/dist/browser/logging.js +29 -0
- package/dist/browser/logging.js.map +1 -0
- package/dist/browser/lru-cache.d.ts +17 -0
- package/dist/browser/lru-cache.d.ts.map +1 -0
- package/dist/browser/lru-cache.js +50 -0
- package/dist/browser/lru-cache.js.map +1 -0
- package/dist/browser/merge-segment-loaders.d.ts +39 -0
- package/dist/browser/merge-segment-loaders.d.ts.map +1 -0
- package/dist/browser/merge-segment-loaders.js +102 -0
- package/dist/browser/merge-segment-loaders.js.map +1 -0
- package/dist/browser/navigation-bridge.d.ts +102 -0
- package/dist/browser/navigation-bridge.d.ts.map +1 -0
- package/dist/browser/navigation-bridge.js +708 -0
- package/dist/browser/navigation-bridge.js.map +1 -0
- package/dist/browser/navigation-client.d.ts +25 -0
- package/dist/browser/navigation-client.d.ts.map +1 -0
- package/dist/browser/navigation-client.js +157 -0
- package/dist/browser/navigation-client.js.map +1 -0
- package/dist/browser/navigation-store.d.ts +101 -0
- package/dist/browser/navigation-store.d.ts.map +1 -0
- package/dist/browser/navigation-store.js +625 -0
- package/dist/browser/navigation-store.js.map +1 -0
- package/dist/browser/partial-update.d.ts +75 -0
- package/dist/browser/partial-update.d.ts.map +1 -0
- package/dist/browser/partial-update.js +426 -0
- package/dist/browser/partial-update.js.map +1 -0
- package/dist/browser/react/Link.d.ts +86 -0
- package/dist/browser/react/Link.d.ts.map +1 -0
- package/dist/browser/react/Link.js +128 -0
- package/dist/browser/react/Link.js.map +1 -0
- package/dist/browser/react/NavigationProvider.d.ts +63 -0
- package/dist/browser/react/NavigationProvider.d.ts.map +1 -0
- package/dist/browser/react/NavigationProvider.js +216 -0
- package/dist/browser/react/NavigationProvider.js.map +1 -0
- package/dist/browser/react/ScrollRestoration.d.ts +75 -0
- package/dist/browser/react/ScrollRestoration.d.ts.map +1 -0
- package/dist/browser/react/ScrollRestoration.js +57 -0
- package/dist/browser/react/ScrollRestoration.js.map +1 -0
- package/dist/browser/react/context.d.ts +46 -0
- package/dist/browser/react/context.d.ts.map +1 -0
- package/dist/browser/react/context.js +10 -0
- package/dist/browser/react/context.js.map +1 -0
- package/dist/browser/react/index.d.ts +11 -0
- package/dist/browser/react/index.d.ts.map +1 -0
- package/dist/browser/react/index.js +22 -0
- package/dist/browser/react/index.js.map +1 -0
- package/dist/browser/react/location-state-shared.d.ts +63 -0
- package/dist/browser/react/location-state-shared.d.ts.map +1 -0
- package/dist/browser/react/location-state-shared.js +81 -0
- package/dist/browser/react/location-state-shared.js.map +1 -0
- package/dist/browser/react/location-state.d.ts +23 -0
- package/dist/browser/react/location-state.d.ts.map +1 -0
- package/dist/browser/react/location-state.js +29 -0
- package/dist/browser/react/location-state.js.map +1 -0
- package/dist/browser/react/mount-context.d.ts +24 -0
- package/dist/browser/react/mount-context.d.ts.map +1 -0
- package/dist/browser/react/mount-context.js +24 -0
- package/dist/browser/react/mount-context.js.map +1 -0
- package/dist/browser/react/use-action.d.ts +64 -0
- package/dist/browser/react/use-action.d.ts.map +1 -0
- package/dist/browser/react/use-action.js +134 -0
- package/dist/browser/react/use-action.js.map +1 -0
- package/dist/browser/react/use-client-cache.d.ts +41 -0
- package/dist/browser/react/use-client-cache.d.ts.map +1 -0
- package/{src/browser/react/use-client-cache.ts → dist/browser/react/use-client-cache.js} +9 -26
- package/dist/browser/react/use-client-cache.js.map +1 -0
- package/dist/browser/react/use-handle.d.ts +31 -0
- package/dist/browser/react/use-handle.d.ts.map +1 -0
- package/dist/browser/react/use-handle.js +144 -0
- package/dist/browser/react/use-handle.js.map +1 -0
- package/dist/browser/react/use-href.d.ts +33 -0
- package/dist/browser/react/use-href.d.ts.map +1 -0
- package/dist/browser/react/use-href.js +39 -0
- package/dist/browser/react/use-href.js.map +1 -0
- package/dist/browser/react/use-link-status.d.ts +37 -0
- package/dist/browser/react/use-link-status.d.ts.map +1 -0
- package/dist/browser/react/use-link-status.js +99 -0
- package/dist/browser/react/use-link-status.js.map +1 -0
- package/dist/browser/react/use-mount.d.ts +25 -0
- package/dist/browser/react/use-mount.d.ts.map +1 -0
- package/dist/browser/react/use-mount.js +30 -0
- package/dist/browser/react/use-mount.js.map +1 -0
- package/dist/browser/react/use-navigation.d.ts +27 -0
- package/dist/browser/react/use-navigation.d.ts.map +1 -0
- package/dist/browser/react/use-navigation.js +87 -0
- package/dist/browser/react/use-navigation.js.map +1 -0
- package/dist/browser/react/use-segments.d.ts +38 -0
- package/dist/browser/react/use-segments.d.ts.map +1 -0
- package/dist/browser/react/use-segments.js +130 -0
- package/dist/browser/react/use-segments.js.map +1 -0
- package/dist/browser/request-controller.d.ts +26 -0
- package/dist/browser/request-controller.d.ts.map +1 -0
- package/dist/browser/request-controller.js +147 -0
- package/dist/browser/request-controller.js.map +1 -0
- package/dist/browser/rsc-router.d.ts +129 -0
- package/dist/browser/rsc-router.d.ts.map +1 -0
- package/dist/browser/rsc-router.js +195 -0
- package/dist/browser/rsc-router.js.map +1 -0
- package/dist/browser/scroll-restoration.d.ts +93 -0
- package/dist/browser/scroll-restoration.d.ts.map +1 -0
- package/dist/browser/scroll-restoration.js +321 -0
- package/dist/browser/scroll-restoration.js.map +1 -0
- package/dist/browser/segment-structure-assert.d.ts +17 -0
- package/dist/browser/segment-structure-assert.d.ts.map +1 -0
- package/dist/browser/segment-structure-assert.js +59 -0
- package/dist/browser/segment-structure-assert.js.map +1 -0
- package/dist/browser/server-action-bridge.d.ts +26 -0
- package/dist/browser/server-action-bridge.d.ts.map +1 -0
- package/dist/browser/server-action-bridge.js +668 -0
- package/dist/browser/server-action-bridge.js.map +1 -0
- package/dist/browser/shallow.d.ts +12 -0
- package/dist/browser/shallow.d.ts.map +1 -0
- package/dist/browser/shallow.js +34 -0
- package/dist/browser/shallow.js.map +1 -0
- package/dist/browser/types.d.ts +369 -0
- package/dist/browser/types.d.ts.map +1 -0
- package/dist/browser/types.js +2 -0
- package/dist/browser/types.js.map +1 -0
- package/dist/build/__tests__/generate-cli.test.d.ts +2 -0
- package/dist/build/__tests__/generate-cli.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-cli.test.js +237 -0
- package/dist/build/__tests__/generate-cli.test.js.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.js +119 -0
- package/dist/build/__tests__/generate-manifest.test.js.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts +2 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.js +620 -0
- package/dist/build/__tests__/generate-route-types.test.js.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.js +308 -0
- package/dist/build/__tests__/per-router-manifest.test.js.map +1 -0
- package/dist/build/generate-manifest.d.ts +81 -0
- package/dist/build/generate-manifest.d.ts.map +1 -0
- package/dist/build/generate-manifest.js +276 -0
- package/dist/build/generate-manifest.js.map +1 -0
- package/dist/build/generate-route-types.d.ts +115 -0
- package/dist/build/generate-route-types.d.ts.map +1 -0
- package/dist/build/generate-route-types.js +740 -0
- package/dist/build/generate-route-types.js.map +1 -0
- package/dist/build/index.d.ts +21 -0
- package/dist/build/index.d.ts.map +1 -0
- package/dist/build/index.js +21 -0
- package/dist/build/index.js.map +1 -0
- package/dist/build/route-trie.d.ts +71 -0
- package/dist/build/route-trie.d.ts.map +1 -0
- package/dist/build/route-trie.js +175 -0
- package/dist/build/route-trie.js.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts +2 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.js +208 -0
- package/dist/cache/__tests__/cache-scope.test.js.map +1 -0
- package/dist/cache/__tests__/document-cache.test.d.ts +2 -0
- package/dist/cache/__tests__/document-cache.test.d.ts.map +1 -0
- package/dist/cache/__tests__/document-cache.test.js +345 -0
- package/dist/cache/__tests__/document-cache.test.js.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.js +425 -0
- package/dist/cache/__tests__/memory-segment-store.test.js.map +1 -0
- package/dist/cache/__tests__/memory-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-store.test.js +367 -0
- package/dist/cache/__tests__/memory-store.test.js.map +1 -0
- package/dist/cache/cache-scope.d.ts +102 -0
- package/dist/cache/cache-scope.d.ts.map +1 -0
- package/dist/cache/cache-scope.js +440 -0
- package/dist/cache/cache-scope.js.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts +2 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js +330 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js.map +1 -0
- package/dist/cache/cf/cf-cache-store.d.ts +165 -0
- package/dist/cache/cf/cf-cache-store.d.ts.map +1 -0
- package/dist/cache/cf/cf-cache-store.js +242 -0
- package/dist/cache/cf/cf-cache-store.js.map +1 -0
- package/dist/cache/cf/index.d.ts +14 -0
- package/dist/cache/cf/index.d.ts.map +1 -0
- package/dist/cache/cf/index.js +17 -0
- package/dist/cache/cf/index.js.map +1 -0
- package/dist/cache/document-cache.d.ts +64 -0
- package/dist/cache/document-cache.d.ts.map +1 -0
- package/dist/cache/document-cache.js +228 -0
- package/dist/cache/document-cache.js.map +1 -0
- package/dist/cache/index.d.ts +19 -0
- package/dist/cache/index.d.ts.map +1 -0
- package/dist/cache/index.js +21 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/cache/memory-segment-store.d.ts +110 -0
- package/dist/cache/memory-segment-store.d.ts.map +1 -0
- package/dist/cache/memory-segment-store.js +117 -0
- package/dist/cache/memory-segment-store.js.map +1 -0
- package/dist/cache/memory-store.d.ts +41 -0
- package/dist/cache/memory-store.d.ts.map +1 -0
- package/dist/cache/memory-store.js +191 -0
- package/dist/cache/memory-store.js.map +1 -0
- package/dist/cache/types.d.ts +317 -0
- package/dist/cache/types.d.ts.map +1 -0
- package/dist/cache/types.js +12 -0
- package/dist/cache/types.js.map +1 -0
- package/dist/client.d.ts +248 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +367 -0
- package/dist/client.js.map +1 -0
- package/dist/client.rsc.d.ts +26 -0
- package/dist/client.rsc.d.ts.map +1 -0
- package/dist/client.rsc.js +46 -0
- package/dist/client.rsc.js.map +1 -0
- package/dist/component-utils.d.ts +36 -0
- package/dist/component-utils.d.ts.map +1 -0
- package/dist/component-utils.js +61 -0
- package/dist/component-utils.js.map +1 -0
- package/dist/components/DefaultDocument.d.ts +13 -0
- package/dist/components/DefaultDocument.d.ts.map +1 -0
- package/dist/components/DefaultDocument.js +15 -0
- package/dist/components/DefaultDocument.js.map +1 -0
- package/dist/debug.d.ts +58 -0
- package/dist/debug.d.ts.map +1 -0
- package/dist/debug.js +157 -0
- package/dist/debug.js.map +1 -0
- package/dist/default-error-boundary.d.ts +11 -0
- package/dist/default-error-boundary.d.ts.map +1 -0
- package/dist/default-error-boundary.js +45 -0
- package/dist/default-error-boundary.js.map +1 -0
- package/dist/deps/browser.d.ts +2 -0
- package/dist/deps/browser.d.ts.map +1 -0
- package/dist/deps/browser.js +3 -0
- package/dist/deps/browser.js.map +1 -0
- package/dist/deps/html-stream-client.d.ts +2 -0
- package/dist/deps/html-stream-client.d.ts.map +1 -0
- package/dist/deps/html-stream-client.js +3 -0
- package/dist/deps/html-stream-client.js.map +1 -0
- package/dist/deps/html-stream-server.d.ts +2 -0
- package/dist/deps/html-stream-server.d.ts.map +1 -0
- package/dist/deps/html-stream-server.js +3 -0
- package/dist/deps/html-stream-server.js.map +1 -0
- package/dist/deps/rsc.d.ts +2 -0
- package/dist/deps/rsc.d.ts.map +1 -0
- package/dist/deps/rsc.js +4 -0
- package/dist/deps/rsc.js.map +1 -0
- package/dist/deps/ssr.d.ts +2 -0
- package/dist/deps/ssr.d.ts.map +1 -0
- package/dist/deps/ssr.js +3 -0
- package/dist/deps/ssr.js.map +1 -0
- package/dist/errors.d.ts +174 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +241 -0
- package/dist/errors.js.map +1 -0
- package/dist/handle.d.ts +78 -0
- package/dist/handle.d.ts.map +1 -0
- package/dist/handle.js +82 -0
- package/dist/handle.js.map +1 -0
- package/dist/handles/MetaTags.d.ts +14 -0
- package/dist/handles/MetaTags.d.ts.map +1 -0
- package/dist/handles/MetaTags.js +136 -0
- package/dist/handles/MetaTags.js.map +1 -0
- package/dist/handles/index.d.ts +6 -0
- package/dist/handles/index.d.ts.map +1 -0
- package/{src/handles/index.ts → dist/handles/index.js} +1 -1
- package/dist/handles/index.js.map +1 -0
- package/dist/handles/meta.d.ts +39 -0
- package/dist/handles/meta.d.ts.map +1 -0
- package/dist/handles/meta.js +202 -0
- package/dist/handles/meta.js.map +1 -0
- package/dist/host/__tests__/errors.test.d.ts +2 -0
- package/dist/host/__tests__/errors.test.d.ts.map +1 -0
- package/dist/host/__tests__/errors.test.js +76 -0
- package/dist/host/__tests__/errors.test.js.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js +732 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.js +251 -0
- package/dist/host/__tests__/pattern-matcher.test.js.map +1 -0
- package/dist/host/__tests__/router.test.d.ts +2 -0
- package/dist/host/__tests__/router.test.d.ts.map +1 -0
- package/dist/host/__tests__/router.test.js +241 -0
- package/dist/host/__tests__/router.test.js.map +1 -0
- package/dist/host/__tests__/testing.test.d.ts +2 -0
- package/dist/host/__tests__/testing.test.d.ts.map +1 -0
- package/dist/host/__tests__/testing.test.js +64 -0
- package/dist/host/__tests__/testing.test.js.map +1 -0
- package/dist/host/__tests__/utils.test.d.ts +2 -0
- package/dist/host/__tests__/utils.test.d.ts.map +1 -0
- package/dist/host/__tests__/utils.test.js +29 -0
- package/dist/host/__tests__/utils.test.js.map +1 -0
- package/dist/host/cookie-handler.d.ts +34 -0
- package/dist/host/cookie-handler.d.ts.map +1 -0
- package/dist/host/cookie-handler.js +124 -0
- package/dist/host/cookie-handler.js.map +1 -0
- package/dist/host/errors.d.ts +56 -0
- package/dist/host/errors.d.ts.map +1 -0
- package/dist/host/errors.js +79 -0
- package/dist/host/errors.js.map +1 -0
- package/dist/host/index.d.ts +29 -0
- package/dist/host/index.d.ts.map +1 -0
- package/dist/host/index.js +32 -0
- package/dist/host/index.js.map +1 -0
- package/dist/host/pattern-matcher.d.ts +36 -0
- package/dist/host/pattern-matcher.d.ts.map +1 -0
- package/dist/host/pattern-matcher.js +172 -0
- package/dist/host/pattern-matcher.js.map +1 -0
- package/dist/host/router.d.ts +26 -0
- package/dist/host/router.d.ts.map +1 -0
- package/dist/host/router.js +218 -0
- package/dist/host/router.js.map +1 -0
- package/dist/host/testing.d.ts +36 -0
- package/dist/host/testing.d.ts.map +1 -0
- package/dist/host/testing.js +55 -0
- package/dist/host/testing.js.map +1 -0
- package/dist/host/types.d.ts +115 -0
- package/dist/host/types.d.ts.map +1 -0
- package/dist/host/types.js +7 -0
- package/dist/host/types.js.map +1 -0
- package/dist/host/utils.d.ts +21 -0
- package/dist/host/utils.d.ts.map +1 -0
- package/dist/host/utils.js +23 -0
- package/dist/host/utils.js.map +1 -0
- package/dist/href-client.d.ts +131 -0
- package/dist/href-client.d.ts.map +1 -0
- package/dist/href-client.js +64 -0
- package/dist/href-client.js.map +1 -0
- package/{src/href-context.ts → dist/href-context.d.ts} +7 -11
- package/dist/href-context.d.ts.map +1 -0
- package/dist/href-context.js +21 -0
- package/dist/href-context.js.map +1 -0
- package/dist/index.d.ts +73 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +91 -0
- package/dist/index.js.map +1 -0
- package/dist/index.rsc.d.ts +32 -0
- package/dist/index.rsc.d.ts.map +1 -0
- package/dist/index.rsc.js +40 -0
- package/dist/index.rsc.js.map +1 -0
- package/dist/internal-debug.d.ts +2 -0
- package/dist/internal-debug.d.ts.map +1 -0
- package/dist/internal-debug.js +5 -0
- package/dist/internal-debug.js.map +1 -0
- package/dist/loader.d.ts +14 -0
- package/dist/loader.d.ts.map +1 -0
- package/dist/loader.js +20 -0
- package/dist/loader.js.map +1 -0
- package/dist/loader.rsc.d.ts +19 -0
- package/dist/loader.rsc.d.ts.map +1 -0
- package/dist/loader.rsc.js +99 -0
- package/dist/loader.rsc.js.map +1 -0
- package/dist/network-error-thrower.d.ts +17 -0
- package/dist/network-error-thrower.d.ts.map +1 -0
- package/dist/network-error-thrower.js +14 -0
- package/dist/network-error-thrower.js.map +1 -0
- package/dist/outlet-context.d.ts +13 -0
- package/dist/outlet-context.d.ts.map +1 -0
- package/dist/outlet-context.js +3 -0
- package/dist/outlet-context.js.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts +2 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.js +148 -0
- package/dist/prerender/__tests__/param-hash.test.js.map +1 -0
- package/dist/prerender/param-hash.d.ts +16 -0
- package/dist/prerender/param-hash.d.ts.map +1 -0
- package/dist/prerender/param-hash.js +36 -0
- package/dist/prerender/param-hash.js.map +1 -0
- package/dist/prerender/store.d.ts +38 -0
- package/dist/prerender/store.d.ts.map +1 -0
- package/dist/prerender/store.js +61 -0
- package/dist/prerender/store.js.map +1 -0
- package/dist/prerender.d.ts +66 -0
- package/dist/prerender.d.ts.map +1 -0
- package/dist/prerender.js +57 -0
- package/dist/prerender.js.map +1 -0
- package/dist/reverse.d.ts +196 -0
- package/dist/reverse.d.ts.map +1 -0
- package/dist/reverse.js +78 -0
- package/dist/reverse.js.map +1 -0
- package/dist/root-error-boundary.d.ts +33 -0
- package/dist/root-error-boundary.d.ts.map +1 -0
- package/dist/root-error-boundary.js +165 -0
- package/dist/root-error-boundary.js.map +1 -0
- package/dist/route-content-wrapper.d.ts +46 -0
- package/dist/route-content-wrapper.d.ts.map +1 -0
- package/dist/route-content-wrapper.js +77 -0
- package/dist/route-content-wrapper.js.map +1 -0
- package/dist/route-definition.d.ts +421 -0
- package/dist/route-definition.d.ts.map +1 -0
- package/dist/route-definition.js +868 -0
- package/dist/route-definition.js.map +1 -0
- package/dist/route-map-builder.d.ts +155 -0
- package/dist/route-map-builder.d.ts.map +1 -0
- package/dist/route-map-builder.js +237 -0
- package/dist/route-map-builder.js.map +1 -0
- package/dist/route-types.d.ts +165 -0
- package/dist/route-types.d.ts.map +1 -0
- package/dist/route-types.js +7 -0
- package/dist/route-types.js.map +1 -0
- package/dist/router/__tests__/handler-context.test.d.ts +2 -0
- package/dist/router/__tests__/handler-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/handler-context.test.js +65 -0
- package/dist/router/__tests__/handler-context.test.js.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts +2 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js +221 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js.map +1 -0
- package/dist/router/__tests__/match-context.test.d.ts +2 -0
- package/dist/router/__tests__/match-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-context.test.js +92 -0
- package/dist/router/__tests__/match-context.test.js.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts +2 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.js +417 -0
- package/dist/router/__tests__/match-pipelines.test.js.map +1 -0
- package/dist/router/__tests__/match-result.test.d.ts +2 -0
- package/dist/router/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-result.test.js +457 -0
- package/dist/router/__tests__/match-result.test.js.map +1 -0
- package/dist/router/__tests__/on-error.test.d.ts +2 -0
- package/dist/router/__tests__/on-error.test.d.ts.map +1 -0
- package/dist/router/__tests__/on-error.test.js +678 -0
- package/dist/router/__tests__/on-error.test.js.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts +2 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.js +629 -0
- package/dist/router/__tests__/pattern-matching.test.js.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts +2 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js +155 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js.map +1 -0
- package/dist/router/error-handling.d.ts +77 -0
- package/dist/router/error-handling.d.ts.map +1 -0
- package/dist/router/error-handling.js +202 -0
- package/dist/router/error-handling.js.map +1 -0
- package/dist/router/handler-context.d.ts +20 -0
- package/dist/router/handler-context.d.ts.map +1 -0
- package/dist/router/handler-context.js +198 -0
- package/dist/router/handler-context.js.map +1 -0
- package/dist/router/intercept-resolution.d.ts +66 -0
- package/dist/router/intercept-resolution.d.ts.map +1 -0
- package/dist/router/intercept-resolution.js +246 -0
- package/dist/router/intercept-resolution.js.map +1 -0
- package/dist/router/loader-resolution.d.ts +64 -0
- package/dist/router/loader-resolution.d.ts.map +1 -0
- package/dist/router/loader-resolution.js +284 -0
- package/dist/router/loader-resolution.js.map +1 -0
- package/dist/router/logging.d.ts +15 -0
- package/dist/router/logging.d.ts.map +1 -0
- package/dist/router/logging.js +99 -0
- package/dist/router/logging.js.map +1 -0
- package/dist/router/manifest.d.ts +22 -0
- package/dist/router/manifest.d.ts.map +1 -0
- package/dist/router/manifest.js +181 -0
- package/dist/router/manifest.js.map +1 -0
- package/dist/router/match-api.d.ts +35 -0
- package/dist/router/match-api.d.ts.map +1 -0
- package/dist/router/match-api.js +406 -0
- package/dist/router/match-api.js.map +1 -0
- package/dist/router/match-context.d.ts +206 -0
- package/dist/router/match-context.d.ts.map +1 -0
- package/dist/router/match-context.js +17 -0
- package/dist/router/match-context.js.map +1 -0
- package/dist/router/match-middleware/background-revalidation.d.ts +127 -0
- package/dist/router/match-middleware/background-revalidation.d.ts.map +1 -0
- package/dist/router/match-middleware/background-revalidation.js +75 -0
- package/dist/router/match-middleware/background-revalidation.js.map +1 -0
- package/dist/router/match-middleware/cache-lookup.d.ts +112 -0
- package/dist/router/match-middleware/cache-lookup.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-lookup.js +257 -0
- package/dist/router/match-middleware/cache-lookup.js.map +1 -0
- package/dist/router/match-middleware/cache-store.d.ts +113 -0
- package/dist/router/match-middleware/cache-store.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-store.js +108 -0
- package/dist/router/match-middleware/cache-store.js.map +1 -0
- package/dist/router/match-middleware/index.d.ts +81 -0
- package/dist/router/match-middleware/index.d.ts.map +1 -0
- package/dist/router/match-middleware/index.js +80 -0
- package/dist/router/match-middleware/index.js.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts +117 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.js +134 -0
- package/dist/router/match-middleware/intercept-resolution.js.map +1 -0
- package/dist/router/match-middleware/segment-resolution.d.ts +99 -0
- package/dist/router/match-middleware/segment-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/segment-resolution.js +53 -0
- package/dist/router/match-middleware/segment-resolution.js.map +1 -0
- package/dist/router/match-pipelines.d.ts +147 -0
- package/dist/router/match-pipelines.d.ts.map +1 -0
- package/dist/router/match-pipelines.js +82 -0
- package/dist/router/match-pipelines.js.map +1 -0
- package/dist/router/match-result.d.ts +126 -0
- package/dist/router/match-result.d.ts.map +1 -0
- package/dist/router/match-result.js +93 -0
- package/dist/router/match-result.js.map +1 -0
- package/dist/router/metrics.d.ts +20 -0
- package/dist/router/metrics.d.ts.map +1 -0
- package/dist/router/metrics.js +47 -0
- package/dist/router/metrics.js.map +1 -0
- package/dist/router/middleware.d.ts +249 -0
- package/dist/router/middleware.d.ts.map +1 -0
- package/dist/router/middleware.js +434 -0
- package/dist/router/middleware.js.map +1 -0
- package/dist/router/middleware.test.d.ts +2 -0
- package/dist/router/middleware.test.d.ts.map +1 -0
- package/dist/router/middleware.test.js +816 -0
- package/dist/router/middleware.test.js.map +1 -0
- package/dist/router/pattern-matching.d.ts +149 -0
- package/dist/router/pattern-matching.d.ts.map +1 -0
- package/dist/router/pattern-matching.js +349 -0
- package/dist/router/pattern-matching.js.map +1 -0
- package/dist/router/revalidation.d.ts +44 -0
- package/dist/router/revalidation.d.ts.map +1 -0
- package/dist/router/revalidation.js +147 -0
- package/dist/router/revalidation.js.map +1 -0
- package/dist/router/router-context.d.ts +135 -0
- package/dist/router/router-context.d.ts.map +1 -0
- package/dist/router/router-context.js +36 -0
- package/dist/router/router-context.js.map +1 -0
- package/dist/router/segment-resolution.d.ts +127 -0
- package/dist/router/segment-resolution.d.ts.map +1 -0
- package/dist/router/segment-resolution.js +919 -0
- package/dist/router/segment-resolution.js.map +1 -0
- package/dist/router/trie-matching.d.ts +40 -0
- package/dist/router/trie-matching.d.ts.map +1 -0
- package/dist/router/trie-matching.js +127 -0
- package/dist/router/trie-matching.js.map +1 -0
- package/dist/router/types.d.ts +136 -0
- package/dist/router/types.d.ts.map +1 -0
- package/dist/router/types.js +7 -0
- package/dist/router/types.js.map +1 -0
- package/dist/router.d.ts +753 -0
- package/dist/router.d.ts.map +1 -0
- package/dist/router.gen.d.ts +6 -0
- package/dist/router.gen.d.ts.map +1 -0
- package/dist/router.gen.js +6 -0
- package/dist/router.gen.js.map +1 -0
- package/dist/router.js +1304 -0
- package/dist/router.js.map +1 -0
- package/dist/rsc/__tests__/helpers.test.d.ts +2 -0
- package/dist/rsc/__tests__/helpers.test.d.ts.map +1 -0
- package/dist/rsc/__tests__/helpers.test.js +140 -0
- package/dist/rsc/__tests__/helpers.test.js.map +1 -0
- package/dist/rsc/handler.d.ts +45 -0
- package/dist/rsc/handler.d.ts.map +1 -0
- package/dist/rsc/handler.js +1172 -0
- package/dist/rsc/handler.js.map +1 -0
- package/dist/rsc/helpers.d.ts +16 -0
- package/dist/rsc/helpers.d.ts.map +1 -0
- package/dist/rsc/helpers.js +55 -0
- package/dist/rsc/helpers.js.map +1 -0
- package/dist/rsc/index.d.ts +22 -0
- package/dist/rsc/index.d.ts.map +1 -0
- package/dist/rsc/index.js +23 -0
- package/dist/rsc/index.js.map +1 -0
- package/dist/rsc/nonce.d.ts +9 -0
- package/dist/rsc/nonce.d.ts.map +1 -0
- package/dist/rsc/nonce.js +18 -0
- package/dist/rsc/nonce.js.map +1 -0
- package/dist/rsc/types.d.ts +206 -0
- package/dist/rsc/types.d.ts.map +1 -0
- package/dist/rsc/types.js +8 -0
- package/dist/rsc/types.js.map +1 -0
- package/dist/search-params.d.ts +103 -0
- package/dist/search-params.d.ts.map +1 -0
- package/dist/search-params.js +74 -0
- package/dist/search-params.js.map +1 -0
- package/dist/segment-system.d.ts +75 -0
- package/dist/segment-system.d.ts.map +1 -0
- package/dist/segment-system.js +336 -0
- package/dist/segment-system.js.map +1 -0
- package/dist/server/context.d.ts +245 -0
- package/dist/server/context.d.ts.map +1 -0
- package/dist/server/context.js +197 -0
- package/dist/server/context.js.map +1 -0
- package/dist/server/fetchable-loader-store.d.ts +18 -0
- package/dist/server/fetchable-loader-store.d.ts.map +1 -0
- package/dist/server/fetchable-loader-store.js +18 -0
- package/dist/server/fetchable-loader-store.js.map +1 -0
- package/dist/server/handle-store.d.ts +85 -0
- package/dist/server/handle-store.d.ts.map +1 -0
- package/dist/server/handle-store.js +142 -0
- package/dist/server/handle-store.js.map +1 -0
- package/dist/server/loader-registry.d.ts +55 -0
- package/dist/server/loader-registry.d.ts.map +1 -0
- package/dist/server/loader-registry.js +132 -0
- package/dist/server/loader-registry.js.map +1 -0
- package/dist/server/request-context.d.ts +226 -0
- package/dist/server/request-context.d.ts.map +1 -0
- package/dist/server/request-context.js +290 -0
- package/dist/server/request-context.js.map +1 -0
- package/dist/server/root-layout.d.ts +4 -0
- package/dist/server/root-layout.d.ts.map +1 -0
- package/dist/server/root-layout.js +5 -0
- package/dist/server/root-layout.js.map +1 -0
- package/dist/server.d.ts +15 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +20 -0
- package/dist/server.js.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts +2 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.js +132 -0
- package/dist/ssr/__tests__/ssr-handler.test.js.map +1 -0
- package/dist/ssr/index.d.ts +98 -0
- package/dist/ssr/index.d.ts.map +1 -0
- package/dist/ssr/index.js +158 -0
- package/dist/ssr/index.js.map +1 -0
- package/dist/static-handler.d.ts +50 -0
- package/dist/static-handler.d.ts.map +1 -0
- package/dist/static-handler.gen.d.ts +5 -0
- package/dist/static-handler.gen.d.ts.map +1 -0
- package/dist/static-handler.gen.js +5 -0
- package/dist/static-handler.gen.js.map +1 -0
- package/dist/static-handler.js +29 -0
- package/dist/static-handler.js.map +1 -0
- package/dist/testing/vitest.js +82 -0
- package/dist/theme/ThemeProvider.d.ts +20 -0
- package/dist/theme/ThemeProvider.d.ts.map +1 -0
- package/dist/theme/ThemeProvider.js +240 -0
- package/dist/theme/ThemeProvider.js.map +1 -0
- package/dist/theme/ThemeScript.d.ts +48 -0
- package/dist/theme/ThemeScript.d.ts.map +1 -0
- package/dist/theme/ThemeScript.js +13 -0
- package/dist/theme/ThemeScript.js.map +1 -0
- package/dist/theme/__tests__/theme.test.d.ts +2 -0
- package/dist/theme/__tests__/theme.test.d.ts.map +1 -0
- package/dist/theme/__tests__/theme.test.js +103 -0
- package/dist/theme/__tests__/theme.test.js.map +1 -0
- package/dist/theme/constants.d.ts +29 -0
- package/dist/theme/constants.d.ts.map +1 -0
- package/dist/theme/constants.js +48 -0
- package/dist/theme/constants.js.map +1 -0
- package/dist/theme/index.d.ts +31 -0
- package/dist/theme/index.d.ts.map +1 -0
- package/dist/theme/index.js +36 -0
- package/dist/theme/index.js.map +1 -0
- package/dist/theme/theme-context.d.ts +40 -0
- package/dist/theme/theme-context.d.ts.map +1 -0
- package/dist/theme/theme-context.js +60 -0
- package/dist/theme/theme-context.js.map +1 -0
- package/dist/theme/theme-script.d.ts +27 -0
- package/dist/theme/theme-script.d.ts.map +1 -0
- package/dist/theme/theme-script.js +147 -0
- package/dist/theme/theme-script.js.map +1 -0
- package/dist/theme/types.d.ts +163 -0
- package/dist/theme/types.d.ts.map +1 -0
- package/dist/theme/types.js +11 -0
- package/dist/theme/types.js.map +1 -0
- package/dist/theme/use-theme.d.ts +12 -0
- package/dist/theme/use-theme.d.ts.map +1 -0
- package/dist/theme/use-theme.js +40 -0
- package/dist/theme/use-theme.js.map +1 -0
- package/dist/types.d.ts +1479 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +10 -0
- package/dist/types.js.map +1 -0
- package/dist/urls.d.ts +441 -0
- package/dist/urls.d.ts.map +1 -0
- package/dist/urls.gen.d.ts +8 -0
- package/dist/urls.gen.d.ts.map +1 -0
- package/dist/urls.gen.js +8 -0
- package/dist/urls.gen.js.map +1 -0
- package/dist/urls.js +443 -0
- package/dist/urls.js.map +1 -0
- package/dist/use-loader.d.ts +127 -0
- package/dist/use-loader.d.ts.map +1 -0
- package/dist/use-loader.js +237 -0
- package/dist/use-loader.js.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts +2 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js +294 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.js +224 -0
- package/dist/vite/__tests__/expose-id-utils.test.js.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js +647 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.js +39 -0
- package/dist/vite/__tests__/expose-router-id.test.js.map +1 -0
- package/dist/vite/ast-handler-extract.d.ts +49 -0
- package/dist/vite/ast-handler-extract.d.ts.map +1 -0
- package/dist/vite/ast-handler-extract.js +249 -0
- package/dist/vite/ast-handler-extract.js.map +1 -0
- package/dist/vite/expose-action-id.d.ts +19 -0
- package/dist/vite/expose-action-id.d.ts.map +1 -0
- package/dist/vite/expose-action-id.js +250 -0
- package/dist/vite/expose-action-id.js.map +1 -0
- package/dist/vite/expose-id-utils.d.ts +69 -0
- package/dist/vite/expose-id-utils.d.ts.map +1 -0
- package/dist/vite/expose-id-utils.js +289 -0
- package/dist/vite/expose-id-utils.js.map +1 -0
- package/dist/vite/expose-internal-ids.d.ts +22 -0
- package/dist/vite/expose-internal-ids.d.ts.map +1 -0
- package/dist/vite/expose-internal-ids.js +886 -0
- package/dist/vite/expose-internal-ids.js.map +1 -0
- package/dist/vite/index.d.ts +149 -0
- package/dist/vite/index.d.ts.map +1 -0
- package/dist/vite/index.js +6263 -2733
- package/dist/vite/index.js.bak +5448 -0
- package/dist/vite/index.js.map +1 -0
- package/dist/vite/package-resolution.d.ts +43 -0
- package/dist/vite/package-resolution.d.ts.map +1 -0
- package/{src/vite/package-resolution.ts → dist/vite/package-resolution.js} +53 -66
- package/dist/vite/package-resolution.js.map +1 -0
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/dist/vite/virtual-entries.d.ts +25 -0
- package/dist/vite/virtual-entries.d.ts.map +1 -0
- package/{src/vite/virtual-entries.ts → dist/vite/virtual-entries.js} +12 -16
- package/dist/vite/virtual-entries.js.map +1 -0
- package/package.json +123 -68
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +312 -0
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +486 -0
- package/skills/caching/SKILL.md +349 -24
- package/skills/composability/SKILL.md +197 -0
- package/skills/css/SKILL.md +76 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +87 -62
- package/skills/fonts/SKILL.md +6 -4
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +557 -79
- package/skills/host-router/SKILL.md +278 -0
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +175 -8
- package/skills/layout/SKILL.md +128 -5
- package/skills/links/SKILL.md +304 -25
- package/skills/loader/SKILL.md +604 -54
- package/skills/middleware/SKILL.md +213 -37
- package/skills/migrate-nextjs/SKILL.md +584 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +41 -10
- package/skills/observability/SKILL.md +172 -0
- package/skills/parallel/SKILL.md +276 -3
- package/skills/prerender/SKILL.md +432 -52
- package/skills/rango/SKILL.md +313 -21
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +248 -120
- package/skills/route/SKILL.md +287 -21
- package/skills/router-setup/SKILL.md +231 -33
- package/skills/server-actions/SKILL.md +775 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/testing/SKILL.md +129 -0
- package/skills/testing/bindings.md +89 -0
- package/skills/testing/cache-prerender.md +124 -0
- package/skills/testing/client-components.md +122 -0
- package/skills/testing/e2e-parity.md +125 -0
- package/skills/testing/flight.md +92 -0
- package/skills/testing/handles.md +129 -0
- package/skills/testing/loader.md +128 -0
- package/skills/testing/middleware.md +99 -0
- package/skills/testing/render-handler.md +121 -0
- package/skills/testing/response-routes.md +95 -0
- package/skills/testing/reverse-and-types.md +84 -0
- package/skills/testing/server-actions.md +107 -0
- package/skills/testing/server-tree.md +128 -0
- package/skills/testing/setup.md +120 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +547 -107
- package/skills/use-cache/SKILL.md +355 -0
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +116 -0
- package/src/__internal.ts +77 -44
- package/src/bin/rango.ts +312 -15
- package/src/browser/action-coordinator.ts +114 -0
- package/src/browser/action-fence.ts +47 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/cookie-name.ts +140 -0
- package/src/browser/event-controller.ts +162 -200
- package/src/browser/history-state.ts +101 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/invalidate-client-cache.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +11 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +323 -563
- package/src/browser/navigation-client.ts +219 -75
- package/src/browser/navigation-store-handle.ts +38 -0
- package/src/browser/navigation-store.ts +104 -112
- package/src/browser/navigation-transaction.ts +247 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +328 -348
- package/src/browser/prefetch/cache.ts +324 -0
- package/src/browser/prefetch/fetch.ts +357 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +194 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +194 -0
- package/src/browser/react/Link.tsx +253 -71
- package/src/browser/react/NavigationProvider.tsx +155 -34
- package/src/browser/react/ScrollRestoration.tsx +10 -6
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +53 -0
- package/src/browser/react/index.ts +0 -48
- package/src/browser/react/location-state-shared.ts +260 -60
- package/src/browser/react/location-state.ts +90 -20
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +35 -66
- package/src/browser/react/use-handle.ts +39 -126
- package/src/browser/react/use-link-status.ts +6 -9
- package/src/browser/react/use-navigation.ts +44 -68
- package/src/browser/react/use-params.ts +75 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +98 -0
- package/src/browser/react/use-search-params.ts +51 -0
- package/src/browser/react/use-segments.ts +72 -99
- package/src/browser/response-adapter.ts +124 -0
- package/src/browser/rsc-router.tsx +290 -72
- package/src/browser/scroll-restoration.ts +132 -49
- package/src/browser/segment-reconciler.ts +243 -0
- package/src/browser/segment-structure-assert.ts +17 -1
- package/src/browser/server-action-bridge.ts +621 -613
- package/src/browser/types.ts +175 -50
- package/src/browser/validate-redirect-origin.ts +56 -0
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +123 -56
- package/src/build/generate-route-types.ts +41 -1038
- package/src/build/index.ts +9 -6
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +165 -34
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +113 -0
- package/src/build/route-types/include-resolution.ts +418 -0
- package/src/build/route-types/param-extraction.ts +51 -0
- package/src/build/route-types/per-module-writer.ts +131 -0
- package/src/build/route-types/router-processing.ts +651 -0
- package/src/build/route-types/scan-filter.ts +85 -0
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +220 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-error.ts +104 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +165 -0
- package/src/cache/cache-runtime.ts +444 -0
- package/src/cache/cache-scope.ts +231 -325
- package/src/cache/cache-tag.ts +98 -0
- package/src/cache/cf/cf-cache-store.ts +2644 -75
- package/src/cache/cf/index.ts +17 -17
- package/src/cache/document-cache.ts +172 -92
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +104 -0
- package/src/cache/index.ts +11 -35
- package/src/cache/memory-segment-store.ts +307 -30
- package/src/cache/profile-registry.ts +49 -0
- package/src/cache/read-through-swr.ts +164 -0
- package/src/cache/segment-codec.ts +240 -0
- package/src/cache/tag-invalidation.ts +230 -0
- package/src/cache/taint.ts +153 -0
- package/src/cache/types.ts +94 -211
- package/src/client.rsc.tsx +8 -21
- package/src/client.tsx +123 -347
- package/src/cloudflare/index.ts +11 -0
- package/src/cloudflare/tracing.ts +109 -0
- package/src/component-utils.ts +23 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +168 -0
- package/src/debug.ts +19 -9
- package/src/decode-loader-results.ts +36 -0
- package/src/defer.ts +196 -0
- package/src/deps/ssr.ts +0 -1
- package/src/errors.ts +106 -10
- package/src/handle.ts +76 -23
- package/src/handles/MetaTags.tsx +73 -34
- package/src/handles/breadcrumbs.ts +77 -0
- package/src/handles/meta.ts +30 -52
- package/src/host/cookie-handler.ts +21 -51
- package/src/host/errors.ts +8 -32
- package/src/host/index.ts +12 -9
- package/src/host/pattern-matcher.ts +34 -77
- package/src/host/router.ts +151 -121
- package/src/host/testing.ts +45 -32
- package/src/host/types.ts +52 -11
- package/src/host/utils.ts +2 -2
- package/src/href-client.ts +192 -57
- package/src/index.rsc.ts +174 -34
- package/src/index.ts +241 -73
- package/src/internal-debug.ts +8 -4
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +31 -99
- package/src/loader.ts +30 -12
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +4 -7
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +41 -0
- package/src/prerender/param-hash.ts +14 -13
- package/src/prerender/store.ts +121 -21
- package/src/prerender.ts +445 -24
- package/src/redirect-origin.ts +100 -0
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +198 -128
- package/src/root-error-boundary.tsx +42 -48
- package/src/route-content-wrapper.tsx +10 -72
- package/src/route-definition/dsl-helpers.ts +1116 -0
- package/src/route-definition/helper-factories.ts +90 -0
- package/src/route-definition/helpers-types.ts +506 -0
- package/src/route-definition/index.ts +55 -0
- package/src/route-definition/redirect.ts +135 -0
- package/src/route-definition/resolve-handler-use.ts +161 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-definition.ts +1 -1450
- package/src/route-map-builder.ts +82 -144
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +72 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +215 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +22 -26
- package/src/router/find-match.ts +181 -0
- package/src/router/handler-context.ts +372 -125
- package/src/router/instrument.ts +308 -0
- package/src/router/intercept-resolution.ts +34 -27
- package/src/router/lazy-includes.ts +200 -0
- package/src/router/loader-resolution.ts +381 -147
- package/src/router/logging.ts +106 -6
- package/src/router/manifest.ts +88 -51
- package/src/router/match-api.ts +162 -245
- package/src/router/match-context.ts +4 -24
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +90 -89
- package/src/router/match-middleware/cache-lookup.ts +297 -150
- package/src/router/match-middleware/cache-store.ts +75 -33
- package/src/router/match-middleware/intercept-resolution.ts +44 -43
- package/src/router/match-middleware/segment-resolution.ts +64 -22
- package/src/router/match-pipelines.ts +11 -87
- package/src/router/match-result.ts +121 -50
- package/src/router/metrics.ts +219 -28
- package/src/router/middleware-types.ts +93 -0
- package/src/router/middleware.ts +421 -413
- package/src/router/navigation-snapshot.ts +131 -0
- package/src/router/params-util.ts +23 -0
- package/src/router/pattern-matching.ts +263 -79
- package/src/router/prerender-match.ts +541 -0
- package/src/router/preview-match.ts +100 -0
- package/src/router/request-classification.ts +276 -0
- package/src/router/revalidation.ts +143 -44
- package/src/router/route-snapshot.ts +244 -0
- package/src/router/router-context.ts +41 -47
- package/src/router/router-interfaces.ts +525 -0
- package/src/router/router-options.ts +726 -0
- package/src/router/router-registry.ts +21 -0
- package/src/router/segment-resolution/fresh.ts +748 -0
- package/src/router/segment-resolution/helpers.ts +313 -0
- package/src/router/segment-resolution/loader-cache.ts +207 -0
- package/src/router/segment-resolution/revalidation.ts +1322 -0
- package/src/router/segment-resolution/static-store.ts +81 -0
- package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +24 -1354
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/state-cookie-name.ts +33 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +261 -0
- package/src/router/telemetry.ts +377 -0
- package/src/router/timeout.ts +128 -0
- package/src/router/tracing.ts +206 -0
- package/src/router/trie-matching.ts +172 -60
- package/src/router/types.ts +23 -70
- package/src/router/url-params.ts +44 -0
- package/src/router.ts +748 -2376
- package/src/rsc/handler-context.ts +46 -0
- package/src/rsc/handler.ts +861 -1141
- package/src/rsc/helpers.ts +269 -19
- package/src/rsc/index.ts +1 -21
- package/src/rsc/json-route-result.ts +38 -0
- package/src/rsc/loader-fetch.ts +235 -0
- package/src/rsc/manifest-init.ts +77 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +155 -0
- package/src/rsc/progressive-enhancement.ts +413 -0
- package/src/rsc/redirect-guard.ts +99 -0
- package/src/rsc/response-error.ts +104 -0
- package/src/rsc/response-route-handler.ts +374 -0
- package/src/rsc/rsc-rendering.ts +261 -0
- package/src/rsc/runtime-warnings.ts +55 -0
- package/src/rsc/server-action.ts +376 -0
- package/src/rsc/ssr-setup.ts +144 -0
- package/src/rsc/types.ts +58 -12
- package/src/runtime-env.ts +18 -0
- package/src/search-params.ts +70 -74
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +134 -0
- package/src/segment-system.tsx +292 -134
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +439 -85
- package/src/server/cookie-store.ts +265 -0
- package/src/server/fetchable-loader-store.ts +11 -6
- package/src/server/handle-store.ts +112 -31
- package/src/server/loader-registry.ts +23 -82
- package/src/server/request-context.ts +724 -143
- package/src/server.ts +26 -164
- package/src/ssr/index.tsx +113 -36
- package/src/static-handler.ts +45 -18
- package/src/testing/cache-status.ts +162 -0
- package/src/testing/collect-handle.ts +40 -0
- package/src/testing/dispatch.ts +618 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +188 -0
- package/src/testing/e2e/index.ts +128 -0
- package/src/testing/e2e/matchers.ts +35 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +387 -0
- package/src/testing/e2e/server.ts +195 -0
- package/src/testing/flight-matchers.ts +97 -0
- package/src/testing/flight-normalize.ts +11 -0
- package/src/testing/flight-runtime.d.ts +57 -0
- package/src/testing/flight-tree.ts +682 -0
- package/src/testing/flight.entry.ts +52 -0
- package/src/testing/flight.ts +232 -0
- package/src/testing/generated-routes.ts +183 -0
- package/src/testing/index.ts +99 -0
- package/src/testing/internal/context.ts +348 -0
- package/src/testing/internal/flight-client-globals.ts +30 -0
- package/src/testing/internal/seed-vars.ts +54 -0
- package/src/testing/render-handler.ts +330 -0
- package/src/testing/render-route.tsx +566 -0
- package/src/testing/run-loader.ts +378 -0
- package/src/testing/run-middleware.ts +205 -0
- package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
- package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
- package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
- package/src/testing/vitest-stubs/version.ts +5 -0
- package/src/testing/vitest.ts +305 -0
- package/src/theme/ThemeProvider.tsx +21 -67
- package/src/theme/ThemeScript.tsx +5 -11
- package/src/theme/constants.ts +5 -14
- package/src/theme/index.ts +3 -20
- package/src/theme/theme-context.ts +5 -35
- package/src/theme/theme-script.ts +21 -32
- package/src/theme/use-theme.ts +0 -3
- package/src/types/boundaries.ts +123 -0
- package/src/types/cache-types.ts +207 -0
- package/src/types/error-types.ts +132 -0
- package/src/types/global-namespace.ts +113 -0
- package/src/types/handler-context.ts +839 -0
- package/src/types/index.ts +79 -0
- package/src/types/loader-types.ts +212 -0
- package/src/types/request-scope.ts +107 -0
- package/src/types/route-config.ts +126 -0
- package/src/types/route-entry.ts +114 -0
- package/src/types/segments.ts +171 -0
- package/src/types.ts +1 -1795
- package/src/urls/include-helper.ts +160 -0
- package/src/urls/index.ts +43 -0
- package/src/urls/path-helper-types.ts +386 -0
- package/src/urls/path-helper.ts +275 -0
- package/src/urls/pattern-types.ts +124 -0
- package/src/urls/response-types.ts +109 -0
- package/src/urls/type-extraction.ts +291 -0
- package/src/urls/urls-function.ts +81 -0
- package/src/urls.ts +1 -1323
- package/src/use-loader.tsx +406 -141
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +182 -0
- package/src/vite/discovery/discover-routers.ts +389 -0
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +467 -0
- package/src/vite/discovery/route-types-writer.ts +214 -0
- package/src/vite/discovery/self-gen-tracking.ts +73 -0
- package/src/vite/discovery/state.ts +161 -0
- package/src/vite/discovery/virtual-module-codegen.ts +183 -0
- package/src/vite/index.ts +17 -2259
- package/src/vite/plugin-types.ts +221 -0
- package/src/vite/plugins/cjs-to-esm.ts +83 -0
- package/src/vite/plugins/client-ref-dedup.ts +120 -0
- package/src/vite/plugins/client-ref-hashing.ts +118 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +88 -110
- package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +13 -87
- package/src/vite/plugins/expose-ids/export-analysis.ts +338 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +141 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +57 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +124 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +806 -0
- package/src/vite/plugins/performance-tracks.ts +89 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +313 -0
- package/src/vite/plugins/version-injector.ts +79 -0
- package/src/vite/plugins/version-plugin.ts +275 -0
- package/src/vite/plugins/virtual-entries.ts +108 -0
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +444 -0
- package/src/vite/router-discovery.ts +1581 -0
- package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +193 -37
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +132 -0
- package/src/vite/utils/client-chunks.ts +184 -0
- package/src/vite/utils/forward-user-plugins.ts +171 -0
- package/src/vite/utils/manifest-utils.ts +15 -0
- package/src/vite/utils/package-resolution.ts +89 -0
- package/src/vite/utils/prerender-utils.ts +223 -0
- package/src/vite/utils/shared-utils.ts +219 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/browser/shallow.ts +0 -35
- package/src/cache/memory-store.ts +0 -253
- package/src/router.gen.ts +0 -6
- package/src/static-handler.gen.ts +0 -5
- package/src/urls.gen.ts +0 -8
- package/src/vite/expose-internal-ids.ts +0 -1167
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
|
@@ -2,25 +2,41 @@ import type {
|
|
|
2
2
|
ServerActionBridge,
|
|
3
3
|
ServerActionBridgeConfig,
|
|
4
4
|
RscPayload,
|
|
5
|
-
ResolvedSegment,
|
|
6
|
-
NavigationStore,
|
|
7
5
|
} from "./types.js";
|
|
8
6
|
import { createPartialUpdater } from "./partial-update.js";
|
|
9
|
-
import {
|
|
7
|
+
import { enterActionFence, exitActionFence } from "./action-fence.js";
|
|
8
|
+
import { KEEP_CACHE_HEADER } from "./cookie-name.js";
|
|
9
|
+
import { createNavigationTransaction } from "./navigation-transaction.js";
|
|
10
10
|
import {
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
} from "./
|
|
14
|
-
import {
|
|
15
|
-
import {
|
|
16
|
-
import
|
|
17
|
-
|
|
18
|
-
|
|
11
|
+
reconcileSegments,
|
|
12
|
+
reconcileErrorSegments,
|
|
13
|
+
} from "./segment-reconciler.js";
|
|
14
|
+
import { startTransition } from "react";
|
|
15
|
+
import type { EventController } from "./event-controller.js";
|
|
16
|
+
import {
|
|
17
|
+
toNetworkError,
|
|
18
|
+
emitNetworkError,
|
|
19
|
+
isBackgroundSuppressible,
|
|
20
|
+
} from "./network-error-handler.js";
|
|
19
21
|
import {
|
|
20
22
|
browserDebugLog,
|
|
21
23
|
isBrowserDebugEnabled,
|
|
22
24
|
startBrowserTransaction,
|
|
23
25
|
} from "./logging.js";
|
|
26
|
+
import {
|
|
27
|
+
validateRedirectOrigin,
|
|
28
|
+
validateExternalRedirect,
|
|
29
|
+
} from "./validate-redirect-origin.js";
|
|
30
|
+
import {
|
|
31
|
+
extractRscHeaderUrl,
|
|
32
|
+
emptyResponse,
|
|
33
|
+
handleReloadHeader,
|
|
34
|
+
teeWithCompletion,
|
|
35
|
+
isForeignRouterId,
|
|
36
|
+
} from "./response-adapter.js";
|
|
37
|
+
import { mergeLocationState } from "./history-state.js";
|
|
38
|
+
import { classifyActionOutcome } from "./action-coordinator.js";
|
|
39
|
+
import { getAppVersion } from "./app-version.js";
|
|
24
40
|
|
|
25
41
|
// Polyfill Symbol.dispose/asyncDispose for Safari and older browsers
|
|
26
42
|
if (typeof Symbol.dispose === "undefined") {
|
|
@@ -30,26 +46,16 @@ if (typeof Symbol.asyncDispose === "undefined") {
|
|
|
30
46
|
(Symbol as any).asyncDispose = Symbol("Symbol.asyncDispose");
|
|
31
47
|
}
|
|
32
48
|
|
|
33
|
-
/**
|
|
34
|
-
* Normalize action ID - returns the ID as-is
|
|
35
|
-
*
|
|
36
|
-
* Server actions have IDs like "hash#actionName" or "src/actions.ts#actionName".
|
|
37
|
-
* The full ID is used for tracking in the event controller. When subscribing
|
|
38
|
-
* via useAction, both exact matching (full ID) and suffix matching (action name
|
|
39
|
-
* only) are supported by the event controller.
|
|
40
|
-
*/
|
|
41
|
-
function normalizeActionId(actionId: string): string {
|
|
42
|
-
return actionId;
|
|
43
|
-
}
|
|
44
|
-
|
|
45
49
|
/**
|
|
46
50
|
* Extended configuration for server action bridge with event controller
|
|
47
51
|
*/
|
|
48
|
-
export interface ServerActionBridgeConfigWithController
|
|
49
|
-
extends ServerActionBridgeConfig {
|
|
52
|
+
export interface ServerActionBridgeConfigWithController extends ServerActionBridgeConfig {
|
|
50
53
|
eventController: EventController;
|
|
51
|
-
/**
|
|
52
|
-
|
|
54
|
+
/** Callback to trigger SPA navigation (for action redirects) */
|
|
55
|
+
onNavigate?: (
|
|
56
|
+
url: string,
|
|
57
|
+
options?: { state?: unknown; replace?: boolean; _skipCache?: boolean },
|
|
58
|
+
) => Promise<void>;
|
|
53
59
|
}
|
|
54
60
|
|
|
55
61
|
/**
|
|
@@ -66,10 +72,37 @@ export interface ServerActionBridgeConfigWithController
|
|
|
66
72
|
* @returns ServerActionBridge instance
|
|
67
73
|
*/
|
|
68
74
|
export function createServerActionBridge(
|
|
69
|
-
config: ServerActionBridgeConfigWithController
|
|
75
|
+
config: ServerActionBridgeConfigWithController,
|
|
70
76
|
): ServerActionBridge {
|
|
71
|
-
const {
|
|
72
|
-
|
|
77
|
+
const {
|
|
78
|
+
store,
|
|
79
|
+
client,
|
|
80
|
+
eventController,
|
|
81
|
+
deps,
|
|
82
|
+
onUpdate,
|
|
83
|
+
renderSegments,
|
|
84
|
+
onNavigate,
|
|
85
|
+
} = config;
|
|
86
|
+
|
|
87
|
+
// SPA-navigate when onNavigate is set, else hard-reload. state is omitted (not
|
|
88
|
+
// passed as undefined) to match the header path's prior call shape.
|
|
89
|
+
// Callers pass an already same-origin-validated url; the hard-reload fallback
|
|
90
|
+
// re-validates defensively so this leaf cannot become an open redirect if a
|
|
91
|
+
// future caller forgets (the SPA path validates inside the navigation bridge).
|
|
92
|
+
async function dispatchRedirect(url: string, state?: unknown): Promise<void> {
|
|
93
|
+
if (onNavigate) {
|
|
94
|
+
await onNavigate(url, {
|
|
95
|
+
...(state !== undefined ? { state } : {}),
|
|
96
|
+
replace: true,
|
|
97
|
+
_skipCache: true,
|
|
98
|
+
});
|
|
99
|
+
} else {
|
|
100
|
+
const safe = validateRedirectOrigin(url, window.location.origin);
|
|
101
|
+
if (safe) {
|
|
102
|
+
window.location.href = safe;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
73
106
|
|
|
74
107
|
let isRegistered = false;
|
|
75
108
|
|
|
@@ -78,9 +111,46 @@ export function createServerActionBridge(
|
|
|
78
111
|
client,
|
|
79
112
|
onUpdate,
|
|
80
113
|
renderSegments,
|
|
81
|
-
|
|
114
|
+
getVersion: getAppVersion,
|
|
82
115
|
});
|
|
83
116
|
|
|
117
|
+
/**
|
|
118
|
+
* Refetch current route via a navigation transaction.
|
|
119
|
+
* Encapsulates the repeated pattern of creating a navTx + fetchPartialUpdate
|
|
120
|
+
* used by navigated-away, hmr-missing, and consolidation-needed scenarios.
|
|
121
|
+
*/
|
|
122
|
+
async function refetchRoute(opts?: {
|
|
123
|
+
segments?: string[];
|
|
124
|
+
interceptSourceUrl?: string | null;
|
|
125
|
+
}): Promise<void> {
|
|
126
|
+
const src = opts?.interceptSourceUrl ?? null;
|
|
127
|
+
const navTx = createNavigationTransaction(
|
|
128
|
+
store,
|
|
129
|
+
eventController,
|
|
130
|
+
window.location.href,
|
|
131
|
+
{ replace: true, skipLoadingState: true },
|
|
132
|
+
);
|
|
133
|
+
try {
|
|
134
|
+
await fetchPartialUpdate(
|
|
135
|
+
window.location.href,
|
|
136
|
+
opts?.segments ?? [],
|
|
137
|
+
false,
|
|
138
|
+
navTx.handle.signal,
|
|
139
|
+
navTx.with({
|
|
140
|
+
url: window.location.href,
|
|
141
|
+
storeOnly: true,
|
|
142
|
+
...(src ? { intercept: true, interceptSourceUrl: src } : {}),
|
|
143
|
+
}),
|
|
144
|
+
{
|
|
145
|
+
type: "action" as const,
|
|
146
|
+
...(src ? { interceptSourceUrl: src } : {}),
|
|
147
|
+
},
|
|
148
|
+
);
|
|
149
|
+
} finally {
|
|
150
|
+
navTx[Symbol.dispose]();
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
|
|
84
154
|
/**
|
|
85
155
|
* Server action callback handler
|
|
86
156
|
*/
|
|
@@ -88,291 +158,428 @@ export function createServerActionBridge(
|
|
|
88
158
|
const tx = isBrowserDebugEnabled()
|
|
89
159
|
? startBrowserTransaction("action")
|
|
90
160
|
: null;
|
|
91
|
-
|
|
161
|
+
const log = (msg: string, details?: Record<string, unknown>) => {
|
|
162
|
+
if (tx) browserDebugLog(tx, msg, details);
|
|
163
|
+
};
|
|
164
|
+
|
|
92
165
|
const locationKey = window.history.state?.key;
|
|
93
|
-
|
|
94
|
-
if (tx) {
|
|
95
|
-
browserDebugLog(tx, "action start", { id, actionId, argsCount: args.length });
|
|
96
|
-
}
|
|
166
|
+
log("action start", { id, argsCount: args.length });
|
|
97
167
|
|
|
98
168
|
// Start action in event controller - handles lifecycle tracking
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
//
|
|
107
|
-
//
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
//
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
//
|
|
114
|
-
//
|
|
115
|
-
//
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
//
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
169
|
+
const handle = eventController.startAction(id, args);
|
|
170
|
+
// Whether the action's response carried the keepClientCache() directive.
|
|
171
|
+
// Set when the response arrives; gates the deferred invalidation below.
|
|
172
|
+
let keepCache = false;
|
|
173
|
+
// Whether a Response actually settled from the network (the server saw the
|
|
174
|
+
// request). Set true as the first statement in the fetch .then() below.
|
|
175
|
+
// Gates the automatic invalidation: a pre-dispatch failure (encodeReply
|
|
176
|
+
// throw or a fetch rejection — server unreachable/DNS/connection refused)
|
|
177
|
+
// leaves this false, so finalizeAction() must NOT invalidate or broadcast —
|
|
178
|
+
// nothing reached the server, so nothing could have mutated. A failed Flight
|
|
179
|
+
// DECODE after the response arrived keeps it true (the mutation may have
|
|
180
|
+
// committed, so invalidating the now-possibly-stale client cache is correct).
|
|
181
|
+
let responseReceived = false;
|
|
182
|
+
// Single deferred invalidation + fence release, run exactly ONCE however the
|
|
183
|
+
// action terminates (normal, redirect, error, abort, intercept, concurrent).
|
|
184
|
+
// This replaces main's eager clear at action start: every directive-free
|
|
185
|
+
// action invalidates once; keepClientCache() suppresses only the automatic
|
|
186
|
+
// invalidation, so a concurrent directive-free action still invalidates via
|
|
187
|
+
// its own latch. Latched so the finally AND the early SPA-redirect returns
|
|
188
|
+
// (whose Flight stream never settles) can both call it safely.
|
|
189
|
+
let actionFinalized = false;
|
|
190
|
+
// skipInvalidation: the version-mismatch reload terminal released nothing
|
|
191
|
+
// server-side, so it releases the fence without invalidating.
|
|
192
|
+
const finalizeAction = (skipInvalidation = false): void => {
|
|
193
|
+
if (actionFinalized) return;
|
|
194
|
+
actionFinalized = true;
|
|
195
|
+
// finally so a throw in invalidation cannot leak the fence (latch is set).
|
|
196
|
+
try {
|
|
197
|
+
// responseReceived gates the automatic invalidation: a pre-dispatch
|
|
198
|
+
// failure (serialize throw / fetch reject) never reached the server, so
|
|
199
|
+
// marking the cache stale + broadcasting cross-tab would be spurious.
|
|
200
|
+
if (responseReceived && !keepCache && !skipInvalidation) {
|
|
201
|
+
store.markCacheAsStaleAndBroadcast();
|
|
202
|
+
}
|
|
203
|
+
} finally {
|
|
204
|
+
exitActionFence();
|
|
205
|
+
}
|
|
206
|
+
};
|
|
207
|
+
try {
|
|
208
|
+
const segmentState = store.getSegmentState();
|
|
209
|
+
|
|
210
|
+
// Raise the action fence (replaces the old eager clear). Nothing is wiped,
|
|
211
|
+
// rotated, or broadcast yet: navigations during the flight fetch fresh
|
|
212
|
+
// (no-store) and popstate is treated as SWR, but the decision to
|
|
213
|
+
// invalidate is deferred to the response so a no-op action (keepClientCache)
|
|
214
|
+
// can leave the caches and the jar untouched.
|
|
215
|
+
enterActionFence();
|
|
216
|
+
|
|
217
|
+
// Create temporary references for serialization
|
|
218
|
+
const temporaryReferences = deps.createTemporaryReferenceSet();
|
|
219
|
+
|
|
220
|
+
// Capture URL pathname at action start to detect navigation during action
|
|
221
|
+
// Must use window.location (not store.path) because intercepts change URL
|
|
222
|
+
// without changing store.path (e.g., /kanban -> /kanban/card/1)
|
|
223
|
+
const actionStartPathname = window.location.pathname;
|
|
224
|
+
|
|
225
|
+
// Build action request URL with current segments
|
|
226
|
+
const url = new URL(window.location.href);
|
|
227
|
+
url.searchParams.set("_rsc_action", id);
|
|
228
|
+
url.searchParams.set(
|
|
229
|
+
"_rsc_segments",
|
|
230
|
+
segmentState.currentSegmentIds.join(","),
|
|
231
|
+
);
|
|
232
|
+
// Add version param for version mismatch detection
|
|
233
|
+
const version = getAppVersion();
|
|
234
|
+
if (version) {
|
|
235
|
+
url.searchParams.set("_rsc_v", version);
|
|
236
|
+
}
|
|
237
|
+
// Add router ID for app switch detection
|
|
238
|
+
const rid = store.getRouterId?.();
|
|
239
|
+
if (rid) {
|
|
240
|
+
url.searchParams.set("_rsc_rid", rid);
|
|
241
|
+
}
|
|
129
242
|
|
|
130
|
-
|
|
131
|
-
|
|
243
|
+
// Encode arguments
|
|
244
|
+
const encodedBody = await deps.encodeReply(args, { temporaryReferences });
|
|
132
245
|
|
|
133
|
-
|
|
134
|
-
browserDebugLog(tx, "sending action request", {
|
|
246
|
+
log("sending action request", {
|
|
135
247
|
url: url.href,
|
|
136
248
|
bodyType: typeof encodedBody,
|
|
137
249
|
isFormData: encodedBody instanceof FormData,
|
|
138
250
|
segmentCount: segmentState.currentSegmentIds.length,
|
|
139
251
|
});
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
// Track when the stream completes
|
|
143
|
-
let resolveStreamComplete: () => void;
|
|
144
|
-
const streamComplete = new Promise<void>((resolve) => {
|
|
145
|
-
resolveStreamComplete = resolve;
|
|
146
|
-
});
|
|
147
|
-
|
|
148
|
-
// Get intercept source URL if in intercept context
|
|
149
|
-
const interceptSourceUrl = store.getInterceptSourceUrl();
|
|
150
|
-
|
|
151
|
-
// Track streaming token - will be set when response arrives
|
|
152
|
-
let streamingToken: { end(): void } | null = null;
|
|
153
|
-
|
|
154
|
-
// Send action request with stream tracking
|
|
155
|
-
const responsePromise = fetch(url, {
|
|
156
|
-
method: "POST",
|
|
157
|
-
headers: {
|
|
158
|
-
"rsc-action": id,
|
|
159
|
-
"X-RSC-Router-Client-Path": segmentState.currentUrl,
|
|
160
|
-
...(tx && { "X-RSC-Router-Request-Id": tx.requestId }),
|
|
161
|
-
// Send intercept source URL so server can maintain intercept context
|
|
162
|
-
...(interceptSourceUrl && {
|
|
163
|
-
"X-RSC-Router-Intercept-Source": interceptSourceUrl,
|
|
164
|
-
}),
|
|
165
|
-
},
|
|
166
|
-
body: encodedBody,
|
|
167
|
-
}).then(async (response) => {
|
|
168
|
-
// Check for version mismatch - server wants us to reload
|
|
169
|
-
const reloadUrl = response.headers.get("X-RSC-Reload");
|
|
170
|
-
if (reloadUrl) {
|
|
171
|
-
if (tx) {
|
|
172
|
-
browserDebugLog(tx, "version mismatch on action, reloading", { reloadUrl });
|
|
173
|
-
}
|
|
174
|
-
window.location.href = reloadUrl;
|
|
175
|
-
// Return a never-resolving promise to prevent further processing
|
|
176
|
-
return new Promise<Response>(() => {});
|
|
177
|
-
}
|
|
178
252
|
|
|
179
|
-
//
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
if (!response.body) {
|
|
185
|
-
// No body means stream is already complete
|
|
186
|
-
streamingToken?.end();
|
|
187
|
-
resolveStreamComplete();
|
|
188
|
-
return response;
|
|
189
|
-
}
|
|
253
|
+
// Track when the stream completes
|
|
254
|
+
let resolveStreamComplete: () => void;
|
|
255
|
+
const streamComplete = new Promise<void>((resolve) => {
|
|
256
|
+
resolveStreamComplete = resolve;
|
|
257
|
+
});
|
|
190
258
|
|
|
191
|
-
//
|
|
192
|
-
const
|
|
259
|
+
// Get intercept source URL if in intercept context
|
|
260
|
+
const interceptSourceUrl = store.getInterceptSourceUrl();
|
|
261
|
+
|
|
262
|
+
// Track streaming token - will be set when response arrives
|
|
263
|
+
let streamingToken: { end(): void } | null = null;
|
|
264
|
+
|
|
265
|
+
// Use a dedicated abort controller for the fetch so we can cancel network
|
|
266
|
+
// I/O without disrupting the Flight stream once the response has arrived.
|
|
267
|
+
// Aborting a response mid-stream causes React's Flight decoder to throw
|
|
268
|
+
// asynchronous unhandled errors (BodyStreamBuffer was aborted).
|
|
269
|
+
const fetchAbort = new AbortController();
|
|
270
|
+
const onHandleAbort = () => fetchAbort.abort();
|
|
271
|
+
handle.signal.addEventListener("abort", onHandleAbort, { once: true });
|
|
272
|
+
|
|
273
|
+
// Send action request with stream tracking
|
|
274
|
+
const responsePromise = fetch(url, {
|
|
275
|
+
method: "POST",
|
|
276
|
+
headers: {
|
|
277
|
+
"rsc-action": id,
|
|
278
|
+
"X-RSC-Router-Client-Path": segmentState.currentUrl,
|
|
279
|
+
...(tx && { "X-RSC-Router-Request-Id": tx.requestId }),
|
|
280
|
+
...(interceptSourceUrl && {
|
|
281
|
+
"X-RSC-Router-Intercept-Source": interceptSourceUrl,
|
|
282
|
+
}),
|
|
283
|
+
},
|
|
284
|
+
body: encodedBody,
|
|
285
|
+
signal: fetchAbort.signal,
|
|
286
|
+
}).then(async (response) => {
|
|
287
|
+
// A settled fetch promise means the request reached the server and a
|
|
288
|
+
// Response came back (true for 2xx, 4xx, AND 5xx — fetch only rejects
|
|
289
|
+
// on network-layer failure, never on HTTP status). Record it as the
|
|
290
|
+
// first statement so every downstream terminal can invalidate; a
|
|
291
|
+
// pre-dispatch failure never gets here and stays gated out.
|
|
292
|
+
responseReceived = true;
|
|
293
|
+
// Response arrived — disconnect fetch abort from handle abort so
|
|
294
|
+
// abortAllActions() doesn't disrupt the in-progress Flight stream.
|
|
295
|
+
handle.signal.removeEventListener("abort", onHandleAbort);
|
|
296
|
+
|
|
297
|
+
// Did the action call keepClientCache()? If so the deferred invalidation
|
|
298
|
+
// below is suppressed for THIS action (a concurrent directive-free
|
|
299
|
+
// action still invalidates via its own response).
|
|
300
|
+
keepCache = response.headers.get(KEEP_CACHE_HEADER) === "1";
|
|
301
|
+
|
|
302
|
+
// Check for version mismatch - server wants us to reload
|
|
303
|
+
const reloadResult = handleReloadHeader(response, {
|
|
304
|
+
onBlocked: resolveStreamComplete,
|
|
305
|
+
onReload: (url) => {
|
|
306
|
+
log("version mismatch on action, reloading", { reloadUrl: url });
|
|
307
|
+
// Never-settling terminal (navigates away), so the finally never
|
|
308
|
+
// runs: release the fence here. skipInvalidation — the mismatch
|
|
309
|
+
// short-circuits the action server-side, so nothing mutated and a
|
|
310
|
+
// broadcast would only risk hard-reloading a sibling mid-task.
|
|
311
|
+
finalizeAction(true);
|
|
312
|
+
},
|
|
313
|
+
});
|
|
314
|
+
if (reloadResult) return reloadResult;
|
|
315
|
+
|
|
316
|
+
// Simple redirect from action (no state, no RSC payload).
|
|
317
|
+
// Short-circuits before createFromFetch — no Flight deserialization needed.
|
|
318
|
+
// Check handle.signal.aborted to avoid redirecting from a stale action
|
|
319
|
+
// when the user has already navigated away.
|
|
320
|
+
const redirect = extractRscHeaderUrl(response, "X-RSC-Redirect");
|
|
321
|
+
if (redirect && redirect !== "blocked" && !handle.signal.aborted) {
|
|
322
|
+
log("action simple redirect", { url: redirect.url });
|
|
323
|
+
handle.complete(undefined);
|
|
324
|
+
// This path returns a never-settling promise, so the finally never
|
|
325
|
+
// runs: invalidate + release the fence here (the mutation committed
|
|
326
|
+
// and we're navigating away). Latched, so the finally is a no-op.
|
|
327
|
+
finalizeAction();
|
|
328
|
+
await dispatchRedirect(redirect.url);
|
|
329
|
+
return new Promise<Response>(() => {});
|
|
330
|
+
}
|
|
331
|
+
if (redirect === "blocked") {
|
|
332
|
+
resolveStreamComplete();
|
|
333
|
+
return emptyResponse();
|
|
334
|
+
}
|
|
193
335
|
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
336
|
+
// Integrity check (pre-decode): a foreign app's action response must
|
|
337
|
+
// not be decoded + applied here. This is the one decode-and-apply path
|
|
338
|
+
// the post-decode partial-update guard does NOT cover (the action
|
|
339
|
+
// bridge has its own createFromFetch -> onUpdate). Ordered after the
|
|
340
|
+
// reload/redirect handlers, which steer control responses first.
|
|
341
|
+
// Reloads via window.location.reload() rather than navigating to a
|
|
342
|
+
// target (as the navigation-client guard does): an action has no
|
|
343
|
+
// navigation target, so reloading the current URL re-syncs the
|
|
344
|
+
// document against the server-applied action effect.
|
|
345
|
+
if (
|
|
346
|
+
!handle.signal.aborted &&
|
|
347
|
+
isForeignRouterId(response, store.getRouterId?.())
|
|
348
|
+
) {
|
|
349
|
+
log("action router id mismatch, reloading to re-sync");
|
|
350
|
+
handle.complete(undefined);
|
|
208
351
|
resolveStreamComplete();
|
|
352
|
+
// Never-settling return: release the fence before the reload (the
|
|
353
|
+
// reload resets module state anyway, but stay balanced). Latched.
|
|
354
|
+
finalizeAction();
|
|
355
|
+
window.location.reload();
|
|
356
|
+
return new Promise<Response>(() => {});
|
|
209
357
|
}
|
|
210
|
-
})().catch((error) => {
|
|
211
|
-
console.error("[STREAMING] Error reading tracking stream:", error);
|
|
212
|
-
streamingToken?.end();
|
|
213
|
-
});
|
|
214
358
|
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
statusText: response.statusText,
|
|
220
|
-
});
|
|
221
|
-
});
|
|
359
|
+
// Start streaming immediately when response arrives
|
|
360
|
+
if (!handle.signal.aborted) {
|
|
361
|
+
streamingToken = handle.startStreaming();
|
|
362
|
+
}
|
|
222
363
|
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
364
|
+
return teeWithCompletion(response, () => {
|
|
365
|
+
log("stream complete");
|
|
366
|
+
streamingToken?.end();
|
|
367
|
+
resolveStreamComplete();
|
|
368
|
+
});
|
|
228
369
|
});
|
|
229
|
-
} catch (error) {
|
|
230
|
-
// Clean up streaming token on error (may be null if fetch failed before .then() ran)
|
|
231
|
-
// The token is assigned in .then() callback which runs before this catch block,
|
|
232
|
-
// but TypeScript doesn't track cross-async assignments, so use type assertion
|
|
233
|
-
(streamingToken as { end(): void } | null)?.end();
|
|
234
|
-
// resolveStreamComplete is assigned in the Promise constructor so it's safe to call
|
|
235
|
-
resolveStreamComplete!();
|
|
236
|
-
|
|
237
|
-
// Convert network-level errors to NetworkError for proper handling
|
|
238
|
-
if (isNetworkError(error)) {
|
|
239
|
-
const networkError = new NetworkError(
|
|
240
|
-
"Unable to connect to server. Please check your connection.",
|
|
241
|
-
{
|
|
242
|
-
cause: error,
|
|
243
|
-
url: url.toString(),
|
|
244
|
-
operation: "action",
|
|
245
|
-
}
|
|
246
|
-
);
|
|
247
370
|
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
startTransition(() => {
|
|
254
|
-
onUpdate({
|
|
255
|
-
root: createElement(NetworkErrorThrower, { error: networkError }),
|
|
256
|
-
metadata: {
|
|
257
|
-
pathname: segmentState.currentUrl,
|
|
258
|
-
segments: [],
|
|
259
|
-
isError: true,
|
|
260
|
-
},
|
|
261
|
-
});
|
|
371
|
+
// Deserialize response (MUST use same temporaryReferences)
|
|
372
|
+
let payload: RscPayload;
|
|
373
|
+
try {
|
|
374
|
+
payload = await deps.createFromFetch<RscPayload>(responsePromise, {
|
|
375
|
+
temporaryReferences,
|
|
262
376
|
});
|
|
377
|
+
} catch (error) {
|
|
378
|
+
// Clean up streaming token on error (may be null if fetch failed before .then() ran)
|
|
379
|
+
// The token is assigned in .then() callback which runs before this catch block,
|
|
380
|
+
// but TypeScript doesn't track cross-async assignments, so use type assertion
|
|
381
|
+
(streamingToken as { end(): void } | null)?.end();
|
|
382
|
+
// resolveStreamComplete is assigned in the Promise constructor so it's safe to call
|
|
383
|
+
resolveStreamComplete!();
|
|
384
|
+
|
|
385
|
+
// Silently swallow abort errors — the action was intentionally cancelled
|
|
386
|
+
// (e.g., user navigated away or abortAllActions was called).
|
|
387
|
+
// Return undefined instead of throwing to avoid surfacing as a page error.
|
|
388
|
+
// Check both DOMException AbortError and stream-level abort messages
|
|
389
|
+
// (BodyStreamBuffer was aborted) that propagate from the aborted fetch.
|
|
390
|
+
if (handle.signal.aborted) {
|
|
391
|
+
return undefined;
|
|
392
|
+
}
|
|
263
393
|
|
|
264
|
-
|
|
394
|
+
// Convert network-level errors to NetworkError for proper handling
|
|
395
|
+
const networkError = toNetworkError(error, {
|
|
396
|
+
url: url.toString(),
|
|
397
|
+
operation: "action",
|
|
398
|
+
});
|
|
399
|
+
if (networkError) {
|
|
400
|
+
handle.fail(networkError);
|
|
401
|
+
emitNetworkError(onUpdate, networkError, segmentState.currentUrl);
|
|
402
|
+
throw networkError;
|
|
403
|
+
}
|
|
404
|
+
throw error;
|
|
265
405
|
}
|
|
266
|
-
throw error;
|
|
267
|
-
}
|
|
268
406
|
|
|
269
|
-
|
|
270
|
-
browserDebugLog(tx, "action response received", {
|
|
407
|
+
log("action response received", {
|
|
271
408
|
isPartial: payload.metadata?.isPartial,
|
|
272
409
|
isError: payload.metadata?.isError,
|
|
273
410
|
matchedCount: payload.metadata?.matched?.length ?? 0,
|
|
274
411
|
diffCount: payload.metadata?.diff?.length ?? 0,
|
|
275
412
|
});
|
|
276
|
-
|
|
413
|
+
// Guard: if the action was aborted while streaming (e.g., user navigated
|
|
414
|
+
// away or abortAllActions fired), bail out before any reconcile/render/cache
|
|
415
|
+
// writes to avoid overwriting the current UI with stale action results.
|
|
416
|
+
if (handle.signal.aborted) {
|
|
417
|
+
log("action aborted after response, skipping reconciliation");
|
|
418
|
+
return undefined;
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
// Process response
|
|
422
|
+
const { metadata, returnValue } = payload;
|
|
423
|
+
|
|
424
|
+
// Handle action redirect: server converted the redirect to a Flight payload
|
|
425
|
+
// so we can perform SPA navigation instead of a full page reload.
|
|
426
|
+
// Check handle.signal.aborted to avoid redirecting from a stale action
|
|
427
|
+
// when the user has already navigated away.
|
|
428
|
+
if (metadata?.redirect && !handle.signal.aborted) {
|
|
429
|
+
// Explicit off-host redirect (redirect(url, { external: true })):
|
|
430
|
+
// hard-navigate, but still scheme-validate (http/https only). external
|
|
431
|
+
// waives the same-origin check, NOT scheme safety, so a forged payload
|
|
432
|
+
// carrying a javascript:/data: URL cannot script via location.assign.
|
|
433
|
+
if (metadata.redirect.external) {
|
|
434
|
+
const externalUrl = validateExternalRedirect(
|
|
435
|
+
metadata.redirect.url,
|
|
436
|
+
window.location.origin,
|
|
437
|
+
);
|
|
438
|
+
if (!externalUrl) {
|
|
439
|
+
log("blocked external action redirect payload", {
|
|
440
|
+
url: metadata.redirect.url,
|
|
441
|
+
});
|
|
442
|
+
handle.complete(returnValue?.data);
|
|
443
|
+
return returnValue?.data;
|
|
444
|
+
}
|
|
445
|
+
log("external action redirect", { url: externalUrl });
|
|
446
|
+
handle.complete(returnValue?.data);
|
|
447
|
+
window.location.assign(externalUrl);
|
|
448
|
+
return returnValue?.data;
|
|
449
|
+
}
|
|
450
|
+
const redirectUrl = validateRedirectOrigin(
|
|
451
|
+
metadata.redirect.url,
|
|
452
|
+
window.location.origin,
|
|
453
|
+
);
|
|
454
|
+
if (!redirectUrl) {
|
|
455
|
+
log("blocked action redirect payload", {
|
|
456
|
+
url: metadata.redirect.url,
|
|
457
|
+
});
|
|
458
|
+
handle.complete(returnValue?.data);
|
|
459
|
+
return returnValue?.data;
|
|
460
|
+
}
|
|
461
|
+
log("action redirect", { url: redirectUrl });
|
|
462
|
+
handle.complete(returnValue?.data);
|
|
463
|
+
await dispatchRedirect(redirectUrl, metadata.locationState);
|
|
464
|
+
return returnValue?.data;
|
|
465
|
+
}
|
|
277
466
|
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
467
|
+
// Bail out if the action was aborted after deserialization (e.g. user
|
|
468
|
+
// navigated away or abortAllActions was called while the Flight stream
|
|
469
|
+
// was being consumed). Without this check the code below would mutate
|
|
470
|
+
// the store / UI for a stale action.
|
|
471
|
+
if (handle.signal.aborted) {
|
|
472
|
+
log("action aborted after deserialization, skipping mutations");
|
|
473
|
+
return returnValue?.data;
|
|
474
|
+
}
|
|
475
|
+
|
|
476
|
+
const { matched, diff, segments, isPartial, isError } = metadata || {};
|
|
281
477
|
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
console.log(`[Browser] Action result:`, returnValue);
|
|
285
|
-
if (!returnValue.ok) {
|
|
478
|
+
// Log action result
|
|
479
|
+
if (returnValue && !returnValue.ok) {
|
|
286
480
|
console.error(`[Browser] Action failed:`, returnValue.data);
|
|
287
481
|
}
|
|
288
|
-
}
|
|
289
482
|
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
483
|
+
// Handle error responses with error boundary UI
|
|
484
|
+
if (isError && isPartial && segments && diff) {
|
|
485
|
+
log("processing error boundary response");
|
|
293
486
|
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
487
|
+
// Fail current handle BEFORE aborting all actions so the event controller
|
|
488
|
+
// records the error state (abortAllActions clears inflight entries)
|
|
489
|
+
if (returnValue && !returnValue.ok) {
|
|
490
|
+
handle.fail(returnValue.data);
|
|
491
|
+
}
|
|
297
492
|
|
|
298
|
-
|
|
299
|
-
|
|
493
|
+
// Abort all other pending action requests - error takes precedence
|
|
494
|
+
// This prevents other actions from completing and overwriting the error UI
|
|
495
|
+
eventController.abortAllActions();
|
|
300
496
|
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
const cached = store.getCachedSegments(currentKey);
|
|
304
|
-
const cachedSegments = cached?.segments || [];
|
|
497
|
+
// Clear concurrent action tracking - no consolidation needed when showing error
|
|
498
|
+
handle.clearConsolidation();
|
|
305
499
|
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
500
|
+
// Get current page's cached segments
|
|
501
|
+
const currentKey = store.getHistoryKey();
|
|
502
|
+
const cached = store.getCachedSegments(currentKey);
|
|
503
|
+
const cachedSegments = cached?.segments || [];
|
|
504
|
+
|
|
505
|
+
// Reconcile error segments with cached tree
|
|
506
|
+
const errorResult = reconcileErrorSegments(cachedSegments, segments);
|
|
507
|
+
|
|
508
|
+
// Render the full tree with error segment merged with parent layouts
|
|
509
|
+
const errorTree = await renderSegments(errorResult.mainSegments, {
|
|
510
|
+
isAction: true,
|
|
511
|
+
interceptSegments:
|
|
512
|
+
errorResult.interceptSegments.length > 0
|
|
513
|
+
? errorResult.interceptSegments
|
|
514
|
+
: undefined,
|
|
515
|
+
});
|
|
318
516
|
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
});
|
|
517
|
+
// Re-check route stability after async renderSegments — user may have
|
|
518
|
+
// navigated away while the error tree was being prepared.
|
|
519
|
+
if (window.location.pathname !== actionStartPathname) {
|
|
520
|
+
log("user navigated during error render, skipping");
|
|
521
|
+
if (returnValue && !returnValue.ok) {
|
|
522
|
+
throw returnValue.data;
|
|
523
|
+
}
|
|
524
|
+
handle.complete(undefined);
|
|
525
|
+
return undefined;
|
|
526
|
+
}
|
|
527
|
+
const currentKeyNow = store.getHistoryKey();
|
|
528
|
+
if (currentKeyNow !== currentKey) {
|
|
529
|
+
log("history key changed during error render, skipping cache update");
|
|
530
|
+
if (returnValue && !returnValue.ok) {
|
|
531
|
+
throw returnValue.data;
|
|
532
|
+
}
|
|
533
|
+
handle.complete(undefined);
|
|
534
|
+
return undefined;
|
|
535
|
+
}
|
|
339
536
|
|
|
340
|
-
|
|
537
|
+
// Update UI with error boundary
|
|
538
|
+
startTransition(() => {
|
|
539
|
+
onUpdate({ root: errorTree, metadata: metadata! });
|
|
540
|
+
});
|
|
341
541
|
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
542
|
+
// Update segment tracking to exclude error segment IDs
|
|
543
|
+
const errorSegmentIds = new Set(diff);
|
|
544
|
+
const segmentIdsAfterError = segmentState.currentSegmentIds.filter(
|
|
545
|
+
(id) => !errorSegmentIds.has(id),
|
|
546
|
+
);
|
|
347
547
|
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
548
|
+
// Update store state
|
|
549
|
+
store.setSegmentIds(segmentIdsAfterError);
|
|
550
|
+
const currentHandleData = eventController.getHandleState().data;
|
|
551
|
+
store.cacheSegmentsForHistory(
|
|
552
|
+
currentKey,
|
|
553
|
+
errorResult.segments,
|
|
554
|
+
currentHandleData,
|
|
555
|
+
);
|
|
352
556
|
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
557
|
+
// Throw the error so the action promise rejects
|
|
558
|
+
if (returnValue && !returnValue.ok) {
|
|
559
|
+
throw returnValue.data;
|
|
560
|
+
}
|
|
357
561
|
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
throw returnValue.data;
|
|
562
|
+
// No error in returnValue (shouldn't happen with isError: true)
|
|
563
|
+
handle.complete(undefined);
|
|
564
|
+
return undefined;
|
|
362
565
|
}
|
|
363
566
|
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
567
|
+
if (!isPartial) {
|
|
568
|
+
// Protocol invariant: action revalidation responses MUST be partial.
|
|
569
|
+
// The server always sends isPartial: true for successful revalidation
|
|
570
|
+
// and isPartial: true + isError: true for error boundary responses.
|
|
571
|
+
// A non-partial payload here indicates a server-side bug.
|
|
572
|
+
throw new Error(
|
|
573
|
+
`[Browser] Action response missing isPartial — the server must ` +
|
|
574
|
+
`always send partial payloads for action revalidation.`,
|
|
575
|
+
);
|
|
576
|
+
}
|
|
368
577
|
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
);
|
|
375
|
-
console.log(`[Browser] Server expects client to have:`, matched);
|
|
578
|
+
log("processing partial update", {
|
|
579
|
+
serverSegments: segments?.length ?? 0,
|
|
580
|
+
diff: diff?.join(", ") ?? "",
|
|
581
|
+
matched: matched?.join(", ") ?? "",
|
|
582
|
+
});
|
|
376
583
|
|
|
377
584
|
// Record revalidated segments for concurrent action tracking
|
|
378
585
|
if (diff) {
|
|
@@ -383,375 +590,188 @@ export function createServerActionBridge(
|
|
|
383
590
|
const currentKey = store.getHistoryKey();
|
|
384
591
|
const cached = store.getCachedSegments(currentKey);
|
|
385
592
|
const cachedSegments = cached?.segments || [];
|
|
386
|
-
const currentSegmentMap = new Map<string, ResolvedSegment>();
|
|
387
|
-
cachedSegments.forEach((s) => currentSegmentMap.set(s.id, s));
|
|
388
|
-
|
|
389
|
-
console.log(
|
|
390
|
-
`[Browser] Client cache has ${currentSegmentMap.size} entries:`,
|
|
391
|
-
Array.from(currentSegmentMap.keys())
|
|
392
|
-
);
|
|
393
|
-
|
|
394
|
-
// Create lookup for new segments from server
|
|
395
|
-
const newSegmentMap = new Map<string, ResolvedSegment>();
|
|
396
|
-
(segments || []).forEach((s: ResolvedSegment) =>
|
|
397
|
-
newSegmentMap.set(s.id, s)
|
|
398
|
-
);
|
|
399
593
|
|
|
400
594
|
if (!matched) {
|
|
401
|
-
console.log(`[Browser] Matched segments: ${matched}`);
|
|
402
595
|
throw new Error("No matched segments in response");
|
|
403
596
|
}
|
|
404
597
|
|
|
405
|
-
//
|
|
406
|
-
const
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
return mergeSegmentLoaders(fromServer, fromCache);
|
|
415
|
-
}
|
|
416
|
-
// When server returns component: null for a layout segment, it means
|
|
417
|
-
// "this segment doesn't need re-rendering" - preserve the cached component
|
|
418
|
-
// to maintain the outlet chain and prevent React tree changes
|
|
419
|
-
const cached = currentSegmentMap.get(segId); // Re-fetch to avoid type narrowing issues
|
|
420
|
-
if (
|
|
421
|
-
fromServer.component === null &&
|
|
422
|
-
fromServer.type === "layout" &&
|
|
423
|
-
cached?.component != null
|
|
424
|
-
) {
|
|
425
|
-
console.log(
|
|
426
|
-
`[Browser] Preserving cached component for layout ${segId} (server returned null)`
|
|
427
|
-
);
|
|
428
|
-
return { ...fromServer, component: cached.component };
|
|
429
|
-
}
|
|
430
|
-
// Dev-mode assertion: warn if tree structure would change
|
|
431
|
-
if (cached) {
|
|
432
|
-
assertSegmentStructure(cached, fromServer, "action-bridge");
|
|
433
|
-
}
|
|
434
|
-
// Preserve cached loading value to maintain consistent tree structure.
|
|
435
|
-
// SSR may set loading=false for skipSSR routes, but actions set
|
|
436
|
-
// loading=<skeleton> (isSSR=false). Changing loading between renders
|
|
437
|
-
// alters the React tree (with/without RouteContentWrapper), causing
|
|
438
|
-
// remounts that destroy useActionState.
|
|
439
|
-
if (
|
|
440
|
-
cached &&
|
|
441
|
-
cached.loading !== undefined &&
|
|
442
|
-
fromServer.loading !== cached.loading
|
|
443
|
-
) {
|
|
444
|
-
return { ...fromServer, loading: cached.loading };
|
|
445
|
-
}
|
|
446
|
-
return fromServer;
|
|
447
|
-
}
|
|
448
|
-
|
|
449
|
-
// Fall back to current page's cached segments
|
|
450
|
-
if (!fromCache) {
|
|
451
|
-
console.error(`[Browser] MISSING SEGMENT: ${segId} not in cache!`);
|
|
452
|
-
}
|
|
453
|
-
return fromCache;
|
|
454
|
-
})
|
|
455
|
-
.filter(Boolean) as ResolvedSegment[];
|
|
456
|
-
|
|
457
|
-
console.log(
|
|
458
|
-
`[Browser] Rebuilt ${fullSegments.length} segments from matched array`
|
|
459
|
-
);
|
|
598
|
+
// Reconcile server segments with cached segments (single source of truth)
|
|
599
|
+
const reconciled = reconcileSegments({
|
|
600
|
+
actor: "action",
|
|
601
|
+
matched,
|
|
602
|
+
diff: diff || [],
|
|
603
|
+
serverSegments: segments || [],
|
|
604
|
+
cachedSegments,
|
|
605
|
+
});
|
|
606
|
+
const fullSegments = reconciled.segments;
|
|
460
607
|
|
|
461
608
|
const returnData = returnValue?.data;
|
|
462
609
|
|
|
463
|
-
console.log(
|
|
464
|
-
`[Browser] Action complete - UI updated (after action state committed)`
|
|
465
|
-
);
|
|
466
|
-
|
|
467
610
|
if (returnValue && !returnValue.ok) {
|
|
468
611
|
handle.fail(returnValue.data);
|
|
469
612
|
throw returnValue.data;
|
|
470
613
|
}
|
|
471
614
|
|
|
472
|
-
//
|
|
473
|
-
const
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
615
|
+
// Classify the post-reconciliation scenario
|
|
616
|
+
const scenario = classifyActionOutcome({
|
|
617
|
+
handleId: handle.id,
|
|
618
|
+
inflightActions: eventController.getInflightActions(),
|
|
619
|
+
hadAnyConcurrentActions: eventController.hadAnyConcurrentActions(),
|
|
620
|
+
revalidatedSegments: handle.getRevalidatedSegments(),
|
|
621
|
+
actionStartPathname,
|
|
622
|
+
currentPathname: window.location.pathname,
|
|
623
|
+
actionStartLocationKey: locationKey,
|
|
624
|
+
currentLocationKey: window.history.state?.key,
|
|
625
|
+
reconciledSegmentCount: fullSegments.length,
|
|
626
|
+
matchedCount: matched.length,
|
|
627
|
+
currentInterceptSource: store.getInterceptSourceUrl(),
|
|
628
|
+
});
|
|
485
629
|
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
630
|
+
switch (scenario.type) {
|
|
631
|
+
case "navigated-away": {
|
|
632
|
+
log("user navigated away during action", {
|
|
633
|
+
from: actionStartPathname,
|
|
634
|
+
to: window.location.pathname,
|
|
635
|
+
historyKeyChanged: scenario.historyKeyChanged,
|
|
636
|
+
});
|
|
637
|
+
// Clear concurrent action tracking - don't consolidate for old route's segments
|
|
638
|
+
handle.clearConsolidation();
|
|
639
|
+
|
|
640
|
+
if (scenario.historyKeyChanged) {
|
|
641
|
+
// Invalidation is deferred to finalizeAction(); here we only trigger
|
|
642
|
+
// the revalidation refetch of the new route (suppressed on keep).
|
|
643
|
+
if (!scenario.onInterceptRoute && !keepCache) {
|
|
644
|
+
refetchRoute().catch((error) => {
|
|
645
|
+
if (isBackgroundSuppressible(error)) return;
|
|
646
|
+
console.error(
|
|
647
|
+
"[Browser] Background revalidation failed:",
|
|
648
|
+
error,
|
|
649
|
+
);
|
|
650
|
+
});
|
|
651
|
+
}
|
|
652
|
+
break;
|
|
653
|
+
}
|
|
492
654
|
|
|
493
|
-
//
|
|
494
|
-
//
|
|
495
|
-
//
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
// - NO if user is on an intercept route (would lose background segments)
|
|
500
|
-
const currentInterceptSource = store.getInterceptSourceUrl();
|
|
501
|
-
if (currentInterceptSource) {
|
|
502
|
-
// User is on an intercept route - skip revalidation to preserve background
|
|
503
|
-
console.log(
|
|
504
|
-
`[Browser] Skipping background revalidation - user on intercept route`
|
|
505
|
-
);
|
|
506
|
-
} else {
|
|
507
|
-
// User is on a non-intercept route - safe to revalidate
|
|
508
|
-
console.log(
|
|
509
|
-
`[Browser] History key changed, triggering background revalidation`
|
|
510
|
-
);
|
|
511
|
-
store.markCacheAsStaleAndBroadcast();
|
|
512
|
-
using navTx = createNavigationTransaction(
|
|
513
|
-
store,
|
|
514
|
-
eventController,
|
|
515
|
-
window.location.href,
|
|
516
|
-
{ replace: true, skipLoadingState: true }
|
|
517
|
-
);
|
|
518
|
-
fetchPartialUpdate(
|
|
519
|
-
window.location.href,
|
|
520
|
-
[],
|
|
521
|
-
false,
|
|
522
|
-
navTx.handle.signal,
|
|
523
|
-
navTx.with({
|
|
524
|
-
url: window.location.href,
|
|
525
|
-
storeOnly: true,
|
|
526
|
-
}),
|
|
527
|
-
{
|
|
528
|
-
isAction: true,
|
|
529
|
-
}
|
|
530
|
-
).then(() => {
|
|
531
|
-
console.log(`[Browser] Background revalidation complete`);
|
|
655
|
+
// Same history key but different pathname - safe to refetch current
|
|
656
|
+
// route. Invalidation is deferred to finalizeAction(); here we only
|
|
657
|
+
// trigger the revalidation refetch (suppressed on keep).
|
|
658
|
+
if (!keepCache) {
|
|
659
|
+
await refetchRoute({
|
|
660
|
+
interceptSourceUrl: store.getInterceptSourceUrl(),
|
|
532
661
|
});
|
|
533
662
|
}
|
|
534
|
-
|
|
535
|
-
handle.complete(returnData);
|
|
536
|
-
return returnData;
|
|
663
|
+
break;
|
|
537
664
|
}
|
|
538
665
|
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
{
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
const currentInterceptSource = store.getInterceptSourceUrl();
|
|
551
|
-
await fetchPartialUpdate(
|
|
552
|
-
window.location.href,
|
|
553
|
-
[], // Empty array = refetch all segments for current route
|
|
554
|
-
false,
|
|
555
|
-
navTx.handle.signal,
|
|
556
|
-
navTx.with({
|
|
557
|
-
url: window.location.href,
|
|
558
|
-
storeOnly: true,
|
|
559
|
-
intercept: !!currentInterceptSource,
|
|
560
|
-
interceptSourceUrl: currentInterceptSource ?? undefined,
|
|
561
|
-
}),
|
|
562
|
-
{
|
|
563
|
-
isAction: true,
|
|
564
|
-
interceptSourceUrl: currentInterceptSource ?? undefined,
|
|
565
|
-
}
|
|
566
|
-
);
|
|
567
|
-
console.log(`[Browser] Refetch after navigation complete`);
|
|
568
|
-
handle.complete(returnData);
|
|
569
|
-
return returnData;
|
|
570
|
-
}
|
|
571
|
-
|
|
572
|
-
// HMR resilience check - only runs if user DIDN'T navigate away
|
|
573
|
-
if (fullSegments.length < matched.length) {
|
|
574
|
-
console.warn(
|
|
575
|
-
`[Browser] Missing segments after action (HMR detected), refetching...`
|
|
576
|
-
);
|
|
577
|
-
|
|
578
|
-
using navTx = createNavigationTransaction(
|
|
579
|
-
store,
|
|
580
|
-
eventController,
|
|
581
|
-
window.location.href,
|
|
582
|
-
{ replace: true, skipLoadingState: true }
|
|
583
|
-
);
|
|
584
|
-
await fetchPartialUpdate(
|
|
585
|
-
window.location.href,
|
|
586
|
-
[],
|
|
587
|
-
false,
|
|
588
|
-
navTx.handle.signal,
|
|
589
|
-
navTx.with({
|
|
590
|
-
url: window.location.href,
|
|
591
|
-
storeOnly: true,
|
|
592
|
-
intercept: !!interceptSourceUrl,
|
|
593
|
-
interceptSourceUrl: interceptSourceUrl ?? undefined,
|
|
594
|
-
}),
|
|
595
|
-
{
|
|
596
|
-
isAction: true,
|
|
597
|
-
interceptSourceUrl: interceptSourceUrl ?? undefined,
|
|
598
|
-
}
|
|
599
|
-
);
|
|
600
|
-
console.log(
|
|
601
|
-
`[Browser] Refetch complete (HMR), now returning action result`
|
|
602
|
-
);
|
|
666
|
+
case "hmr-missing": {
|
|
667
|
+
console.warn(
|
|
668
|
+
`[Browser] Missing segments after action (HMR detected), refetching...`,
|
|
669
|
+
);
|
|
670
|
+
// Repair (not revalidation), so ungated on keepCache: a keep action
|
|
671
|
+
// resolving last must discharge a directive-free sibling's repair.
|
|
672
|
+
// See the keep row in docs/design/rango-state-cookie.md (the all-keep
|
|
673
|
+
// edge, and the benign re-mark-stale-after-refetch end-state delta).
|
|
674
|
+
await refetchRoute({ interceptSourceUrl });
|
|
675
|
+
break;
|
|
676
|
+
}
|
|
603
677
|
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
678
|
+
case "consolidation-needed": {
|
|
679
|
+
log("consolidation fetch needed", {
|
|
680
|
+
segmentIds: scenario.segmentIds,
|
|
681
|
+
});
|
|
682
|
+
// Calculate segments to send (exclude the ones we want fresh)
|
|
683
|
+
const currentSegmentIds = store.getSegmentState().currentSegmentIds;
|
|
684
|
+
const segmentsToSend = currentSegmentIds.filter(
|
|
685
|
+
(sid) => !scenario.segmentIds.includes(sid),
|
|
686
|
+
);
|
|
609
687
|
|
|
610
|
-
|
|
611
|
-
|
|
688
|
+
// Clear consolidation tracking before fetch
|
|
689
|
+
handle.clearConsolidation();
|
|
612
690
|
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
const currentSegmentIds = store.getSegmentState().currentSegmentIds;
|
|
621
|
-
const segmentsToSend = currentSegmentIds.filter(
|
|
622
|
-
(id) => !consolidationSegments.includes(id)
|
|
623
|
-
);
|
|
691
|
+
// Ungated on keepCache, same as hmr-missing above (see the keep row).
|
|
692
|
+
await refetchRoute({
|
|
693
|
+
segments: segmentsToSend,
|
|
694
|
+
interceptSourceUrl,
|
|
695
|
+
});
|
|
696
|
+
break;
|
|
697
|
+
}
|
|
624
698
|
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
699
|
+
case "concurrent-skip": {
|
|
700
|
+
log("skipping UI update, other actions fetching", {
|
|
701
|
+
otherCount: scenario.otherFetchingCount,
|
|
702
|
+
});
|
|
703
|
+
// Only update store if history key hasn't changed (user didn't navigate away)
|
|
704
|
+
const currentKeyNow = store.getHistoryKey();
|
|
705
|
+
if (currentKeyNow === currentKey) {
|
|
706
|
+
store.setSegmentIds(matched);
|
|
707
|
+
const currentHandleData = eventController.getHandleState().data;
|
|
708
|
+
store.cacheSegmentsForHistory(
|
|
709
|
+
currentKey,
|
|
710
|
+
fullSegments,
|
|
711
|
+
currentHandleData,
|
|
712
|
+
);
|
|
713
|
+
}
|
|
714
|
+
break;
|
|
715
|
+
}
|
|
629
716
|
|
|
630
|
-
|
|
631
|
-
|
|
717
|
+
case "normal": {
|
|
718
|
+
// Prepare new tree (await loader data resolution)
|
|
719
|
+
const newTree = await renderSegments(reconciled.mainSegments, {
|
|
720
|
+
isAction: true,
|
|
721
|
+
interceptSegments:
|
|
722
|
+
reconciled.interceptSegments.length > 0
|
|
723
|
+
? reconciled.interceptSegments
|
|
724
|
+
: undefined,
|
|
725
|
+
});
|
|
632
726
|
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
);
|
|
727
|
+
// Re-check if user navigated away (could happen during async renderSegments)
|
|
728
|
+
if (window.location.pathname !== actionStartPathname) {
|
|
729
|
+
log("user navigated during render, skipping");
|
|
730
|
+
break;
|
|
731
|
+
}
|
|
639
732
|
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
navTx.with({
|
|
647
|
-
url: window.location.href,
|
|
648
|
-
storeOnly: true,
|
|
649
|
-
intercept: !!interceptSourceUrl,
|
|
650
|
-
interceptSourceUrl: interceptSourceUrl ?? undefined,
|
|
651
|
-
}),
|
|
652
|
-
{
|
|
653
|
-
isAction: true,
|
|
654
|
-
interceptSourceUrl: interceptSourceUrl ?? undefined,
|
|
733
|
+
// Verify the store's current key still matches what we captured at action start
|
|
734
|
+
// If they differ, user navigated away and we should NOT cache under the old key
|
|
735
|
+
const currentKeyNow = store.getHistoryKey();
|
|
736
|
+
if (currentKeyNow !== currentKey) {
|
|
737
|
+
log("history key changed during action, skipping cache update");
|
|
738
|
+
break;
|
|
655
739
|
}
|
|
656
|
-
);
|
|
657
740
|
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
console.log(
|
|
662
|
-
`[Browser] Consolidate/Reconcile - Returning to React:`,
|
|
663
|
-
returnData
|
|
664
|
-
);
|
|
741
|
+
startTransition(() => {
|
|
742
|
+
onUpdate({ root: newTree, metadata: metadata! });
|
|
743
|
+
});
|
|
665
744
|
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
745
|
+
// Apply server-set location state to history.state (non-redirect flow)
|
|
746
|
+
const actionLocationState = metadata?.locationState;
|
|
747
|
+
if (actionLocationState) {
|
|
748
|
+
mergeLocationState(actionLocationState);
|
|
749
|
+
}
|
|
669
750
|
|
|
670
|
-
|
|
671
|
-
// Exclude the current action since we already have our response
|
|
672
|
-
// We don't need to wait for streaming to complete - just for the response to arrive
|
|
673
|
-
const otherFetchingActions = [...eventController.getInflightActions().values()].filter(
|
|
674
|
-
(a) => a.phase === "fetching" && a.id !== handle.id
|
|
675
|
-
);
|
|
676
|
-
if (otherFetchingActions.length > 0) {
|
|
677
|
-
console.log(
|
|
678
|
-
`[Browser] Skipping UI update - ${otherFetchingActions.length} other action(s) still fetching`
|
|
679
|
-
);
|
|
680
|
-
console.log(
|
|
681
|
-
`[Browser] Multi actions - Returning to React (no cache clear):`,
|
|
682
|
-
returnData
|
|
683
|
-
);
|
|
684
|
-
// Only update store if history key hasn't changed (user didn't navigate away)
|
|
685
|
-
const currentKeyNow = store.getHistoryKey();
|
|
686
|
-
if (currentKeyNow === currentKey) {
|
|
751
|
+
// Update store state
|
|
687
752
|
store.setSegmentIds(matched);
|
|
688
753
|
const currentHandleData = eventController.getHandleState().data;
|
|
689
|
-
store.cacheSegmentsForHistory(
|
|
690
|
-
|
|
691
|
-
|
|
692
|
-
|
|
754
|
+
store.cacheSegmentsForHistory(
|
|
755
|
+
currentKey,
|
|
756
|
+
fullSegments,
|
|
757
|
+
currentHandleData,
|
|
693
758
|
);
|
|
759
|
+
// Invalidation deferred to finalizeAction() (runs after this caches
|
|
760
|
+
// the fresh segments), suppressed when the action called
|
|
761
|
+
// keepClientCache().
|
|
762
|
+
break;
|
|
694
763
|
}
|
|
695
|
-
handle.complete(returnData);
|
|
696
|
-
return returnData;
|
|
697
|
-
}
|
|
698
|
-
|
|
699
|
-
// No concurrent actions - normal flow with single action
|
|
700
|
-
// INTERCEPT HANDLING: Separate intercept segments for explicit injection
|
|
701
|
-
const isInterceptSegment = (s: ResolvedSegment) =>
|
|
702
|
-
s.namespace?.startsWith("intercept:") ||
|
|
703
|
-
(s.type === "parallel" && s.id.includes(".@"));
|
|
704
|
-
|
|
705
|
-
const interceptSegments = fullSegments.filter(isInterceptSegment);
|
|
706
|
-
const mainSegments = fullSegments.filter((s) => !isInterceptSegment(s));
|
|
707
|
-
|
|
708
|
-
// Prepare new tree (await loader data resolution)
|
|
709
|
-
const renderOptions = {
|
|
710
|
-
isAction: true,
|
|
711
|
-
interceptSegments:
|
|
712
|
-
interceptSegments.length > 0 ? interceptSegments : undefined,
|
|
713
|
-
};
|
|
714
|
-
const newTree = await renderSegments(mainSegments, renderOptions);
|
|
715
|
-
|
|
716
|
-
// Re-check if user navigated away (could happen during async wait)
|
|
717
|
-
const currentPathnameNow = window.location.pathname;
|
|
718
|
-
if (currentPathnameNow !== actionStartPathname) {
|
|
719
|
-
console.log(
|
|
720
|
-
`[Browser] User navigated during UI update scheduling, skipping`
|
|
721
|
-
);
|
|
722
|
-
handle.complete(returnData);
|
|
723
|
-
return returnData;
|
|
724
|
-
}
|
|
725
|
-
|
|
726
|
-
// Verify the store's current key still matches what we captured at action start
|
|
727
|
-
// If they differ, user navigated away and we should NOT cache under the old key
|
|
728
|
-
const currentKeyNow = store.getHistoryKey();
|
|
729
|
-
if (currentKeyNow !== currentKey) {
|
|
730
|
-
console.log(
|
|
731
|
-
`[Browser] History key changed during action (${currentKey} -> ${currentKeyNow}), skipping cache update`
|
|
732
|
-
);
|
|
733
|
-
handle.complete(returnData);
|
|
734
|
-
return returnData;
|
|
735
764
|
}
|
|
736
765
|
|
|
737
|
-
startTransition(() => {
|
|
738
|
-
onUpdate({ root: newTree, metadata: metadata! });
|
|
739
|
-
});
|
|
740
|
-
|
|
741
|
-
// Update store state
|
|
742
|
-
store.setSegmentIds(matched);
|
|
743
|
-
const currentHandleData = eventController.getHandleState().data;
|
|
744
|
-
store.cacheSegmentsForHistory(currentKey, fullSegments, currentHandleData);
|
|
745
|
-
store.markCacheAsStaleAndBroadcast();
|
|
746
|
-
|
|
747
|
-
console.log(`[Browser] Normal - Returning to React:`, returnData);
|
|
748
766
|
handle.complete(returnData);
|
|
749
767
|
return returnData;
|
|
750
|
-
}
|
|
751
|
-
//
|
|
752
|
-
|
|
753
|
-
|
|
754
|
-
|
|
768
|
+
} finally {
|
|
769
|
+
// The single deferred invalidation + fence release for this action. Runs
|
|
770
|
+
// for every terminal that settles (normal, navigated-away, error, abort,
|
|
771
|
+
// intercept, concurrent); the SPA-redirect paths above already ran it.
|
|
772
|
+
// Latched, so it fires exactly once.
|
|
773
|
+
finalizeAction();
|
|
774
|
+
handle[Symbol.dispose]();
|
|
755
775
|
}
|
|
756
776
|
}
|
|
757
777
|
|
|
@@ -766,18 +786,6 @@ export function createServerActionBridge(
|
|
|
766
786
|
}
|
|
767
787
|
deps.setServerCallback(handleServerAction);
|
|
768
788
|
isRegistered = true;
|
|
769
|
-
console.log("[Browser] Server action callback registered");
|
|
770
|
-
},
|
|
771
|
-
|
|
772
|
-
/**
|
|
773
|
-
* Unregister the server action callback
|
|
774
|
-
*/
|
|
775
|
-
unregister(): void {
|
|
776
|
-
if (!isRegistered) {
|
|
777
|
-
return;
|
|
778
|
-
}
|
|
779
|
-
isRegistered = false;
|
|
780
|
-
console.log("[Browser] Server action bridge unregistered");
|
|
781
789
|
},
|
|
782
790
|
};
|
|
783
791
|
}
|