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