@rangojs/router 0.0.0-experimental.259 → 0.0.0-experimental.25dbaac3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/{CLAUDE.md → AGENTS.md} +4 -0
- package/README.md +471 -51
- 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 +490 -94
- package/dist/bin/rango.js.map +1 -0
- package/dist/browser/event-controller.d.ts +191 -0
- package/dist/browser/event-controller.d.ts.map +1 -0
- package/dist/browser/event-controller.js +559 -0
- package/dist/browser/event-controller.js.map +1 -0
- package/dist/browser/index.d.ts +2 -0
- package/dist/browser/index.d.ts.map +1 -0
- package/dist/browser/index.js +14 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/browser/link-interceptor.d.ts +38 -0
- package/dist/browser/link-interceptor.d.ts.map +1 -0
- package/dist/browser/link-interceptor.js +99 -0
- package/dist/browser/link-interceptor.js.map +1 -0
- package/dist/browser/logging.d.ts +10 -0
- package/dist/browser/logging.d.ts.map +1 -0
- package/dist/browser/logging.js +29 -0
- package/dist/browser/logging.js.map +1 -0
- package/dist/browser/lru-cache.d.ts +17 -0
- package/dist/browser/lru-cache.d.ts.map +1 -0
- package/dist/browser/lru-cache.js +50 -0
- package/dist/browser/lru-cache.js.map +1 -0
- package/dist/browser/merge-segment-loaders.d.ts +39 -0
- package/dist/browser/merge-segment-loaders.d.ts.map +1 -0
- package/dist/browser/merge-segment-loaders.js +102 -0
- package/dist/browser/merge-segment-loaders.js.map +1 -0
- package/dist/browser/navigation-bridge.d.ts +102 -0
- package/dist/browser/navigation-bridge.d.ts.map +1 -0
- package/dist/browser/navigation-bridge.js +708 -0
- package/dist/browser/navigation-bridge.js.map +1 -0
- package/dist/browser/navigation-client.d.ts +25 -0
- package/dist/browser/navigation-client.d.ts.map +1 -0
- package/dist/browser/navigation-client.js +157 -0
- package/dist/browser/navigation-client.js.map +1 -0
- package/dist/browser/navigation-store.d.ts +101 -0
- package/dist/browser/navigation-store.d.ts.map +1 -0
- package/dist/browser/navigation-store.js +625 -0
- package/dist/browser/navigation-store.js.map +1 -0
- package/dist/browser/partial-update.d.ts +75 -0
- package/dist/browser/partial-update.d.ts.map +1 -0
- package/dist/browser/partial-update.js +426 -0
- package/dist/browser/partial-update.js.map +1 -0
- package/dist/browser/react/Link.d.ts +86 -0
- package/dist/browser/react/Link.d.ts.map +1 -0
- package/dist/browser/react/Link.js +128 -0
- package/dist/browser/react/Link.js.map +1 -0
- package/dist/browser/react/NavigationProvider.d.ts +63 -0
- package/dist/browser/react/NavigationProvider.d.ts.map +1 -0
- package/dist/browser/react/NavigationProvider.js +216 -0
- package/dist/browser/react/NavigationProvider.js.map +1 -0
- package/dist/browser/react/ScrollRestoration.d.ts +75 -0
- package/dist/browser/react/ScrollRestoration.d.ts.map +1 -0
- package/dist/browser/react/ScrollRestoration.js +57 -0
- package/dist/browser/react/ScrollRestoration.js.map +1 -0
- package/dist/browser/react/context.d.ts +46 -0
- package/dist/browser/react/context.d.ts.map +1 -0
- package/dist/browser/react/context.js +10 -0
- package/dist/browser/react/context.js.map +1 -0
- package/dist/browser/react/index.d.ts +11 -0
- package/dist/browser/react/index.d.ts.map +1 -0
- package/dist/browser/react/index.js +22 -0
- package/dist/browser/react/index.js.map +1 -0
- package/dist/browser/react/location-state-shared.d.ts +63 -0
- package/dist/browser/react/location-state-shared.d.ts.map +1 -0
- package/dist/browser/react/location-state-shared.js +81 -0
- package/dist/browser/react/location-state-shared.js.map +1 -0
- package/dist/browser/react/location-state.d.ts +23 -0
- package/dist/browser/react/location-state.d.ts.map +1 -0
- package/dist/browser/react/location-state.js +29 -0
- package/dist/browser/react/location-state.js.map +1 -0
- package/dist/browser/react/mount-context.d.ts +24 -0
- package/dist/browser/react/mount-context.d.ts.map +1 -0
- package/dist/browser/react/mount-context.js +24 -0
- package/dist/browser/react/mount-context.js.map +1 -0
- package/dist/browser/react/use-action.d.ts +64 -0
- package/dist/browser/react/use-action.d.ts.map +1 -0
- package/dist/browser/react/use-action.js +134 -0
- package/dist/browser/react/use-action.js.map +1 -0
- package/dist/browser/react/use-client-cache.d.ts +41 -0
- package/dist/browser/react/use-client-cache.d.ts.map +1 -0
- package/dist/browser/react/use-client-cache.js +39 -0
- package/dist/browser/react/use-client-cache.js.map +1 -0
- package/dist/browser/react/use-handle.d.ts +31 -0
- package/dist/browser/react/use-handle.d.ts.map +1 -0
- package/dist/browser/react/use-handle.js +144 -0
- package/dist/browser/react/use-handle.js.map +1 -0
- package/dist/browser/react/use-href.d.ts +33 -0
- package/dist/browser/react/use-href.d.ts.map +1 -0
- package/dist/browser/react/use-href.js +39 -0
- package/dist/browser/react/use-href.js.map +1 -0
- package/dist/browser/react/use-link-status.d.ts +37 -0
- package/dist/browser/react/use-link-status.d.ts.map +1 -0
- package/dist/browser/react/use-link-status.js +99 -0
- package/dist/browser/react/use-link-status.js.map +1 -0
- package/dist/browser/react/use-mount.d.ts +25 -0
- package/dist/browser/react/use-mount.d.ts.map +1 -0
- package/dist/browser/react/use-mount.js +30 -0
- package/dist/browser/react/use-mount.js.map +1 -0
- package/dist/browser/react/use-navigation.d.ts +27 -0
- package/dist/browser/react/use-navigation.d.ts.map +1 -0
- package/dist/browser/react/use-navigation.js +87 -0
- package/dist/browser/react/use-navigation.js.map +1 -0
- package/dist/browser/react/use-segments.d.ts +38 -0
- package/dist/browser/react/use-segments.d.ts.map +1 -0
- package/dist/browser/react/use-segments.js +130 -0
- package/dist/browser/react/use-segments.js.map +1 -0
- package/dist/browser/request-controller.d.ts +26 -0
- package/dist/browser/request-controller.d.ts.map +1 -0
- package/dist/browser/request-controller.js +147 -0
- package/dist/browser/request-controller.js.map +1 -0
- package/dist/browser/rsc-router.d.ts +129 -0
- package/dist/browser/rsc-router.d.ts.map +1 -0
- package/dist/browser/rsc-router.js +195 -0
- package/dist/browser/rsc-router.js.map +1 -0
- package/dist/browser/scroll-restoration.d.ts +93 -0
- package/dist/browser/scroll-restoration.d.ts.map +1 -0
- package/dist/browser/scroll-restoration.js +321 -0
- package/dist/browser/scroll-restoration.js.map +1 -0
- package/dist/browser/segment-structure-assert.d.ts +17 -0
- package/dist/browser/segment-structure-assert.d.ts.map +1 -0
- package/dist/browser/segment-structure-assert.js +59 -0
- package/dist/browser/segment-structure-assert.js.map +1 -0
- package/dist/browser/server-action-bridge.d.ts +26 -0
- package/dist/browser/server-action-bridge.d.ts.map +1 -0
- package/dist/browser/server-action-bridge.js +668 -0
- package/dist/browser/server-action-bridge.js.map +1 -0
- package/dist/browser/shallow.d.ts +12 -0
- package/dist/browser/shallow.d.ts.map +1 -0
- package/dist/browser/shallow.js +34 -0
- package/dist/browser/shallow.js.map +1 -0
- package/dist/browser/types.d.ts +369 -0
- package/dist/browser/types.d.ts.map +1 -0
- package/dist/browser/types.js +2 -0
- package/dist/browser/types.js.map +1 -0
- package/dist/build/__tests__/generate-cli.test.d.ts +2 -0
- package/dist/build/__tests__/generate-cli.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-cli.test.js +237 -0
- package/dist/build/__tests__/generate-cli.test.js.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/generate-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-manifest.test.js +119 -0
- package/dist/build/__tests__/generate-manifest.test.js.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts +2 -0
- package/dist/build/__tests__/generate-route-types.test.d.ts.map +1 -0
- package/dist/build/__tests__/generate-route-types.test.js +620 -0
- package/dist/build/__tests__/generate-route-types.test.js.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts +2 -0
- package/dist/build/__tests__/per-router-manifest.test.d.ts.map +1 -0
- package/dist/build/__tests__/per-router-manifest.test.js +308 -0
- package/dist/build/__tests__/per-router-manifest.test.js.map +1 -0
- package/dist/build/generate-manifest.d.ts +81 -0
- package/dist/build/generate-manifest.d.ts.map +1 -0
- package/dist/build/generate-manifest.js +276 -0
- package/dist/build/generate-manifest.js.map +1 -0
- package/dist/build/generate-route-types.d.ts +115 -0
- package/dist/build/generate-route-types.d.ts.map +1 -0
- package/dist/build/generate-route-types.js +740 -0
- package/dist/build/generate-route-types.js.map +1 -0
- package/dist/build/index.d.ts +21 -0
- package/dist/build/index.d.ts.map +1 -0
- package/dist/build/index.js +21 -0
- package/dist/build/index.js.map +1 -0
- package/dist/build/route-trie.d.ts +71 -0
- package/dist/build/route-trie.d.ts.map +1 -0
- package/dist/build/route-trie.js +175 -0
- package/dist/build/route-trie.js.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts +2 -0
- package/dist/cache/__tests__/cache-scope.test.d.ts.map +1 -0
- package/dist/cache/__tests__/cache-scope.test.js +208 -0
- package/dist/cache/__tests__/cache-scope.test.js.map +1 -0
- package/dist/cache/__tests__/document-cache.test.d.ts +2 -0
- package/dist/cache/__tests__/document-cache.test.d.ts.map +1 -0
- package/dist/cache/__tests__/document-cache.test.js +345 -0
- package/dist/cache/__tests__/document-cache.test.js.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-segment-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-segment-store.test.js +425 -0
- package/dist/cache/__tests__/memory-segment-store.test.js.map +1 -0
- package/dist/cache/__tests__/memory-store.test.d.ts +2 -0
- package/dist/cache/__tests__/memory-store.test.d.ts.map +1 -0
- package/dist/cache/__tests__/memory-store.test.js +367 -0
- package/dist/cache/__tests__/memory-store.test.js.map +1 -0
- package/dist/cache/cache-scope.d.ts +102 -0
- package/dist/cache/cache-scope.d.ts.map +1 -0
- package/dist/cache/cache-scope.js +440 -0
- package/dist/cache/cache-scope.js.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts +2 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts.map +1 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js +330 -0
- package/dist/cache/cf/__tests__/cf-cache-store.test.js.map +1 -0
- package/dist/cache/cf/cf-cache-store.d.ts +165 -0
- package/dist/cache/cf/cf-cache-store.d.ts.map +1 -0
- package/dist/cache/cf/cf-cache-store.js +242 -0
- package/dist/cache/cf/cf-cache-store.js.map +1 -0
- package/dist/cache/cf/index.d.ts +14 -0
- package/dist/cache/cf/index.d.ts.map +1 -0
- package/dist/cache/cf/index.js +17 -0
- package/dist/cache/cf/index.js.map +1 -0
- package/dist/cache/document-cache.d.ts +64 -0
- package/dist/cache/document-cache.d.ts.map +1 -0
- package/dist/cache/document-cache.js +228 -0
- package/dist/cache/document-cache.js.map +1 -0
- package/dist/cache/index.d.ts +19 -0
- package/dist/cache/index.d.ts.map +1 -0
- package/dist/cache/index.js +21 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/cache/memory-segment-store.d.ts +110 -0
- package/dist/cache/memory-segment-store.d.ts.map +1 -0
- package/dist/cache/memory-segment-store.js +117 -0
- package/dist/cache/memory-segment-store.js.map +1 -0
- package/dist/cache/memory-store.d.ts +41 -0
- package/dist/cache/memory-store.d.ts.map +1 -0
- package/dist/cache/memory-store.js +191 -0
- package/dist/cache/memory-store.js.map +1 -0
- package/dist/cache/types.d.ts +317 -0
- package/dist/cache/types.d.ts.map +1 -0
- package/dist/cache/types.js +12 -0
- package/dist/cache/types.js.map +1 -0
- package/dist/client.d.ts +248 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +367 -0
- package/dist/client.js.map +1 -0
- package/dist/client.rsc.d.ts +26 -0
- package/dist/client.rsc.d.ts.map +1 -0
- package/dist/client.rsc.js +46 -0
- package/dist/client.rsc.js.map +1 -0
- package/dist/component-utils.d.ts +36 -0
- package/dist/component-utils.d.ts.map +1 -0
- package/dist/component-utils.js +61 -0
- package/dist/component-utils.js.map +1 -0
- package/dist/components/DefaultDocument.d.ts +13 -0
- package/dist/components/DefaultDocument.d.ts.map +1 -0
- package/dist/components/DefaultDocument.js +15 -0
- package/dist/components/DefaultDocument.js.map +1 -0
- package/dist/debug.d.ts +58 -0
- package/dist/debug.d.ts.map +1 -0
- package/dist/debug.js +157 -0
- package/dist/debug.js.map +1 -0
- package/dist/default-error-boundary.d.ts +11 -0
- package/dist/default-error-boundary.d.ts.map +1 -0
- package/dist/default-error-boundary.js +45 -0
- package/dist/default-error-boundary.js.map +1 -0
- package/dist/deps/browser.d.ts +2 -0
- package/dist/deps/browser.d.ts.map +1 -0
- package/dist/deps/browser.js +3 -0
- package/dist/deps/browser.js.map +1 -0
- package/dist/deps/html-stream-client.d.ts +2 -0
- package/dist/deps/html-stream-client.d.ts.map +1 -0
- package/dist/deps/html-stream-client.js +3 -0
- package/dist/deps/html-stream-client.js.map +1 -0
- package/dist/deps/html-stream-server.d.ts +2 -0
- package/dist/deps/html-stream-server.d.ts.map +1 -0
- package/dist/deps/html-stream-server.js +3 -0
- package/dist/deps/html-stream-server.js.map +1 -0
- package/dist/deps/rsc.d.ts +2 -0
- package/dist/deps/rsc.d.ts.map +1 -0
- package/dist/deps/rsc.js +4 -0
- package/dist/deps/rsc.js.map +1 -0
- package/dist/deps/ssr.d.ts +2 -0
- package/dist/deps/ssr.d.ts.map +1 -0
- package/dist/deps/ssr.js +3 -0
- package/dist/deps/ssr.js.map +1 -0
- package/dist/errors.d.ts +174 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +241 -0
- package/dist/errors.js.map +1 -0
- package/dist/handle.d.ts +78 -0
- package/dist/handle.d.ts.map +1 -0
- package/dist/handle.js +82 -0
- package/dist/handle.js.map +1 -0
- package/dist/handles/MetaTags.d.ts +14 -0
- package/dist/handles/MetaTags.d.ts.map +1 -0
- package/dist/handles/MetaTags.js +136 -0
- package/dist/handles/MetaTags.js.map +1 -0
- package/dist/handles/index.d.ts +6 -0
- package/dist/handles/index.d.ts.map +1 -0
- package/dist/handles/index.js +6 -0
- package/dist/handles/index.js.map +1 -0
- package/dist/handles/meta.d.ts +39 -0
- package/dist/handles/meta.d.ts.map +1 -0
- package/dist/handles/meta.js +202 -0
- package/dist/handles/meta.js.map +1 -0
- package/dist/host/__tests__/errors.test.d.ts +2 -0
- package/dist/host/__tests__/errors.test.d.ts.map +1 -0
- package/dist/host/__tests__/errors.test.js +76 -0
- package/dist/host/__tests__/errors.test.js.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-comprehensive.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js +732 -0
- package/dist/host/__tests__/pattern-comprehensive.test.js.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts +2 -0
- package/dist/host/__tests__/pattern-matcher.test.d.ts.map +1 -0
- package/dist/host/__tests__/pattern-matcher.test.js +251 -0
- package/dist/host/__tests__/pattern-matcher.test.js.map +1 -0
- package/dist/host/__tests__/router.test.d.ts +2 -0
- package/dist/host/__tests__/router.test.d.ts.map +1 -0
- package/dist/host/__tests__/router.test.js +241 -0
- package/dist/host/__tests__/router.test.js.map +1 -0
- package/dist/host/__tests__/testing.test.d.ts +2 -0
- package/dist/host/__tests__/testing.test.d.ts.map +1 -0
- package/dist/host/__tests__/testing.test.js +64 -0
- package/dist/host/__tests__/testing.test.js.map +1 -0
- package/dist/host/__tests__/utils.test.d.ts +2 -0
- package/dist/host/__tests__/utils.test.d.ts.map +1 -0
- package/dist/host/__tests__/utils.test.js +29 -0
- package/dist/host/__tests__/utils.test.js.map +1 -0
- package/dist/host/cookie-handler.d.ts +34 -0
- package/dist/host/cookie-handler.d.ts.map +1 -0
- package/dist/host/cookie-handler.js +124 -0
- package/dist/host/cookie-handler.js.map +1 -0
- package/dist/host/errors.d.ts +56 -0
- package/dist/host/errors.d.ts.map +1 -0
- package/dist/host/errors.js +79 -0
- package/dist/host/errors.js.map +1 -0
- package/dist/host/index.d.ts +29 -0
- package/dist/host/index.d.ts.map +1 -0
- package/dist/host/index.js +32 -0
- package/dist/host/index.js.map +1 -0
- package/dist/host/pattern-matcher.d.ts +36 -0
- package/dist/host/pattern-matcher.d.ts.map +1 -0
- package/dist/host/pattern-matcher.js +172 -0
- package/dist/host/pattern-matcher.js.map +1 -0
- package/dist/host/router.d.ts +26 -0
- package/dist/host/router.d.ts.map +1 -0
- package/dist/host/router.js +218 -0
- package/dist/host/router.js.map +1 -0
- package/dist/host/testing.d.ts +36 -0
- package/dist/host/testing.d.ts.map +1 -0
- package/dist/host/testing.js +55 -0
- package/dist/host/testing.js.map +1 -0
- package/dist/host/types.d.ts +115 -0
- package/dist/host/types.d.ts.map +1 -0
- package/dist/host/types.js +7 -0
- package/dist/host/types.js.map +1 -0
- package/dist/host/utils.d.ts +21 -0
- package/dist/host/utils.d.ts.map +1 -0
- package/dist/host/utils.js +23 -0
- package/dist/host/utils.js.map +1 -0
- package/dist/href-client.d.ts +131 -0
- package/dist/href-client.d.ts.map +1 -0
- package/dist/href-client.js +64 -0
- package/dist/href-client.js.map +1 -0
- package/{src/href-context.ts → dist/href-context.d.ts} +7 -11
- package/dist/href-context.d.ts.map +1 -0
- package/dist/href-context.js +21 -0
- package/dist/href-context.js.map +1 -0
- package/dist/index.d.ts +73 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +91 -0
- package/dist/index.js.map +1 -0
- package/dist/index.rsc.d.ts +32 -0
- package/dist/index.rsc.d.ts.map +1 -0
- package/dist/index.rsc.js +40 -0
- package/dist/index.rsc.js.map +1 -0
- package/dist/internal-debug.d.ts +2 -0
- package/dist/internal-debug.d.ts.map +1 -0
- package/dist/internal-debug.js +5 -0
- package/dist/internal-debug.js.map +1 -0
- package/dist/loader.d.ts +14 -0
- package/dist/loader.d.ts.map +1 -0
- package/dist/loader.js +20 -0
- package/dist/loader.js.map +1 -0
- package/dist/loader.rsc.d.ts +19 -0
- package/dist/loader.rsc.d.ts.map +1 -0
- package/dist/loader.rsc.js +99 -0
- package/dist/loader.rsc.js.map +1 -0
- package/dist/network-error-thrower.d.ts +17 -0
- package/dist/network-error-thrower.d.ts.map +1 -0
- package/dist/network-error-thrower.js +14 -0
- package/dist/network-error-thrower.js.map +1 -0
- package/dist/outlet-context.d.ts +13 -0
- package/dist/outlet-context.d.ts.map +1 -0
- package/dist/outlet-context.js +3 -0
- package/dist/outlet-context.js.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts +2 -0
- package/dist/prerender/__tests__/param-hash.test.d.ts.map +1 -0
- package/dist/prerender/__tests__/param-hash.test.js +148 -0
- package/dist/prerender/__tests__/param-hash.test.js.map +1 -0
- package/dist/prerender/param-hash.d.ts +16 -0
- package/dist/prerender/param-hash.d.ts.map +1 -0
- package/dist/prerender/param-hash.js +36 -0
- package/dist/prerender/param-hash.js.map +1 -0
- package/dist/prerender/store.d.ts +38 -0
- package/dist/prerender/store.d.ts.map +1 -0
- package/dist/prerender/store.js +61 -0
- package/dist/prerender/store.js.map +1 -0
- package/dist/prerender.d.ts +66 -0
- package/dist/prerender.d.ts.map +1 -0
- package/dist/prerender.js +57 -0
- package/dist/prerender.js.map +1 -0
- package/dist/reverse.d.ts +196 -0
- package/dist/reverse.d.ts.map +1 -0
- package/dist/reverse.js +78 -0
- package/dist/reverse.js.map +1 -0
- package/dist/root-error-boundary.d.ts +33 -0
- package/dist/root-error-boundary.d.ts.map +1 -0
- package/dist/root-error-boundary.js +165 -0
- package/dist/root-error-boundary.js.map +1 -0
- package/dist/route-content-wrapper.d.ts +46 -0
- package/dist/route-content-wrapper.d.ts.map +1 -0
- package/dist/route-content-wrapper.js +77 -0
- package/dist/route-content-wrapper.js.map +1 -0
- package/dist/route-definition.d.ts +421 -0
- package/dist/route-definition.d.ts.map +1 -0
- package/dist/route-definition.js +868 -0
- package/dist/route-definition.js.map +1 -0
- package/dist/route-map-builder.d.ts +155 -0
- package/dist/route-map-builder.d.ts.map +1 -0
- package/dist/route-map-builder.js +237 -0
- package/dist/route-map-builder.js.map +1 -0
- package/dist/route-types.d.ts +165 -0
- package/dist/route-types.d.ts.map +1 -0
- package/dist/route-types.js +7 -0
- package/dist/route-types.js.map +1 -0
- package/dist/router/__tests__/handler-context.test.d.ts +2 -0
- package/dist/router/__tests__/handler-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/handler-context.test.js +65 -0
- package/dist/router/__tests__/handler-context.test.js.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts +2 -0
- package/dist/router/__tests__/loader-cycle-detection.test.d.ts.map +1 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js +221 -0
- package/dist/router/__tests__/loader-cycle-detection.test.js.map +1 -0
- package/dist/router/__tests__/match-context.test.d.ts +2 -0
- package/dist/router/__tests__/match-context.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-context.test.js +92 -0
- package/dist/router/__tests__/match-context.test.js.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts +2 -0
- package/dist/router/__tests__/match-pipelines.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-pipelines.test.js +417 -0
- package/dist/router/__tests__/match-pipelines.test.js.map +1 -0
- package/dist/router/__tests__/match-result.test.d.ts +2 -0
- package/dist/router/__tests__/match-result.test.d.ts.map +1 -0
- package/dist/router/__tests__/match-result.test.js +457 -0
- package/dist/router/__tests__/match-result.test.js.map +1 -0
- package/dist/router/__tests__/on-error.test.d.ts +2 -0
- package/dist/router/__tests__/on-error.test.d.ts.map +1 -0
- package/dist/router/__tests__/on-error.test.js +678 -0
- package/dist/router/__tests__/on-error.test.js.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts +2 -0
- package/dist/router/__tests__/pattern-matching.test.d.ts.map +1 -0
- package/dist/router/__tests__/pattern-matching.test.js +629 -0
- package/dist/router/__tests__/pattern-matching.test.js.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts +2 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts.map +1 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js +155 -0
- package/dist/router/__tests__/segment-resolution-parallel-loading.test.js.map +1 -0
- package/dist/router/error-handling.d.ts +77 -0
- package/dist/router/error-handling.d.ts.map +1 -0
- package/dist/router/error-handling.js +202 -0
- package/dist/router/error-handling.js.map +1 -0
- package/dist/router/handler-context.d.ts +20 -0
- package/dist/router/handler-context.d.ts.map +1 -0
- package/dist/router/handler-context.js +198 -0
- package/dist/router/handler-context.js.map +1 -0
- package/dist/router/intercept-resolution.d.ts +66 -0
- package/dist/router/intercept-resolution.d.ts.map +1 -0
- package/dist/router/intercept-resolution.js +246 -0
- package/dist/router/intercept-resolution.js.map +1 -0
- package/dist/router/loader-resolution.d.ts +64 -0
- package/dist/router/loader-resolution.d.ts.map +1 -0
- package/dist/router/loader-resolution.js +284 -0
- package/dist/router/loader-resolution.js.map +1 -0
- package/dist/router/logging.d.ts +15 -0
- package/dist/router/logging.d.ts.map +1 -0
- package/dist/router/logging.js +99 -0
- package/dist/router/logging.js.map +1 -0
- package/dist/router/manifest.d.ts +22 -0
- package/dist/router/manifest.d.ts.map +1 -0
- package/dist/router/manifest.js +181 -0
- package/dist/router/manifest.js.map +1 -0
- package/dist/router/match-api.d.ts +35 -0
- package/dist/router/match-api.d.ts.map +1 -0
- package/dist/router/match-api.js +406 -0
- package/dist/router/match-api.js.map +1 -0
- package/dist/router/match-context.d.ts +206 -0
- package/dist/router/match-context.d.ts.map +1 -0
- package/dist/router/match-context.js +17 -0
- package/dist/router/match-context.js.map +1 -0
- package/dist/router/match-middleware/background-revalidation.d.ts +127 -0
- package/dist/router/match-middleware/background-revalidation.d.ts.map +1 -0
- package/dist/router/match-middleware/background-revalidation.js +75 -0
- package/dist/router/match-middleware/background-revalidation.js.map +1 -0
- package/dist/router/match-middleware/cache-lookup.d.ts +112 -0
- package/dist/router/match-middleware/cache-lookup.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-lookup.js +257 -0
- package/dist/router/match-middleware/cache-lookup.js.map +1 -0
- package/dist/router/match-middleware/cache-store.d.ts +113 -0
- package/dist/router/match-middleware/cache-store.d.ts.map +1 -0
- package/dist/router/match-middleware/cache-store.js +108 -0
- package/dist/router/match-middleware/cache-store.js.map +1 -0
- package/dist/router/match-middleware/index.d.ts +81 -0
- package/dist/router/match-middleware/index.d.ts.map +1 -0
- package/dist/router/match-middleware/index.js +80 -0
- package/dist/router/match-middleware/index.js.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts +117 -0
- package/dist/router/match-middleware/intercept-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/intercept-resolution.js +134 -0
- package/dist/router/match-middleware/intercept-resolution.js.map +1 -0
- package/dist/router/match-middleware/segment-resolution.d.ts +99 -0
- package/dist/router/match-middleware/segment-resolution.d.ts.map +1 -0
- package/dist/router/match-middleware/segment-resolution.js +53 -0
- package/dist/router/match-middleware/segment-resolution.js.map +1 -0
- package/dist/router/match-pipelines.d.ts +147 -0
- package/dist/router/match-pipelines.d.ts.map +1 -0
- package/dist/router/match-pipelines.js +82 -0
- package/dist/router/match-pipelines.js.map +1 -0
- package/dist/router/match-result.d.ts +126 -0
- package/dist/router/match-result.d.ts.map +1 -0
- package/dist/router/match-result.js +93 -0
- package/dist/router/match-result.js.map +1 -0
- package/dist/router/metrics.d.ts +20 -0
- package/dist/router/metrics.d.ts.map +1 -0
- package/dist/router/metrics.js +47 -0
- package/dist/router/metrics.js.map +1 -0
- package/dist/router/middleware.d.ts +249 -0
- package/dist/router/middleware.d.ts.map +1 -0
- package/dist/router/middleware.js +434 -0
- package/dist/router/middleware.js.map +1 -0
- package/dist/router/middleware.test.d.ts +2 -0
- package/dist/router/middleware.test.d.ts.map +1 -0
- package/dist/router/middleware.test.js +816 -0
- package/dist/router/middleware.test.js.map +1 -0
- package/dist/router/pattern-matching.d.ts +149 -0
- package/dist/router/pattern-matching.d.ts.map +1 -0
- package/dist/router/pattern-matching.js +349 -0
- package/dist/router/pattern-matching.js.map +1 -0
- package/dist/router/revalidation.d.ts +44 -0
- package/dist/router/revalidation.d.ts.map +1 -0
- package/dist/router/revalidation.js +147 -0
- package/dist/router/revalidation.js.map +1 -0
- package/dist/router/router-context.d.ts +135 -0
- package/dist/router/router-context.d.ts.map +1 -0
- package/dist/router/router-context.js +36 -0
- package/dist/router/router-context.js.map +1 -0
- package/dist/router/segment-resolution.d.ts +127 -0
- package/dist/router/segment-resolution.d.ts.map +1 -0
- package/dist/router/segment-resolution.js +919 -0
- package/dist/router/segment-resolution.js.map +1 -0
- package/dist/router/trie-matching.d.ts +40 -0
- package/dist/router/trie-matching.d.ts.map +1 -0
- package/dist/router/trie-matching.js +127 -0
- package/dist/router/trie-matching.js.map +1 -0
- package/dist/router/types.d.ts +136 -0
- package/dist/router/types.d.ts.map +1 -0
- package/dist/router/types.js +7 -0
- package/dist/router/types.js.map +1 -0
- package/dist/router.d.ts +753 -0
- package/dist/router.d.ts.map +1 -0
- package/dist/router.gen.d.ts +6 -0
- package/dist/router.gen.d.ts.map +1 -0
- package/dist/router.gen.js +6 -0
- package/dist/router.gen.js.map +1 -0
- package/dist/router.js +1304 -0
- package/dist/router.js.map +1 -0
- package/dist/rsc/__tests__/helpers.test.d.ts +2 -0
- package/dist/rsc/__tests__/helpers.test.d.ts.map +1 -0
- package/dist/rsc/__tests__/helpers.test.js +140 -0
- package/dist/rsc/__tests__/helpers.test.js.map +1 -0
- package/dist/rsc/handler.d.ts +45 -0
- package/dist/rsc/handler.d.ts.map +1 -0
- package/dist/rsc/handler.js +1172 -0
- package/dist/rsc/handler.js.map +1 -0
- package/dist/rsc/helpers.d.ts +16 -0
- package/dist/rsc/helpers.d.ts.map +1 -0
- package/dist/rsc/helpers.js +55 -0
- package/dist/rsc/helpers.js.map +1 -0
- package/dist/rsc/index.d.ts +22 -0
- package/dist/rsc/index.d.ts.map +1 -0
- package/dist/rsc/index.js +23 -0
- package/dist/rsc/index.js.map +1 -0
- package/dist/rsc/nonce.d.ts +9 -0
- package/dist/rsc/nonce.d.ts.map +1 -0
- package/dist/rsc/nonce.js +18 -0
- package/dist/rsc/nonce.js.map +1 -0
- package/dist/rsc/types.d.ts +206 -0
- package/dist/rsc/types.d.ts.map +1 -0
- package/dist/rsc/types.js +8 -0
- package/dist/rsc/types.js.map +1 -0
- package/dist/search-params.d.ts +103 -0
- package/dist/search-params.d.ts.map +1 -0
- package/dist/search-params.js +74 -0
- package/dist/search-params.js.map +1 -0
- package/dist/segment-system.d.ts +75 -0
- package/dist/segment-system.d.ts.map +1 -0
- package/dist/segment-system.js +336 -0
- package/dist/segment-system.js.map +1 -0
- package/dist/server/context.d.ts +245 -0
- package/dist/server/context.d.ts.map +1 -0
- package/dist/server/context.js +197 -0
- package/dist/server/context.js.map +1 -0
- package/dist/server/fetchable-loader-store.d.ts +18 -0
- package/dist/server/fetchable-loader-store.d.ts.map +1 -0
- package/dist/server/fetchable-loader-store.js +18 -0
- package/dist/server/fetchable-loader-store.js.map +1 -0
- package/dist/server/handle-store.d.ts +85 -0
- package/dist/server/handle-store.d.ts.map +1 -0
- package/dist/server/handle-store.js +142 -0
- package/dist/server/handle-store.js.map +1 -0
- package/dist/server/loader-registry.d.ts +55 -0
- package/dist/server/loader-registry.d.ts.map +1 -0
- package/dist/server/loader-registry.js +132 -0
- package/dist/server/loader-registry.js.map +1 -0
- package/dist/server/request-context.d.ts +226 -0
- package/dist/server/request-context.d.ts.map +1 -0
- package/dist/server/request-context.js +290 -0
- package/dist/server/request-context.js.map +1 -0
- package/dist/server/root-layout.d.ts +4 -0
- package/dist/server/root-layout.d.ts.map +1 -0
- package/dist/server/root-layout.js +5 -0
- package/dist/server/root-layout.js.map +1 -0
- package/dist/server.d.ts +15 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +20 -0
- package/dist/server.js.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts +2 -0
- package/dist/ssr/__tests__/ssr-handler.test.d.ts.map +1 -0
- package/dist/ssr/__tests__/ssr-handler.test.js +132 -0
- package/dist/ssr/__tests__/ssr-handler.test.js.map +1 -0
- package/dist/ssr/index.d.ts +98 -0
- package/dist/ssr/index.d.ts.map +1 -0
- package/dist/ssr/index.js +158 -0
- package/dist/ssr/index.js.map +1 -0
- package/dist/static-handler.d.ts +50 -0
- package/dist/static-handler.d.ts.map +1 -0
- package/dist/static-handler.gen.d.ts +5 -0
- package/dist/static-handler.gen.d.ts.map +1 -0
- package/dist/static-handler.gen.js +5 -0
- package/dist/static-handler.gen.js.map +1 -0
- package/dist/static-handler.js +29 -0
- package/dist/static-handler.js.map +1 -0
- package/dist/theme/ThemeProvider.d.ts +20 -0
- package/dist/theme/ThemeProvider.d.ts.map +1 -0
- package/dist/theme/ThemeProvider.js +240 -0
- package/dist/theme/ThemeProvider.js.map +1 -0
- package/dist/theme/ThemeScript.d.ts +48 -0
- package/dist/theme/ThemeScript.d.ts.map +1 -0
- package/dist/theme/ThemeScript.js +13 -0
- package/dist/theme/ThemeScript.js.map +1 -0
- package/dist/theme/__tests__/theme.test.d.ts +2 -0
- package/dist/theme/__tests__/theme.test.d.ts.map +1 -0
- package/dist/theme/__tests__/theme.test.js +103 -0
- package/dist/theme/__tests__/theme.test.js.map +1 -0
- package/dist/theme/constants.d.ts +29 -0
- package/dist/theme/constants.d.ts.map +1 -0
- package/dist/theme/constants.js +48 -0
- package/dist/theme/constants.js.map +1 -0
- package/dist/theme/index.d.ts +31 -0
- package/dist/theme/index.d.ts.map +1 -0
- package/dist/theme/index.js +36 -0
- package/dist/theme/index.js.map +1 -0
- package/dist/theme/theme-context.d.ts +40 -0
- package/dist/theme/theme-context.d.ts.map +1 -0
- package/dist/theme/theme-context.js +60 -0
- package/dist/theme/theme-context.js.map +1 -0
- package/dist/theme/theme-script.d.ts +27 -0
- package/dist/theme/theme-script.d.ts.map +1 -0
- package/dist/theme/theme-script.js +147 -0
- package/dist/theme/theme-script.js.map +1 -0
- package/dist/theme/types.d.ts +163 -0
- package/dist/theme/types.d.ts.map +1 -0
- package/dist/theme/types.js +11 -0
- package/dist/theme/types.js.map +1 -0
- package/dist/theme/use-theme.d.ts +12 -0
- package/dist/theme/use-theme.d.ts.map +1 -0
- package/dist/theme/use-theme.js +40 -0
- package/dist/theme/use-theme.js.map +1 -0
- package/dist/types.d.ts +1479 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +10 -0
- package/dist/types.js.map +1 -0
- package/dist/urls.d.ts +441 -0
- package/dist/urls.d.ts.map +1 -0
- package/dist/urls.gen.d.ts +8 -0
- package/dist/urls.gen.d.ts.map +1 -0
- package/dist/urls.gen.js +8 -0
- package/dist/urls.gen.js.map +1 -0
- package/dist/urls.js +443 -0
- package/dist/urls.js.map +1 -0
- package/dist/use-loader.d.ts +127 -0
- package/dist/use-loader.d.ts.map +1 -0
- package/dist/use-loader.js +237 -0
- package/dist/use-loader.js.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts +2 -0
- package/dist/vite/__tests__/ast-handler-extract.test.d.ts.map +1 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js +294 -0
- package/dist/vite/__tests__/ast-handler-extract.test.js.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-id-utils.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-id-utils.test.js +224 -0
- package/dist/vite/__tests__/expose-id-utils.test.js.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-internal-ids.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js +647 -0
- package/dist/vite/__tests__/expose-internal-ids.test.js.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts +2 -0
- package/dist/vite/__tests__/expose-router-id.test.d.ts.map +1 -0
- package/dist/vite/__tests__/expose-router-id.test.js +39 -0
- package/dist/vite/__tests__/expose-router-id.test.js.map +1 -0
- package/dist/vite/ast-handler-extract.d.ts +49 -0
- package/dist/vite/ast-handler-extract.d.ts.map +1 -0
- package/dist/vite/ast-handler-extract.js +249 -0
- package/dist/vite/ast-handler-extract.js.map +1 -0
- package/dist/vite/expose-action-id.d.ts +19 -0
- package/dist/vite/expose-action-id.d.ts.map +1 -0
- package/dist/vite/expose-action-id.js +250 -0
- package/dist/vite/expose-action-id.js.map +1 -0
- package/dist/vite/expose-id-utils.d.ts +69 -0
- package/dist/vite/expose-id-utils.d.ts.map +1 -0
- package/dist/vite/expose-id-utils.js +289 -0
- package/dist/vite/expose-id-utils.js.map +1 -0
- package/dist/vite/expose-internal-ids.d.ts +22 -0
- package/dist/vite/expose-internal-ids.d.ts.map +1 -0
- package/dist/vite/expose-internal-ids.js +886 -0
- package/dist/vite/expose-internal-ids.js.map +1 -0
- package/dist/vite/index.d.ts +149 -0
- package/dist/vite/index.d.ts.map +1 -0
- package/dist/vite/index.js +3423 -1718
- package/dist/vite/index.js.bak +5448 -0
- package/dist/vite/index.js.map +1 -0
- package/dist/vite/index.named-routes.gen.ts +103 -0
- package/dist/vite/package-resolution.d.ts +43 -0
- package/dist/vite/package-resolution.d.ts.map +1 -0
- package/dist/vite/package-resolution.js +112 -0
- package/dist/vite/package-resolution.js.map +1 -0
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/dist/vite/virtual-entries.d.ts +25 -0
- package/dist/vite/virtual-entries.d.ts.map +1 -0
- package/dist/vite/virtual-entries.js +110 -0
- package/dist/vite/virtual-entries.js.map +1 -0
- package/package.json +21 -18
- package/skills/breadcrumbs/SKILL.md +252 -0
- package/skills/cache-guide/SKILL.md +41 -5
- package/skills/caching/SKILL.md +49 -8
- package/skills/document-cache/SKILL.md +2 -2
- package/skills/handler-use/SKILL.md +362 -0
- package/skills/hooks/SKILL.md +94 -52
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +99 -0
- package/skills/layout/SKILL.md +84 -2
- package/skills/links/SKILL.md +93 -17
- package/skills/loader/SKILL.md +367 -36
- package/skills/middleware/SKILL.md +145 -33
- package/skills/migrate-nextjs/SKILL.md +562 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/parallel/SKILL.md +249 -2
- package/skills/prerender/SKILL.md +261 -49
- package/skills/rango/SKILL.md +26 -24
- package/skills/response-routes/SKILL.md +11 -3
- package/skills/route/SKILL.md +100 -5
- package/skills/router-setup/SKILL.md +172 -5
- package/skills/server-actions/SKILL.md +739 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/theme/SKILL.md +5 -4
- package/skills/typesafety/SKILL.md +95 -38
- package/skills/use-cache/SKILL.md +16 -2
- package/src/__internal.ts +93 -1
- package/src/bin/rango.ts +56 -19
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/app-shell.ts +52 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +78 -52
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +1 -1
- package/src/browser/link-interceptor.ts +23 -3
- package/src/browser/merge-segment-loaders.ts +9 -2
- package/src/browser/navigation-bridge.ts +234 -453
- package/src/browser/navigation-client.ts +191 -86
- package/src/browser/navigation-store.ts +71 -41
- package/src/browser/navigation-transaction.ts +297 -0
- package/src/browser/partial-update.ts +215 -167
- package/src/browser/prefetch/cache.ts +314 -0
- package/src/browser/prefetch/fetch.ts +282 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +191 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +152 -0
- package/src/browser/react/Link.tsx +241 -47
- package/src/browser/react/NavigationProvider.tsx +136 -16
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +55 -0
- package/src/browser/react/index.ts +2 -6
- package/src/browser/react/location-state-shared.ts +29 -11
- package/src/browser/react/location-state.ts +6 -4
- 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 +23 -45
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +28 -120
- package/src/browser/react/use-navigation.ts +29 -34
- package/src/browser/react/use-params.ts +16 -35
- package/src/browser/react/use-pathname.ts +2 -3
- package/src/browser/react/use-router.ts +31 -14
- package/src/browser/react/use-search-params.ts +2 -1
- package/src/browser/react/use-segments.ts +82 -118
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +202 -67
- package/src/browser/scroll-restoration.ts +47 -45
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/server-action-bridge.ts +464 -409
- package/src/browser/types.ts +100 -40
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +44 -19
- package/src/build/generate-route-types.ts +7 -0
- package/src/build/index.ts +1 -0
- package/src/build/route-trie.ts +67 -25
- package/src/build/route-types/codegen.ts +13 -4
- package/src/build/route-types/include-resolution.ts +21 -1
- package/src/build/route-types/per-module-writer.ts +15 -3
- package/src/build/route-types/router-processing.ts +391 -90
- package/src/build/route-types/scan-filter.ts +8 -1
- package/src/build/runtime-discovery.ts +13 -1
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +140 -123
- package/src/cache/cache-scope.ts +124 -90
- package/src/cache/cf/cf-cache-store.ts +467 -22
- package/src/cache/cf/index.ts +5 -1
- package/src/cache/document-cache.ts +109 -75
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/index.ts +1 -15
- package/src/cache/memory-segment-store.ts +43 -69
- package/src/cache/profile-registry.ts +43 -8
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +140 -117
- package/src/cache/taint.ts +85 -3
- package/src/cache/types.ts +1 -115
- package/src/client.rsc.tsx +2 -1
- package/src/client.tsx +86 -254
- package/src/context-var.ts +72 -2
- package/src/debug.ts +2 -2
- package/src/errors.ts +6 -1
- package/src/handle.ts +41 -1
- package/src/handles/MetaTags.tsx +5 -2
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/host/cookie-handler.ts +8 -3
- package/src/host/index.ts +0 -3
- package/src/host/router.ts +14 -1
- package/src/href-client.ts +3 -1
- package/src/index.rsc.ts +48 -32
- package/src/index.ts +104 -64
- package/src/loader.rsc.ts +12 -4
- package/src/loader.ts +8 -0
- package/src/outlet-context.ts +1 -1
- package/src/prerender/store.ts +61 -18
- package/src/prerender.ts +207 -88
- package/src/response-utils.ts +28 -0
- package/src/reverse.ts +36 -7
- package/src/root-error-boundary.tsx +30 -26
- package/src/route-definition/dsl-helpers.ts +249 -46
- package/src/route-definition/helpers-types.ts +73 -20
- package/src/route-definition/index.ts +3 -3
- package/src/route-definition/redirect.ts +24 -4
- package/src/route-definition/resolve-handler-use.ts +155 -0
- package/src/route-map-builder.ts +45 -3
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +25 -0
- package/src/router/content-negotiation.ts +101 -2
- package/src/router/debug-manifest.ts +16 -3
- package/src/router/find-match.ts +4 -2
- package/src/router/handler-context.ts +221 -39
- package/src/router/intercept-resolution.ts +17 -8
- package/src/router/lazy-includes.ts +12 -5
- package/src/router/loader-resolution.ts +164 -31
- package/src/router/logging.ts +104 -4
- package/src/router/manifest.ts +53 -9
- package/src/router/match-api.ts +150 -206
- package/src/router/match-context.ts +3 -0
- package/src/router/match-handlers.ts +186 -11
- package/src/router/match-middleware/background-revalidation.ts +94 -86
- package/src/router/match-middleware/cache-lookup.ts +164 -19
- package/src/router/match-middleware/cache-store.ts +55 -10
- package/src/router/match-middleware/intercept-resolution.ts +9 -7
- package/src/router/match-middleware/segment-resolution.ts +61 -5
- package/src/router/match-pipelines.ts +8 -43
- package/src/router/match-result.ts +122 -18
- package/src/router/metrics.ts +238 -13
- package/src/router/middleware-types.ts +64 -61
- package/src/router/middleware.ts +316 -145
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +134 -15
- package/src/router/prerender-match.ts +150 -16
- package/src/router/preview-match.ts +30 -96
- package/src/router/request-classification.ts +310 -0
- package/src/router/revalidation.ts +102 -9
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +20 -0
- package/src/router/router-interfaces.ts +200 -43
- package/src/router/router-options.ts +265 -12
- package/src/router/router-registry.ts +5 -2
- package/src/router/segment-resolution/fresh.ts +365 -256
- package/src/router/segment-resolution/helpers.ts +268 -0
- package/src/router/segment-resolution/loader-cache.ts +103 -98
- package/src/router/segment-resolution/revalidation.ts +711 -423
- package/src/router/segment-resolution/static-store.ts +2 -2
- package/src/router/segment-resolution.ts +1 -3
- package/src/router/segment-wrappers.ts +5 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +30 -6
- package/src/router/types.ts +8 -1
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +269 -25
- package/src/rsc/handler-context.ts +13 -2
- package/src/rsc/handler.ts +784 -629
- package/src/rsc/helpers.ts +168 -20
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +105 -43
- package/src/rsc/manifest-init.ts +7 -2
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +262 -20
- package/src/rsc/response-route-handler.ts +360 -0
- package/src/rsc/rsc-rendering.ts +70 -44
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +178 -66
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +37 -3
- package/src/search-params.ts +38 -23
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +109 -23
- package/src/server/context.ts +227 -24
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +11 -6
- package/src/server/handle-store.ts +103 -12
- package/src/server/loader-registry.ts +20 -54
- package/src/server/request-context.ts +475 -105
- package/src/server.ts +6 -0
- package/src/ssr/index.tsx +74 -27
- package/src/static-handler.ts +25 -6
- package/src/theme/ThemeProvider.tsx +6 -1
- package/src/theme/index.ts +4 -18
- package/src/theme/theme-context.ts +1 -28
- package/src/theme/theme-script.ts +2 -1
- package/src/types/cache-types.ts +10 -5
- package/src/types/error-types.ts +3 -0
- package/src/types/global-namespace.ts +22 -0
- package/src/types/handler-context.ts +242 -82
- package/src/types/index.ts +1 -1
- package/src/types/loader-types.ts +50 -21
- package/src/types/request-scope.ts +126 -0
- package/src/types/route-config.ts +17 -26
- package/src/types/route-entry.ts +46 -0
- package/src/types/segments.ts +19 -5
- package/src/urls/include-helper.ts +72 -21
- package/src/urls/index.ts +1 -0
- package/src/urls/path-helper-types.ts +69 -18
- package/src/urls/path-helper.ts +65 -15
- package/src/urls/pattern-types.ts +33 -1
- package/src/urls/response-types.ts +47 -23
- package/src/urls/type-extraction.ts +23 -15
- package/src/use-loader.tsx +103 -13
- package/src/vite/debug.ts +184 -0
- package/src/vite/discovery/bundle-postprocess.ts +62 -85
- package/src/vite/discovery/discover-routers.ts +87 -29
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +224 -106
- package/src/vite/discovery/route-types-writer.ts +7 -7
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/discovery/state.ts +20 -13
- package/src/vite/discovery/virtual-module-codegen.ts +5 -2
- package/src/vite/index.ts +14 -51
- package/src/vite/plugin-types.ts +51 -79
- package/src/vite/plugins/cjs-to-esm.ts +5 -0
- package/src/vite/plugins/client-ref-dedup.ts +131 -0
- package/src/vite/plugins/client-ref-hashing.ts +19 -7
- package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
- package/src/vite/plugins/expose-action-id.ts +53 -31
- package/src/vite/plugins/expose-id-utils.ts +12 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +30 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
- package/src/vite/plugins/expose-internal-ids.ts +567 -319
- package/src/vite/plugins/performance-tracks.ts +96 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +139 -38
- package/src/vite/plugins/version-injector.ts +37 -11
- package/src/vite/plugins/version-plugin.ts +200 -18
- package/src/vite/rango.ts +233 -221
- package/src/vite/router-discovery.ts +887 -176
- package/src/vite/utils/banner.ts +4 -4
- package/src/vite/utils/package-resolution.ts +41 -1
- package/src/vite/utils/prerender-utils.ts +113 -0
- package/src/vite/utils/shared-utils.ts +22 -11
- package/skills/testing/SKILL.md +0 -226
- package/src/browser/lru-cache.ts +0 -61
- package/src/browser/react/prefetch.ts +0 -27
- package/src/browser/request-controller.ts +0 -164
- package/src/cache/memory-store.ts +0 -253
- package/src/route-definition/route-function.ts +0 -119
- package/src/router.gen.ts +0 -6
- package/src/static-handler.gen.ts +0 -5
- package/src/urls.gen.ts +0 -8
|
@@ -10,12 +10,19 @@ import type { Plugin } from "vite";
|
|
|
10
10
|
import { createServer as createViteServer } from "vite";
|
|
11
11
|
import { resolve } from "node:path";
|
|
12
12
|
import { readFileSync } from "node:fs";
|
|
13
|
+
import { createRequire, register } from "node:module";
|
|
14
|
+
import { pathToFileURL } from "node:url";
|
|
13
15
|
import {
|
|
16
|
+
formatNestedRouterConflictError,
|
|
17
|
+
findNestedRouterConflict,
|
|
14
18
|
findRouterFiles,
|
|
15
|
-
createScanFilter,
|
|
16
19
|
} from "../build/generate-route-types.js";
|
|
17
20
|
import { createVersionPlugin } from "./plugins/version-plugin.js";
|
|
18
21
|
import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
|
|
22
|
+
import {
|
|
23
|
+
BUILD_ENV_GLOBAL_KEY,
|
|
24
|
+
createCloudflareProtocolStubPlugin,
|
|
25
|
+
} from "./plugins/cloudflare-protocol-stub.js";
|
|
19
26
|
import {
|
|
20
27
|
exposeInternalIds,
|
|
21
28
|
exposeRouterId,
|
|
@@ -25,9 +32,13 @@ import { extractHandlerExportsFromChunk } from "./utils/bundle-analysis.js";
|
|
|
25
32
|
import {
|
|
26
33
|
createDiscoveryState,
|
|
27
34
|
VIRTUAL_ROUTES_MANIFEST_ID,
|
|
35
|
+
type DiscoveryState,
|
|
28
36
|
type PluginOptions,
|
|
29
37
|
} from "./discovery/state.js";
|
|
30
|
-
import {
|
|
38
|
+
import {
|
|
39
|
+
consumeSelfGenWrite,
|
|
40
|
+
peekSelfGenWrite,
|
|
41
|
+
} from "./discovery/self-gen-tracking.js";
|
|
31
42
|
import { discoverRouters } from "./discovery/discover-routers.js";
|
|
32
43
|
import {
|
|
33
44
|
writeCombinedRouteTypesWithTracking,
|
|
@@ -39,9 +50,220 @@ import {
|
|
|
39
50
|
generatePerRouterModule,
|
|
40
51
|
} from "./discovery/virtual-module-codegen.js";
|
|
41
52
|
import { postprocessBundle } from "./discovery/bundle-postprocess.js";
|
|
53
|
+
import { createDiscoveryGate } from "./discovery/gate-state.js";
|
|
54
|
+
import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
|
|
55
|
+
import { createRangoDebugger, timed, timedSync, NS } from "./debug.js";
|
|
56
|
+
|
|
57
|
+
const debugDiscovery = createRangoDebugger(NS.discovery);
|
|
58
|
+
const debugRoutes = createRangoDebugger(NS.routes);
|
|
59
|
+
const debugBuild = createRangoDebugger(NS.build);
|
|
60
|
+
const debugDev = createRangoDebugger(NS.dev);
|
|
42
61
|
|
|
43
62
|
export { VIRTUAL_ROUTES_MANIFEST_ID };
|
|
44
63
|
|
|
64
|
+
// ============================================================================
|
|
65
|
+
// Node ESM Loader Hook Registration
|
|
66
|
+
// ============================================================================
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* Registers a Node ESM loader hook that resolves `cloudflare:*` specifiers
|
|
70
|
+
* to a data: URL stub. Defense-in-depth alongside the Vite transform in
|
|
71
|
+
* `cloudflare-protocol-stub.ts`:
|
|
72
|
+
*
|
|
73
|
+
* - The Vite transform catches `cloudflare:*` imports in modules that flow
|
|
74
|
+
* through Vite's plugin pipeline. That's the vast majority of cases.
|
|
75
|
+
* - The Node loader catches imports in modules that Vite/Rollup externalize
|
|
76
|
+
* (e.g. the `partyserver` package, which has a top-level
|
|
77
|
+
* `import { DurableObject, env } from "cloudflare:workers"` and ships
|
|
78
|
+
* shapes plugin-rsc marks as external). Externalized modules are loaded
|
|
79
|
+
* via Node's native ESM loader, which rejects URL schemes.
|
|
80
|
+
*
|
|
81
|
+
* Registration is process-global and one-shot. The hook only intercepts
|
|
82
|
+
* `cloudflare:*` specifiers; everything else passes through via
|
|
83
|
+
* `nextResolve()`. It runs in a separate worker thread (Node ESM loader
|
|
84
|
+
* architecture), so it can't read the `globalThis[BUILD_ENV_GLOBAL_KEY]`
|
|
85
|
+
* bridge that the Vite transform uses — the stubs served here always
|
|
86
|
+
* return `env = {}`. That's fine because externalized libraries don't
|
|
87
|
+
* typically access `env` at module top level; user source (where real
|
|
88
|
+
* `env` matters at build time) flows through the Vite transform.
|
|
89
|
+
*/
|
|
90
|
+
let loaderHookRegistered = false;
|
|
91
|
+
function ensureCloudflareProtocolLoaderRegistered(): void {
|
|
92
|
+
if (loaderHookRegistered) return;
|
|
93
|
+
loaderHookRegistered = true;
|
|
94
|
+
try {
|
|
95
|
+
register(
|
|
96
|
+
new URL("./plugins/cloudflare-protocol-loader-hook.mjs", import.meta.url),
|
|
97
|
+
);
|
|
98
|
+
} catch (err: any) {
|
|
99
|
+
// register() requires Node 18.19+ / 20.6+. Older Node still has the
|
|
100
|
+
// Vite transform as primary defense.
|
|
101
|
+
console.warn(
|
|
102
|
+
`[rsc-router] Could not register Node ESM loader hook for cloudflare:* imports (${err?.message ?? err}). Falling back to Vite transform only.`,
|
|
103
|
+
);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// ============================================================================
|
|
108
|
+
// Temp Server Factory
|
|
109
|
+
// ============================================================================
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Create a minimal Vite server for router discovery.
|
|
113
|
+
*
|
|
114
|
+
* Both dev-mode prerender and build-mode discovery need a temp RSC server
|
|
115
|
+
* to import user router files via module runner. This factory centralizes
|
|
116
|
+
* the shared config and the mode-specific differences:
|
|
117
|
+
* - Dev: path-based IDs (no forceBuild), separate cacheDir
|
|
118
|
+
* - Build: hashed IDs (forceBuild), hashClientRefs for production bundles
|
|
119
|
+
*
|
|
120
|
+
* Returns the ViteDevServer instance. Callers access .environments.rsc as needed.
|
|
121
|
+
*/
|
|
122
|
+
async function createTempRscServer(
|
|
123
|
+
state: DiscoveryState,
|
|
124
|
+
options: { forceBuild?: boolean; cacheDir?: string } = {},
|
|
125
|
+
) {
|
|
126
|
+
// Install the Node ESM loader hook before any module evaluation so
|
|
127
|
+
// `cloudflare:*` specifiers in externalized/loader-delegated modules
|
|
128
|
+
// (e.g. packages plugin-rsc marks as external) resolve to stubs
|
|
129
|
+
// instead of crashing Node's native loader.
|
|
130
|
+
ensureCloudflareProtocolLoaderRegistered();
|
|
131
|
+
const { default: rsc } = await import("@vitejs/plugin-rsc");
|
|
132
|
+
return createViteServer({
|
|
133
|
+
root: state.projectRoot,
|
|
134
|
+
configFile: false,
|
|
135
|
+
server: { middlewareMode: true },
|
|
136
|
+
appType: "custom",
|
|
137
|
+
logLevel: "silent",
|
|
138
|
+
resolve: { alias: state.userResolveAlias },
|
|
139
|
+
esbuild: { jsx: "automatic", jsxImportSource: "react" },
|
|
140
|
+
...(options.cacheDir && { cacheDir: options.cacheDir }),
|
|
141
|
+
plugins: [
|
|
142
|
+
rsc({
|
|
143
|
+
entries: {
|
|
144
|
+
client: "virtual:entry-client",
|
|
145
|
+
ssr: "virtual:entry-ssr",
|
|
146
|
+
rsc: state.resolvedEntryPath!,
|
|
147
|
+
},
|
|
148
|
+
}),
|
|
149
|
+
// hashClientRefs only in build mode — production bundles need hashed refs
|
|
150
|
+
...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
|
|
151
|
+
createVersionPlugin(),
|
|
152
|
+
createVirtualStubPlugin(),
|
|
153
|
+
createCloudflareProtocolStubPlugin(),
|
|
154
|
+
// Dev prerender must use dev-mode IDs (path-based) to match the workerd
|
|
155
|
+
// runtime. forceBuild produces hashed IDs for production bundle consistency.
|
|
156
|
+
exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
|
|
157
|
+
exposeRouterId(),
|
|
158
|
+
],
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// ============================================================================
|
|
163
|
+
// Build-Time Env Resolution
|
|
164
|
+
// ============================================================================
|
|
165
|
+
|
|
166
|
+
import type {
|
|
167
|
+
BuildEnvOption,
|
|
168
|
+
BuildEnvFactoryContext,
|
|
169
|
+
BuildEnvResult,
|
|
170
|
+
} from "./plugin-types.js";
|
|
171
|
+
|
|
172
|
+
/**
|
|
173
|
+
* Resolve the buildEnv option into a concrete { env, dispose? } result.
|
|
174
|
+
* Handles all four input shapes: false, "auto", factory, plain object.
|
|
175
|
+
*/
|
|
176
|
+
async function resolveBuildEnv(
|
|
177
|
+
option: BuildEnvOption | undefined,
|
|
178
|
+
factoryCtx: BuildEnvFactoryContext,
|
|
179
|
+
): Promise<BuildEnvResult | null> {
|
|
180
|
+
if (!option) return null;
|
|
181
|
+
|
|
182
|
+
if (option === "auto") {
|
|
183
|
+
if (factoryCtx.preset !== "cloudflare") {
|
|
184
|
+
throw new Error(
|
|
185
|
+
'[rsc-router] buildEnv: "auto" is only supported with preset: "cloudflare". ' +
|
|
186
|
+
"Use a factory function or plain object for other presets.",
|
|
187
|
+
);
|
|
188
|
+
}
|
|
189
|
+
try {
|
|
190
|
+
// Resolve wrangler from the user's project root (not the router package)
|
|
191
|
+
const userRequire = createRequire(
|
|
192
|
+
resolve(factoryCtx.root, "package.json"),
|
|
193
|
+
);
|
|
194
|
+
const wranglerPath = userRequire.resolve("wrangler");
|
|
195
|
+
const { getPlatformProxy } = (await import(
|
|
196
|
+
pathToFileURL(wranglerPath).href
|
|
197
|
+
)) as {
|
|
198
|
+
getPlatformProxy: (opts?: any) => Promise<any>;
|
|
199
|
+
};
|
|
200
|
+
const proxy = await getPlatformProxy();
|
|
201
|
+
return {
|
|
202
|
+
env: proxy.env as Record<string, unknown>,
|
|
203
|
+
dispose: proxy.dispose,
|
|
204
|
+
};
|
|
205
|
+
} catch (err: any) {
|
|
206
|
+
throw new Error(
|
|
207
|
+
'[rsc-router] buildEnv: "auto" requires wrangler to be installed.\n' +
|
|
208
|
+
`Install it with: pnpm add -D wrangler\n${err.message}`,
|
|
209
|
+
);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
if (typeof option === "function") {
|
|
214
|
+
return await option(factoryCtx);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
// Plain object
|
|
218
|
+
return { env: option };
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Acquire build-time env bindings and store on discovery state.
|
|
223
|
+
* Returns true if env was acquired, false if buildEnv is disabled.
|
|
224
|
+
*/
|
|
225
|
+
async function acquireBuildEnv(
|
|
226
|
+
s: DiscoveryState,
|
|
227
|
+
command: "serve" | "build",
|
|
228
|
+
mode: string,
|
|
229
|
+
): Promise<boolean> {
|
|
230
|
+
const option = s.opts?.buildEnv;
|
|
231
|
+
if (!option) return false;
|
|
232
|
+
|
|
233
|
+
const result = await resolveBuildEnv(option, {
|
|
234
|
+
root: s.projectRoot,
|
|
235
|
+
mode,
|
|
236
|
+
command,
|
|
237
|
+
preset: s.opts?.preset ?? "node",
|
|
238
|
+
});
|
|
239
|
+
if (!result) return false;
|
|
240
|
+
|
|
241
|
+
s.resolvedBuildEnv = result.env;
|
|
242
|
+
s.buildEnvDispose = result.dispose ?? null;
|
|
243
|
+
// Bridge the resolved env into `cloudflare:workers`'s stubbed `env`
|
|
244
|
+
// export so user code that does `import { env } from "cloudflare:workers"`
|
|
245
|
+
// sees the real bindings proxy during discovery + prerender instead of
|
|
246
|
+
// an empty object. The stub reads this global at module-evaluation time.
|
|
247
|
+
(globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY] = result.env;
|
|
248
|
+
return true;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* Release build-time env resources and clear state.
|
|
253
|
+
*/
|
|
254
|
+
async function releaseBuildEnv(s: DiscoveryState): Promise<void> {
|
|
255
|
+
if (s.buildEnvDispose) {
|
|
256
|
+
try {
|
|
257
|
+
await s.buildEnvDispose();
|
|
258
|
+
} catch (err: any) {
|
|
259
|
+
console.warn(`[rsc-router] buildEnv dispose failed: ${err.message}`);
|
|
260
|
+
}
|
|
261
|
+
s.buildEnvDispose = null;
|
|
262
|
+
}
|
|
263
|
+
s.resolvedBuildEnv = undefined;
|
|
264
|
+
delete (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY];
|
|
265
|
+
}
|
|
266
|
+
|
|
45
267
|
/**
|
|
46
268
|
* Plugin that discovers router instances at dev/build time via the RSC environment.
|
|
47
269
|
*
|
|
@@ -59,6 +281,8 @@ export function createRouterDiscoveryPlugin(
|
|
|
59
281
|
opts?: PluginOptions,
|
|
60
282
|
): Plugin {
|
|
61
283
|
const s = createDiscoveryState(entryPath, opts);
|
|
284
|
+
let viteCommand: "serve" | "build" = "build";
|
|
285
|
+
let viteMode = "production";
|
|
62
286
|
|
|
63
287
|
return {
|
|
64
288
|
name: "@rangojs/router:discovery",
|
|
@@ -69,34 +293,28 @@ export function createRouterDiscoveryPlugin(
|
|
|
69
293
|
__RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
|
|
70
294
|
},
|
|
71
295
|
};
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
manualChunks(id: string) {
|
|
79
|
-
if (s.resolvedPrerenderModules?.has(id)) {
|
|
80
|
-
return "__prerender-handlers";
|
|
81
|
-
}
|
|
82
|
-
if (s.resolvedStaticModules?.has(id)) {
|
|
83
|
-
return "__static-handlers";
|
|
84
|
-
}
|
|
85
|
-
},
|
|
86
|
-
},
|
|
87
|
-
},
|
|
88
|
-
},
|
|
89
|
-
},
|
|
90
|
-
};
|
|
91
|
-
}
|
|
296
|
+
// Prerender/static handler modules are bundled naturally with the
|
|
297
|
+
// rest of the RSC entry. A previous design forced them into dedicated
|
|
298
|
+
// __prerender-handlers / __static-handlers chunks via manualChunks,
|
|
299
|
+
// but Rollup hoisted all shared dependencies into those chunks,
|
|
300
|
+
// inflating them to ~1 MB with active runtime code. Handler code is
|
|
301
|
+
// evicted in closeBundle regardless of which chunk it lands in.
|
|
92
302
|
return config;
|
|
93
303
|
},
|
|
94
304
|
|
|
95
305
|
configResolved(config) {
|
|
96
306
|
s.projectRoot = config.root;
|
|
97
307
|
s.isBuildMode = config.command === "build";
|
|
308
|
+
viteCommand = config.command as "serve" | "build";
|
|
309
|
+
viteMode = config.mode;
|
|
98
310
|
// Capture user's resolve aliases for the temp server
|
|
99
311
|
s.userResolveAlias = config.resolve.alias;
|
|
312
|
+
// Node preset: pick up auto-discovered router path from the config() hook.
|
|
313
|
+
// The auto-discover plugin runs in config() using Vite's resolved root,
|
|
314
|
+
// populating the mutable ref before configResolved fires.
|
|
315
|
+
if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
|
|
316
|
+
s.resolvedEntryPath = opts.routerPathRef.path;
|
|
317
|
+
}
|
|
100
318
|
// Cloudflare preset: read entry from resolved environment config.
|
|
101
319
|
// The @cloudflare/vite-plugin reads wrangler config (toml/json/jsonc)
|
|
102
320
|
// and sets optimizeDeps.entries on the RSC environment.
|
|
@@ -109,13 +327,6 @@ export function createRouterDiscoveryPlugin(
|
|
|
109
327
|
s.resolvedEntryPath = entries[0];
|
|
110
328
|
}
|
|
111
329
|
}
|
|
112
|
-
// Compile include/exclude patterns into a scan filter
|
|
113
|
-
if (opts?.include || opts?.exclude) {
|
|
114
|
-
s.scanFilter = createScanFilter(s.projectRoot, {
|
|
115
|
-
include: opts.include,
|
|
116
|
-
exclude: opts.exclude,
|
|
117
|
-
});
|
|
118
|
-
}
|
|
119
330
|
// Generate combined named-routes.gen.ts from static source parsing.
|
|
120
331
|
// Runs before the dev server starts so the gen file exists immediately for IDE.
|
|
121
332
|
// In build mode, the runtime discovery in buildStart produces the definitive
|
|
@@ -153,6 +364,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
153
364
|
resolveDiscovery = resolve;
|
|
154
365
|
});
|
|
155
366
|
|
|
367
|
+
// Manifest-readiness gate + rediscovery scheduler.
|
|
368
|
+
// The virtual:rsc-router/routes-manifest module's `load()` hook
|
|
369
|
+
// awaits `s.discoveryDone`; the gate is reset on each discovery
|
|
370
|
+
// cycle so workerd's HMR reloads block until the new gen file is
|
|
371
|
+
// written. State machine + transitions are extracted into
|
|
372
|
+
// ./discovery/gate-state.ts and unit-tested there — see the
|
|
373
|
+
// module's JSDoc for the four-flag contract.
|
|
374
|
+
const gate = createDiscoveryGate(s, debugDiscovery);
|
|
375
|
+
const beginDiscoveryGate = gate.beginGate;
|
|
376
|
+
const resolveDiscoveryGate = gate.resolveGate;
|
|
377
|
+
|
|
156
378
|
// Compute dev server origin from resolved URLs (preferred) or config port (fallback).
|
|
157
379
|
// Called after discovery (or in the load hook) when the server may be listening.
|
|
158
380
|
const getDevServerOrigin = () =>
|
|
@@ -166,57 +388,130 @@ export function createRouterDiscoveryPlugin(
|
|
|
166
388
|
let prerenderTempServer: any = null;
|
|
167
389
|
let prerenderNodeRegistry: Map<string, any> | null = null;
|
|
168
390
|
|
|
169
|
-
// Clean up the temporary server when the dev server shuts down
|
|
391
|
+
// Clean up the temporary server and build env when the dev server shuts down
|
|
170
392
|
server.httpServer?.on("close", () => {
|
|
171
393
|
if (prerenderTempServer) {
|
|
172
394
|
prerenderTempServer.close().catch(() => {});
|
|
173
395
|
prerenderTempServer = null;
|
|
174
396
|
}
|
|
397
|
+
releaseBuildEnv(s).catch(() => {});
|
|
175
398
|
});
|
|
176
399
|
|
|
400
|
+
// Mirror the build-path contract (router-discovery.ts ~line 878):
|
|
401
|
+
// set __rscRouterDiscoveryActive before running user modules so any
|
|
402
|
+
// module-level router.reverse() calls return a placeholder instead
|
|
403
|
+
// of throwing. The temp Vite server's module runner has its own
|
|
404
|
+
// module context; the flag must be on globalThis to cross that
|
|
405
|
+
// boundary. Cleared in finally so the dev request handlers run with
|
|
406
|
+
// strict reverse() semantics afterwards.
|
|
407
|
+
async function importEntryAndRegistry(tempRscEnv: any): Promise<void> {
|
|
408
|
+
const flagAlreadySet = !!(globalThis as any).__rscRouterDiscoveryActive;
|
|
409
|
+
if (!flagAlreadySet) {
|
|
410
|
+
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
411
|
+
}
|
|
412
|
+
try {
|
|
413
|
+
debugDiscovery?.(
|
|
414
|
+
"importEntryAndRegistry: importing entry (flag=%s)",
|
|
415
|
+
(globalThis as any).__rscRouterDiscoveryActive ?? false,
|
|
416
|
+
);
|
|
417
|
+
await tempRscEnv.runner.import(s.resolvedEntryPath!);
|
|
418
|
+
debugDiscovery?.(
|
|
419
|
+
"importEntryAndRegistry: entry import OK, fetching RouterRegistry",
|
|
420
|
+
);
|
|
421
|
+
const serverMod = await tempRscEnv.runner.import(
|
|
422
|
+
"@rangojs/router/server",
|
|
423
|
+
);
|
|
424
|
+
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
425
|
+
debugDiscovery?.(
|
|
426
|
+
"importEntryAndRegistry: registry size=%d",
|
|
427
|
+
prerenderNodeRegistry?.size ?? 0,
|
|
428
|
+
);
|
|
429
|
+
} finally {
|
|
430
|
+
if (!flagAlreadySet) {
|
|
431
|
+
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
432
|
+
debugDiscovery?.(
|
|
433
|
+
"importEntryAndRegistry: cleared __rscRouterDiscoveryActive",
|
|
434
|
+
);
|
|
435
|
+
}
|
|
436
|
+
}
|
|
437
|
+
}
|
|
438
|
+
|
|
177
439
|
async function getOrCreateTempServer(): Promise<any | null> {
|
|
178
|
-
if
|
|
179
|
-
|
|
440
|
+
// Reuse path: if a temp server is already alive, prefer reusing
|
|
441
|
+
// it over orphaning the existing instance and spinning up a new
|
|
442
|
+
// one. This handles two cases:
|
|
443
|
+
//
|
|
444
|
+
// 1. Steady-state cache hit (cold-start completed, registry
|
|
445
|
+
// cached) — return the env immediately.
|
|
446
|
+
// 2. Recovery from a failed refresh: refreshTempRscEnv() may
|
|
447
|
+
// have invalidated and nulled the registry, then thrown
|
|
448
|
+
// during importEntryAndRegistry. Without reuse, the next
|
|
449
|
+
// call would `createTempRscServer` and overwrite the
|
|
450
|
+
// handle, leaking the previous server. Try to re-import on
|
|
451
|
+
// the existing runner first; only if THAT fails do we
|
|
452
|
+
// close the orphan and create new.
|
|
453
|
+
if (prerenderTempServer) {
|
|
454
|
+
const existingEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
455
|
+
if (existingEnv?.runner) {
|
|
456
|
+
if (prerenderNodeRegistry) {
|
|
457
|
+
debugDiscovery?.(
|
|
458
|
+
"getOrCreateTempServer: cached temp runner reused",
|
|
459
|
+
);
|
|
460
|
+
return existingEnv;
|
|
461
|
+
}
|
|
462
|
+
// Server alive but registry missing — likely after a prior
|
|
463
|
+
// refresh's invalidate + import threw. Try to re-import.
|
|
464
|
+
debugDiscovery?.(
|
|
465
|
+
"getOrCreateTempServer: server alive but registry missing — re-importing",
|
|
466
|
+
);
|
|
467
|
+
try {
|
|
468
|
+
await importEntryAndRegistry(existingEnv);
|
|
469
|
+
return existingEnv;
|
|
470
|
+
} catch (err: any) {
|
|
471
|
+
debugDiscovery?.(
|
|
472
|
+
"getOrCreateTempServer: reuse import failed (%s) — closing orphan and creating fresh",
|
|
473
|
+
err?.message ?? String(err),
|
|
474
|
+
);
|
|
475
|
+
await prerenderTempServer.close().catch(() => {});
|
|
476
|
+
prerenderTempServer = null;
|
|
477
|
+
prerenderNodeRegistry = null;
|
|
478
|
+
// Fall through to create-new path below.
|
|
479
|
+
}
|
|
480
|
+
} else {
|
|
481
|
+
// Server reference exists but its rsc env is unhealthy
|
|
482
|
+
// (no runner). Close and recreate.
|
|
483
|
+
debugDiscovery?.(
|
|
484
|
+
"getOrCreateTempServer: existing server has no rsc.runner — closing and recreating",
|
|
485
|
+
);
|
|
486
|
+
await prerenderTempServer.close().catch(() => {});
|
|
487
|
+
prerenderTempServer = null;
|
|
488
|
+
prerenderNodeRegistry = null;
|
|
489
|
+
}
|
|
180
490
|
}
|
|
491
|
+
|
|
492
|
+
// Create path: no existing temp server (or just nullified above).
|
|
493
|
+
debugDiscovery?.(
|
|
494
|
+
"getOrCreateTempServer: creating new temp server, entry=%s",
|
|
495
|
+
s.resolvedEntryPath ?? "(unset)",
|
|
496
|
+
);
|
|
181
497
|
try {
|
|
182
|
-
|
|
183
|
-
prerenderTempServer = await createViteServer({
|
|
184
|
-
root: s.projectRoot,
|
|
185
|
-
configFile: false,
|
|
186
|
-
server: { middlewareMode: true },
|
|
187
|
-
appType: "custom",
|
|
188
|
-
logLevel: "silent",
|
|
498
|
+
prerenderTempServer = await createTempRscServer(s, {
|
|
189
499
|
cacheDir: "node_modules/.vite_prerender",
|
|
190
|
-
resolve: { alias: s.userResolveAlias },
|
|
191
|
-
esbuild: { jsx: "automatic", jsxImportSource: "react" },
|
|
192
|
-
plugins: [
|
|
193
|
-
rsc({
|
|
194
|
-
entries: {
|
|
195
|
-
client: "virtual:entry-client",
|
|
196
|
-
ssr: "virtual:entry-ssr",
|
|
197
|
-
rsc: s.resolvedEntryPath!,
|
|
198
|
-
},
|
|
199
|
-
}),
|
|
200
|
-
createVersionPlugin(),
|
|
201
|
-
createVirtualStubPlugin(),
|
|
202
|
-
// Dev prerender must use dev-mode IDs (path-based) to match the
|
|
203
|
-
// workerd runtime. forceBuild would produce hashed IDs causing
|
|
204
|
-
// handle data key mismatches when replayed into the runtime store.
|
|
205
|
-
exposeInternalIds(),
|
|
206
|
-
exposeRouterId(),
|
|
207
|
-
],
|
|
208
500
|
});
|
|
209
501
|
|
|
210
502
|
const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
211
503
|
if (tempRscEnv?.runner) {
|
|
212
|
-
await tempRscEnv
|
|
213
|
-
const serverMod = await tempRscEnv.runner.import(
|
|
214
|
-
"@rangojs/router/server",
|
|
215
|
-
);
|
|
216
|
-
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
504
|
+
await importEntryAndRegistry(tempRscEnv);
|
|
217
505
|
return tempRscEnv;
|
|
218
506
|
}
|
|
507
|
+
debugDiscovery?.(
|
|
508
|
+
"getOrCreateTempServer: tempRscEnv.runner unavailable",
|
|
509
|
+
);
|
|
219
510
|
} catch (err: any) {
|
|
511
|
+
debugDiscovery?.(
|
|
512
|
+
"getOrCreateTempServer: FAILED message=%s",
|
|
513
|
+
err.message,
|
|
514
|
+
);
|
|
220
515
|
console.warn(
|
|
221
516
|
`[rsc-router] Failed to create temp runner: ${err.message}`,
|
|
222
517
|
);
|
|
@@ -224,21 +519,137 @@ export function createRouterDiscoveryPlugin(
|
|
|
224
519
|
return null;
|
|
225
520
|
}
|
|
226
521
|
|
|
522
|
+
// Clear the package-level singleton registries that survive a Vite
|
|
523
|
+
// moduleGraph.invalidateAll(). createRouter() / createHostRouter()
|
|
524
|
+
// call .set(id, ...) on these Maps; for "router removed" or
|
|
525
|
+
// "router id changed" edits, the OLD entry would persist after
|
|
526
|
+
// re-import without an explicit .clear(), leaving ghost routes
|
|
527
|
+
// in discoverRouters' output.
|
|
528
|
+
//
|
|
529
|
+
// We import the same module the runner imports, so the .clear()
|
|
530
|
+
// here mutates the same Map the freshly re-imported entry will
|
|
531
|
+
// populate.
|
|
532
|
+
async function clearTempRegistries(tempRscEnv: any): Promise<void> {
|
|
533
|
+
try {
|
|
534
|
+
const serverMod = await tempRscEnv.runner.import(
|
|
535
|
+
"@rangojs/router/server",
|
|
536
|
+
);
|
|
537
|
+
if (typeof serverMod?.RouterRegistry?.clear === "function") {
|
|
538
|
+
serverMod.RouterRegistry.clear();
|
|
539
|
+
}
|
|
540
|
+
if (typeof serverMod?.HostRouterRegistry?.clear === "function") {
|
|
541
|
+
serverMod.HostRouterRegistry.clear();
|
|
542
|
+
}
|
|
543
|
+
debugDiscovery?.(
|
|
544
|
+
"clearTempRegistries: cleared RouterRegistry + HostRouterRegistry",
|
|
545
|
+
);
|
|
546
|
+
} catch (err: any) {
|
|
547
|
+
// Non-fatal: if the import fails here, importEntryAndRegistry
|
|
548
|
+
// below will fail loudly with the same root cause and the
|
|
549
|
+
// caller will surface it.
|
|
550
|
+
debugDiscovery?.(
|
|
551
|
+
"clearTempRegistries: import @rangojs/router/server failed (%s)",
|
|
552
|
+
err?.message ?? String(err),
|
|
553
|
+
);
|
|
554
|
+
}
|
|
555
|
+
}
|
|
556
|
+
|
|
557
|
+
// HMR refresh: keep the temp Vite server alive across HMR cycles and
|
|
558
|
+
// invalidate its module graph instead of close+recreate. Closing the
|
|
559
|
+
// temp server during workerd's first post-cold-start module-fetch
|
|
560
|
+
// window disrupted the main dev server's transport — the user-visible
|
|
561
|
+
// symptom was a `transport was disconnected, cannot call "fetchModule"`
|
|
562
|
+
// error on the first urls.tsx edit (workerd's cache was cold, so its
|
|
563
|
+
// eval was still in flight when our close() ran). Module-graph
|
|
564
|
+
// invalidation is the architecturally cleaner refresh: same Vite
|
|
565
|
+
// instance, same transport, fresh source.
|
|
566
|
+
//
|
|
567
|
+
// Falls back to close+recreate when neither the env-level nor
|
|
568
|
+
// server-level moduleGraph exposes invalidateAll() (defensive — Vite
|
|
569
|
+
// versions / preset configurations may differ in which graph carries
|
|
570
|
+
// the module-runner cache).
|
|
571
|
+
async function refreshTempRscEnv(): Promise<any | null> {
|
|
572
|
+
let tempRscEnv = await getOrCreateTempServer();
|
|
573
|
+
if (!tempRscEnv) return null;
|
|
574
|
+
|
|
575
|
+
// Module-runner cache is on the per-environment graph in Vite 6+;
|
|
576
|
+
// older / non-environments setups carry it on the server graph.
|
|
577
|
+
// Try env first, server second.
|
|
578
|
+
const envGraph = (tempRscEnv as any).moduleGraph;
|
|
579
|
+
const serverGraph = (prerenderTempServer as any)?.moduleGraph;
|
|
580
|
+
const target = envGraph?.invalidateAll
|
|
581
|
+
? envGraph
|
|
582
|
+
: serverGraph?.invalidateAll
|
|
583
|
+
? serverGraph
|
|
584
|
+
: null;
|
|
585
|
+
|
|
586
|
+
if (!target) {
|
|
587
|
+
// No invalidate method available — fall back to close+recreate.
|
|
588
|
+
// This preserves the previous behavior in case a Vite version
|
|
589
|
+
// doesn't expose invalidateAll on either graph.
|
|
590
|
+
debugDiscovery?.(
|
|
591
|
+
"refreshTempRscEnv: invalidateAll unavailable on env+server graphs, falling back to close+recreate",
|
|
592
|
+
);
|
|
593
|
+
if (prerenderTempServer) {
|
|
594
|
+
await prerenderTempServer.close().catch(() => {});
|
|
595
|
+
prerenderTempServer = null;
|
|
596
|
+
prerenderNodeRegistry = null;
|
|
597
|
+
}
|
|
598
|
+
return await getOrCreateTempServer();
|
|
599
|
+
}
|
|
600
|
+
|
|
601
|
+
debugDiscovery?.(
|
|
602
|
+
"refreshTempRscEnv: invalidating module graph (%s)",
|
|
603
|
+
envGraph?.invalidateAll ? "env" : "server",
|
|
604
|
+
);
|
|
605
|
+
target.invalidateAll();
|
|
606
|
+
// Drop the cached registry so importEntryAndRegistry re-reads it
|
|
607
|
+
// through the now-invalidated module runner.
|
|
608
|
+
prerenderNodeRegistry = null;
|
|
609
|
+
// Clear singleton Maps that Vite's moduleGraph invalidation can't
|
|
610
|
+
// reach (RouterRegistry / HostRouterRegistry). Without this, an
|
|
611
|
+
// edit that REMOVES a createRouter() call or CHANGES a router id
|
|
612
|
+
// would leave the old entry in the registry, and discoverRouters
|
|
613
|
+
// would still emit its routes alongside whatever the new source
|
|
614
|
+
// declares.
|
|
615
|
+
await clearTempRegistries(tempRscEnv);
|
|
616
|
+
await importEntryAndRegistry(tempRscEnv);
|
|
617
|
+
return tempRscEnv;
|
|
618
|
+
}
|
|
619
|
+
|
|
227
620
|
const discover = async () => {
|
|
621
|
+
const discoverStart = performance.now();
|
|
228
622
|
const rscEnv = (server.environments as any)?.rsc;
|
|
229
623
|
if (!rscEnv?.runner) {
|
|
230
624
|
// Cloudflare dev: no module runner available (workerd-based RSC env).
|
|
231
625
|
// Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
|
|
232
626
|
// for on-demand prerender via the /__rsc_prerender endpoint.
|
|
627
|
+
debugDiscovery?.(
|
|
628
|
+
"dev: cloudflare path start, __rscRouterDiscoveryActive=%s",
|
|
629
|
+
(globalThis as any).__rscRouterDiscoveryActive ?? false,
|
|
630
|
+
);
|
|
233
631
|
s.devServerOrigin = getDevServerOrigin();
|
|
234
632
|
|
|
235
633
|
// Create a temp Node.js server to run runtime discovery and generate
|
|
236
634
|
// named route types (static parser can't resolve factory calls).
|
|
237
635
|
try {
|
|
238
|
-
|
|
636
|
+
// Acquire build-time env bindings for dev prerender
|
|
637
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
638
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
639
|
+
);
|
|
640
|
+
|
|
641
|
+
const tempRscEnv = await timed(
|
|
642
|
+
debugDiscovery,
|
|
643
|
+
"getOrCreateTempServer",
|
|
644
|
+
() => getOrCreateTempServer(),
|
|
645
|
+
);
|
|
239
646
|
if (tempRscEnv) {
|
|
240
|
-
await discoverRouters(
|
|
241
|
-
|
|
647
|
+
await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
|
|
648
|
+
discoverRouters(s, tempRscEnv),
|
|
649
|
+
);
|
|
650
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
651
|
+
writeRouteTypesFiles(s),
|
|
652
|
+
);
|
|
242
653
|
}
|
|
243
654
|
} catch (err: any) {
|
|
244
655
|
console.warn(
|
|
@@ -246,24 +657,35 @@ export function createRouterDiscoveryPlugin(
|
|
|
246
657
|
);
|
|
247
658
|
}
|
|
248
659
|
|
|
660
|
+
debugDiscovery?.(
|
|
661
|
+
"dev discovery done (%sms)",
|
|
662
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
663
|
+
);
|
|
249
664
|
resolveDiscovery!();
|
|
250
665
|
return;
|
|
251
666
|
}
|
|
252
667
|
|
|
253
668
|
try {
|
|
669
|
+
// Acquire build-time env bindings for dev prerender (Node.js path)
|
|
670
|
+
debugDiscovery?.("dev: node path start");
|
|
671
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
672
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
673
|
+
);
|
|
674
|
+
|
|
254
675
|
// Set the readiness gate BEFORE discovery so early requests
|
|
255
676
|
// block until manifest is populated
|
|
256
|
-
const serverMod = await
|
|
257
|
-
|
|
677
|
+
const serverMod = await timed(
|
|
678
|
+
debugDiscovery,
|
|
679
|
+
"import @rangojs/router/server",
|
|
680
|
+
() => rscEnv.runner.import("@rangojs/router/server"),
|
|
258
681
|
);
|
|
259
682
|
if (serverMod?.setManifestReadyPromise) {
|
|
260
683
|
serverMod.setManifestReadyPromise(discoveryPromise);
|
|
261
684
|
}
|
|
262
685
|
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
mainRegistry = serverModAfterDiscovery?.RouterRegistry ?? null;
|
|
686
|
+
await timed(debugDiscovery, "discoverRouters", () =>
|
|
687
|
+
discoverRouters(s, rscEnv),
|
|
688
|
+
);
|
|
267
689
|
|
|
268
690
|
// Store server origin for dev prerender endpoint (virtual module injection)
|
|
269
691
|
s.devServerOrigin = getDevServerOrigin();
|
|
@@ -273,53 +695,36 @@ export function createRouterDiscoveryPlugin(
|
|
|
273
695
|
// routes (e.g. Array.from loops) that the static parser cannot see.
|
|
274
696
|
// writeRouteTypesFiles() only writes when content changes, so this
|
|
275
697
|
// won't cause unnecessary HMR triggers.
|
|
276
|
-
writeRouteTypesFiles(
|
|
698
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
699
|
+
writeRouteTypesFiles(s),
|
|
700
|
+
);
|
|
277
701
|
|
|
278
|
-
// Populate the route map in the RSC env
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
if (
|
|
283
|
-
s.mergedPrecomputedEntries &&
|
|
284
|
-
s.mergedPrecomputedEntries.length > 0 &&
|
|
285
|
-
serverMod?.setPrecomputedEntries
|
|
286
|
-
) {
|
|
287
|
-
serverMod.setPrecomputedEntries(s.mergedPrecomputedEntries);
|
|
288
|
-
}
|
|
289
|
-
if (s.mergedRouteTrie && serverMod?.setRouteTrie) {
|
|
290
|
-
serverMod.setRouteTrie(s.mergedRouteTrie);
|
|
291
|
-
}
|
|
292
|
-
// Populate per-router isolated data eagerly in dev (HMR).
|
|
293
|
-
// In production builds, per-router data is loaded lazily via import().
|
|
294
|
-
if (serverMod?.setRouterManifest) {
|
|
295
|
-
for (const [routerId, manifest] of s.perRouterManifestDataMap) {
|
|
296
|
-
serverMod.setRouterManifest(routerId, manifest);
|
|
297
|
-
}
|
|
298
|
-
}
|
|
299
|
-
if (serverMod?.setRouterTrie) {
|
|
300
|
-
for (const [routerId, trie] of s.perRouterTrieMap) {
|
|
301
|
-
serverMod.setRouterTrie(routerId, trie);
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
if (serverMod?.setRouterPrecomputedEntries) {
|
|
305
|
-
for (const [routerId, entries] of s.perRouterPrecomputedMap) {
|
|
306
|
-
serverMod.setRouterPrecomputedEntries(routerId, entries);
|
|
307
|
-
}
|
|
308
|
-
}
|
|
702
|
+
// Populate the route map and per-router data in the RSC env
|
|
703
|
+
await timed(debugDiscovery, "propagateDiscoveryState", () =>
|
|
704
|
+
propagateDiscoveryState(rscEnv),
|
|
705
|
+
);
|
|
309
706
|
} catch (err: any) {
|
|
310
707
|
console.warn(
|
|
311
708
|
`[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
|
|
312
709
|
);
|
|
313
710
|
} finally {
|
|
711
|
+
debugDiscovery?.(
|
|
712
|
+
"dev discovery done (%sms)",
|
|
713
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
714
|
+
);
|
|
314
715
|
resolveDiscovery!();
|
|
315
716
|
}
|
|
316
717
|
};
|
|
317
718
|
|
|
318
719
|
// Schedule after all plugins have finished configureServer.
|
|
319
|
-
//
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
720
|
+
// The gate (s.discoveryDone) is reset via beginDiscoveryGate() and
|
|
721
|
+
// resolved when discover() finishes, so the virtual manifest module's
|
|
722
|
+
// load() awaits the populated state.
|
|
723
|
+
beginDiscoveryGate();
|
|
724
|
+
setTimeout(
|
|
725
|
+
() => discover().then(resolveDiscoveryGate, resolveDiscoveryGate),
|
|
726
|
+
0,
|
|
727
|
+
);
|
|
323
728
|
|
|
324
729
|
// Dev-mode on-demand prerender endpoint.
|
|
325
730
|
// When workerd hits a prerender route, it fetches this endpoint instead of
|
|
@@ -333,7 +738,61 @@ export function createRouterDiscoveryPlugin(
|
|
|
333
738
|
// Registry from the main server's RSC environment (populated by discoverRouters)
|
|
334
739
|
let mainRegistry: Map<string, any> | null = null;
|
|
335
740
|
|
|
741
|
+
// Push discovery state (manifest, trie, precomputed entries) to the
|
|
742
|
+
// server module so runtime request handling uses the current routes.
|
|
743
|
+
// Shared by initial discovery and HMR-triggered re-discovery.
|
|
744
|
+
const propagateDiscoveryState = async (rscEnv: any) => {
|
|
745
|
+
const serverMod = await rscEnv.runner.import("@rangojs/router/server");
|
|
746
|
+
if (!serverMod) return;
|
|
747
|
+
// Clear stale per-router and global route data before repopulating.
|
|
748
|
+
// Without this, removed routers/routes survive in the per-router maps
|
|
749
|
+
// and shrunk precomputed entries or tries are never purged.
|
|
750
|
+
if (serverMod.clearAllRouterData) {
|
|
751
|
+
serverMod.clearAllRouterData();
|
|
752
|
+
}
|
|
753
|
+
mainRegistry = serverMod.RouterRegistry ?? null;
|
|
754
|
+
if (s.mergedRouteManifest && serverMod.setCachedManifest) {
|
|
755
|
+
serverMod.setCachedManifest(s.mergedRouteManifest);
|
|
756
|
+
}
|
|
757
|
+
if (
|
|
758
|
+
s.mergedPrecomputedEntries &&
|
|
759
|
+
s.mergedPrecomputedEntries.length > 0 &&
|
|
760
|
+
serverMod.setPrecomputedEntries
|
|
761
|
+
) {
|
|
762
|
+
serverMod.setPrecomputedEntries(s.mergedPrecomputedEntries);
|
|
763
|
+
}
|
|
764
|
+
if (s.mergedRouteTrie && serverMod.setRouteTrie) {
|
|
765
|
+
serverMod.setRouteTrie(s.mergedRouteTrie);
|
|
766
|
+
}
|
|
767
|
+
if (serverMod.setRouterManifest) {
|
|
768
|
+
for (const [routerId, manifest] of s.perRouterManifestDataMap) {
|
|
769
|
+
serverMod.setRouterManifest(routerId, manifest);
|
|
770
|
+
}
|
|
771
|
+
}
|
|
772
|
+
if (serverMod.setRouterTrie) {
|
|
773
|
+
for (const [routerId, trie] of s.perRouterTrieMap) {
|
|
774
|
+
serverMod.setRouterTrie(routerId, trie);
|
|
775
|
+
}
|
|
776
|
+
}
|
|
777
|
+
if (serverMod.setRouterPrecomputedEntries) {
|
|
778
|
+
for (const [routerId, entries] of s.perRouterPrecomputedMap) {
|
|
779
|
+
serverMod.setRouterPrecomputedEntries(routerId, entries);
|
|
780
|
+
}
|
|
781
|
+
}
|
|
782
|
+
};
|
|
783
|
+
|
|
336
784
|
server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
|
|
785
|
+
const reqStart = debugDev ? performance.now() : 0;
|
|
786
|
+
const logResult = (status: number, note: string) => {
|
|
787
|
+
debugDev?.(
|
|
788
|
+
"/__rsc_prerender %s -> %d %s (%sms)",
|
|
789
|
+
req.url,
|
|
790
|
+
status,
|
|
791
|
+
note,
|
|
792
|
+
(performance.now() - reqStart).toFixed(1),
|
|
793
|
+
);
|
|
794
|
+
};
|
|
795
|
+
|
|
337
796
|
if (s.discoveryDone) await s.discoveryDone;
|
|
338
797
|
|
|
339
798
|
const url = new URL(req.url || "/", "http://localhost");
|
|
@@ -341,12 +800,36 @@ export function createRouterDiscoveryPlugin(
|
|
|
341
800
|
if (!pathname) {
|
|
342
801
|
res.statusCode = 400;
|
|
343
802
|
res.end("Missing pathname");
|
|
803
|
+
logResult(400, "missing pathname");
|
|
344
804
|
return;
|
|
345
805
|
}
|
|
346
806
|
|
|
347
|
-
//
|
|
348
|
-
//
|
|
349
|
-
|
|
807
|
+
// Import the user's entry module to force re-evaluation of any
|
|
808
|
+
// HMR-invalidated modules in the chain (entry → router → urls → handlers).
|
|
809
|
+
// This ensures createRouter() re-runs with updated handler code before
|
|
810
|
+
// we read RouterRegistry. Without this, edits to prerender handler files
|
|
811
|
+
// produce stale content because the old router instance remains registered.
|
|
812
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
813
|
+
let registry: Map<string, any> | null = null;
|
|
814
|
+
if (rscEnv?.runner && s.resolvedEntryPath) {
|
|
815
|
+
try {
|
|
816
|
+
await rscEnv.runner.import(s.resolvedEntryPath);
|
|
817
|
+
const serverMod = await rscEnv.runner.import(
|
|
818
|
+
"@rangojs/router/server",
|
|
819
|
+
);
|
|
820
|
+
registry = serverMod.RouterRegistry ?? null;
|
|
821
|
+
} catch (err: any) {
|
|
822
|
+
console.warn(
|
|
823
|
+
`[rsc-router] Dev prerender module refresh failed: ${err.message}`,
|
|
824
|
+
);
|
|
825
|
+
res.statusCode = 500;
|
|
826
|
+
res.end(`Prerender handler error: ${err.message}`);
|
|
827
|
+
logResult(500, "module refresh failed");
|
|
828
|
+
return;
|
|
829
|
+
}
|
|
830
|
+
} else {
|
|
831
|
+
registry = mainRegistry;
|
|
832
|
+
}
|
|
350
833
|
|
|
351
834
|
if (!registry) {
|
|
352
835
|
// No main registry: the RSC env has no module runner (Cloudflare dev).
|
|
@@ -360,16 +843,31 @@ export function createRouterDiscoveryPlugin(
|
|
|
360
843
|
if (!registry || registry.size === 0) {
|
|
361
844
|
res.statusCode = 503;
|
|
362
845
|
res.end("Prerender runner not available");
|
|
846
|
+
logResult(503, "no registry");
|
|
363
847
|
return;
|
|
364
848
|
}
|
|
365
849
|
|
|
366
850
|
const wantIntercept = url.searchParams.get("intercept") === "1";
|
|
851
|
+
const wantRouteName = url.searchParams.get("routeName");
|
|
852
|
+
const wantPassthrough = url.searchParams.get("passthrough") === "1";
|
|
367
853
|
|
|
368
854
|
for (const [, routerInstance] of registry) {
|
|
369
855
|
if (!routerInstance.matchForPrerender) continue;
|
|
370
856
|
try {
|
|
371
|
-
const result = await routerInstance.matchForPrerender(
|
|
857
|
+
const result = await routerInstance.matchForPrerender(
|
|
858
|
+
pathname,
|
|
859
|
+
{},
|
|
860
|
+
undefined,
|
|
861
|
+
wantPassthrough,
|
|
862
|
+
s.resolvedBuildEnv,
|
|
863
|
+
true, // devMode: check getParams for passthrough routes
|
|
864
|
+
);
|
|
372
865
|
if (!result) continue;
|
|
866
|
+
if (result.passthrough) continue;
|
|
867
|
+
// When routeName is specified, only accept a match for that route.
|
|
868
|
+
// This prevents returning the wrong entry when multiple routers
|
|
869
|
+
// have prerenderable routes sharing the same pathname.
|
|
870
|
+
if (wantRouteName && result.routeName !== wantRouteName) continue;
|
|
373
871
|
res.setHeader("content-type", "application/json");
|
|
374
872
|
let payload: Record<string, unknown>;
|
|
375
873
|
if (wantIntercept && result.interceptSegments?.length) {
|
|
@@ -384,6 +882,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
384
882
|
payload = { segments: result.segments, handles: result.handles };
|
|
385
883
|
}
|
|
386
884
|
res.end(JSON.stringify(payload));
|
|
885
|
+
logResult(200, `match ${result.routeName}`);
|
|
387
886
|
return;
|
|
388
887
|
} catch (err: any) {
|
|
389
888
|
console.warn(
|
|
@@ -394,6 +893,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
394
893
|
|
|
395
894
|
res.statusCode = 404;
|
|
396
895
|
res.end("No prerender match");
|
|
896
|
+
logResult(404, "no match");
|
|
397
897
|
});
|
|
398
898
|
|
|
399
899
|
// Watch url module and router files for changes and regenerate named-routes.gen.ts.
|
|
@@ -417,6 +917,13 @@ export function createRouterDiscoveryPlugin(
|
|
|
417
917
|
): boolean => {
|
|
418
918
|
if (!isGeneratedRouteFile(filePath)) return false;
|
|
419
919
|
if (consumeSelfGenWrite(s, filePath)) return true;
|
|
920
|
+
// In Cloudflare dev (no module runner), perRouterManifests is never
|
|
921
|
+
// refreshed after HMR so regenerateGeneratedRouteFiles() would use
|
|
922
|
+
// stale data and revert user edits. Source files own route state;
|
|
923
|
+
// gen files are derived output. Skip regeneration and let the next
|
|
924
|
+
// source-file change rebuild them from the static parser.
|
|
925
|
+
const hasRunner = !!(server.environments as any)?.rsc?.runner;
|
|
926
|
+
if (!hasRunner) return true;
|
|
420
927
|
regenerateGeneratedRouteFiles();
|
|
421
928
|
return true;
|
|
422
929
|
};
|
|
@@ -427,20 +934,121 @@ export function createRouterDiscoveryPlugin(
|
|
|
427
934
|
// only the expensive regeneration is debounced.
|
|
428
935
|
let routeChangeTimer: ReturnType<typeof setTimeout> | undefined;
|
|
429
936
|
|
|
937
|
+
// Re-run runtime discovery so factory-generated routes that the
|
|
938
|
+
// static parser cannot see are refreshed after source changes.
|
|
939
|
+
// The state-machine concerns (queued/pending/gatePending) are
|
|
940
|
+
// owned by the gate created above (./discovery/gate-state.ts).
|
|
941
|
+
// Here we provide just the env-specific work.
|
|
942
|
+
const refreshRuntimeDiscovery = async () => {
|
|
943
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
944
|
+
const hasMainRunner = !!rscEnv?.runner;
|
|
945
|
+
// Cloudflare HMR has no main RSC runner (workerd is a separate
|
|
946
|
+
// runtime). When we have a populated runtime manifest from cold
|
|
947
|
+
// start, we can re-discover via the temp Node runner — the same
|
|
948
|
+
// mechanism getOrCreateTempServer() uses at startup. Without a
|
|
949
|
+
// populated manifest there's nothing useful to do, so bail
|
|
950
|
+
// before involving the gate machine at all.
|
|
951
|
+
if (!hasMainRunner && s.perRouterManifests.length === 0) return;
|
|
952
|
+
await gate.runRefreshCycle(async () => {
|
|
953
|
+
const hmrStart = performance.now();
|
|
954
|
+
try {
|
|
955
|
+
if (hasMainRunner) {
|
|
956
|
+
await timed(debugDiscovery, "hmr discoverRouters", () =>
|
|
957
|
+
discoverRouters(s, rscEnv),
|
|
958
|
+
);
|
|
959
|
+
timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
|
|
960
|
+
writeRouteTypesFiles(s),
|
|
961
|
+
);
|
|
962
|
+
await timed(debugDiscovery, "hmr propagateDiscoveryState", () =>
|
|
963
|
+
propagateDiscoveryState(rscEnv),
|
|
964
|
+
);
|
|
965
|
+
} else {
|
|
966
|
+
// Cloudflare HMR: invalidate the temp server's RSC module
|
|
967
|
+
// graph (or close+recreate as a fallback) so the runner
|
|
968
|
+
// re-reads the freshly edited source. Keeping the same
|
|
969
|
+
// Vite instance alive avoids disrupting workerd's transport
|
|
970
|
+
// during the first post-cold-start module-fetch window.
|
|
971
|
+
const tempRscEnv = await timed(
|
|
972
|
+
debugDiscovery,
|
|
973
|
+
"hmr refreshTempRscEnv (cloudflare)",
|
|
974
|
+
() => refreshTempRscEnv(),
|
|
975
|
+
);
|
|
976
|
+
if (!tempRscEnv) {
|
|
977
|
+
throw new Error(
|
|
978
|
+
"temp runner unavailable for cloudflare HMR rediscovery",
|
|
979
|
+
);
|
|
980
|
+
}
|
|
981
|
+
await timed(
|
|
982
|
+
debugDiscovery,
|
|
983
|
+
"hmr discoverRouters (cloudflare)",
|
|
984
|
+
() => discoverRouters(s, tempRscEnv),
|
|
985
|
+
);
|
|
986
|
+
timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
|
|
987
|
+
writeRouteTypesFiles(s),
|
|
988
|
+
);
|
|
989
|
+
}
|
|
990
|
+
} catch (err: any) {
|
|
991
|
+
console.warn(
|
|
992
|
+
`[rsc-router] Runtime re-discovery failed: ${err.message}`,
|
|
993
|
+
);
|
|
994
|
+
} finally {
|
|
995
|
+
debugDiscovery?.(
|
|
996
|
+
"hmr re-discovery done (%sms)",
|
|
997
|
+
(performance.now() - hmrStart).toFixed(1),
|
|
998
|
+
);
|
|
999
|
+
}
|
|
1000
|
+
});
|
|
1001
|
+
};
|
|
1002
|
+
|
|
430
1003
|
const scheduleRouteRegeneration = () => {
|
|
431
1004
|
clearTimeout(routeChangeTimer);
|
|
432
1005
|
routeChangeTimer = setTimeout(() => {
|
|
433
1006
|
routeChangeTimer = undefined;
|
|
1007
|
+
const regenStart = debugDiscovery ? performance.now() : 0;
|
|
1008
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
1009
|
+
const skipStaticWrite =
|
|
1010
|
+
!rscEnv?.runner && s.perRouterManifests.length > 0;
|
|
434
1011
|
try {
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
1012
|
+
// In cloudflare dev with a populated runtime manifest, the
|
|
1013
|
+
// static parser produces a strictly smaller (and actively
|
|
1014
|
+
// wrong) gen file — supplementGenFilesWithRuntimeRoutes can
|
|
1015
|
+
// only restore factory-only prefixes, and apps with mixed
|
|
1016
|
+
// static+factory routes under shared prefixes (cf-stress)
|
|
1017
|
+
// collapse to the 19-route static view. Skip the static
|
|
1018
|
+
// write entirely; runtime rediscovery below will overwrite
|
|
1019
|
+
// the gen file with the authoritative manifest.
|
|
1020
|
+
if (skipStaticWrite) {
|
|
1021
|
+
debugDiscovery?.(
|
|
1022
|
+
"watcher: skipping static write (cloudflare HMR — runtime rediscovery owns gen file)",
|
|
1023
|
+
);
|
|
1024
|
+
} else {
|
|
1025
|
+
writeCombinedRouteTypesWithTracking(s);
|
|
1026
|
+
if (s.perRouterManifests.length > 0) {
|
|
1027
|
+
supplementGenFilesWithRuntimeRoutes(s);
|
|
1028
|
+
}
|
|
438
1029
|
}
|
|
439
1030
|
} catch (err: any) {
|
|
440
1031
|
console.error(
|
|
441
1032
|
`[rsc-router] Route regeneration error: ${err.message}`,
|
|
442
1033
|
);
|
|
443
1034
|
}
|
|
1035
|
+
debugDiscovery?.(
|
|
1036
|
+
"watcher: regenerated gen files (%sms)",
|
|
1037
|
+
(performance.now() - regenStart).toFixed(1),
|
|
1038
|
+
);
|
|
1039
|
+
// Async: re-run runtime discovery to refresh factory-generated
|
|
1040
|
+
// routes that the static parser cannot resolve. Resolves the
|
|
1041
|
+
// discovery gate when complete.
|
|
1042
|
+
if (s.perRouterManifests.length > 0) {
|
|
1043
|
+
refreshRuntimeDiscovery().catch((err: any) => {
|
|
1044
|
+
console.warn(
|
|
1045
|
+
`[rsc-router] Runtime re-discovery error: ${err.message}`,
|
|
1046
|
+
);
|
|
1047
|
+
// Even on error, unblock the gate so workerd's reload
|
|
1048
|
+
// doesn't hang indefinitely against the previous manifest.
|
|
1049
|
+
resolveDiscoveryGate();
|
|
1050
|
+
});
|
|
1051
|
+
}
|
|
444
1052
|
}, 100);
|
|
445
1053
|
};
|
|
446
1054
|
|
|
@@ -466,10 +1074,37 @@ export function createRouterDiscoveryPlugin(
|
|
|
466
1074
|
const hasUrls = source.includes("urls(");
|
|
467
1075
|
const hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
|
|
468
1076
|
if (!hasUrls && !hasCreateRouter) return;
|
|
1077
|
+
debugDiscovery?.(
|
|
1078
|
+
"watcher: %s matches (urls=%s, router=%s)",
|
|
1079
|
+
filePath,
|
|
1080
|
+
hasUrls,
|
|
1081
|
+
hasCreateRouter,
|
|
1082
|
+
);
|
|
469
1083
|
// Invalidate cache when a router file changes (new router added/removed)
|
|
470
1084
|
if (hasCreateRouter) {
|
|
1085
|
+
const nestedRouterConflict = findNestedRouterConflict([
|
|
1086
|
+
...(s.cachedRouterFiles ?? []),
|
|
1087
|
+
resolve(filePath),
|
|
1088
|
+
]);
|
|
1089
|
+
if (nestedRouterConflict) {
|
|
1090
|
+
server.config.logger.error(
|
|
1091
|
+
formatNestedRouterConflictError(nestedRouterConflict),
|
|
1092
|
+
);
|
|
1093
|
+
return;
|
|
1094
|
+
}
|
|
471
1095
|
s.cachedRouterFiles = undefined;
|
|
472
1096
|
}
|
|
1097
|
+
// Note the event in the gate machine IMMEDIATELY (before the
|
|
1098
|
+
// 100ms debounce and any downstream HMR fanout). This sets
|
|
1099
|
+
// both `pendingEvents` (so refresh's finally holds the gate
|
|
1100
|
+
// through the tail window even if no rediscovery is queued)
|
|
1101
|
+
// and resets `discoveryDone` to a fresh pending promise (so
|
|
1102
|
+
// workerd reloads triggered by the same source change can't
|
|
1103
|
+
// observe a stale resolved gate from cold-start). Resolved
|
|
1104
|
+
// by the trailing refreshRuntimeDiscovery() cycle.
|
|
1105
|
+
if (s.perRouterManifests.length > 0) {
|
|
1106
|
+
gate.noteRouteEvent();
|
|
1107
|
+
}
|
|
473
1108
|
scheduleRouteRegeneration();
|
|
474
1109
|
} catch {
|
|
475
1110
|
// Ignore read errors for deleted/moved files
|
|
@@ -483,8 +1118,12 @@ export function createRouterDiscoveryPlugin(
|
|
|
483
1118
|
server.watcher.on("change", handleRouteFileChange);
|
|
484
1119
|
|
|
485
1120
|
// Regenerate gen files when they are deleted (e.g. manual cleanup).
|
|
1121
|
+
// Same no-runner guard as change/add: stale perRouterManifests would
|
|
1122
|
+
// reintroduce reverted content.
|
|
486
1123
|
server.watcher.on("unlink", (filePath) => {
|
|
487
1124
|
if (!isGeneratedRouteFile(filePath)) return;
|
|
1125
|
+
const hasRunner = !!(server.environments as any)?.rsc?.runner;
|
|
1126
|
+
if (!hasRunner) return;
|
|
488
1127
|
regenerateGeneratedRouteFiles();
|
|
489
1128
|
});
|
|
490
1129
|
}
|
|
@@ -496,7 +1135,23 @@ export function createRouterDiscoveryPlugin(
|
|
|
496
1135
|
async buildStart() {
|
|
497
1136
|
if (!s.isBuildMode) return;
|
|
498
1137
|
// Only run once across environment builds
|
|
499
|
-
if (s.mergedRouteManifest !== null)
|
|
1138
|
+
if (s.mergedRouteManifest !== null) {
|
|
1139
|
+
debugDiscovery?.(
|
|
1140
|
+
"build: skip (already discovered, env=%s)",
|
|
1141
|
+
this.environment?.name ?? "?",
|
|
1142
|
+
);
|
|
1143
|
+
return;
|
|
1144
|
+
}
|
|
1145
|
+
const buildStartTime = performance.now();
|
|
1146
|
+
debugDiscovery?.("build: start (env=%s)", this.environment?.name ?? "?");
|
|
1147
|
+
resetStagedBuildAssets(s.projectRoot);
|
|
1148
|
+
s.prerenderManifestEntries = null;
|
|
1149
|
+
s.staticManifestEntries = null;
|
|
1150
|
+
|
|
1151
|
+
// Acquire build-time env bindings if configured
|
|
1152
|
+
await timed(debugDiscovery, "build acquireBuildEnv", () =>
|
|
1153
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
1154
|
+
);
|
|
500
1155
|
|
|
501
1156
|
let tempServer: any = null;
|
|
502
1157
|
// Signal to user-space code (e.g. reverse.ts) that build-time discovery
|
|
@@ -505,45 +1160,11 @@ export function createRouterDiscoveryPlugin(
|
|
|
505
1160
|
// between the vite plugin and user code loaded via runner.import().
|
|
506
1161
|
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
507
1162
|
try {
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
const { default: rsc } = await import("@vitejs/plugin-rsc");
|
|
514
|
-
tempServer = await createViteServer({
|
|
515
|
-
root: s.projectRoot,
|
|
516
|
-
configFile: false,
|
|
517
|
-
server: { middlewareMode: true },
|
|
518
|
-
appType: "custom",
|
|
519
|
-
logLevel: "silent",
|
|
520
|
-
// Use the resolved aliases from the real config (includes user's path aliases
|
|
521
|
-
// like @/ -> src/ AND package aliases from rsc-router)
|
|
522
|
-
resolve: { alias: s.userResolveAlias },
|
|
523
|
-
// Enable automatic JSX runtime so .tsx files don't need `import React`.
|
|
524
|
-
// Without this, esbuild defaults to classic mode (React.createElement)
|
|
525
|
-
// which fails when lazy host-router handlers load sub-app modules with JSX.
|
|
526
|
-
esbuild: { jsx: "automatic", jsxImportSource: "react" },
|
|
527
|
-
plugins: [
|
|
528
|
-
rsc({
|
|
529
|
-
entries: {
|
|
530
|
-
client: "virtual:entry-client",
|
|
531
|
-
ssr: "virtual:entry-ssr",
|
|
532
|
-
rsc: s.resolvedEntryPath!,
|
|
533
|
-
},
|
|
534
|
-
}),
|
|
535
|
-
hashClientRefs(s.projectRoot),
|
|
536
|
-
createVersionPlugin(),
|
|
537
|
-
// Stub virtual modules that the RSC entry may import
|
|
538
|
-
// (e.g., virtual:rsc-router/routes-manifest, virtual:rsc-router/loader-manifest)
|
|
539
|
-
createVirtualStubPlugin(),
|
|
540
|
-
// Inject handle + router IDs so prerender-collected handle data uses
|
|
541
|
-
// the same hashed keys as the production client/SSR bundles, and
|
|
542
|
-
// build-time router IDs match runtime IDs across environments.
|
|
543
|
-
exposeInternalIds({ forceBuild: true }),
|
|
544
|
-
exposeRouterId(),
|
|
545
|
-
],
|
|
546
|
-
});
|
|
1163
|
+
tempServer = await timed(
|
|
1164
|
+
debugDiscovery,
|
|
1165
|
+
"build createTempRscServer",
|
|
1166
|
+
() => createTempRscServer(s, { forceBuild: true }),
|
|
1167
|
+
);
|
|
547
1168
|
|
|
548
1169
|
const rscEnv = (tempServer.environments as any)?.rsc;
|
|
549
1170
|
if (!rscEnv?.runner) {
|
|
@@ -563,11 +1184,15 @@ export function createRouterDiscoveryPlugin(
|
|
|
563
1184
|
s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
|
|
564
1185
|
}
|
|
565
1186
|
|
|
566
|
-
await
|
|
1187
|
+
await timed(debugDiscovery, "build discoverRouters", () =>
|
|
1188
|
+
discoverRouters(s, rscEnv),
|
|
1189
|
+
);
|
|
567
1190
|
// Update named-routes.gen.ts from runtime discovery.
|
|
568
1191
|
// The runtime manifest includes dynamically generated routes
|
|
569
1192
|
// that the static parser cannot extract from source code.
|
|
570
|
-
writeRouteTypesFiles(
|
|
1193
|
+
timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
|
|
1194
|
+
writeRouteTypesFiles(s),
|
|
1195
|
+
);
|
|
571
1196
|
} catch (err: any) {
|
|
572
1197
|
// Extract the user source file from the stack trace (skip internal frames)
|
|
573
1198
|
const sourceFile = err.stack
|
|
@@ -592,8 +1217,44 @@ export function createRouterDiscoveryPlugin(
|
|
|
592
1217
|
} finally {
|
|
593
1218
|
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
594
1219
|
if (tempServer) {
|
|
595
|
-
await tempServer.close()
|
|
1220
|
+
await timed(debugDiscovery, "build tempServer.close", () =>
|
|
1221
|
+
tempServer.close(),
|
|
1222
|
+
);
|
|
596
1223
|
}
|
|
1224
|
+
await releaseBuildEnv(s);
|
|
1225
|
+
debugDiscovery?.(
|
|
1226
|
+
"build discovery done (%sms)",
|
|
1227
|
+
(performance.now() - buildStartTime).toFixed(1),
|
|
1228
|
+
);
|
|
1229
|
+
}
|
|
1230
|
+
},
|
|
1231
|
+
|
|
1232
|
+
// Suppress vite's HMR cascade for our own gen-file writes.
|
|
1233
|
+
//
|
|
1234
|
+
// After every cf HMR cycle, refreshTempRscEnv → writeRouteTypesFiles
|
|
1235
|
+
// writes the configured gen files (default `router.named-routes.gen.ts`,
|
|
1236
|
+
// but the source filenames and gen suffix are user-configurable). The
|
|
1237
|
+
// chokidar watcher then fires twice independently: our
|
|
1238
|
+
// `handleRouteFileChange` (already short-circuited by
|
|
1239
|
+
// `consumeSelfGenWrite` inside `maybeHandleGeneratedRouteFileMutation`),
|
|
1240
|
+
// AND vite's own HMR pipeline (which invalidates the gen file's
|
|
1241
|
+
// importers and triggers a second workerd full reload — visible to the
|
|
1242
|
+
// user as a duplicate "[RSCRouter] HMR: version changed" on the client).
|
|
1243
|
+
//
|
|
1244
|
+
// `peekSelfGenWrite` is the authoritative filter: its map only contains
|
|
1245
|
+
// paths that `markSelfGenWrite` has registered, so it natively works
|
|
1246
|
+
// for any configured gen-file name. It is non-consuming so the chokidar
|
|
1247
|
+
// handler that fires later can still consume the same entry. Returning
|
|
1248
|
+
// [] tells vite "no modules invalidated by this change" — safe because
|
|
1249
|
+
// `s.perRouterManifests` is already up-to-date (the write that just
|
|
1250
|
+
// happened is the consequence of our just-completed rediscovery).
|
|
1251
|
+
handleHotUpdate(ctx) {
|
|
1252
|
+
if (peekSelfGenWrite(s, ctx.file)) {
|
|
1253
|
+
debugDiscovery?.(
|
|
1254
|
+
"handleHotUpdate: suppressing self-write HMR cascade for %s",
|
|
1255
|
+
ctx.file,
|
|
1256
|
+
);
|
|
1257
|
+
return [];
|
|
597
1258
|
}
|
|
598
1259
|
},
|
|
599
1260
|
|
|
@@ -617,19 +1278,38 @@ export function createRouterDiscoveryPlugin(
|
|
|
617
1278
|
// This is critical for Cloudflare dev where the worker runs in a separate
|
|
618
1279
|
// Miniflare process and can only receive manifest data via the virtual module.
|
|
619
1280
|
if (s.discoveryDone) {
|
|
620
|
-
await
|
|
1281
|
+
await timed(
|
|
1282
|
+
debugRoutes,
|
|
1283
|
+
"await discoveryDone (manifest)",
|
|
1284
|
+
() => s.discoveryDone,
|
|
1285
|
+
);
|
|
621
1286
|
}
|
|
622
|
-
|
|
1287
|
+
const code = await timed(
|
|
1288
|
+
debugRoutes,
|
|
1289
|
+
"generateRoutesManifestModule",
|
|
1290
|
+
() => generateRoutesManifestModule(s),
|
|
1291
|
+
);
|
|
1292
|
+
debugRoutes?.("manifest module emitted (%d bytes)", code?.length ?? 0);
|
|
1293
|
+
return code;
|
|
623
1294
|
}
|
|
624
1295
|
// Per-router virtual modules: pure data exports (no side effects).
|
|
625
1296
|
// ensureRouterManifest() imports the module and stores the data.
|
|
626
1297
|
const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
|
|
627
1298
|
if (id.startsWith(perRouterPrefix)) {
|
|
628
1299
|
if (s.discoveryDone) {
|
|
629
|
-
await
|
|
1300
|
+
await timed(
|
|
1301
|
+
debugRoutes,
|
|
1302
|
+
"await discoveryDone (per-router)",
|
|
1303
|
+
() => s.discoveryDone,
|
|
1304
|
+
);
|
|
630
1305
|
}
|
|
631
1306
|
const routerId = id.slice(perRouterPrefix.length);
|
|
632
|
-
|
|
1307
|
+
const code = await timed(
|
|
1308
|
+
debugRoutes,
|
|
1309
|
+
`generatePerRouterModule ${routerId}`,
|
|
1310
|
+
() => generatePerRouterModule(s, routerId),
|
|
1311
|
+
);
|
|
1312
|
+
return code;
|
|
633
1313
|
}
|
|
634
1314
|
// virtual:rsc-router/prerender-paths load handler removed
|
|
635
1315
|
return null;
|
|
@@ -639,6 +1319,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
639
1319
|
// Used by closeBundle for handler code eviction and prerender data injection.
|
|
640
1320
|
generateBundle(_options: any, bundle: any) {
|
|
641
1321
|
if (this.environment?.name !== "rsc") return;
|
|
1322
|
+
const genStart = debugBuild ? performance.now() : 0;
|
|
642
1323
|
|
|
643
1324
|
// Record RSC entry chunk filename for closeBundle injection
|
|
644
1325
|
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
@@ -651,8 +1332,19 @@ export function createRouterDiscoveryPlugin(
|
|
|
651
1332
|
}
|
|
652
1333
|
}
|
|
653
1334
|
|
|
654
|
-
if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
|
|
1335
|
+
if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size) {
|
|
1336
|
+
debugBuild?.(
|
|
1337
|
+
"generateBundle (rsc): no handlers to scan (%sms)",
|
|
1338
|
+
(performance.now() - genStart).toFixed(1),
|
|
1339
|
+
);
|
|
655
1340
|
return;
|
|
1341
|
+
}
|
|
1342
|
+
|
|
1343
|
+
// Clear maps at the start of each RSC generateBundle pass.
|
|
1344
|
+
// Vite 6 multi-environment builds run RSC twice (analysis + production);
|
|
1345
|
+
// clearing prevents stale/duplicate records from the analysis pass.
|
|
1346
|
+
s.handlerChunkInfoMap.clear();
|
|
1347
|
+
s.staticHandlerChunkInfoMap.clear();
|
|
656
1348
|
|
|
657
1349
|
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
658
1350
|
string,
|
|
@@ -660,27 +1352,28 @@ export function createRouterDiscoveryPlugin(
|
|
|
660
1352
|
][]) {
|
|
661
1353
|
if (chunk.type !== "chunk") continue;
|
|
662
1354
|
|
|
663
|
-
//
|
|
664
|
-
if (
|
|
665
|
-
fileName.includes("__prerender-handlers") &&
|
|
666
|
-
s.resolvedPrerenderModules?.size
|
|
667
|
-
) {
|
|
1355
|
+
// Scan all chunks for handler exports (handlers may land in any chunk)
|
|
1356
|
+
if (s.resolvedPrerenderModules?.size) {
|
|
668
1357
|
const handlers = extractHandlerExportsFromChunk(
|
|
669
1358
|
chunk.code,
|
|
670
1359
|
s.resolvedPrerenderModules,
|
|
671
1360
|
"Prerender",
|
|
672
|
-
|
|
1361
|
+
false,
|
|
673
1362
|
);
|
|
674
1363
|
if (handlers.length > 0) {
|
|
675
|
-
|
|
1364
|
+
const existing = s.handlerChunkInfoMap.get(fileName);
|
|
1365
|
+
if (existing) {
|
|
1366
|
+
existing.exports.push(...handlers);
|
|
1367
|
+
} else {
|
|
1368
|
+
s.handlerChunkInfoMap.set(fileName, {
|
|
1369
|
+
fileName,
|
|
1370
|
+
exports: handlers,
|
|
1371
|
+
});
|
|
1372
|
+
}
|
|
676
1373
|
}
|
|
677
1374
|
}
|
|
678
1375
|
|
|
679
|
-
|
|
680
|
-
if (
|
|
681
|
-
fileName.includes("__static-handlers") &&
|
|
682
|
-
s.resolvedStaticModules?.size
|
|
683
|
-
) {
|
|
1376
|
+
if (s.resolvedStaticModules?.size) {
|
|
684
1377
|
const handlers = extractHandlerExportsFromChunk(
|
|
685
1378
|
chunk.code,
|
|
686
1379
|
s.resolvedStaticModules,
|
|
@@ -688,10 +1381,26 @@ export function createRouterDiscoveryPlugin(
|
|
|
688
1381
|
false,
|
|
689
1382
|
);
|
|
690
1383
|
if (handlers.length > 0) {
|
|
691
|
-
|
|
1384
|
+
const existing = s.staticHandlerChunkInfoMap.get(fileName);
|
|
1385
|
+
if (existing) {
|
|
1386
|
+
existing.exports.push(...handlers);
|
|
1387
|
+
} else {
|
|
1388
|
+
s.staticHandlerChunkInfoMap.set(fileName, {
|
|
1389
|
+
fileName,
|
|
1390
|
+
exports: handlers,
|
|
1391
|
+
});
|
|
1392
|
+
}
|
|
692
1393
|
}
|
|
693
1394
|
}
|
|
694
1395
|
}
|
|
1396
|
+
|
|
1397
|
+
debugBuild?.(
|
|
1398
|
+
"generateBundle (rsc): scanned %d chunks, %d prerender chunk(s), %d static chunk(s) (%sms)",
|
|
1399
|
+
Object.keys(bundle).length,
|
|
1400
|
+
s.handlerChunkInfoMap.size,
|
|
1401
|
+
s.staticHandlerChunkInfoMap.size,
|
|
1402
|
+
(performance.now() - genStart).toFixed(1),
|
|
1403
|
+
);
|
|
695
1404
|
},
|
|
696
1405
|
|
|
697
1406
|
// Build-time pre-rendering: evict handler code and inject collected prerender data.
|
|
@@ -705,7 +1414,9 @@ export function createRouterDiscoveryPlugin(
|
|
|
705
1414
|
// Only run for the RSC environment — other environments (client, ssr) have
|
|
706
1415
|
// no prerender/static data to process and would just do redundant file I/O.
|
|
707
1416
|
if (this.environment && this.environment.name !== "rsc") return;
|
|
708
|
-
postprocessBundle(
|
|
1417
|
+
timedSync(debugBuild, "closeBundle postprocessBundle", () =>
|
|
1418
|
+
postprocessBundle(s),
|
|
1419
|
+
);
|
|
709
1420
|
},
|
|
710
1421
|
},
|
|
711
1422
|
};
|