@rangojs/router 0.0.0-experimental.0b82333f
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 +899 -0
- package/dist/__internal.d.ts +83 -0
- package/dist/__internal.d.ts.map +1 -0
- package/dist/__internal.js +19 -0
- package/dist/__internal.js.map +1 -0
- package/dist/__mocks__/version.d.ts +7 -0
- package/dist/__mocks__/version.d.ts.map +1 -0
- package/dist/__mocks__/version.js +7 -0
- package/dist/__mocks__/version.js.map +1 -0
- package/dist/__tests__/client-href.test.d.ts +2 -0
- package/dist/__tests__/client-href.test.d.ts.map +1 -0
- package/dist/__tests__/client-href.test.js +74 -0
- package/dist/__tests__/client-href.test.js.map +1 -0
- package/dist/__tests__/component-utils.test.d.ts +2 -0
- package/dist/__tests__/component-utils.test.d.ts.map +1 -0
- package/dist/__tests__/component-utils.test.js +51 -0
- package/dist/__tests__/component-utils.test.js.map +1 -0
- package/dist/__tests__/event-controller.test.d.ts +2 -0
- package/dist/__tests__/event-controller.test.d.ts.map +1 -0
- package/dist/__tests__/event-controller.test.js +538 -0
- package/dist/__tests__/event-controller.test.js.map +1 -0
- package/dist/__tests__/helpers/route-tree.d.ts +118 -0
- package/dist/__tests__/helpers/route-tree.d.ts.map +1 -0
- package/dist/__tests__/helpers/route-tree.js +374 -0
- package/dist/__tests__/helpers/route-tree.js.map +1 -0
- package/dist/__tests__/match-result.test.d.ts +2 -0
- package/dist/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/__tests__/match-result.test.js +154 -0
- package/dist/__tests__/match-result.test.js.map +1 -0
- package/dist/__tests__/navigation-store.test.d.ts +2 -0
- package/dist/__tests__/navigation-store.test.d.ts.map +1 -0
- package/dist/__tests__/navigation-store.test.js +440 -0
- package/dist/__tests__/navigation-store.test.js.map +1 -0
- package/dist/__tests__/partial-update.test.d.ts +2 -0
- package/dist/__tests__/partial-update.test.d.ts.map +1 -0
- package/dist/__tests__/partial-update.test.js +1009 -0
- package/dist/__tests__/partial-update.test.js.map +1 -0
- package/dist/__tests__/reverse-types.test.d.ts +8 -0
- package/dist/__tests__/reverse-types.test.d.ts.map +1 -0
- package/dist/__tests__/reverse-types.test.js +656 -0
- package/dist/__tests__/reverse-types.test.js.map +1 -0
- package/dist/__tests__/route-definition.test.d.ts +2 -0
- package/dist/__tests__/route-definition.test.d.ts.map +1 -0
- package/dist/__tests__/route-definition.test.js +55 -0
- package/dist/__tests__/route-definition.test.js.map +1 -0
- package/dist/__tests__/router-helpers.test.d.ts +2 -0
- package/dist/__tests__/router-helpers.test.d.ts.map +1 -0
- package/dist/__tests__/router-helpers.test.js +377 -0
- package/dist/__tests__/router-helpers.test.js.map +1 -0
- package/dist/__tests__/router-integration-2.test.d.ts +2 -0
- package/dist/__tests__/router-integration-2.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration-2.test.js +426 -0
- package/dist/__tests__/router-integration-2.test.js.map +1 -0
- package/dist/__tests__/router-integration.test.d.ts +2 -0
- package/dist/__tests__/router-integration.test.d.ts.map +1 -0
- package/dist/__tests__/router-integration.test.js +1051 -0
- package/dist/__tests__/router-integration.test.js.map +1 -0
- package/dist/__tests__/search-params.test.d.ts +5 -0
- package/dist/__tests__/search-params.test.d.ts.map +1 -0
- package/dist/__tests__/search-params.test.js +306 -0
- package/dist/__tests__/search-params.test.js.map +1 -0
- package/dist/__tests__/segment-system.test.d.ts +2 -0
- package/dist/__tests__/segment-system.test.d.ts.map +1 -0
- package/dist/__tests__/segment-system.test.js +627 -0
- package/dist/__tests__/segment-system.test.js.map +1 -0
- package/dist/__tests__/static-handler-types.test.d.ts +8 -0
- package/dist/__tests__/static-handler-types.test.d.ts.map +1 -0
- package/dist/__tests__/static-handler-types.test.js +63 -0
- package/dist/__tests__/static-handler-types.test.js.map +1 -0
- package/dist/__tests__/urls.test.d.ts +2 -0
- package/dist/__tests__/urls.test.d.ts.map +1 -0
- package/dist/__tests__/urls.test.js +421 -0
- package/dist/__tests__/urls.test.js.map +1 -0
- package/dist/__tests__/use-mount.test.d.ts +2 -0
- package/dist/__tests__/use-mount.test.d.ts.map +1 -0
- package/dist/__tests__/use-mount.test.js +35 -0
- package/dist/__tests__/use-mount.test.js.map +1 -0
- package/dist/bin/rango.d.ts +2 -0
- package/dist/bin/rango.d.ts.map +1 -0
- package/dist/bin/rango.js +1601 -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/dist/href-context.d.ts +29 -0
- 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 +5214 -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/dist/vite/package-resolution.js +112 -0
- package/dist/vite/package-resolution.js.map +1 -0
- package/dist/vite/virtual-entries.d.ts +25 -0
- package/dist/vite/virtual-entries.d.ts.map +1 -0
- package/dist/vite/virtual-entries.js +110 -0
- package/dist/vite/virtual-entries.js.map +1 -0
- package/package.json +177 -0
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +220 -0
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +112 -0
- package/skills/document-cache/SKILL.md +182 -0
- package/skills/fonts/SKILL.md +167 -0
- package/skills/hooks/SKILL.md +704 -0
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +313 -0
- package/skills/layout/SKILL.md +310 -0
- package/skills/links/SKILL.md +239 -0
- package/skills/loader/SKILL.md +596 -0
- package/skills/middleware/SKILL.md +339 -0
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +305 -0
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +118 -0
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +385 -0
- package/skills/router-setup/SKILL.md +439 -0
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +79 -0
- package/skills/typesafety/SKILL.md +623 -0
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +273 -0
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/event-controller.ts +899 -0
- package/src/browser/history-state.ts +80 -0
- package/src/browser/index.ts +18 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +141 -0
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +134 -0
- package/src/browser/navigation-bridge.ts +640 -0
- package/src/browser/navigation-client.ts +215 -0
- package/src/browser/navigation-store.ts +806 -0
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +552 -0
- package/src/browser/prefetch/cache.ts +154 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +360 -0
- package/src/browser/react/NavigationProvider.tsx +386 -0
- package/src/browser/react/ScrollRestoration.tsx +94 -0
- package/src/browser/react/context.ts +59 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +52 -0
- package/src/browser/react/location-state-shared.ts +162 -0
- package/src/browser/react/location-state.ts +107 -0
- 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 +218 -0
- package/src/browser/react/use-client-cache.ts +58 -0
- package/src/browser/react/use-handle.ts +162 -0
- package/src/browser/react/use-href.tsx +40 -0
- package/src/browser/react/use-link-status.ts +135 -0
- package/src/browser/react/use-mount.ts +31 -0
- package/src/browser/react/use-navigation.ts +99 -0
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +171 -0
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +431 -0
- package/src/browser/scroll-restoration.ts +384 -0
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +83 -0
- package/src/browser/server-action-bridge.ts +667 -0
- package/src/browser/shallow.ts +40 -0
- package/src/browser/types.ts +538 -0
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +438 -0
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +35 -0
- package/src/build/route-trie.ts +265 -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 +411 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +469 -0
- package/src/build/route-types/scan-filter.ts +78 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +338 -0
- package/src/cache/cache-scope.ts +382 -0
- package/src/cache/cf/cf-cache-store.ts +540 -0
- package/src/cache/cf/index.ts +25 -0
- package/src/cache/document-cache.ts +369 -0
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +43 -0
- package/src/cache/memory-segment-store.ts +328 -0
- 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 +98 -0
- package/src/cache/types.ts +342 -0
- package/src/client.rsc.tsx +85 -0
- package/src/client.tsx +601 -0
- package/src/component-utils.ts +76 -0
- package/src/components/DefaultDocument.tsx +27 -0
- package/src/context-var.ts +86 -0
- package/src/debug.ts +243 -0
- package/src/default-error-boundary.tsx +88 -0
- package/src/deps/browser.ts +8 -0
- package/src/deps/html-stream-client.ts +2 -0
- package/src/deps/html-stream-server.ts +2 -0
- package/src/deps/rsc.ts +10 -0
- package/src/deps/ssr.ts +2 -0
- package/src/errors.ts +365 -0
- package/src/handle.ts +135 -0
- package/src/handles/MetaTags.tsx +246 -0
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +7 -0
- package/src/handles/meta.ts +264 -0
- package/src/host/cookie-handler.ts +165 -0
- package/src/host/errors.ts +97 -0
- package/src/host/index.ts +53 -0
- package/src/host/pattern-matcher.ts +214 -0
- package/src/host/router.ts +352 -0
- package/src/host/testing.ts +79 -0
- package/src/host/types.ts +146 -0
- package/src/host/utils.ts +25 -0
- package/src/href-client.ts +222 -0
- package/src/index.rsc.ts +233 -0
- package/src/index.ts +277 -0
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +89 -0
- package/src/loader.ts +64 -0
- package/src/network-error-thrower.tsx +23 -0
- package/src/outlet-context.ts +15 -0
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +289 -0
- package/src/route-content-wrapper.tsx +196 -0
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -0
- package/src/route-map-builder.ts +275 -0
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +259 -0
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +287 -0
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +451 -0
- package/src/router/intercept-resolution.ts +395 -0
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +420 -0
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +267 -0
- package/src/router/match-api.ts +620 -0
- package/src/router/match-context.ts +266 -0
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +223 -0
- package/src/router/match-middleware/cache-lookup.ts +634 -0
- package/src/router/match-middleware/cache-store.ts +295 -0
- package/src/router/match-middleware/index.ts +81 -0
- package/src/router/match-middleware/intercept-resolution.ts +306 -0
- package/src/router/match-middleware/segment-resolution.ts +192 -0
- package/src/router/match-pipelines.ts +179 -0
- package/src/router/match-result.ts +219 -0
- package/src/router/metrics.ts +282 -0
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +748 -0
- package/src/router/pattern-matching.ts +563 -0
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +289 -0
- package/src/router/router-context.ts +316 -0
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1239 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +170 -0
- package/src/router.ts +1002 -0
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +1089 -0
- package/src/rsc/helpers.ts +198 -0
- package/src/rsc/index.ts +36 -0
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +32 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +237 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +263 -0
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +454 -0
- package/src/server/context.ts +591 -0
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +308 -0
- package/src/server/loader-registry.ts +133 -0
- package/src/server/request-context.ts +914 -0
- package/src/server/root-layout.tsx +10 -0
- package/src/server/tsconfig.json +14 -0
- package/src/server.ts +51 -0
- package/src/ssr/index.tsx +365 -0
- package/src/static-handler.ts +114 -0
- package/src/theme/ThemeProvider.tsx +297 -0
- package/src/theme/ThemeScript.tsx +61 -0
- package/src/theme/constants.ts +62 -0
- package/src/theme/index.ts +48 -0
- package/src/theme/theme-context.ts +44 -0
- package/src/theme/theme-script.ts +155 -0
- package/src/theme/types.ts +182 -0
- package/src/theme/use-theme.ts +44 -0
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -0
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -0
- package/src/use-loader.tsx +354 -0
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +16 -0
- package/src/vite/plugin-types.ts +131 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/plugins/expose-action-id.ts +365 -0
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +254 -0
- package/src/vite/plugins/version.d.ts +12 -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 +510 -0
- package/src/vite/router-discovery.ts +785 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/utils/package-resolution.ts +121 -0
- package/src/vite/utils/prerender-utils.ts +189 -0
- package/src/vite/utils/shared-utils.ts +169 -0
|
@@ -0,0 +1,1009 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from "vitest";
|
|
2
|
+
// Mock startTransition to run callbacks synchronously
|
|
3
|
+
vi.mock("react", async () => {
|
|
4
|
+
const actual = await vi.importActual("react");
|
|
5
|
+
return {
|
|
6
|
+
...actual,
|
|
7
|
+
startTransition: (fn) => fn(),
|
|
8
|
+
};
|
|
9
|
+
});
|
|
10
|
+
// Let merge helpers run as real code (pure logic)
|
|
11
|
+
// Mock assertSegmentStructure to suppress dev warnings
|
|
12
|
+
vi.mock("../browser/segment-structure-assert.js", () => ({
|
|
13
|
+
assertSegmentStructure: vi.fn(),
|
|
14
|
+
}));
|
|
15
|
+
import { createPartialUpdater } from "../browser/partial-update";
|
|
16
|
+
// ---------------------------------------------------------------------------
|
|
17
|
+
// Test helpers
|
|
18
|
+
// ---------------------------------------------------------------------------
|
|
19
|
+
function seg(id, overrides) {
|
|
20
|
+
return {
|
|
21
|
+
id,
|
|
22
|
+
namespace: "",
|
|
23
|
+
index: 0,
|
|
24
|
+
type: "route",
|
|
25
|
+
component: `component-${id}`,
|
|
26
|
+
...overrides,
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
function createMockStore(opts) {
|
|
30
|
+
const cache = new Map();
|
|
31
|
+
const historyKey = opts?.historyKey ?? "/";
|
|
32
|
+
if (opts?.cachedSegments) {
|
|
33
|
+
cache.set(historyKey, { segments: opts.cachedSegments, stale: false });
|
|
34
|
+
}
|
|
35
|
+
let currentHistoryKey = historyKey;
|
|
36
|
+
let interceptSourceUrl = null;
|
|
37
|
+
return {
|
|
38
|
+
getHistoryKey: vi.fn(() => currentHistoryKey),
|
|
39
|
+
setHistoryKey: vi.fn((k) => {
|
|
40
|
+
currentHistoryKey = k;
|
|
41
|
+
}),
|
|
42
|
+
getCachedSegments: vi.fn((key) => cache.get(key)),
|
|
43
|
+
getSegmentState: vi.fn(() => ({
|
|
44
|
+
path: "/",
|
|
45
|
+
currentUrl: opts?.currentUrl ?? "http://localhost/",
|
|
46
|
+
currentSegmentIds: opts?.segmentIds ?? [],
|
|
47
|
+
})),
|
|
48
|
+
setInterceptSourceUrl: vi.fn((url) => {
|
|
49
|
+
interceptSourceUrl = url;
|
|
50
|
+
}),
|
|
51
|
+
getInterceptSourceUrl: vi.fn(() => interceptSourceUrl),
|
|
52
|
+
// Stubs for methods not directly used in partial-update
|
|
53
|
+
getState: vi.fn(() => ({ state: "idle" })),
|
|
54
|
+
setState: vi.fn(),
|
|
55
|
+
subscribe: vi.fn(),
|
|
56
|
+
cacheSegmentsForHistory: vi.fn(),
|
|
57
|
+
hasHistoryCache: vi.fn(() => false),
|
|
58
|
+
setPath: vi.fn(),
|
|
59
|
+
setCurrentUrl: vi.fn(),
|
|
60
|
+
setSegmentIds: vi.fn(),
|
|
61
|
+
markCacheAsStale: vi.fn(),
|
|
62
|
+
clearHistoryCache: vi.fn(),
|
|
63
|
+
markCacheAsStaleAndBroadcast: vi.fn(),
|
|
64
|
+
broadcastCacheInvalidation: vi.fn(),
|
|
65
|
+
setCrossTabRefreshCallback: vi.fn(),
|
|
66
|
+
addInflightAction: vi.fn(),
|
|
67
|
+
removeInflightAction: vi.fn(),
|
|
68
|
+
isActionInProgress: vi.fn(() => false),
|
|
69
|
+
setActionInProgress: vi.fn(),
|
|
70
|
+
updateCacheHandleData: vi.fn(),
|
|
71
|
+
onUpdate: vi.fn(),
|
|
72
|
+
emitUpdate: vi.fn(),
|
|
73
|
+
getActionState: vi.fn(),
|
|
74
|
+
setActionState: vi.fn(),
|
|
75
|
+
subscribeToAction: vi.fn(),
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
function createMockClient(payload, opts) {
|
|
79
|
+
let resolveStream;
|
|
80
|
+
// By default, streamComplete resolves immediately.
|
|
81
|
+
// The async function's return Promise adopts streamComplete (JS promise flattening),
|
|
82
|
+
// so a hanging stream would hang the entire await.
|
|
83
|
+
const streamComplete = opts?.hangStream
|
|
84
|
+
? new Promise((resolve) => {
|
|
85
|
+
resolveStream = resolve;
|
|
86
|
+
})
|
|
87
|
+
: Promise.resolve();
|
|
88
|
+
return {
|
|
89
|
+
client: {
|
|
90
|
+
fetchPartial: vi.fn(async () => ({
|
|
91
|
+
payload,
|
|
92
|
+
streamComplete,
|
|
93
|
+
})),
|
|
94
|
+
},
|
|
95
|
+
resolveStream: () => resolveStream(),
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
function createMockTx(currentUrl = "http://localhost/") {
|
|
99
|
+
return {
|
|
100
|
+
currentUrl,
|
|
101
|
+
startStreaming: vi.fn(() => ({ end: vi.fn() })),
|
|
102
|
+
commit: vi.fn(),
|
|
103
|
+
};
|
|
104
|
+
}
|
|
105
|
+
// ---------------------------------------------------------------------------
|
|
106
|
+
// Tests
|
|
107
|
+
// ---------------------------------------------------------------------------
|
|
108
|
+
describe("partial-update", () => {
|
|
109
|
+
describe("partial update (isPartial=true)", () => {
|
|
110
|
+
it("merges server diff segments with cached segments", async () => {
|
|
111
|
+
const cachedLayout = seg("L0", { type: "layout", component: "cached-layout" });
|
|
112
|
+
const cachedRoute = seg("L0R0", { component: "cached-route" });
|
|
113
|
+
const newRoute = seg("L0R0", { component: "new-route" });
|
|
114
|
+
const store = createMockStore({
|
|
115
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
116
|
+
});
|
|
117
|
+
const { client } = createMockClient({
|
|
118
|
+
metadata: {
|
|
119
|
+
isPartial: true,
|
|
120
|
+
segments: [newRoute],
|
|
121
|
+
matched: ["L0", "L0R0"],
|
|
122
|
+
diff: ["L0R0"],
|
|
123
|
+
},
|
|
124
|
+
});
|
|
125
|
+
const renderSegments = vi.fn(async (segs) => `tree-${segs.length}`);
|
|
126
|
+
const onUpdate = vi.fn();
|
|
127
|
+
const tx = createMockTx();
|
|
128
|
+
const updater = createPartialUpdater({
|
|
129
|
+
store: store,
|
|
130
|
+
client: client,
|
|
131
|
+
onUpdate,
|
|
132
|
+
renderSegments,
|
|
133
|
+
});
|
|
134
|
+
await updater("http://localhost/page", ["L0", "L0R0"], false, undefined, tx);
|
|
135
|
+
// renderSegments should be called with merged segments
|
|
136
|
+
expect(renderSegments).toHaveBeenCalledTimes(1);
|
|
137
|
+
const renderedSegments = renderSegments.mock.calls[0][0];
|
|
138
|
+
expect(renderedSegments).toHaveLength(2);
|
|
139
|
+
// L0 should come from cache (not in diff)
|
|
140
|
+
expect(renderedSegments.find((s) => s.id === "L0").component).toBe("cached-layout");
|
|
141
|
+
// L0R0 should come from server (in diff)
|
|
142
|
+
expect(renderedSegments.find((s) => s.id === "L0R0").component).toBe("new-route");
|
|
143
|
+
// tx.commit called with all segment IDs
|
|
144
|
+
expect(tx.commit).toHaveBeenCalledWith(["L0", "L0R0"], expect.any(Array), undefined);
|
|
145
|
+
// onUpdate called with rendered tree
|
|
146
|
+
expect(onUpdate).toHaveBeenCalledWith({
|
|
147
|
+
root: `tree-2`,
|
|
148
|
+
metadata: expect.objectContaining({ isPartial: true }),
|
|
149
|
+
});
|
|
150
|
+
});
|
|
151
|
+
it("preserves cached component when server returns null for layout", async () => {
|
|
152
|
+
const cachedLayout = seg("L0", { type: "layout", component: "my-layout" });
|
|
153
|
+
const newLayout = seg("L0", { type: "layout", component: null });
|
|
154
|
+
const cachedRoute = seg("L0R0");
|
|
155
|
+
const store = createMockStore({
|
|
156
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
157
|
+
});
|
|
158
|
+
const { client } = createMockClient({
|
|
159
|
+
metadata: {
|
|
160
|
+
isPartial: true,
|
|
161
|
+
segments: [newLayout],
|
|
162
|
+
matched: ["L0", "L0R0"],
|
|
163
|
+
diff: ["L0"],
|
|
164
|
+
},
|
|
165
|
+
});
|
|
166
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
167
|
+
const tx = createMockTx();
|
|
168
|
+
const updater = createPartialUpdater({
|
|
169
|
+
store: store,
|
|
170
|
+
client: client,
|
|
171
|
+
onUpdate: vi.fn(),
|
|
172
|
+
renderSegments,
|
|
173
|
+
});
|
|
174
|
+
await updater("http://localhost/", ["L0", "L0R0"], false, undefined, tx);
|
|
175
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
176
|
+
const layout = rendered.find((s) => s.id === "L0");
|
|
177
|
+
// Should preserve cached component, not null
|
|
178
|
+
expect(layout.component).toBe("my-layout");
|
|
179
|
+
});
|
|
180
|
+
it("clears loading state for cached (unchanged) segments", async () => {
|
|
181
|
+
const cached = seg("R0", { loading: "skeleton" });
|
|
182
|
+
const store = createMockStore({ cachedSegments: [cached] });
|
|
183
|
+
const { client } = createMockClient({
|
|
184
|
+
metadata: {
|
|
185
|
+
isPartial: true,
|
|
186
|
+
segments: [],
|
|
187
|
+
matched: ["R0"],
|
|
188
|
+
diff: ["R0"],
|
|
189
|
+
},
|
|
190
|
+
});
|
|
191
|
+
// No server segment for R0 in newSegmentMap, but it's in matched.
|
|
192
|
+
// Since it's not found in newSegmentMap, it falls back to cache.
|
|
193
|
+
// Actually, let me fix this: R0 is in diff but not in segments from server.
|
|
194
|
+
// So newSegmentMap won't have R0. It'll use cache.
|
|
195
|
+
// Actually - let me re-read. segments: [] means newSegmentMap has nothing.
|
|
196
|
+
// matched: ["R0"] means we try to find R0 in newSegmentMap (miss) then cache (hit).
|
|
197
|
+
// Cache has loading set → should be cleared to undefined.
|
|
198
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
199
|
+
const tx = createMockTx();
|
|
200
|
+
const updater = createPartialUpdater({
|
|
201
|
+
store: store,
|
|
202
|
+
client: client,
|
|
203
|
+
onUpdate: vi.fn(),
|
|
204
|
+
renderSegments,
|
|
205
|
+
});
|
|
206
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx);
|
|
207
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
208
|
+
expect(rendered[0].loading).toBeUndefined();
|
|
209
|
+
});
|
|
210
|
+
// BUG P0-1: cached segment loading cleared from false to undefined
|
|
211
|
+
//
|
|
212
|
+
// When a cached segment has loading=false (suppressed), navigating to a route
|
|
213
|
+
// where the server doesn't re-render that segment (matched but not in diff)
|
|
214
|
+
// should preserve loading=false. Instead, the current code unconditionally
|
|
215
|
+
// sets it to undefined, which changes the React tree structure:
|
|
216
|
+
// loading: false -> LoaderBoundary + OutletProvider (structural boundary)
|
|
217
|
+
// loading: undefined -> OutletProvider directly (no boundary)
|
|
218
|
+
// This causes React to remount components, destroying client state.
|
|
219
|
+
//
|
|
220
|
+
// The correct behavior (see server-action-bridge.ts lines 439-445) preserves
|
|
221
|
+
// the cached loading value instead of clearing it.
|
|
222
|
+
it("preserves loading=false for cached segment (does not clear to undefined)", async () => {
|
|
223
|
+
// Layout cached with loading=false (suppressed via { ssr: false } in loading())
|
|
224
|
+
const cachedLayout = seg("L0", {
|
|
225
|
+
type: "layout",
|
|
226
|
+
component: "cached-layout",
|
|
227
|
+
loading: false,
|
|
228
|
+
});
|
|
229
|
+
const cachedRoute = seg("L0R0", { component: "cached-route" });
|
|
230
|
+
const store = createMockStore({
|
|
231
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
232
|
+
});
|
|
233
|
+
// Server returns a new route segment but NOT the layout.
|
|
234
|
+
// The layout is in matched (server expects client to have it) but NOT in diff
|
|
235
|
+
// (server decided it doesn't need re-rendering). This triggers the cache
|
|
236
|
+
// fallback path where the bug exists.
|
|
237
|
+
const newRoute = seg("L0R0", { component: "new-route" });
|
|
238
|
+
const { client } = createMockClient({
|
|
239
|
+
metadata: {
|
|
240
|
+
isPartial: true,
|
|
241
|
+
segments: [newRoute],
|
|
242
|
+
matched: ["L0", "L0R0"],
|
|
243
|
+
diff: ["L0R0"],
|
|
244
|
+
},
|
|
245
|
+
});
|
|
246
|
+
const renderSegments = vi.fn(async (segs) => `tree-${segs.length}`);
|
|
247
|
+
const tx = createMockTx();
|
|
248
|
+
const updater = createPartialUpdater({
|
|
249
|
+
store: store,
|
|
250
|
+
client: client,
|
|
251
|
+
onUpdate: vi.fn(),
|
|
252
|
+
renderSegments,
|
|
253
|
+
});
|
|
254
|
+
await updater("http://localhost/page", ["L0", "L0R0"], false, undefined, tx);
|
|
255
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
256
|
+
const layout = rendered.find((s) => s.id === "L0");
|
|
257
|
+
// The layout should preserve loading=false (suppressed boundary).
|
|
258
|
+
// BUG: current code clears it to undefined, changing the tree structure.
|
|
259
|
+
expect(layout.loading).toBe(false);
|
|
260
|
+
});
|
|
261
|
+
});
|
|
262
|
+
describe("empty diff handling", () => {
|
|
263
|
+
it("skips UI update when diff is empty (same-route revalidation)", async () => {
|
|
264
|
+
const cached = seg("R0");
|
|
265
|
+
const store = createMockStore({ cachedSegments: [cached] });
|
|
266
|
+
const { client } = createMockClient({
|
|
267
|
+
metadata: {
|
|
268
|
+
isPartial: true,
|
|
269
|
+
segments: [],
|
|
270
|
+
matched: ["R0"],
|
|
271
|
+
diff: [],
|
|
272
|
+
},
|
|
273
|
+
});
|
|
274
|
+
const onUpdate = vi.fn();
|
|
275
|
+
const tx = createMockTx();
|
|
276
|
+
const updater = createPartialUpdater({
|
|
277
|
+
store: store,
|
|
278
|
+
client: client,
|
|
279
|
+
onUpdate,
|
|
280
|
+
renderSegments: vi.fn(),
|
|
281
|
+
});
|
|
282
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx);
|
|
283
|
+
// Should commit but NOT call onUpdate (no UI change)
|
|
284
|
+
expect(tx.commit).toHaveBeenCalled();
|
|
285
|
+
expect(onUpdate).not.toHaveBeenCalled();
|
|
286
|
+
});
|
|
287
|
+
it("renders cached segments when navigating with targetCacheSegments", async () => {
|
|
288
|
+
const targetSegs = [seg("L0", { type: "layout" }), seg("L0R0")];
|
|
289
|
+
const store = createMockStore();
|
|
290
|
+
const { client } = createMockClient({
|
|
291
|
+
metadata: {
|
|
292
|
+
isPartial: true,
|
|
293
|
+
segments: [],
|
|
294
|
+
matched: ["L0", "L0R0"],
|
|
295
|
+
diff: [],
|
|
296
|
+
},
|
|
297
|
+
});
|
|
298
|
+
const renderSegments = vi.fn(async () => "cached-tree");
|
|
299
|
+
const onUpdate = vi.fn();
|
|
300
|
+
const tx = createMockTx();
|
|
301
|
+
const updater = createPartialUpdater({
|
|
302
|
+
store: store,
|
|
303
|
+
client: client,
|
|
304
|
+
onUpdate,
|
|
305
|
+
renderSegments,
|
|
306
|
+
});
|
|
307
|
+
await updater("http://localhost/page", undefined, false, undefined, tx, {
|
|
308
|
+
targetCacheSegments: targetSegs,
|
|
309
|
+
});
|
|
310
|
+
// Should render even with empty diff because targetCacheSegments provided
|
|
311
|
+
expect(renderSegments).toHaveBeenCalledWith(expect.any(Array), expect.objectContaining({ forceAwait: true }));
|
|
312
|
+
expect(onUpdate).toHaveBeenCalled();
|
|
313
|
+
});
|
|
314
|
+
it("forces re-render when leavingIntercept even with empty diff", async () => {
|
|
315
|
+
const cached = seg("L0", { type: "layout" });
|
|
316
|
+
const store = createMockStore({
|
|
317
|
+
cachedSegments: [cached],
|
|
318
|
+
segmentIds: ["L0", "L0.@modal"],
|
|
319
|
+
});
|
|
320
|
+
const { client } = createMockClient({
|
|
321
|
+
metadata: {
|
|
322
|
+
isPartial: true,
|
|
323
|
+
segments: [],
|
|
324
|
+
matched: ["L0"],
|
|
325
|
+
diff: [],
|
|
326
|
+
},
|
|
327
|
+
});
|
|
328
|
+
const renderSegments = vi.fn(async () => "tree-no-modal");
|
|
329
|
+
const onUpdate = vi.fn();
|
|
330
|
+
const tx = createMockTx();
|
|
331
|
+
const updater = createPartialUpdater({
|
|
332
|
+
store: store,
|
|
333
|
+
client: client,
|
|
334
|
+
onUpdate,
|
|
335
|
+
renderSegments,
|
|
336
|
+
});
|
|
337
|
+
await updater("http://localhost/", undefined, false, undefined, tx, {
|
|
338
|
+
leavingIntercept: true,
|
|
339
|
+
});
|
|
340
|
+
// Should render and update UI to remove modal
|
|
341
|
+
expect(renderSegments).toHaveBeenCalled();
|
|
342
|
+
expect(onUpdate).toHaveBeenCalled();
|
|
343
|
+
});
|
|
344
|
+
});
|
|
345
|
+
describe("intercept with target cache segments", () => {
|
|
346
|
+
// Regression test for REVIEW #29: The deleted dead-code block attempted to
|
|
347
|
+
// rebuild currentSegmentMap from getCurrentSegmentMap() when an intercept
|
|
348
|
+
// response arrived with targetCacheSegments. Because the reassignment came
|
|
349
|
+
// AFTER all currentSegmentMap reads, it was dead and was removed.
|
|
350
|
+
//
|
|
351
|
+
// This test confirms: unmodified segments come from targetCacheSegments
|
|
352
|
+
// (not the source page's cache) even when the server returns an intercept response.
|
|
353
|
+
it("uses targetCacheSegments for unmodified layout when server returns intercept response", async () => {
|
|
354
|
+
const sourceLayout = seg("L0", { type: "layout", component: "source-layout" });
|
|
355
|
+
const sourceRoute = seg("L0R0", { component: "source-route" });
|
|
356
|
+
const targetLayout = seg("L0", { type: "layout", component: "target-layout" });
|
|
357
|
+
const targetRoute = seg("L0R0", { component: "target-route" });
|
|
358
|
+
const modalSegment = seg("L0.@modal", { type: "parallel", namespace: "intercept:modal" });
|
|
359
|
+
// Source page is cached at the current history key
|
|
360
|
+
const store = createMockStore({
|
|
361
|
+
cachedSegments: [sourceLayout, sourceRoute],
|
|
362
|
+
segmentIds: ["L0", "L0R0"],
|
|
363
|
+
});
|
|
364
|
+
// Server returns an intercept response: only the modal is in diff.
|
|
365
|
+
// The layout is in matched but not re-rendered (should come from cache).
|
|
366
|
+
const { client } = createMockClient({
|
|
367
|
+
metadata: {
|
|
368
|
+
isPartial: true,
|
|
369
|
+
segments: [modalSegment],
|
|
370
|
+
matched: ["L0", "L0.@modal"],
|
|
371
|
+
diff: ["L0.@modal"],
|
|
372
|
+
slots: { "@modal": { active: true } },
|
|
373
|
+
},
|
|
374
|
+
});
|
|
375
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
376
|
+
const tx = createMockTx();
|
|
377
|
+
const updater = createPartialUpdater({
|
|
378
|
+
store: store,
|
|
379
|
+
client: client,
|
|
380
|
+
onUpdate: vi.fn(),
|
|
381
|
+
renderSegments,
|
|
382
|
+
});
|
|
383
|
+
await updater("http://localhost/target", ["L0", "L0R0"], false, undefined, tx, { targetCacheSegments: [targetLayout, targetRoute] });
|
|
384
|
+
const mainSegs = renderSegments.mock.calls[0][0];
|
|
385
|
+
const layout = mainSegs.find((s) => s.id === "L0");
|
|
386
|
+
// Layout must come from targetCacheSegments ("target-layout"),
|
|
387
|
+
// not the source page's store cache ("source-layout").
|
|
388
|
+
expect(layout?.component).toBe("target-layout");
|
|
389
|
+
});
|
|
390
|
+
});
|
|
391
|
+
describe("intercept segment filtering", () => {
|
|
392
|
+
it("filters .@ segments from segmentIds when leavingIntercept", async () => {
|
|
393
|
+
const store = createMockStore({
|
|
394
|
+
segmentIds: ["L0", "L0R0", "L0.@modal"],
|
|
395
|
+
});
|
|
396
|
+
const { client } = createMockClient({
|
|
397
|
+
metadata: {
|
|
398
|
+
isPartial: true,
|
|
399
|
+
segments: [],
|
|
400
|
+
matched: ["L0"],
|
|
401
|
+
diff: [],
|
|
402
|
+
},
|
|
403
|
+
});
|
|
404
|
+
const tx = createMockTx();
|
|
405
|
+
const updater = createPartialUpdater({
|
|
406
|
+
store: store,
|
|
407
|
+
client: client,
|
|
408
|
+
onUpdate: vi.fn(),
|
|
409
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
410
|
+
});
|
|
411
|
+
await updater("http://localhost/", undefined, false, undefined, tx, {
|
|
412
|
+
leavingIntercept: true,
|
|
413
|
+
});
|
|
414
|
+
// fetchPartial should be called with filtered segments (no .@)
|
|
415
|
+
const fetchCall = client.fetchPartial.mock.calls[0][0];
|
|
416
|
+
expect(fetchCall.segmentIds).toEqual(["L0", "L0R0"]);
|
|
417
|
+
expect(fetchCall.segmentIds).not.toContain("L0.@modal");
|
|
418
|
+
});
|
|
419
|
+
it("separates intercept segments from main segments for rendering", async () => {
|
|
420
|
+
const layout = seg("L0", { type: "layout" });
|
|
421
|
+
const route = seg("L0R0");
|
|
422
|
+
const modal = seg("L0.@modal", { type: "parallel", namespace: "intercept:modal" });
|
|
423
|
+
const store = createMockStore({ cachedSegments: [layout, route] });
|
|
424
|
+
const { client } = createMockClient({
|
|
425
|
+
metadata: {
|
|
426
|
+
isPartial: true,
|
|
427
|
+
segments: [layout, route, modal],
|
|
428
|
+
matched: ["L0", "L0R0", "L0.@modal"],
|
|
429
|
+
diff: ["L0", "L0R0", "L0.@modal"],
|
|
430
|
+
slots: { "@modal": { active: true } },
|
|
431
|
+
},
|
|
432
|
+
});
|
|
433
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
434
|
+
const tx = createMockTx();
|
|
435
|
+
const updater = createPartialUpdater({
|
|
436
|
+
store: store,
|
|
437
|
+
client: client,
|
|
438
|
+
onUpdate: vi.fn(),
|
|
439
|
+
renderSegments,
|
|
440
|
+
});
|
|
441
|
+
await updater("http://localhost/modal", ["L0", "L0R0"], false, undefined, tx);
|
|
442
|
+
// renderSegments should receive main segments and intercept segments separately
|
|
443
|
+
const call = renderSegments.mock.calls[0];
|
|
444
|
+
const mainSegs = call[0];
|
|
445
|
+
const opts = call[1];
|
|
446
|
+
// Main segments should NOT include the parallel intercept
|
|
447
|
+
expect(mainSegs.find((s) => s.id === "L0.@modal")).toBeUndefined();
|
|
448
|
+
// Intercept segments should be passed via options
|
|
449
|
+
expect(opts?.interceptSegments).toHaveLength(1);
|
|
450
|
+
expect(opts?.interceptSegments[0].id).toBe("L0.@modal");
|
|
451
|
+
});
|
|
452
|
+
});
|
|
453
|
+
describe("stale navigation detection", () => {
|
|
454
|
+
it("returns early when signal is aborted before processing", async () => {
|
|
455
|
+
const store = createMockStore();
|
|
456
|
+
const controller = new AbortController();
|
|
457
|
+
controller.abort();
|
|
458
|
+
const { client } = createMockClient({
|
|
459
|
+
metadata: {
|
|
460
|
+
isPartial: true,
|
|
461
|
+
segments: [],
|
|
462
|
+
matched: [],
|
|
463
|
+
diff: ["R0"],
|
|
464
|
+
},
|
|
465
|
+
});
|
|
466
|
+
const renderSegments = vi.fn();
|
|
467
|
+
const onUpdate = vi.fn();
|
|
468
|
+
const tx = createMockTx();
|
|
469
|
+
const updater = createPartialUpdater({
|
|
470
|
+
store: store,
|
|
471
|
+
client: client,
|
|
472
|
+
onUpdate,
|
|
473
|
+
renderSegments,
|
|
474
|
+
});
|
|
475
|
+
await updater("http://localhost/", [], false, controller.signal, tx);
|
|
476
|
+
// Should not render or update
|
|
477
|
+
expect(renderSegments).not.toHaveBeenCalled();
|
|
478
|
+
expect(onUpdate).not.toHaveBeenCalled();
|
|
479
|
+
});
|
|
480
|
+
it("skips UI update for stale revalidation when history key changed", async () => {
|
|
481
|
+
const store = createMockStore({ historyKey: "/page1" });
|
|
482
|
+
// Simulate history key changing mid-request
|
|
483
|
+
let callCount = 0;
|
|
484
|
+
store.getHistoryKey.mockImplementation(() => {
|
|
485
|
+
callCount++;
|
|
486
|
+
return callCount <= 1 ? "/page1" : "/page2";
|
|
487
|
+
});
|
|
488
|
+
const { client } = createMockClient({
|
|
489
|
+
metadata: {
|
|
490
|
+
isPartial: true,
|
|
491
|
+
segments: [seg("R0")],
|
|
492
|
+
matched: ["R0"],
|
|
493
|
+
diff: ["R0"],
|
|
494
|
+
},
|
|
495
|
+
});
|
|
496
|
+
const onUpdate = vi.fn();
|
|
497
|
+
const tx = createMockTx();
|
|
498
|
+
const updater = createPartialUpdater({
|
|
499
|
+
store: store,
|
|
500
|
+
client: client,
|
|
501
|
+
onUpdate,
|
|
502
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
503
|
+
});
|
|
504
|
+
await updater("http://localhost/page1", [], false, undefined, tx, {
|
|
505
|
+
staleRevalidation: true,
|
|
506
|
+
});
|
|
507
|
+
// Should render but NOT call onUpdate because history key changed
|
|
508
|
+
expect(onUpdate).not.toHaveBeenCalled();
|
|
509
|
+
});
|
|
510
|
+
});
|
|
511
|
+
describe("full update fallback", () => {
|
|
512
|
+
it("renders all segments client-side when isPartial=false", async () => {
|
|
513
|
+
const serverSegments = [
|
|
514
|
+
seg("L0", { type: "layout" }),
|
|
515
|
+
seg("L0R0"),
|
|
516
|
+
];
|
|
517
|
+
const store = createMockStore();
|
|
518
|
+
const { client } = createMockClient({
|
|
519
|
+
metadata: {
|
|
520
|
+
isPartial: false,
|
|
521
|
+
segments: serverSegments,
|
|
522
|
+
},
|
|
523
|
+
});
|
|
524
|
+
const renderSegments = vi.fn(async () => "full-tree");
|
|
525
|
+
const onUpdate = vi.fn();
|
|
526
|
+
const tx = createMockTx();
|
|
527
|
+
const updater = createPartialUpdater({
|
|
528
|
+
store: store,
|
|
529
|
+
client: client,
|
|
530
|
+
onUpdate,
|
|
531
|
+
renderSegments,
|
|
532
|
+
});
|
|
533
|
+
await updater("http://localhost/new", [], false, undefined, tx);
|
|
534
|
+
// Should render ALL segments (not merge with cache)
|
|
535
|
+
expect(renderSegments).toHaveBeenCalledWith(serverSegments);
|
|
536
|
+
// Commit with segment IDs from server
|
|
537
|
+
expect(tx.commit).toHaveBeenCalledWith(["L0", "L0R0"], serverSegments);
|
|
538
|
+
expect(onUpdate).toHaveBeenCalledWith({
|
|
539
|
+
root: "full-tree",
|
|
540
|
+
metadata: expect.objectContaining({ isPartial: false }),
|
|
541
|
+
});
|
|
542
|
+
});
|
|
543
|
+
it("returns early for full update when signal is aborted", async () => {
|
|
544
|
+
const controller = new AbortController();
|
|
545
|
+
controller.abort();
|
|
546
|
+
const store = createMockStore();
|
|
547
|
+
const { client } = createMockClient({
|
|
548
|
+
metadata: {
|
|
549
|
+
isPartial: false,
|
|
550
|
+
segments: [seg("R0")],
|
|
551
|
+
},
|
|
552
|
+
});
|
|
553
|
+
const renderSegments = vi.fn();
|
|
554
|
+
const onUpdate = vi.fn();
|
|
555
|
+
const tx = createMockTx();
|
|
556
|
+
const updater = createPartialUpdater({
|
|
557
|
+
store: store,
|
|
558
|
+
client: client,
|
|
559
|
+
onUpdate,
|
|
560
|
+
renderSegments,
|
|
561
|
+
});
|
|
562
|
+
await updater("http://localhost/", [], false, controller.signal, tx);
|
|
563
|
+
expect(renderSegments).not.toHaveBeenCalled();
|
|
564
|
+
expect(onUpdate).not.toHaveBeenCalled();
|
|
565
|
+
});
|
|
566
|
+
});
|
|
567
|
+
describe("HMR resilience", () => {
|
|
568
|
+
it("retries with empty segments when matched IDs are missing from cache", async () => {
|
|
569
|
+
// No cached segments → server's matched["L0", "L0R0"] won't be found
|
|
570
|
+
const store = createMockStore({ cachedSegments: [] });
|
|
571
|
+
// First call: partial with missing segments
|
|
572
|
+
// Second call (retry): full update
|
|
573
|
+
const { client } = createMockClient({
|
|
574
|
+
metadata: {
|
|
575
|
+
isPartial: true,
|
|
576
|
+
segments: [],
|
|
577
|
+
matched: ["L0", "L0R0"],
|
|
578
|
+
diff: ["L0R0"],
|
|
579
|
+
},
|
|
580
|
+
});
|
|
581
|
+
// Override to return full update on retry
|
|
582
|
+
let callCount = 0;
|
|
583
|
+
client.fetchPartial.mockImplementation(async () => {
|
|
584
|
+
callCount++;
|
|
585
|
+
if (callCount === 1) {
|
|
586
|
+
return {
|
|
587
|
+
payload: {
|
|
588
|
+
metadata: {
|
|
589
|
+
isPartial: true,
|
|
590
|
+
segments: [],
|
|
591
|
+
matched: ["L0", "L0R0"],
|
|
592
|
+
diff: ["L0R0"],
|
|
593
|
+
},
|
|
594
|
+
},
|
|
595
|
+
streamComplete: Promise.resolve(),
|
|
596
|
+
};
|
|
597
|
+
}
|
|
598
|
+
return {
|
|
599
|
+
payload: {
|
|
600
|
+
metadata: {
|
|
601
|
+
isPartial: false,
|
|
602
|
+
segments: [
|
|
603
|
+
seg("L0", { type: "layout" }),
|
|
604
|
+
seg("L0R0"),
|
|
605
|
+
],
|
|
606
|
+
},
|
|
607
|
+
},
|
|
608
|
+
streamComplete: Promise.resolve(),
|
|
609
|
+
};
|
|
610
|
+
});
|
|
611
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
612
|
+
const tx = createMockTx();
|
|
613
|
+
const updater = createPartialUpdater({
|
|
614
|
+
store: store,
|
|
615
|
+
client: client,
|
|
616
|
+
onUpdate: vi.fn(),
|
|
617
|
+
renderSegments,
|
|
618
|
+
});
|
|
619
|
+
await updater("http://localhost/page", [], false, undefined, tx);
|
|
620
|
+
// Should have fetched twice: first partial (missing), then retry with empty segments
|
|
621
|
+
expect(client.fetchPartial).toHaveBeenCalledTimes(2);
|
|
622
|
+
const retryCall = client.fetchPartial.mock.calls[1][0];
|
|
623
|
+
expect(retryCall.segmentIds).toEqual([]);
|
|
624
|
+
});
|
|
625
|
+
it("throws on retry failure", async () => {
|
|
626
|
+
const store = createMockStore({ cachedSegments: [] });
|
|
627
|
+
const { client } = createMockClient({
|
|
628
|
+
metadata: {
|
|
629
|
+
isPartial: true,
|
|
630
|
+
segments: [],
|
|
631
|
+
matched: ["L0"],
|
|
632
|
+
diff: ["L0"],
|
|
633
|
+
},
|
|
634
|
+
});
|
|
635
|
+
const tx = createMockTx();
|
|
636
|
+
const updater = createPartialUpdater({
|
|
637
|
+
store: store,
|
|
638
|
+
client: client,
|
|
639
|
+
onUpdate: vi.fn(),
|
|
640
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
641
|
+
});
|
|
642
|
+
// isRetry=true, missing segments → should throw
|
|
643
|
+
await expect(updater("http://localhost/", [], true, undefined, tx)).rejects.toThrow("Failed to fetch segments after retry");
|
|
644
|
+
});
|
|
645
|
+
});
|
|
646
|
+
describe("isAction and staleRevalidation options", () => {
|
|
647
|
+
it("wraps onUpdate in startTransition when isAction=true", async () => {
|
|
648
|
+
const store = createMockStore({
|
|
649
|
+
cachedSegments: [seg("R0")],
|
|
650
|
+
});
|
|
651
|
+
const { client } = createMockClient({
|
|
652
|
+
metadata: {
|
|
653
|
+
isPartial: true,
|
|
654
|
+
segments: [seg("R0", { component: "updated" })],
|
|
655
|
+
matched: ["R0"],
|
|
656
|
+
diff: ["R0"],
|
|
657
|
+
},
|
|
658
|
+
});
|
|
659
|
+
const onUpdate = vi.fn();
|
|
660
|
+
const tx = createMockTx();
|
|
661
|
+
const updater = createPartialUpdater({
|
|
662
|
+
store: store,
|
|
663
|
+
client: client,
|
|
664
|
+
onUpdate,
|
|
665
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
666
|
+
});
|
|
667
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx, {
|
|
668
|
+
isAction: true,
|
|
669
|
+
});
|
|
670
|
+
// Our mocked startTransition runs synchronously, so onUpdate should be called
|
|
671
|
+
expect(onUpdate).toHaveBeenCalled();
|
|
672
|
+
});
|
|
673
|
+
it("passes forceAwait to renderSegments when staleRevalidation=true", async () => {
|
|
674
|
+
const store = createMockStore({
|
|
675
|
+
cachedSegments: [seg("R0")],
|
|
676
|
+
});
|
|
677
|
+
const { client } = createMockClient({
|
|
678
|
+
metadata: {
|
|
679
|
+
isPartial: true,
|
|
680
|
+
segments: [seg("R0", { component: "updated" })],
|
|
681
|
+
matched: ["R0"],
|
|
682
|
+
diff: ["R0"],
|
|
683
|
+
},
|
|
684
|
+
});
|
|
685
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
686
|
+
const tx = createMockTx();
|
|
687
|
+
const updater = createPartialUpdater({
|
|
688
|
+
store: store,
|
|
689
|
+
client: client,
|
|
690
|
+
onUpdate: vi.fn(),
|
|
691
|
+
renderSegments,
|
|
692
|
+
});
|
|
693
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx, {
|
|
694
|
+
staleRevalidation: true,
|
|
695
|
+
});
|
|
696
|
+
expect(renderSegments).toHaveBeenCalledWith(expect.any(Array), expect.objectContaining({ forceAwait: true }));
|
|
697
|
+
});
|
|
698
|
+
});
|
|
699
|
+
describe("structural preservation on server-returned segments", () => {
|
|
700
|
+
// These tests verify the fromServer path in partial-update preserves
|
|
701
|
+
// cached loading/mountPath values when the server returns different ones.
|
|
702
|
+
// Without this preservation, the React tree depth changes between SSR
|
|
703
|
+
// and action revalidation, causing component remounts that destroy
|
|
704
|
+
// useActionState results.
|
|
705
|
+
it("preserves cached loading value when server returns different loading", async () => {
|
|
706
|
+
// SSR set loading=false (skipSSR route). Server action revalidation
|
|
707
|
+
// returns loading=<skeleton> because isSSR=false. Without preservation,
|
|
708
|
+
// tree changes from LoaderBoundary+Outlet to LoaderBoundary+Outlet+RouteContentWrapper.
|
|
709
|
+
const cachedLayout = seg("L0", {
|
|
710
|
+
type: "layout",
|
|
711
|
+
component: "my-layout",
|
|
712
|
+
loading: false,
|
|
713
|
+
});
|
|
714
|
+
const cachedRoute = seg("L0R0", { component: "my-route" });
|
|
715
|
+
// Server returns the SAME layout but with loading=<skeleton> instead of false
|
|
716
|
+
const serverLayout = seg("L0", {
|
|
717
|
+
type: "layout",
|
|
718
|
+
component: "my-layout",
|
|
719
|
+
loading: "skeleton",
|
|
720
|
+
});
|
|
721
|
+
const serverRoute = seg("L0R0", { component: "updated-route" });
|
|
722
|
+
const store = createMockStore({
|
|
723
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
724
|
+
});
|
|
725
|
+
const { client } = createMockClient({
|
|
726
|
+
metadata: {
|
|
727
|
+
isPartial: true,
|
|
728
|
+
segments: [serverLayout, serverRoute],
|
|
729
|
+
matched: ["L0", "L0R0"],
|
|
730
|
+
diff: ["L0", "L0R0"],
|
|
731
|
+
},
|
|
732
|
+
});
|
|
733
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
734
|
+
const tx = createMockTx();
|
|
735
|
+
const updater = createPartialUpdater({
|
|
736
|
+
store: store,
|
|
737
|
+
client: client,
|
|
738
|
+
onUpdate: vi.fn(),
|
|
739
|
+
renderSegments,
|
|
740
|
+
});
|
|
741
|
+
await updater("http://localhost/page", ["L0", "L0R0"], false, undefined, tx, {
|
|
742
|
+
isAction: true,
|
|
743
|
+
});
|
|
744
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
745
|
+
const layout = rendered.find((s) => s.id === "L0");
|
|
746
|
+
// loading must be preserved from cache (false), not server (skeleton)
|
|
747
|
+
expect(layout.loading).toBe(false);
|
|
748
|
+
// Component should come from server (updated)
|
|
749
|
+
const route = rendered.find((s) => s.id === "L0R0");
|
|
750
|
+
expect(route.component).toBe("updated-route");
|
|
751
|
+
});
|
|
752
|
+
it("preserves cached mountPath when server returns different mountPath", async () => {
|
|
753
|
+
// SSR may not include mountPath, but action revalidation may include it
|
|
754
|
+
// (from include() scope). The conditional MountContextProvider wrapper
|
|
755
|
+
// in renderSegments changes tree depth.
|
|
756
|
+
const cachedLayout = seg("L0", {
|
|
757
|
+
type: "layout",
|
|
758
|
+
component: "my-layout",
|
|
759
|
+
mountPath: undefined,
|
|
760
|
+
});
|
|
761
|
+
const cachedRoute = seg("L0R0", { component: "my-route" });
|
|
762
|
+
const serverLayout = seg("L0", {
|
|
763
|
+
type: "layout",
|
|
764
|
+
component: "my-layout",
|
|
765
|
+
mountPath: "/shop",
|
|
766
|
+
});
|
|
767
|
+
const store = createMockStore({
|
|
768
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
769
|
+
});
|
|
770
|
+
const { client } = createMockClient({
|
|
771
|
+
metadata: {
|
|
772
|
+
isPartial: true,
|
|
773
|
+
segments: [serverLayout],
|
|
774
|
+
matched: ["L0", "L0R0"],
|
|
775
|
+
diff: ["L0"],
|
|
776
|
+
},
|
|
777
|
+
});
|
|
778
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
779
|
+
const tx = createMockTx();
|
|
780
|
+
const updater = createPartialUpdater({
|
|
781
|
+
store: store,
|
|
782
|
+
client: client,
|
|
783
|
+
onUpdate: vi.fn(),
|
|
784
|
+
renderSegments,
|
|
785
|
+
});
|
|
786
|
+
await updater("http://localhost/page", ["L0", "L0R0"], false, undefined, tx);
|
|
787
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
788
|
+
const layout = rendered.find((s) => s.id === "L0");
|
|
789
|
+
// mountPath must be preserved from cache (undefined), not server ("/shop")
|
|
790
|
+
expect(layout.mountPath).toBeUndefined();
|
|
791
|
+
});
|
|
792
|
+
it("preserves loading + mountPath + null component together during action", async () => {
|
|
793
|
+
// Combined scenario: layout has all three structural properties differing
|
|
794
|
+
const cachedLayout = seg("L0", {
|
|
795
|
+
type: "layout",
|
|
796
|
+
component: "my-layout",
|
|
797
|
+
loading: false,
|
|
798
|
+
mountPath: undefined,
|
|
799
|
+
});
|
|
800
|
+
const cachedRoute = seg("L0R0", { component: "my-route" });
|
|
801
|
+
const serverLayout = seg("L0", {
|
|
802
|
+
type: "layout",
|
|
803
|
+
component: null, // null = don't re-render
|
|
804
|
+
loading: "skeleton", // different from cached
|
|
805
|
+
mountPath: "/shop", // different from cached
|
|
806
|
+
});
|
|
807
|
+
const serverRoute = seg("L0R0", { component: "updated" });
|
|
808
|
+
const store = createMockStore({
|
|
809
|
+
cachedSegments: [cachedLayout, cachedRoute],
|
|
810
|
+
});
|
|
811
|
+
const { client } = createMockClient({
|
|
812
|
+
metadata: {
|
|
813
|
+
isPartial: true,
|
|
814
|
+
segments: [serverLayout, serverRoute],
|
|
815
|
+
matched: ["L0", "L0R0"],
|
|
816
|
+
diff: ["L0", "L0R0"],
|
|
817
|
+
},
|
|
818
|
+
});
|
|
819
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
820
|
+
const tx = createMockTx();
|
|
821
|
+
const updater = createPartialUpdater({
|
|
822
|
+
store: store,
|
|
823
|
+
client: client,
|
|
824
|
+
onUpdate: vi.fn(),
|
|
825
|
+
renderSegments,
|
|
826
|
+
});
|
|
827
|
+
await updater("http://localhost/page", ["L0", "L0R0"], false, undefined, tx, {
|
|
828
|
+
isAction: true,
|
|
829
|
+
});
|
|
830
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
831
|
+
const layout = rendered.find((s) => s.id === "L0");
|
|
832
|
+
// All structural properties preserved from cache
|
|
833
|
+
expect(layout.component).toBe("my-layout"); // preserved (server sent null)
|
|
834
|
+
expect(layout.loading).toBe(false); // preserved (server sent skeleton)
|
|
835
|
+
expect(layout.mountPath).toBeUndefined(); // preserved (server sent /shop)
|
|
836
|
+
});
|
|
837
|
+
it("does not alter loading when cached and server values match", async () => {
|
|
838
|
+
const cachedRoute = seg("R0", {
|
|
839
|
+
component: "my-route",
|
|
840
|
+
loading: "skeleton",
|
|
841
|
+
});
|
|
842
|
+
const serverRoute = seg("R0", {
|
|
843
|
+
component: "updated",
|
|
844
|
+
loading: "skeleton",
|
|
845
|
+
});
|
|
846
|
+
const store = createMockStore({
|
|
847
|
+
cachedSegments: [cachedRoute],
|
|
848
|
+
});
|
|
849
|
+
const { client } = createMockClient({
|
|
850
|
+
metadata: {
|
|
851
|
+
isPartial: true,
|
|
852
|
+
segments: [serverRoute],
|
|
853
|
+
matched: ["R0"],
|
|
854
|
+
diff: ["R0"],
|
|
855
|
+
},
|
|
856
|
+
});
|
|
857
|
+
const renderSegments = vi.fn(async () => "tree");
|
|
858
|
+
const tx = createMockTx();
|
|
859
|
+
const updater = createPartialUpdater({
|
|
860
|
+
store: store,
|
|
861
|
+
client: client,
|
|
862
|
+
onUpdate: vi.fn(),
|
|
863
|
+
renderSegments,
|
|
864
|
+
});
|
|
865
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx);
|
|
866
|
+
const rendered = renderSegments.mock.calls[0][0];
|
|
867
|
+
expect(rendered[0].loading).toBe("skeleton");
|
|
868
|
+
expect(rendered[0].component).toBe("updated");
|
|
869
|
+
});
|
|
870
|
+
});
|
|
871
|
+
describe("intercept context tracking", () => {
|
|
872
|
+
it("sets intercept source URL when slots are active", async () => {
|
|
873
|
+
const store = createMockStore({
|
|
874
|
+
cachedSegments: [seg("L0", { type: "layout" }), seg("L0R0")],
|
|
875
|
+
currentUrl: "http://localhost/shop",
|
|
876
|
+
});
|
|
877
|
+
const { client } = createMockClient({
|
|
878
|
+
metadata: {
|
|
879
|
+
isPartial: true,
|
|
880
|
+
segments: [
|
|
881
|
+
seg("L0", { type: "layout" }),
|
|
882
|
+
seg("L0R0"),
|
|
883
|
+
],
|
|
884
|
+
matched: ["L0", "L0R0"],
|
|
885
|
+
diff: ["L0", "L0R0"],
|
|
886
|
+
slots: { "@modal": { active: true } },
|
|
887
|
+
},
|
|
888
|
+
});
|
|
889
|
+
const tx = createMockTx();
|
|
890
|
+
const updater = createPartialUpdater({
|
|
891
|
+
store: store,
|
|
892
|
+
client: client,
|
|
893
|
+
onUpdate: vi.fn(),
|
|
894
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
895
|
+
});
|
|
896
|
+
await updater("http://localhost/modal", ["L0", "L0R0"], false, undefined, tx);
|
|
897
|
+
// Should set intercept source URL from current URL (segmentState.currentUrl)
|
|
898
|
+
expect(store.setInterceptSourceUrl).toHaveBeenCalledWith("http://localhost/shop");
|
|
899
|
+
});
|
|
900
|
+
it("clears intercept source URL for non-intercept navigation", async () => {
|
|
901
|
+
const store = createMockStore({
|
|
902
|
+
cachedSegments: [seg("R0")],
|
|
903
|
+
});
|
|
904
|
+
const { client } = createMockClient({
|
|
905
|
+
metadata: {
|
|
906
|
+
isPartial: true,
|
|
907
|
+
segments: [seg("R0", { component: "updated" })],
|
|
908
|
+
matched: ["R0"],
|
|
909
|
+
diff: ["R0"],
|
|
910
|
+
},
|
|
911
|
+
});
|
|
912
|
+
const tx = createMockTx();
|
|
913
|
+
const updater = createPartialUpdater({
|
|
914
|
+
store: store,
|
|
915
|
+
client: client,
|
|
916
|
+
onUpdate: vi.fn(),
|
|
917
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
918
|
+
});
|
|
919
|
+
await updater("http://localhost/page", ["R0"], false, undefined, tx);
|
|
920
|
+
expect(store.setInterceptSourceUrl).toHaveBeenCalledWith(null);
|
|
921
|
+
});
|
|
922
|
+
it("does not update intercept context for actions", async () => {
|
|
923
|
+
const store = createMockStore({
|
|
924
|
+
cachedSegments: [seg("R0")],
|
|
925
|
+
});
|
|
926
|
+
const { client } = createMockClient({
|
|
927
|
+
metadata: {
|
|
928
|
+
isPartial: true,
|
|
929
|
+
segments: [seg("R0")],
|
|
930
|
+
matched: ["R0"],
|
|
931
|
+
diff: ["R0"],
|
|
932
|
+
},
|
|
933
|
+
});
|
|
934
|
+
const tx = createMockTx();
|
|
935
|
+
const updater = createPartialUpdater({
|
|
936
|
+
store: store,
|
|
937
|
+
client: client,
|
|
938
|
+
onUpdate: vi.fn(),
|
|
939
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
940
|
+
});
|
|
941
|
+
await updater("http://localhost/", ["R0"], false, undefined, tx, {
|
|
942
|
+
isAction: true,
|
|
943
|
+
});
|
|
944
|
+
expect(store.setInterceptSourceUrl).not.toHaveBeenCalled();
|
|
945
|
+
});
|
|
946
|
+
});
|
|
947
|
+
describe("streaming lifecycle", () => {
|
|
948
|
+
it("starts and ends streaming token", async () => {
|
|
949
|
+
const store = createMockStore({ cachedSegments: [seg("R0")] });
|
|
950
|
+
const endFn = vi.fn();
|
|
951
|
+
const { client, resolveStream } = createMockClient({
|
|
952
|
+
metadata: {
|
|
953
|
+
isPartial: true,
|
|
954
|
+
segments: [seg("R0")],
|
|
955
|
+
matched: ["R0"],
|
|
956
|
+
diff: ["R0"],
|
|
957
|
+
},
|
|
958
|
+
}, { hangStream: true });
|
|
959
|
+
const tx = createMockTx();
|
|
960
|
+
tx.startStreaming.mockReturnValue({ end: endFn });
|
|
961
|
+
const updater = createPartialUpdater({
|
|
962
|
+
store: store,
|
|
963
|
+
client: client,
|
|
964
|
+
onUpdate: vi.fn(),
|
|
965
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
966
|
+
});
|
|
967
|
+
// Don't await directly since stream hangs (promise flattening blocks it).
|
|
968
|
+
// Instead, capture the promise and resolve the stream, then await.
|
|
969
|
+
const resultPromise = updater("http://localhost/", ["R0"], false, undefined, tx);
|
|
970
|
+
// Let microtasks run so the function progresses to the return statement
|
|
971
|
+
await new Promise((r) => setTimeout(r, 10));
|
|
972
|
+
// startStreaming called
|
|
973
|
+
expect(tx.startStreaming).toHaveBeenCalled();
|
|
974
|
+
// end not called yet (stream still open)
|
|
975
|
+
expect(endFn).not.toHaveBeenCalled();
|
|
976
|
+
// Resolve stream
|
|
977
|
+
resolveStream();
|
|
978
|
+
await resultPromise;
|
|
979
|
+
// end called after stream completes
|
|
980
|
+
expect(endFn).toHaveBeenCalled();
|
|
981
|
+
});
|
|
982
|
+
it("ends streaming token when fetchPartial throws", async () => {
|
|
983
|
+
const store = createMockStore({ cachedSegments: [seg("R0")] });
|
|
984
|
+
const endFn = vi.fn();
|
|
985
|
+
const fetchError = new Error("network failure");
|
|
986
|
+
const client = {
|
|
987
|
+
fetchPartial: vi.fn(async () => {
|
|
988
|
+
throw fetchError;
|
|
989
|
+
}),
|
|
990
|
+
};
|
|
991
|
+
const tx = createMockTx();
|
|
992
|
+
tx.startStreaming.mockReturnValue({ end: endFn });
|
|
993
|
+
const updater = createPartialUpdater({
|
|
994
|
+
store: store,
|
|
995
|
+
client: client,
|
|
996
|
+
onUpdate: vi.fn(),
|
|
997
|
+
renderSegments: vi.fn(async () => "tree"),
|
|
998
|
+
});
|
|
999
|
+
// fetchPartial throws, so the updater should re-throw
|
|
1000
|
+
await expect(updater("http://localhost/", ["R0"], false, undefined, tx)).rejects.toThrow("network failure");
|
|
1001
|
+
// startStreaming was called before the fetch
|
|
1002
|
+
expect(tx.startStreaming).toHaveBeenCalled();
|
|
1003
|
+
// streamingToken.end() must be called even though fetchPartial threw,
|
|
1004
|
+
// otherwise isStreaming is permanently stuck as true
|
|
1005
|
+
expect(endFn).toHaveBeenCalled();
|
|
1006
|
+
});
|
|
1007
|
+
});
|
|
1008
|
+
});
|
|
1009
|
+
//# sourceMappingURL=partial-update.test.js.map
|