@ecopages/core 0.2.0-alpha.5 → 0.2.0-alpha.51
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/README.md +267 -14
- package/package.json +98 -108
- package/src/adapters/README.md +39 -0
- package/src/adapters/abstract/application-adapter.d.ts +28 -2
- package/src/adapters/abstract/application-adapter.js +14 -2
- package/src/adapters/abstract/router-adapter.d.ts +1 -1
- package/src/adapters/abstract/server-adapter.d.ts +2 -2
- package/src/adapters/bun/client-bridge.d.ts +1 -1
- package/src/adapters/bun/create-app.d.ts +12 -12
- package/src/adapters/bun/create-app.js +64 -41
- package/src/adapters/bun/hmr-manager.d.ts +30 -62
- package/src/adapters/bun/hmr-manager.js +30 -194
- package/src/adapters/bun/index.d.ts +2 -3
- package/src/adapters/bun/index.js +3 -3
- package/src/adapters/bun/runtime-host.d.ts +52 -0
- package/src/adapters/bun/runtime-host.js +56 -0
- package/src/adapters/bun/server-adapter.d.ts +93 -32
- package/src/adapters/bun/server-adapter.js +166 -89
- package/src/adapters/bun/static-preview-host.d.ts +28 -0
- package/src/adapters/bun/static-preview-host.js +45 -0
- package/src/{create-app.d.ts → adapters/create-app.d.ts} +9 -6
- package/src/{create-app.js → adapters/create-app.js} +4 -4
- package/src/adapters/index.d.ts +2 -6
- package/src/adapters/index.js +2 -8
- package/src/adapters/node/create-app.d.ts +15 -12
- package/src/adapters/node/create-app.js +34 -85
- package/src/adapters/node/http-request-bridge.d.ts +57 -0
- package/src/adapters/node/http-request-bridge.js +118 -0
- package/src/adapters/node/node-client-bridge.d.ts +1 -1
- package/src/adapters/node/node-hmr-manager.d.ts +38 -48
- package/src/adapters/node/node-hmr-manager.js +31 -203
- package/src/adapters/node/runtime-host.d.ts +57 -0
- package/src/adapters/node/runtime-host.js +92 -0
- package/src/adapters/node/server-adapter-dependencies.d.ts +19 -0
- package/src/adapters/node/server-adapter-dependencies.js +18 -0
- package/src/adapters/node/server-adapter.d.ts +20 -72
- package/src/adapters/node/server-adapter.js +98 -203
- package/src/adapters/node/static-content-server.d.ts +37 -1
- package/src/adapters/node/static-content-server.js +29 -1
- package/src/adapters/node/static-preview-host.d.ts +55 -0
- package/src/adapters/node/static-preview-host.js +68 -0
- package/src/adapters/shared/application-adapter.d.ts +1 -1
- package/src/{define-api-handler.d.ts → adapters/shared/define-api-handler.d.ts} +1 -1
- package/src/adapters/shared/explicit-static-render-preparation.d.ts +25 -0
- package/src/adapters/shared/explicit-static-render-preparation.js +26 -0
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +7 -4
- package/src/adapters/shared/explicit-static-route-matcher.js +14 -13
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +7 -10
- package/src/adapters/shared/file-route-middleware-pipeline.js +3 -11
- package/src/adapters/shared/fs-server-response-factory.d.ts +14 -10
- package/src/adapters/shared/fs-server-response-factory.js +11 -27
- package/src/adapters/shared/fs-server-response-matcher.d.ts +20 -16
- package/src/adapters/shared/fs-server-response-matcher.js +76 -45
- package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
- package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
- package/src/adapters/shared/hmr-html-response.d.ts +22 -0
- package/src/adapters/shared/hmr-html-response.js +33 -0
- package/src/adapters/shared/render-context.d.ts +5 -3
- package/src/adapters/shared/render-context.js +27 -3
- package/src/adapters/shared/runtime-app-bootstrap.d.ts +26 -0
- package/src/adapters/shared/runtime-app-bootstrap.js +46 -0
- package/src/adapters/shared/runtime-host.d.ts +12 -0
- package/src/adapters/shared/server-adapter.d.ts +33 -12
- package/src/adapters/shared/server-adapter.js +215 -132
- package/src/adapters/shared/server-route-handler.d.ts +5 -5
- package/src/adapters/shared/server-route-handler.js +7 -16
- package/src/adapters/shared/server-static-builder.d.ts +41 -8
- package/src/adapters/shared/server-static-builder.js +65 -11
- package/src/adapters/shared/shared-hmr-manager.d.ts +59 -0
- package/src/adapters/shared/shared-hmr-manager.js +240 -0
- package/src/adapters/shared/static-preview-host.d.ts +10 -0
- package/src/build/README.md +107 -0
- package/src/build/browser-runtime-import-rewrite-plugin.d.ts +26 -0
- package/src/build/browser-runtime-import-rewrite-plugin.js +162 -0
- package/src/build/browser-runtime-manifest.d.ts +31 -0
- package/src/build/browser-runtime-manifest.js +61 -0
- package/src/build/build-adapter.d.ts +175 -3
- package/src/build/build-adapter.js +633 -16
- package/src/build/build-manifest.d.ts +33 -0
- package/src/build/build-manifest.js +52 -0
- package/src/build/dev-build-coordinator.d.ts +72 -0
- package/src/build/dev-build-coordinator.js +154 -0
- package/src/build/esbuild-build-adapter.d.ts +16 -6
- package/src/build/esbuild-build-adapter.js +205 -75
- package/src/build/runtime-build-executor.d.ts +14 -0
- package/src/build/runtime-build-executor.js +22 -0
- package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
- package/src/build/runtime-specifier-alias-plugin.js +31 -0
- package/src/config/README.md +36 -0
- package/src/config/config-builder.d.ts +54 -29
- package/src/config/config-builder.js +256 -50
- package/src/{constants.d.ts → config/constants.d.ts} +13 -0
- package/src/{constants.js → config/constants.js} +4 -0
- package/src/declarations.d.ts +19 -14
- package/src/dev/host-runtime.d.ts +10 -0
- package/src/dev/host-runtime.js +24 -0
- package/src/dev/sc-server.d.ts +1 -1
- package/src/dev/sc-server.js +1 -1
- package/src/eco/README.md +70 -16
- package/src/eco/eco.browser.d.ts +2 -0
- package/src/eco/eco.browser.js +88 -0
- package/src/eco/eco.js +63 -54
- package/src/eco/eco.types.d.ts +69 -6
- package/src/eco/eco.utils.d.ts +1 -40
- package/src/eco/eco.utils.js +5 -35
- package/src/eco/global-injector-map.d.ts +3 -3
- package/src/eco/global-injector-map.js +2 -2
- package/src/eco/lazy-injector-map.d.ts +2 -2
- package/src/errors/index.d.ts +1 -0
- package/src/errors/index.js +3 -1
- package/src/hmr/README.md +26 -0
- package/src/hmr/client/hmr-runtime.d.ts +1 -6
- package/src/hmr/client/hmr-runtime.js +38 -7
- package/src/hmr/hmr-strategy.d.ts +16 -13
- package/src/hmr/hmr-strategy.js +22 -7
- package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.postcss.test.e2e.js +31 -0
- package/src/hmr/hmr.test.e2e.js +26 -33
- package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
- package/src/hmr/strategies/default-hmr-strategy.js +1 -1
- package/src/hmr/strategies/js-hmr-strategy.d.ts +40 -42
- package/src/hmr/strategies/js-hmr-strategy.js +24 -43
- package/src/index.browser.d.ts +2 -2
- package/src/index.browser.js +1 -1
- package/src/index.d.ts +4 -3
- package/src/index.js +16 -5
- package/src/integrations/ghtml/ghtml-renderer.d.ts +3 -2
- package/src/integrations/ghtml/ghtml-renderer.js +27 -30
- package/src/integrations/ghtml/ghtml.constants.d.ts +1 -0
- package/src/integrations/ghtml/ghtml.constants.js +4 -0
- package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -6
- package/src/integrations/ghtml/ghtml.plugin.js +3 -4
- package/src/plugins/README.md +35 -0
- package/src/plugins/alias-resolver-plugin.d.ts +1 -0
- package/src/plugins/alias-resolver-plugin.js +27 -5
- package/src/plugins/eco-component-meta-plugin.d.ts +14 -1
- package/src/plugins/eco-component-meta-plugin.js +42 -24
- package/src/plugins/foreign-jsx-override-plugin.d.ts +33 -0
- package/src/plugins/foreign-jsx-override-plugin.js +41 -0
- package/src/plugins/integration-plugin.d.ts +157 -29
- package/src/plugins/integration-plugin.js +115 -14
- package/src/plugins/processor.d.ts +17 -2
- package/src/plugins/processor.js +22 -3
- package/src/plugins/runtime-capability.d.ts +9 -0
- package/src/plugins/source-transform.d.ts +46 -0
- package/src/plugins/source-transform.js +71 -0
- package/src/route-renderer/GRAPH.md +83 -325
- package/src/route-renderer/README.md +73 -90
- package/src/route-renderer/orchestration/component-render-context.d.ts +97 -0
- package/src/route-renderer/orchestration/component-render-context.js +147 -0
- package/src/route-renderer/orchestration/declared-ownership-graph.d.ts +18 -0
- package/src/route-renderer/orchestration/declared-ownership-graph.js +34 -0
- package/src/route-renderer/orchestration/foreign-subtree-execution.service.d.ts +110 -0
- package/src/route-renderer/orchestration/foreign-subtree-execution.service.js +233 -0
- package/src/route-renderer/orchestration/integration-renderer.d.ts +534 -0
- package/src/route-renderer/orchestration/integration-renderer.js +991 -0
- package/src/route-renderer/orchestration/ownership-planning.service.d.ts +24 -0
- package/src/route-renderer/orchestration/ownership-planning.service.js +63 -0
- package/src/route-renderer/orchestration/ownership-validation.service.d.ts +29 -0
- package/src/route-renderer/orchestration/ownership-validation.service.js +53 -0
- package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +3 -0
- package/src/route-renderer/orchestration/processed-asset-dedupe.js +27 -0
- package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.d.ts +91 -0
- package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.js +170 -0
- package/src/route-renderer/orchestration/render-output.utils.d.ts +66 -0
- package/src/route-renderer/orchestration/render-output.utils.js +171 -0
- package/src/route-renderer/orchestration/route-render-orchestrator.d.ts +156 -0
- package/src/route-renderer/orchestration/route-render-orchestrator.js +577 -0
- package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
- package/src/route-renderer/orchestration/template-serialization.js +45 -0
- package/src/route-renderer/page-loading/component-dependency-collection.d.ts +37 -0
- package/src/route-renderer/page-loading/component-dependency-collection.js +132 -0
- package/src/route-renderer/page-loading/declared-asset-collection.d.ts +24 -0
- package/src/route-renderer/page-loading/declared-asset-collection.js +106 -0
- package/src/route-renderer/{dependency-resolver.d.ts → page-loading/dependency-resolver.d.ts} +15 -4
- package/src/route-renderer/page-loading/dependency-resolver.js +115 -0
- package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +11 -0
- package/src/route-renderer/page-loading/ecopages-virtual-imports.js +57 -0
- package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +45 -0
- package/src/route-renderer/page-loading/lazy-entry-collection.js +105 -0
- package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +19 -0
- package/src/route-renderer/page-loading/lazy-trigger-planning.js +40 -0
- package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +5 -0
- package/src/route-renderer/page-loading/module-declaration-aggregation.js +33 -0
- package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +3 -0
- package/src/route-renderer/page-loading/module-declaration-scripts.js +18 -0
- package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +13 -0
- package/src/route-renderer/page-loading/page-dependency-bundling.js +137 -0
- package/src/route-renderer/page-loading/page-module-loader.d.ts +90 -0
- package/src/route-renderer/{page-module-loader.js → page-loading/page-module-loader.js} +39 -14
- package/src/route-renderer/route-renderer.d.ts +57 -14
- package/src/route-renderer/route-renderer.js +30 -18
- package/src/router/README.md +94 -0
- package/src/router/client/link-intent.d.ts +53 -0
- package/src/router/client/link-intent.js +34 -0
- package/src/router/client/link-intent.test.browser.d.ts +1 -0
- package/src/router/client/link-intent.test.browser.js +43 -0
- package/src/router/client/navigation-coordinator.d.ts +169 -0
- package/src/router/client/navigation-coordinator.js +215 -0
- package/src/router/server/route-registry.d.ts +78 -0
- package/src/router/server/route-registry.js +262 -0
- package/src/services/README.md +28 -0
- package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +3 -0
- package/src/services/assets/asset-processing-service/asset-dependency-keys.js +56 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +103 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.js +124 -89
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.d.ts +1 -1
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.js +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.d.ts +16 -1
- package/src/services/assets/asset-processing-service/assets.types.js +0 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +57 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +49 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
- package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +30 -0
- package/src/services/assets/asset-processing-service/grouped-content-bundles.js +65 -0
- package/src/services/assets/asset-processing-service/index.d.ts +6 -0
- package/src/services/assets/asset-processing-service/index.js +6 -0
- package/src/services/assets/asset-processing-service/page-package.d.ts +6 -0
- package/src/services/assets/asset-processing-service/page-package.js +80 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.d.ts +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.d.ts +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.d.ts +1 -1
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.js +11 -5
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +22 -0
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +136 -0
- package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/index.js +5 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.d.ts +3 -2
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +120 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.d.ts +4 -3
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.js +28 -7
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +42 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +126 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.d.ts +5 -2
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +59 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.js +9 -3
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +18 -0
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +45 -0
- package/src/services/assets/browser-bundle.service.d.ts +73 -0
- package/src/services/assets/browser-bundle.service.js +41 -0
- package/src/services/{page-request-cache-coordinator.service.d.ts → cache/page-request-cache-coordinator.service.d.ts} +2 -2
- package/src/services/{page-request-cache-coordinator.service.js → cache/page-request-cache-coordinator.service.js} +3 -1
- package/src/services/html/html-rewriter-provider.service.d.ts +40 -0
- package/src/services/html/html-rewriter-provider.service.js +68 -0
- package/src/services/html/html-transformer.service.d.ts +96 -0
- package/src/services/html/html-transformer.service.js +287 -0
- package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
- package/src/services/invalidation/development-invalidation.service.js +190 -0
- package/src/services/module-loading/app-module-loader.service.d.ts +7 -0
- package/src/services/module-loading/app-module-loader.service.js +0 -0
- package/src/services/module-loading/app-server-module-transpiler.service.d.ts +24 -0
- package/src/services/module-loading/app-server-module-transpiler.service.js +151 -0
- package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
- package/src/services/module-loading/host-module-loader-registry.js +15 -0
- package/src/services/module-loading/module-loading-types.d.ts +2 -0
- package/src/services/module-loading/module-loading-types.js +0 -0
- package/src/services/module-loading/node-bootstrap-plugin.d.ts +38 -0
- package/src/services/module-loading/node-bootstrap-plugin.js +215 -0
- package/src/services/module-loading/page-module-import.service.d.ts +95 -0
- package/src/services/module-loading/page-module-import.service.js +191 -0
- package/src/services/module-loading/server-module-transpiler.service.d.ts +63 -0
- package/src/services/module-loading/server-module-transpiler.service.js +64 -0
- package/src/services/module-loading/source-module-support.d.ts +5 -0
- package/src/services/module-loading/source-module-support.js +8 -0
- package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
- package/src/services/runtime-state/dev-graph.service.js +162 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
- package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
- package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
- package/src/services/{schema-validation-service.d.ts → validation/schema-validation-service.d.ts} +1 -1
- package/src/static-site-generator/README.md +26 -0
- package/src/static-site-generator/static-site-generator.d.ts +67 -20
- package/src/static-site-generator/static-site-generator.js +182 -138
- package/src/{internal-types.d.ts → types/internal-types.d.ts} +62 -30
- package/src/types/internal-types.js +0 -0
- package/src/{public-types.d.ts → types/public-types.d.ts} +239 -31
- package/src/types/public-types.js +0 -0
- package/src/utils/html-escaping.d.ts +7 -0
- package/src/utils/html-escaping.js +6 -0
- package/src/utils/locals-utils.d.ts +1 -1
- package/src/utils/parse-cli-args.d.ts +4 -1
- package/src/utils/parse-cli-args.js +16 -1
- package/src/utils/resolve-work-dir.d.ts +11 -0
- package/src/utils/resolve-work-dir.js +31 -0
- package/src/watchers/project-watcher.d.ts +11 -7
- package/src/watchers/project-watcher.js +69 -75
- package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
- package/src/watchers/project-watcher.test-helpers.js +6 -5
- package/CHANGELOG.md +0 -94
- package/src/adapters/abstract/application-adapter.ts +0 -337
- package/src/adapters/abstract/router-adapter.ts +0 -30
- package/src/adapters/abstract/server-adapter.ts +0 -79
- package/src/adapters/bun/client-bridge.ts +0 -62
- package/src/adapters/bun/create-app.ts +0 -189
- package/src/adapters/bun/define-api-handler.d.ts +0 -61
- package/src/adapters/bun/define-api-handler.ts +0 -114
- package/src/adapters/bun/hmr-manager.ts +0 -276
- package/src/adapters/bun/index.ts +0 -3
- package/src/adapters/bun/server-adapter.ts +0 -492
- package/src/adapters/bun/server-lifecycle.d.ts +0 -52
- package/src/adapters/bun/server-lifecycle.js +0 -120
- package/src/adapters/bun/server-lifecycle.ts +0 -154
- package/src/adapters/index.ts +0 -6
- package/src/adapters/node/create-app.ts +0 -179
- package/src/adapters/node/index.d.ts +0 -4
- package/src/adapters/node/index.js +0 -8
- package/src/adapters/node/index.ts +0 -9
- package/src/adapters/node/node-client-bridge.ts +0 -79
- package/src/adapters/node/node-hmr-manager.ts +0 -271
- package/src/adapters/node/server-adapter.ts +0 -561
- package/src/adapters/node/static-content-server.ts +0 -203
- package/src/adapters/shared/api-response.ts +0 -104
- package/src/adapters/shared/application-adapter.ts +0 -199
- package/src/adapters/shared/explicit-static-route-matcher.ts +0 -134
- package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -123
- package/src/adapters/shared/fs-server-response-factory.ts +0 -118
- package/src/adapters/shared/fs-server-response-matcher.ts +0 -198
- package/src/adapters/shared/render-context.ts +0 -105
- package/src/adapters/shared/server-adapter.ts +0 -442
- package/src/adapters/shared/server-route-handler.ts +0 -166
- package/src/adapters/shared/server-static-builder.ts +0 -82
- package/src/build/build-adapter.ts +0 -133
- package/src/build/build-types.ts +0 -83
- package/src/build/esbuild-build-adapter.ts +0 -511
- package/src/config/config-builder.ts +0 -474
- package/src/constants.ts +0 -39
- package/src/create-app.ts +0 -87
- package/src/define-api-handler.js +0 -15
- package/src/define-api-handler.ts +0 -66
- package/src/dev/sc-server.ts +0 -143
- package/src/eco/component-render-context.d.ts +0 -105
- package/src/eco/component-render-context.js +0 -77
- package/src/eco/component-render-context.ts +0 -202
- package/src/eco/eco.ts +0 -221
- package/src/eco/eco.types.ts +0 -202
- package/src/eco/eco.utils.ts +0 -89
- package/src/eco/global-injector-map.ts +0 -112
- package/src/eco/lazy-injector-map.ts +0 -120
- package/src/eco/module-dependencies.ts +0 -75
- package/src/errors/http-error.ts +0 -72
- package/src/errors/index.ts +0 -2
- package/src/errors/locals-access-error.ts +0 -7
- package/src/global/app-logger.ts +0 -4
- package/src/hmr/client/hmr-runtime.ts +0 -121
- package/src/hmr/hmr-strategy.ts +0 -172
- package/src/hmr/hmr.test.e2e.ts +0 -75
- package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
- package/src/hmr/strategies/js-hmr-strategy.ts +0 -327
- package/src/index.browser.ts +0 -3
- package/src/index.ts +0 -5
- package/src/integrations/ghtml/ghtml-renderer.ts +0 -93
- package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
- package/src/internal-types.ts +0 -212
- package/src/plugins/alias-resolver-plugin.ts +0 -45
- package/src/plugins/eco-component-meta-plugin.ts +0 -474
- package/src/plugins/integration-plugin.ts +0 -184
- package/src/plugins/processor.ts +0 -220
- package/src/public-types.ts +0 -1255
- package/src/route-renderer/component-graph-executor.d.ts +0 -32
- package/src/route-renderer/component-graph-executor.js +0 -31
- package/src/route-renderer/component-graph-executor.ts +0 -84
- package/src/route-renderer/component-graph.d.ts +0 -42
- package/src/route-renderer/component-graph.js +0 -72
- package/src/route-renderer/component-graph.ts +0 -159
- package/src/route-renderer/component-marker.d.ts +0 -52
- package/src/route-renderer/component-marker.js +0 -46
- package/src/route-renderer/component-marker.ts +0 -117
- package/src/route-renderer/dependency-resolver.js +0 -428
- package/src/route-renderer/dependency-resolver.ts +0 -596
- package/src/route-renderer/html-post-processing.service.d.ts +0 -40
- package/src/route-renderer/html-post-processing.service.js +0 -86
- package/src/route-renderer/html-post-processing.service.ts +0 -103
- package/src/route-renderer/integration-renderer.d.ts +0 -339
- package/src/route-renderer/integration-renderer.js +0 -526
- package/src/route-renderer/integration-renderer.ts +0 -696
- package/src/route-renderer/marker-graph-resolver.d.ts +0 -76
- package/src/route-renderer/marker-graph-resolver.js +0 -93
- package/src/route-renderer/marker-graph-resolver.ts +0 -153
- package/src/route-renderer/page-module-loader.d.ts +0 -61
- package/src/route-renderer/page-module-loader.ts +0 -153
- package/src/route-renderer/render-execution.service.d.ts +0 -69
- package/src/route-renderer/render-execution.service.js +0 -91
- package/src/route-renderer/render-execution.service.ts +0 -158
- package/src/route-renderer/render-preparation.service.d.ts +0 -112
- package/src/route-renderer/render-preparation.service.js +0 -243
- package/src/route-renderer/render-preparation.service.ts +0 -358
- package/src/route-renderer/route-renderer.ts +0 -80
- package/src/router/fs-router-scanner.d.ts +0 -41
- package/src/router/fs-router-scanner.js +0 -155
- package/src/router/fs-router-scanner.ts +0 -217
- package/src/router/fs-router.d.ts +0 -26
- package/src/router/fs-router.js +0 -100
- package/src/router/fs-router.ts +0 -122
- package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
- package/src/services/asset-processing-service/asset-processing.service.ts +0 -306
- package/src/services/asset-processing-service/asset.factory.ts +0 -105
- package/src/services/asset-processing-service/assets.types.ts +0 -112
- package/src/services/asset-processing-service/index.d.ts +0 -3
- package/src/services/asset-processing-service/index.js +0 -3
- package/src/services/asset-processing-service/index.ts +0 -3
- package/src/services/asset-processing-service/processor.interface.ts +0 -27
- package/src/services/asset-processing-service/processor.registry.ts +0 -18
- package/src/services/asset-processing-service/processors/base/base-processor.ts +0 -76
- package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +0 -16
- package/src/services/asset-processing-service/processors/base/base-script-processor.js +0 -80
- package/src/services/asset-processing-service/processors/base/base-script-processor.ts +0 -105
- package/src/services/asset-processing-service/processors/index.d.ts +0 -5
- package/src/services/asset-processing-service/processors/index.js +0 -5
- package/src/services/asset-processing-service/processors/index.ts +0 -5
- package/src/services/asset-processing-service/processors/script/content-script.processor.js +0 -57
- package/src/services/asset-processing-service/processors/script/content-script.processor.ts +0 -66
- package/src/services/asset-processing-service/processors/script/file-script.processor.ts +0 -88
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +0 -74
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +0 -84
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -77
- package/src/services/cache/cache.types.ts +0 -126
- package/src/services/cache/index.ts +0 -18
- package/src/services/cache/memory-cache-store.ts +0 -130
- package/src/services/cache/page-cache-service.ts +0 -202
- package/src/services/html-transformer.service.d.ts +0 -50
- package/src/services/html-transformer.service.js +0 -163
- package/src/services/html-transformer.service.ts +0 -217
- package/src/services/page-module-import.service.d.ts +0 -37
- package/src/services/page-module-import.service.js +0 -88
- package/src/services/page-module-import.service.ts +0 -129
- package/src/services/page-request-cache-coordinator.service.ts +0 -128
- package/src/services/schema-validation-service.ts +0 -204
- package/src/services/validation/standard-schema.types.ts +0 -68
- package/src/static-site-generator/static-site-generator.ts +0 -359
- package/src/utils/css.d.ts +0 -1
- package/src/utils/css.js +0 -7
- package/src/utils/css.ts +0 -5
- package/src/utils/deep-merge.ts +0 -47
- package/src/utils/hash.ts +0 -5
- package/src/utils/html.ts +0 -1
- package/src/utils/invariant.ts +0 -15
- package/src/utils/locals-utils.ts +0 -37
- package/src/utils/parse-cli-args.ts +0 -83
- package/src/utils/path-utils.module.ts +0 -14
- package/src/utils/runtime.ts +0 -44
- package/src/utils/server-utils.module.ts +0 -67
- package/src/watchers/project-watcher.test-helpers.ts +0 -40
- package/src/watchers/project-watcher.ts +0 -364
- /package/src/adapters/{bun → shared}/define-api-handler.js +0 -0
- /package/src/{internal-types.js → adapters/shared/runtime-host.js} +0 -0
- /package/src/{public-types.js → adapters/shared/static-preview-host.js} +0 -0
- /package/src/{services/asset-processing-service/assets.types.js → plugins/runtime-capability.js} +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.js +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.js +0 -0
- /package/src/services/{schema-validation-service.js → validation/schema-validation-service.js} +0 -0
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
# Router Layer
|
|
2
|
+
|
|
3
|
+
This directory contains route discovery, matching, and browser-side navigation infrastructure.
|
|
4
|
+
|
|
5
|
+
## Purpose
|
|
6
|
+
|
|
7
|
+
The router layer determines what route is being handled and how the client runtime coordinates navigation.
|
|
8
|
+
|
|
9
|
+
It is responsible for:
|
|
10
|
+
|
|
11
|
+
- filesystem route discovery and classification (`exact`, `dynamic`, `catch-all`)
|
|
12
|
+
- matching incoming request URLs to canonical template routes
|
|
13
|
+
- server-side static-path expansion for dynamic template routes
|
|
14
|
+
- client-side navigation ownership and cross-runtime handoff
|
|
15
|
+
- keeping route discovery separate from rendering execution
|
|
16
|
+
|
|
17
|
+
## Directory Structure
|
|
18
|
+
|
|
19
|
+
```
|
|
20
|
+
router/
|
|
21
|
+
├── server/ # Server-side route discovery and matching
|
|
22
|
+
│ └── route-registry.ts # Owns template routes, request matching, static expansion, and reload
|
|
23
|
+
└── client/ # Browser-side navigation coordination
|
|
24
|
+
├── navigation-coordinator.ts # Singleton runtime coordinator
|
|
25
|
+
└── link-intent.ts # Shared anchor detection and intent recovery helpers
|
|
26
|
+
```
|
|
27
|
+
|
|
28
|
+
## `server/`
|
|
29
|
+
|
|
30
|
+
### `RouteRegistry`
|
|
31
|
+
|
|
32
|
+
Owns the canonical set of filesystem-discovered template routes for one application.
|
|
33
|
+
|
|
34
|
+
File patterns determine route kind:
|
|
35
|
+
|
|
36
|
+
| Pattern | Kind | Example |
|
|
37
|
+
| --------------- | ----------- | ----------------- |
|
|
38
|
+
| `page.tsx` | `exact` | `/about` |
|
|
39
|
+
| `[slug].tsx` | `dynamic` | `/blog/[slug]` |
|
|
40
|
+
| `[...slug].tsx` | `catch-all` | `/docs/[...slug]` |
|
|
41
|
+
|
|
42
|
+
The registry stores canonical template routes only. It compiles request-time matching metadata during `init()` and `reload()`, but it does not execute `staticPaths()` during discovery.
|
|
43
|
+
|
|
44
|
+
Build-time static expansion is a separate operation. The registry invokes `staticPaths()` lazily through an injected page-module adapter and returns concrete static path expansions for the static generator.
|
|
45
|
+
|
|
46
|
+
The public interface is intentionally small:
|
|
47
|
+
|
|
48
|
+
- `templateRoutes` — ordered readonly template routes
|
|
49
|
+
- `init()` / `reload()` — rebuild discovery state and match metadata
|
|
50
|
+
- `matchRequest(requestUrl)` — request-time matching result with requested pathname, matched template route, params, and query
|
|
51
|
+
- `listStaticPathExpansions()` — build-time expansion for dynamic routes
|
|
52
|
+
- `listStaticGenerationRoutes()` — build-time route planning for static generation across exact and expanded routes
|
|
53
|
+
|
|
54
|
+
Match priority:
|
|
55
|
+
|
|
56
|
+
1. `exact` — the pathname must equal the route pathname exactly.
|
|
57
|
+
2. `dynamic` — the clean (bracket-stripped) prefix must appear in the pathname, and the segment counts must match.
|
|
58
|
+
3. `catch-all` — the clean prefix must appear in the pathname.
|
|
59
|
+
|
|
60
|
+
## `client/`
|
|
61
|
+
|
|
62
|
+
### Navigation Coordinator (`navigation-coordinator.ts`)
|
|
63
|
+
|
|
64
|
+
A singleton browser-side runtime stored on `window.__ECO_PAGES__.navigation`.
|
|
65
|
+
|
|
66
|
+
Access it with:
|
|
67
|
+
|
|
68
|
+
```ts
|
|
69
|
+
import { getEcoNavigationRuntime } from '@ecopages/core/router/navigation-coordinator';
|
|
70
|
+
const runtime = getEcoNavigationRuntime();
|
|
71
|
+
```
|
|
72
|
+
|
|
73
|
+
The coordinator is framework-agnostic. Browser runtimes (e.g. `browser-router`, `react-router`) register themselves and the coordinator arbitrates:
|
|
74
|
+
|
|
75
|
+
- **Ownership** — which runtime currently drives SPA navigation (`claimOwnership`, `releaseOwnership`, `setOwner`).
|
|
76
|
+
- **Document owner marker** — an HTML attribute (`data-eco-document-owner`) written into rendered markup so the coordinator can `adoptDocumentOwner` on the incoming page.
|
|
77
|
+
- **Navigation transactions** — each navigation begins a transaction with an `AbortSignal`; superseded navigations are automatically cancelled.
|
|
78
|
+
- **Cross-runtime handoff** — `requestHandoff` passes a pre-fetched `Document` to the target runtime without tearing down the current page prematurely.
|
|
79
|
+
- **Reload** — `reloadCurrentPage` delegates to whichever runtime currently owns the document.
|
|
80
|
+
- **Events** — `subscribe` lets runtimes react to `owner-change` and `registration-change` events.
|
|
81
|
+
|
|
82
|
+
### Link Intent (`link-intent.ts`)
|
|
83
|
+
|
|
84
|
+
Shared helpers for locating anchors and recovering stale navigation intent.
|
|
85
|
+
|
|
86
|
+
- `getAnchorFromNavigationEvent(event, linkSelector)` — finds the nearest matching anchor in the event's composed path, including across Shadow DOM boundaries.
|
|
87
|
+
- `recoverPendingNavigationHref(intent, hasInFlightNavigation, now, maxAgeMs?)` — resolves a previously captured pointer or hover target when the DOM changes before the click lands. Intents expire after `maxAgeMs` (default 1000 ms).
|
|
88
|
+
|
|
89
|
+
## Relationship To Rendering
|
|
90
|
+
|
|
91
|
+
The router layer answers **which route should run**.
|
|
92
|
+
The route-renderer layer answers **how that route gets rendered**.
|
|
93
|
+
|
|
94
|
+
Keeping those seams separate avoids mixing route ownership, module loading, and component orchestration into one service.
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared client-side navigation intent helpers.
|
|
3
|
+
*
|
|
4
|
+
* Browser runtimes use the same low-level mechanics to:
|
|
5
|
+
* - locate the anchor associated with a click, pointer, or hover event,
|
|
6
|
+
* - persist the last valid pointer or hover target while a navigation is in flight,
|
|
7
|
+
* - recover the final intended href when the DOM changes before the click lands.
|
|
8
|
+
*
|
|
9
|
+
* Keeping those mechanics here lets browser-router and react-router share one
|
|
10
|
+
* implementation while preserving their router-specific interception rules.
|
|
11
|
+
*
|
|
12
|
+
* @module
|
|
13
|
+
*/
|
|
14
|
+
/**
|
|
15
|
+
* Timestamped navigation intent captured from pointer or hover activity.
|
|
16
|
+
*/
|
|
17
|
+
export type EcoPendingNavigationIntent = {
|
|
18
|
+
href: string;
|
|
19
|
+
timestamp: number;
|
|
20
|
+
};
|
|
21
|
+
/**
|
|
22
|
+
* Finds the nearest matching anchor within an event's composed path.
|
|
23
|
+
*
|
|
24
|
+
* This works across Shadow DOM boundaries, which is required for delegated
|
|
25
|
+
* navigation handling when links are rendered inside custom elements.
|
|
26
|
+
*
|
|
27
|
+
* @param event - Pointer or mouse event being inspected.
|
|
28
|
+
* @param linkSelector - Selector that identifies navigable anchors.
|
|
29
|
+
* @returns The matched anchor element, or `null` when no matching anchor exists.
|
|
30
|
+
*/
|
|
31
|
+
export declare function getAnchorFromNavigationEvent(
|
|
32
|
+
event: MouseEvent | PointerEvent,
|
|
33
|
+
linkSelector: string,
|
|
34
|
+
): HTMLAnchorElement | null;
|
|
35
|
+
/**
|
|
36
|
+
* Resolves a previously captured intent while a navigation is still in flight.
|
|
37
|
+
*
|
|
38
|
+
* Pending intents expire quickly because they are only meant to bridge the gap
|
|
39
|
+
* between pointer or hover capture and the later click event when the DOM or
|
|
40
|
+
* active runtime changes during a rapid navigation sequence.
|
|
41
|
+
*
|
|
42
|
+
* @param intent - Previously captured pointer or hover intent.
|
|
43
|
+
* @param hasInFlightNavigation - Whether a router navigation is still active.
|
|
44
|
+
* @param now - Current monotonic timestamp, usually from `performance.now()`.
|
|
45
|
+
* @param maxAgeMs - Maximum allowed age for the recovered intent.
|
|
46
|
+
* @returns The intended href when still valid, otherwise `null`.
|
|
47
|
+
*/
|
|
48
|
+
export declare function recoverPendingNavigationHref(
|
|
49
|
+
intent: EcoPendingNavigationIntent | null,
|
|
50
|
+
hasInFlightNavigation: boolean,
|
|
51
|
+
now: number,
|
|
52
|
+
maxAgeMs?: number,
|
|
53
|
+
): string | null;
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
function getAnchorFromNavigationEvent(event, linkSelector) {
|
|
2
|
+
const eventTarget = event.target;
|
|
3
|
+
if (eventTarget instanceof HTMLAnchorElement && eventTarget.matches(linkSelector)) {
|
|
4
|
+
return eventTarget;
|
|
5
|
+
}
|
|
6
|
+
if (eventTarget instanceof Element) {
|
|
7
|
+
const closestAnchor = eventTarget.closest(linkSelector);
|
|
8
|
+
if (closestAnchor instanceof HTMLAnchorElement) {
|
|
9
|
+
return closestAnchor;
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
if (eventTarget instanceof Text) {
|
|
13
|
+
const parentAnchor = eventTarget.parentElement?.closest(linkSelector);
|
|
14
|
+
if (parentAnchor instanceof HTMLAnchorElement) {
|
|
15
|
+
return parentAnchor;
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
return event.composedPath().find(
|
|
19
|
+
(target) => target instanceof HTMLAnchorElement && target.matches(linkSelector)
|
|
20
|
+
);
|
|
21
|
+
}
|
|
22
|
+
function recoverPendingNavigationHref(intent, hasInFlightNavigation, now, maxAgeMs = 1e3) {
|
|
23
|
+
if (!intent || !hasInFlightNavigation) {
|
|
24
|
+
return null;
|
|
25
|
+
}
|
|
26
|
+
if (now - intent.timestamp > maxAgeMs) {
|
|
27
|
+
return null;
|
|
28
|
+
}
|
|
29
|
+
return intent.href;
|
|
30
|
+
}
|
|
31
|
+
export {
|
|
32
|
+
getAnchorFromNavigationEvent,
|
|
33
|
+
recoverPendingNavigationHref
|
|
34
|
+
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { describe, expect, it } from "vitest";
|
|
2
|
+
import { getAnchorFromNavigationEvent, recoverPendingNavigationHref } from "./link-intent.js";
|
|
3
|
+
describe("getAnchorFromNavigationEvent", () => {
|
|
4
|
+
it("returns the anchor when the event target is a text node inside it", () => {
|
|
5
|
+
const anchor = document.createElement("a");
|
|
6
|
+
anchor.href = "/fast";
|
|
7
|
+
anchor.setAttribute("data-eco-link", "true");
|
|
8
|
+
const textNode = document.createTextNode("fast-link");
|
|
9
|
+
anchor.append(textNode);
|
|
10
|
+
document.body.append(anchor);
|
|
11
|
+
const event = new MouseEvent("click", { bubbles: true, cancelable: true, composed: true });
|
|
12
|
+
Object.defineProperty(event, "target", {
|
|
13
|
+
configurable: true,
|
|
14
|
+
value: textNode
|
|
15
|
+
});
|
|
16
|
+
expect(getAnchorFromNavigationEvent(event, "a[data-eco-link]")).toBe(anchor);
|
|
17
|
+
});
|
|
18
|
+
it("returns the closest matching anchor for nested element targets", () => {
|
|
19
|
+
const anchor = document.createElement("a");
|
|
20
|
+
anchor.href = "/fast";
|
|
21
|
+
anchor.setAttribute("data-eco-link", "true");
|
|
22
|
+
const span = document.createElement("span");
|
|
23
|
+
span.textContent = "fast-link";
|
|
24
|
+
anchor.append(span);
|
|
25
|
+
document.body.append(anchor);
|
|
26
|
+
const event = new MouseEvent("click", { bubbles: true, cancelable: true, composed: true });
|
|
27
|
+
Object.defineProperty(event, "target", {
|
|
28
|
+
configurable: true,
|
|
29
|
+
value: span
|
|
30
|
+
});
|
|
31
|
+
expect(getAnchorFromNavigationEvent(event, "a[data-eco-link]")).toBe(anchor);
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
describe("recoverPendingNavigationHref", () => {
|
|
35
|
+
it("returns null for stale or missing pending intent state", () => {
|
|
36
|
+
expect(recoverPendingNavigationHref(null, true, 10)).toBeNull();
|
|
37
|
+
expect(recoverPendingNavigationHref({ href: "/fast", timestamp: 0 }, false, 10)).toBeNull();
|
|
38
|
+
expect(recoverPendingNavigationHref({ href: "/fast", timestamp: 0 }, true, 2e3, 1e3)).toBeNull();
|
|
39
|
+
});
|
|
40
|
+
it("returns the captured href while a navigation is still in flight", () => {
|
|
41
|
+
expect(recoverPendingNavigationHref({ href: "/fast", timestamp: 10 }, true, 20, 1e3)).toBe("/fast");
|
|
42
|
+
});
|
|
43
|
+
});
|
|
@@ -0,0 +1,169 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared browser-side navigation coordinator.
|
|
3
|
+
*
|
|
4
|
+
* This module is the client runtime contract used by browser-router,
|
|
5
|
+
* react-router, and HMR code to coordinate ownership, cross-runtime handoff,
|
|
6
|
+
* current-page reloads, and stale-navigation cancellation.
|
|
7
|
+
*
|
|
8
|
+
* The coordinator stays framework-agnostic: browser runtimes register their
|
|
9
|
+
* capabilities here, and the coordinator arbitrates which runtime currently
|
|
10
|
+
* owns the document and which navigation transaction is still current.
|
|
11
|
+
*
|
|
12
|
+
* @module
|
|
13
|
+
*/
|
|
14
|
+
/** Logical owner name for a browser navigation runtime. */
|
|
15
|
+
export type EcoNavigationOwner = 'none' | 'browser-router' | 'react-router' | (string & {});
|
|
16
|
+
/** HTML attribute used to persist the rendered document owner across navigations. */
|
|
17
|
+
export declare const ECO_DOCUMENT_OWNER_ATTRIBUTE = "data-eco-document-owner";
|
|
18
|
+
/** High-level navigation direction understood by browser runtimes. */
|
|
19
|
+
export type EcoNavigationDirection = 'forward' | 'back' | 'replace';
|
|
20
|
+
/** Navigation request sent between browser runtimes. */
|
|
21
|
+
export type EcoNavigationRequest = {
|
|
22
|
+
href: string;
|
|
23
|
+
direction?: EcoNavigationDirection;
|
|
24
|
+
source?: EcoNavigationOwner;
|
|
25
|
+
};
|
|
26
|
+
/** Navigation handoff request that includes a pre-fetched document. */
|
|
27
|
+
export type EcoNavigationHandoffRequest = EcoNavigationRequest & {
|
|
28
|
+
finalHref?: string;
|
|
29
|
+
targetOwner: EcoNavigationOwner;
|
|
30
|
+
document: Document;
|
|
31
|
+
html?: string;
|
|
32
|
+
/**
|
|
33
|
+
* Reports whether the source runtime's original navigation has already been
|
|
34
|
+
* superseded.
|
|
35
|
+
*
|
|
36
|
+
* Target runtimes use this to ignore handoff work that arrives after a newer
|
|
37
|
+
* navigation has already claimed ownership.
|
|
38
|
+
*/
|
|
39
|
+
isStaleSourceNavigation?: () => boolean;
|
|
40
|
+
};
|
|
41
|
+
/** Request to reload the current page through the active runtime. */
|
|
42
|
+
export type EcoReloadRequest = {
|
|
43
|
+
/**
|
|
44
|
+
* Clears runtime-owned caches before reloading.
|
|
45
|
+
*
|
|
46
|
+
* Runtimes use this when a reload must discard persisted client state such as
|
|
47
|
+
* cached layouts or route-local module instances.
|
|
48
|
+
*/
|
|
49
|
+
clearCache?: boolean;
|
|
50
|
+
/**
|
|
51
|
+
* Explicit page module URL to load for the reload.
|
|
52
|
+
*
|
|
53
|
+
* This is primarily used by HMR-aware runtimes so a current-page reload can
|
|
54
|
+
* reuse the active hot module entry instead of rediscovering the static page
|
|
55
|
+
* bootstrap asset from the fetched HTML document.
|
|
56
|
+
*/
|
|
57
|
+
moduleUrl?: string;
|
|
58
|
+
source?: EcoNavigationOwner;
|
|
59
|
+
};
|
|
60
|
+
/** Snapshot of the coordinator's current runtime ownership state. */
|
|
61
|
+
export type EcoNavigationOwnerState = {
|
|
62
|
+
owner: EcoNavigationOwner;
|
|
63
|
+
canHandleSpaNavigation: boolean;
|
|
64
|
+
};
|
|
65
|
+
/**
|
|
66
|
+
* Coordinator-managed navigation transaction.
|
|
67
|
+
*
|
|
68
|
+
* Runtimes use this to determine whether async work has become stale and to
|
|
69
|
+
* cancel or complete the active navigation sequence.
|
|
70
|
+
*/
|
|
71
|
+
export type EcoNavigationTransaction = {
|
|
72
|
+
id: number;
|
|
73
|
+
signal: AbortSignal;
|
|
74
|
+
isCurrent: () => boolean;
|
|
75
|
+
cancel: () => void;
|
|
76
|
+
complete: () => void;
|
|
77
|
+
};
|
|
78
|
+
export type EcoNavigationRuntimeEvent = {
|
|
79
|
+
type: 'owner-change';
|
|
80
|
+
owner: EcoNavigationOwner;
|
|
81
|
+
previousOwner: EcoNavigationOwner;
|
|
82
|
+
reason: 'set' | 'claim' | 'release' | 'document' | 'unregister';
|
|
83
|
+
} | {
|
|
84
|
+
type: 'registration-change';
|
|
85
|
+
owner: EcoNavigationOwner;
|
|
86
|
+
status: 'registered' | 'unregistered';
|
|
87
|
+
};
|
|
88
|
+
export type EcoNavigationRuntimeListener = (event: EcoNavigationRuntimeEvent) => void;
|
|
89
|
+
export type EcoNavigationRuntimeRegistration = {
|
|
90
|
+
owner: EcoNavigationOwner;
|
|
91
|
+
navigate?: (request: EcoNavigationRequest) => Promise<boolean | void>;
|
|
92
|
+
handoffNavigation?: (request: EcoNavigationHandoffRequest) => Promise<boolean | void>;
|
|
93
|
+
/**
|
|
94
|
+
* Reloads the current page through the runtime's local navigation mechanism.
|
|
95
|
+
*
|
|
96
|
+
* Implementations may honor `request.moduleUrl` to force a specific page entry
|
|
97
|
+
* and `request.clearCache` to discard persisted runtime state before reloading.
|
|
98
|
+
*/
|
|
99
|
+
reloadCurrentPage?: (request?: EcoReloadRequest) => Promise<void>;
|
|
100
|
+
/**
|
|
101
|
+
* Releases runtime-owned client state before another runtime commits a new
|
|
102
|
+
* document.
|
|
103
|
+
*
|
|
104
|
+
* This hook intentionally does not run as part of `requestHandoff()`. The
|
|
105
|
+
* accepting runtime decides when cleanup is safe so cross-runtime handoffs do
|
|
106
|
+
* not blank the current page before the incoming document is ready.
|
|
107
|
+
*/
|
|
108
|
+
cleanupBeforeHandoff?: () => void | Promise<void>;
|
|
109
|
+
};
|
|
110
|
+
/** Public browser-side navigation coordinator interface. */
|
|
111
|
+
export interface EcoNavigationRuntime {
|
|
112
|
+
/** Returns the currently active runtime owner and whether it can handle SPA navigation. */
|
|
113
|
+
getOwnerState(): EcoNavigationOwnerState;
|
|
114
|
+
/** Starts a new navigation transaction, invalidating the previously active one. */
|
|
115
|
+
beginNavigationTransaction(): EcoNavigationTransaction;
|
|
116
|
+
/** Reports whether a navigation transaction is still in flight. */
|
|
117
|
+
hasPendingNavigationTransaction(): boolean;
|
|
118
|
+
/** Cancels the active navigation transaction, if one exists. */
|
|
119
|
+
cancelCurrentNavigationTransaction(): void;
|
|
120
|
+
/** Forces the current owner value without checking registrations. */
|
|
121
|
+
setOwner(owner: EcoNavigationOwner): void;
|
|
122
|
+
/** Claims ownership for a runtime that is ready to drive SPA navigation. */
|
|
123
|
+
claimOwnership(owner: EcoNavigationOwner): void;
|
|
124
|
+
/** Releases ownership when the given runtime no longer controls the document. */
|
|
125
|
+
releaseOwnership(owner: EcoNavigationOwner): void;
|
|
126
|
+
/** Resolves document ownership from the rendered owner marker or fallback. */
|
|
127
|
+
resolveDocumentOwner(doc: Document, fallbackOwner?: EcoNavigationOwner): EcoNavigationOwner;
|
|
128
|
+
/** Reads and adopts the rendered document owner as the active runtime owner. */
|
|
129
|
+
adoptDocumentOwner(doc: Document, fallbackOwner?: EcoNavigationOwner): EcoNavigationOwner;
|
|
130
|
+
/** Returns whether the active owner is some runtime other than the given owner. */
|
|
131
|
+
isOwnedByAnotherRuntime(owner: EcoNavigationOwner): boolean;
|
|
132
|
+
/** Subscribes to ownership and registration change events. */
|
|
133
|
+
subscribe(listener: EcoNavigationRuntimeListener): () => void;
|
|
134
|
+
/** Registers a runtime implementation with the coordinator. */
|
|
135
|
+
register(runtime: EcoNavigationRuntimeRegistration): () => void;
|
|
136
|
+
/** Requests navigation through another eligible registered runtime. */
|
|
137
|
+
requestNavigation(request: EcoNavigationRequest): Promise<boolean>;
|
|
138
|
+
/**
|
|
139
|
+
* Hands a pre-fetched document to the target runtime.
|
|
140
|
+
*
|
|
141
|
+
* The coordinator delegates the document but does not clean up the current
|
|
142
|
+
* owner first. Cleanup timing belongs to the accepting runtime so a stale or
|
|
143
|
+
* superseded handoff cannot tear down the current page prematurely.
|
|
144
|
+
*/
|
|
145
|
+
requestHandoff(request: EcoNavigationHandoffRequest): Promise<boolean>;
|
|
146
|
+
/** Requests the active runtime to reload the current page. */
|
|
147
|
+
reloadCurrentPage(request?: EcoReloadRequest): Promise<boolean>;
|
|
148
|
+
/** Runs a target runtime's cleanup hook before handoff. */
|
|
149
|
+
cleanupOwner(owner: EcoNavigationOwner): Promise<void>;
|
|
150
|
+
/** Runs cleanup for whichever runtime currently owns the document. */
|
|
151
|
+
cleanupCurrentOwner(): Promise<void>;
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Reads the explicit browser document owner marker from a rendered HTML document.
|
|
155
|
+
*
|
|
156
|
+
* Documents without a marker return `null`, allowing runtimes to fall back to
|
|
157
|
+
* their local default behavior without scanning hydration scripts.
|
|
158
|
+
*/
|
|
159
|
+
export declare function getEcoDocumentOwner(doc: Document): EcoNavigationOwner | null;
|
|
160
|
+
/**
|
|
161
|
+
* Returns the singleton browser-side navigation coordinator.
|
|
162
|
+
*
|
|
163
|
+
* The coordinator centralizes ownership, handoff, and current-page reload
|
|
164
|
+
* requests across browser runtimes through one internal protocol.
|
|
165
|
+
*
|
|
166
|
+
* @param windowObject - Window-like object that stores the singleton runtime.
|
|
167
|
+
* @returns The shared browser navigation coordinator.
|
|
168
|
+
*/
|
|
169
|
+
export declare function getEcoNavigationRuntime(windowObject?: Window & typeof globalThis): EcoNavigationRuntime;
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
const ECO_DOCUMENT_OWNER_ATTRIBUTE = "data-eco-document-owner";
|
|
2
|
+
function getEcoDocumentOwner(doc) {
|
|
3
|
+
const owner = doc.documentElement.getAttribute(ECO_DOCUMENT_OWNER_ATTRIBUTE);
|
|
4
|
+
return owner && owner.length > 0 ? owner : null;
|
|
5
|
+
}
|
|
6
|
+
function getCandidateOwners(currentOwner, registrations, excludedOwner) {
|
|
7
|
+
const owners = [];
|
|
8
|
+
if (currentOwner !== "none" && currentOwner !== excludedOwner) {
|
|
9
|
+
owners.push(currentOwner);
|
|
10
|
+
}
|
|
11
|
+
for (const owner of registrations.keys()) {
|
|
12
|
+
if (owner === currentOwner || owner === excludedOwner) {
|
|
13
|
+
continue;
|
|
14
|
+
}
|
|
15
|
+
owners.push(owner);
|
|
16
|
+
}
|
|
17
|
+
return owners;
|
|
18
|
+
}
|
|
19
|
+
function createEcoNavigationRuntime() {
|
|
20
|
+
const registrations = /* @__PURE__ */ new Map();
|
|
21
|
+
const listeners = /* @__PURE__ */ new Set();
|
|
22
|
+
let owner = "none";
|
|
23
|
+
let navigationSequence = 0;
|
|
24
|
+
let navigationAbortController = null;
|
|
25
|
+
const emit = (event) => {
|
|
26
|
+
for (const listener of listeners) {
|
|
27
|
+
listener(event);
|
|
28
|
+
}
|
|
29
|
+
};
|
|
30
|
+
const updateOwner = (nextOwner, reason) => {
|
|
31
|
+
if (owner === nextOwner) {
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
const previousOwner = owner;
|
|
35
|
+
owner = nextOwner;
|
|
36
|
+
emit({
|
|
37
|
+
type: "owner-change",
|
|
38
|
+
owner: nextOwner,
|
|
39
|
+
previousOwner,
|
|
40
|
+
reason
|
|
41
|
+
});
|
|
42
|
+
};
|
|
43
|
+
const cancelNavigationTransaction = (navigationId, abortController) => {
|
|
44
|
+
if (navigationSequence !== navigationId || navigationAbortController !== abortController) {
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
navigationSequence += 1;
|
|
48
|
+
navigationAbortController = null;
|
|
49
|
+
abortController.abort();
|
|
50
|
+
};
|
|
51
|
+
const completeNavigationTransaction = (navigationId, abortController) => {
|
|
52
|
+
if (navigationSequence !== navigationId || navigationAbortController !== abortController) {
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
navigationAbortController = null;
|
|
56
|
+
};
|
|
57
|
+
const runtime = {
|
|
58
|
+
getOwnerState() {
|
|
59
|
+
const activeRuntime = registrations.get(owner);
|
|
60
|
+
return {
|
|
61
|
+
owner,
|
|
62
|
+
canHandleSpaNavigation: typeof activeRuntime?.navigate === "function"
|
|
63
|
+
};
|
|
64
|
+
},
|
|
65
|
+
hasPendingNavigationTransaction() {
|
|
66
|
+
return navigationAbortController !== null;
|
|
67
|
+
},
|
|
68
|
+
beginNavigationTransaction() {
|
|
69
|
+
navigationAbortController?.abort();
|
|
70
|
+
const abortController = new AbortController();
|
|
71
|
+
const navigationId = ++navigationSequence;
|
|
72
|
+
navigationAbortController = abortController;
|
|
73
|
+
return {
|
|
74
|
+
id: navigationId,
|
|
75
|
+
signal: abortController.signal,
|
|
76
|
+
isCurrent: () => navigationSequence === navigationId && navigationAbortController === abortController,
|
|
77
|
+
cancel: () => {
|
|
78
|
+
cancelNavigationTransaction(navigationId, abortController);
|
|
79
|
+
},
|
|
80
|
+
complete: () => {
|
|
81
|
+
completeNavigationTransaction(navigationId, abortController);
|
|
82
|
+
}
|
|
83
|
+
};
|
|
84
|
+
},
|
|
85
|
+
cancelCurrentNavigationTransaction() {
|
|
86
|
+
if (!navigationAbortController) {
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
cancelNavigationTransaction(navigationSequence, navigationAbortController);
|
|
90
|
+
},
|
|
91
|
+
setOwner(nextOwner) {
|
|
92
|
+
updateOwner(nextOwner, "set");
|
|
93
|
+
},
|
|
94
|
+
claimOwnership(nextOwner) {
|
|
95
|
+
updateOwner(nextOwner, "claim");
|
|
96
|
+
},
|
|
97
|
+
releaseOwnership(currentOwner) {
|
|
98
|
+
if (owner !== currentOwner) {
|
|
99
|
+
return;
|
|
100
|
+
}
|
|
101
|
+
updateOwner("none", "release");
|
|
102
|
+
},
|
|
103
|
+
resolveDocumentOwner(doc, fallbackOwner = "none") {
|
|
104
|
+
return getEcoDocumentOwner(doc) ?? fallbackOwner;
|
|
105
|
+
},
|
|
106
|
+
adoptDocumentOwner(doc, fallbackOwner = "none") {
|
|
107
|
+
const nextOwner = runtime.resolveDocumentOwner(doc, fallbackOwner);
|
|
108
|
+
updateOwner(nextOwner, "document");
|
|
109
|
+
return nextOwner;
|
|
110
|
+
},
|
|
111
|
+
isOwnedByAnotherRuntime(candidateOwner) {
|
|
112
|
+
return owner !== "none" && owner !== candidateOwner;
|
|
113
|
+
},
|
|
114
|
+
subscribe(listener) {
|
|
115
|
+
listeners.add(listener);
|
|
116
|
+
return () => {
|
|
117
|
+
listeners.delete(listener);
|
|
118
|
+
};
|
|
119
|
+
},
|
|
120
|
+
register(registration) {
|
|
121
|
+
registrations.set(registration.owner, registration);
|
|
122
|
+
emit({
|
|
123
|
+
type: "registration-change",
|
|
124
|
+
owner: registration.owner,
|
|
125
|
+
status: "registered"
|
|
126
|
+
});
|
|
127
|
+
return () => {
|
|
128
|
+
const currentRegistration = registrations.get(registration.owner);
|
|
129
|
+
if (currentRegistration !== registration) {
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
registrations.delete(registration.owner);
|
|
133
|
+
emit({
|
|
134
|
+
type: "registration-change",
|
|
135
|
+
owner: registration.owner,
|
|
136
|
+
status: "unregistered"
|
|
137
|
+
});
|
|
138
|
+
if (owner === registration.owner) {
|
|
139
|
+
updateOwner("none", "unregister");
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
},
|
|
143
|
+
async requestNavigation(request) {
|
|
144
|
+
for (const candidateOwner of getCandidateOwners(owner, registrations, request.source)) {
|
|
145
|
+
const registration = registrations.get(candidateOwner);
|
|
146
|
+
if (!registration?.navigate) {
|
|
147
|
+
continue;
|
|
148
|
+
}
|
|
149
|
+
const handled = await registration.navigate(request);
|
|
150
|
+
if (handled !== false) {
|
|
151
|
+
return true;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
return false;
|
|
155
|
+
},
|
|
156
|
+
async requestHandoff(request) {
|
|
157
|
+
if (request.targetOwner === "none") {
|
|
158
|
+
return false;
|
|
159
|
+
}
|
|
160
|
+
if (request.isStaleSourceNavigation?.()) {
|
|
161
|
+
return true;
|
|
162
|
+
}
|
|
163
|
+
const registration = registrations.get(request.targetOwner);
|
|
164
|
+
if (!registration?.handoffNavigation) {
|
|
165
|
+
return false;
|
|
166
|
+
}
|
|
167
|
+
if (request.isStaleSourceNavigation?.()) {
|
|
168
|
+
return true;
|
|
169
|
+
}
|
|
170
|
+
const handled = await registration.handoffNavigation(request);
|
|
171
|
+
return handled !== false;
|
|
172
|
+
},
|
|
173
|
+
async reloadCurrentPage(request) {
|
|
174
|
+
for (const candidateOwner of getCandidateOwners(owner, registrations)) {
|
|
175
|
+
const registration = registrations.get(candidateOwner);
|
|
176
|
+
if (!registration?.reloadCurrentPage) {
|
|
177
|
+
continue;
|
|
178
|
+
}
|
|
179
|
+
await registration.reloadCurrentPage(request);
|
|
180
|
+
return true;
|
|
181
|
+
}
|
|
182
|
+
return false;
|
|
183
|
+
},
|
|
184
|
+
async cleanupOwner(targetOwner) {
|
|
185
|
+
if (targetOwner === "none") {
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
const registration = registrations.get(targetOwner);
|
|
189
|
+
if (!registration?.cleanupBeforeHandoff) {
|
|
190
|
+
return;
|
|
191
|
+
}
|
|
192
|
+
await registration.cleanupBeforeHandoff();
|
|
193
|
+
if (owner === targetOwner) {
|
|
194
|
+
updateOwner("none", "release");
|
|
195
|
+
}
|
|
196
|
+
},
|
|
197
|
+
async cleanupCurrentOwner() {
|
|
198
|
+
await runtime.cleanupOwner(owner);
|
|
199
|
+
}
|
|
200
|
+
};
|
|
201
|
+
return runtime;
|
|
202
|
+
}
|
|
203
|
+
function getEcoNavigationRuntime(windowObject = window) {
|
|
204
|
+
const runtimeWindow = windowObject;
|
|
205
|
+
runtimeWindow.__ECO_PAGES__ = runtimeWindow.__ECO_PAGES__ || {};
|
|
206
|
+
if (!runtimeWindow.__ECO_PAGES__.navigation) {
|
|
207
|
+
runtimeWindow.__ECO_PAGES__.navigation = createEcoNavigationRuntime();
|
|
208
|
+
}
|
|
209
|
+
return runtimeWindow.__ECO_PAGES__.navigation;
|
|
210
|
+
}
|
|
211
|
+
export {
|
|
212
|
+
ECO_DOCUMENT_OWNER_ATTRIBUTE,
|
|
213
|
+
getEcoDocumentOwner,
|
|
214
|
+
getEcoNavigationRuntime
|
|
215
|
+
};
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import type { EcoPagesAppConfig, RouteKind } from '../../types/internal-types.js';
|
|
2
|
+
export type RouteParams = Record<string, string | string[]>;
|
|
3
|
+
export type RouteQuery = Record<string, string>;
|
|
4
|
+
export type TemplateRoute = {
|
|
5
|
+
readonly pathname: string;
|
|
6
|
+
readonly kind: RouteKind;
|
|
7
|
+
readonly filePath: string;
|
|
8
|
+
readonly paramNames: readonly string[];
|
|
9
|
+
};
|
|
10
|
+
export type RouteMatch = {
|
|
11
|
+
readonly requestedPathname: string;
|
|
12
|
+
readonly templateRoute: TemplateRoute;
|
|
13
|
+
readonly params: RouteParams;
|
|
14
|
+
readonly query: RouteQuery;
|
|
15
|
+
};
|
|
16
|
+
export type StaticPathExpansion = {
|
|
17
|
+
readonly pathname: string;
|
|
18
|
+
readonly templateRoute: TemplateRoute;
|
|
19
|
+
readonly params: RouteParams;
|
|
20
|
+
};
|
|
21
|
+
export type StaticGenerationRoute = {
|
|
22
|
+
readonly requestUrl: string;
|
|
23
|
+
readonly pathname: string;
|
|
24
|
+
readonly templateRoute: TemplateRoute;
|
|
25
|
+
readonly params: RouteParams;
|
|
26
|
+
};
|
|
27
|
+
export type StaticPathsContext = {
|
|
28
|
+
readonly appConfig: EcoPagesAppConfig;
|
|
29
|
+
readonly runtimeOrigin: string;
|
|
30
|
+
};
|
|
31
|
+
export type RouteRegistryPageModule = {
|
|
32
|
+
readonly staticPaths?: (context: StaticPathsContext) => Promise<{
|
|
33
|
+
paths: Array<{
|
|
34
|
+
params: RouteParams;
|
|
35
|
+
}>;
|
|
36
|
+
}>;
|
|
37
|
+
readonly staticProps?: unknown;
|
|
38
|
+
};
|
|
39
|
+
export interface RouteRegistryPageModuleAdapter {
|
|
40
|
+
loadPageModule(filePath: string): Promise<RouteRegistryPageModule>;
|
|
41
|
+
}
|
|
42
|
+
export type RouteRegistryOptions = {
|
|
43
|
+
pagesDir: string;
|
|
44
|
+
appConfig: EcoPagesAppConfig;
|
|
45
|
+
origin: string;
|
|
46
|
+
templatesExt: readonly string[];
|
|
47
|
+
buildMode: boolean;
|
|
48
|
+
pageModuleAdapter: RouteRegistryPageModuleAdapter;
|
|
49
|
+
};
|
|
50
|
+
export declare class RouteRegistry {
|
|
51
|
+
readonly origin: string;
|
|
52
|
+
readonly appConfig: EcoPagesAppConfig;
|
|
53
|
+
readonly pagesDir: string;
|
|
54
|
+
readonly templatesExt: readonly string[];
|
|
55
|
+
readonly buildMode: boolean;
|
|
56
|
+
private readonly pageModuleAdapter;
|
|
57
|
+
private templateRouteList;
|
|
58
|
+
private readonly reloadListeners;
|
|
59
|
+
constructor(options: RouteRegistryOptions);
|
|
60
|
+
get templateRoutes(): readonly TemplateRoute[];
|
|
61
|
+
init(): Promise<void>;
|
|
62
|
+
reload(): Promise<void>;
|
|
63
|
+
onReload(listener: () => void): () => void;
|
|
64
|
+
matchRequest(requestUrl: string): RouteMatch | null;
|
|
65
|
+
listStaticPathExpansions(input: {
|
|
66
|
+
runtimeOrigin: string;
|
|
67
|
+
}): Promise<readonly StaticPathExpansion[]>;
|
|
68
|
+
listStaticGenerationRoutes(input: {
|
|
69
|
+
runtimeOrigin: string;
|
|
70
|
+
}): Promise<readonly StaticGenerationRoute[]>;
|
|
71
|
+
private scanTemplateRoutes;
|
|
72
|
+
private getRoutePath;
|
|
73
|
+
private classifyRouteKind;
|
|
74
|
+
private getParamNames;
|
|
75
|
+
private tryExtractParams;
|
|
76
|
+
private getSearchParams;
|
|
77
|
+
private resolveTemplatePath;
|
|
78
|
+
}
|