@ecopages/core 0.2.0-alpha.23 → 0.2.0-alpha.24
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/package.json +93 -226
- package/src/adapters/abstract/application-adapter.test.ts +172 -0
- package/src/adapters/abstract/application-adapter.ts +379 -0
- package/src/adapters/abstract/router-adapter.ts +30 -0
- package/src/adapters/abstract/server-adapter.ts +79 -0
- package/src/adapters/bun/client-bridge.ts +62 -0
- package/src/adapters/bun/create-app.ts +180 -0
- package/src/adapters/bun/hmr-manager.test.ts +267 -0
- package/src/adapters/bun/hmr-manager.ts +406 -0
- package/src/adapters/bun/index.ts +2 -0
- package/src/adapters/bun/server-adapter.ts +500 -0
- package/src/adapters/bun/server-lifecycle.ts +124 -0
- package/src/adapters/create-app.test.ts +10 -0
- package/src/adapters/create-app.ts +91 -0
- package/src/adapters/index.ts +2 -0
- package/src/adapters/node/create-app.test.ts +53 -0
- package/src/adapters/node/create-app.ts +183 -0
- package/src/adapters/node/node-client-bridge.test.ts +198 -0
- package/src/adapters/node/node-client-bridge.ts +79 -0
- package/src/adapters/node/node-hmr-manager.test.ts +322 -0
- package/src/adapters/node/node-hmr-manager.ts +378 -0
- package/src/adapters/node/server-adapter.ts +502 -0
- package/src/adapters/node/static-content-server.test.ts +60 -0
- package/src/adapters/node/static-content-server.ts +239 -0
- package/src/adapters/shared/api-response.test.ts +97 -0
- package/src/adapters/shared/api-response.ts +104 -0
- package/src/adapters/shared/application-adapter.ts +199 -0
- package/src/adapters/shared/define-api-handler.ts +66 -0
- package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +140 -0
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +90 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +127 -0
- package/src/adapters/shared/fs-server-response-factory.test.ts +187 -0
- package/src/adapters/shared/fs-server-response-factory.ts +118 -0
- package/src/adapters/shared/fs-server-response-matcher.test.ts +285 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +189 -0
- package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
- package/src/adapters/shared/hmr-html-response.ts +52 -0
- package/src/adapters/shared/hmr-manager.contract.test.ts +232 -0
- package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
- package/src/adapters/shared/render-context.test.ts +150 -0
- package/src/adapters/shared/render-context.ts +123 -0
- package/src/adapters/shared/runtime-bootstrap.ts +79 -0
- package/src/adapters/shared/server-adapter.test.ts +77 -0
- package/src/adapters/shared/server-adapter.ts +493 -0
- package/src/adapters/shared/server-route-handler.test.ts +110 -0
- package/src/adapters/shared/server-route-handler.ts +153 -0
- package/src/adapters/shared/server-static-builder.test.ts +338 -0
- package/src/adapters/shared/server-static-builder.ts +170 -0
- package/src/build/build-adapter-serialization.test.ts +281 -0
- package/src/build/build-adapter.test.ts +1240 -0
- package/src/build/build-adapter.ts +1012 -0
- package/src/build/build-manifest.ts +54 -0
- package/src/build/build-types.ts +83 -0
- package/src/build/dev-build-coordinator.ts +220 -0
- package/src/build/esbuild-build-adapter.ts +660 -0
- package/src/build/runtime-build-executor.test.ts +81 -0
- package/src/build/runtime-build-executor.ts +40 -0
- package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
- package/src/build/runtime-specifier-alias-plugin.ts +62 -0
- package/src/build/runtime-specifier-aliases.ts +135 -0
- package/src/config/config-builder.test.ts +443 -0
- package/src/config/config-builder.ts +742 -0
- package/src/config/config-builder.typecheck.test.ts +96 -0
- package/src/config/{constants.d.ts → constants.ts} +22 -13
- package/src/dev/sc-server.ts +143 -0
- package/src/eco/eco.browser.test.ts +43 -0
- package/src/eco/eco.browser.ts +118 -0
- package/src/eco/eco.test.ts +654 -0
- package/src/eco/eco.ts +205 -0
- package/src/eco/eco.types.ts +221 -0
- package/src/eco/eco.utils.test.ts +219 -0
- package/src/eco/eco.utils.ts +5 -0
- package/src/eco/global-injector-map.test.ts +42 -0
- package/src/eco/global-injector-map.ts +112 -0
- package/src/eco/lazy-injector-map.test.ts +66 -0
- package/src/eco/lazy-injector-map.ts +120 -0
- package/src/eco/module-dependencies.test.ts +30 -0
- package/src/eco/module-dependencies.ts +75 -0
- package/src/errors/http-error.test.ts +134 -0
- package/src/errors/http-error.ts +72 -0
- package/src/errors/{index.d.ts → index.ts} +2 -2
- package/src/errors/locals-access-error.ts +7 -0
- package/src/global/app-logger.ts +4 -0
- package/src/global/utils.test.ts +12 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
- package/src/hmr/client/hmr-runtime.ts +160 -0
- package/src/hmr/hmr-strategy.test.ts +124 -0
- package/src/hmr/hmr-strategy.ts +177 -0
- package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
- package/src/hmr/hmr.test.e2e.ts +66 -0
- package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
- package/src/hmr/strategies/js-hmr-strategy.test.ts +335 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +320 -0
- package/src/index.browser.ts +3 -0
- package/src/index.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +97 -0
- package/src/integrations/ghtml/ghtml.constants.ts +1 -0
- package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
- package/src/plugins/alias-resolver-plugin.test.ts +41 -0
- package/src/plugins/alias-resolver-plugin.ts +63 -0
- package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
- package/src/plugins/eco-component-meta-plugin.ts +495 -0
- package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
- package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
- package/src/plugins/integration-plugin.test.ts +156 -0
- package/src/plugins/integration-plugin.ts +311 -0
- package/src/plugins/processor.test.ts +148 -0
- package/src/plugins/processor.ts +240 -0
- package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
- package/src/plugins/source-transform.test.ts +82 -0
- package/src/plugins/source-transform.ts +123 -0
- package/src/route-renderer/orchestration/boundary-planning.service.ts +146 -0
- package/src/route-renderer/orchestration/component-render-context.ts +318 -0
- package/src/route-renderer/orchestration/integration-renderer.test.ts +2088 -0
- package/src/route-renderer/orchestration/integration-renderer.ts +1285 -0
- package/src/route-renderer/orchestration/page-packaging.service.test.ts +76 -0
- package/src/route-renderer/orchestration/page-packaging.service.ts +85 -0
- package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.test.ts +319 -0
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.ts +289 -0
- package/src/route-renderer/orchestration/render-execution.service.test.ts +196 -0
- package/src/route-renderer/orchestration/render-execution.service.ts +182 -0
- package/src/route-renderer/orchestration/render-output.utils.ts +302 -0
- package/src/route-renderer/orchestration/render-preparation.service.test.ts +569 -0
- package/src/route-renderer/orchestration/render-preparation.service.ts +508 -0
- package/src/route-renderer/orchestration/route-shell-composer.service.ts +162 -0
- package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
- package/src/route-renderer/orchestration/template-serialization.ts +117 -0
- package/src/route-renderer/page-loading/component-dependency-collection.ts +196 -0
- package/src/route-renderer/page-loading/declared-asset-collection.ts +156 -0
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +665 -0
- package/src/route-renderer/page-loading/dependency-resolver.ts +150 -0
- package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
- package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
- package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
- package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
- package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
- package/src/route-renderer/page-loading/page-dependency-bundling.ts +205 -0
- package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
- package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
- package/src/route-renderer/route-renderer.ts +136 -0
- package/src/router/client/link-intent.test.browser.ts +51 -0
- package/src/router/client/link-intent.ts +92 -0
- package/src/router/client/navigation-coordinator.test.ts +237 -0
- package/src/router/client/navigation-coordinator.ts +453 -0
- package/src/router/server/fs-router-scanner.test.ts +83 -0
- package/src/router/server/fs-router-scanner.ts +224 -0
- package/src/router/server/fs-router.test.ts +214 -0
- package/src/router/server/fs-router.ts +122 -0
- package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +476 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.ts +345 -0
- package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
- package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
- package/src/services/assets/asset-processing-service/assets.types.ts +125 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
- package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
- package/src/services/assets/asset-processing-service/index.ts +5 -0
- package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
- package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +174 -0
- package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +192 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +134 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +110 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +87 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +71 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +81 -0
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
- package/src/services/assets/browser-bundle.service.test.ts +66 -0
- package/src/services/assets/browser-bundle.service.ts +109 -0
- package/src/services/cache/cache.types.ts +126 -0
- package/src/services/cache/index.ts +18 -0
- package/src/services/cache/memory-cache-store.test.ts +225 -0
- package/src/services/cache/memory-cache-store.ts +130 -0
- package/src/services/cache/page-cache-service.test.ts +175 -0
- package/src/services/cache/page-cache-service.ts +202 -0
- package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
- package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
- package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
- package/src/services/html/html-rewriter-provider.service.ts +104 -0
- package/src/services/html/html-transformer.service.test.ts +479 -0
- package/src/services/html/html-transformer.service.ts +275 -0
- package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
- package/src/services/invalidation/development-invalidation.service.ts +262 -0
- package/src/services/module-loading/app-module-loader.service.ts +9 -0
- package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
- package/src/services/module-loading/app-server-module-transpiler.service.ts +143 -0
- package/src/services/module-loading/host-module-loader-registry.ts +15 -0
- package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
- package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
- package/src/services/module-loading/node-bootstrap-plugin.ts +297 -0
- package/src/services/module-loading/page-module-import.service.test.ts +504 -0
- package/src/services/module-loading/page-module-import.service.ts +252 -0
- package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
- package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
- package/src/services/module-loading/source-module-support.ts +19 -0
- package/src/services/runtime-state/dev-graph.service.ts +217 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
- package/src/services/runtime-state/runtime-specifier-registry.service.ts +96 -0
- package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
- package/src/services/validation/schema-validation-service.test.ts +223 -0
- package/src/services/validation/schema-validation-service.ts +204 -0
- package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
- package/src/static-site-generator/static-site-generator.test.ts +316 -0
- package/src/static-site-generator/static-site-generator.ts +462 -0
- package/src/types/internal-types.ts +242 -0
- package/src/types/public-types.ts +1443 -0
- package/src/utils/deep-merge.test.ts +114 -0
- package/src/utils/deep-merge.ts +47 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/html-escaping.ts +9 -0
- package/src/utils/invariant.test.ts +22 -0
- package/src/utils/invariant.ts +15 -0
- package/src/utils/locals-utils.ts +37 -0
- package/src/utils/parse-cli-args.test.ts +69 -0
- package/src/utils/parse-cli-args.ts +105 -0
- package/src/utils/path-utils.module.ts +14 -0
- package/src/utils/path-utils.test.ts +15 -0
- package/src/utils/resolve-work-dir.ts +45 -0
- package/src/utils/runtime.ts +44 -0
- package/src/utils/server-utils.module.ts +67 -0
- package/src/utils/server-utils.test.ts +38 -0
- package/src/watchers/project-watcher.integration.test.ts +337 -0
- package/src/watchers/project-watcher.test-helpers.ts +41 -0
- package/src/watchers/project-watcher.test.ts +768 -0
- package/src/watchers/project-watcher.ts +357 -0
- package/CHANGELOG.md +0 -51
- package/src/adapters/abstract/application-adapter.d.ts +0 -194
- package/src/adapters/abstract/application-adapter.js +0 -121
- package/src/adapters/abstract/router-adapter.d.ts +0 -26
- package/src/adapters/abstract/router-adapter.js +0 -5
- package/src/adapters/abstract/server-adapter.d.ts +0 -69
- package/src/adapters/abstract/server-adapter.js +0 -15
- package/src/adapters/bun/client-bridge.d.ts +0 -34
- package/src/adapters/bun/client-bridge.js +0 -48
- package/src/adapters/bun/create-app.d.ts +0 -52
- package/src/adapters/bun/create-app.js +0 -116
- package/src/adapters/bun/hmr-manager.d.ts +0 -143
- package/src/adapters/bun/hmr-manager.js +0 -333
- package/src/adapters/bun/index.d.ts +0 -2
- package/src/adapters/bun/index.js +0 -8
- package/src/adapters/bun/server-adapter.d.ts +0 -155
- package/src/adapters/bun/server-adapter.js +0 -374
- package/src/adapters/bun/server-lifecycle.d.ts +0 -63
- package/src/adapters/bun/server-lifecycle.js +0 -92
- package/src/adapters/create-app.d.ts +0 -20
- package/src/adapters/create-app.js +0 -66
- package/src/adapters/index.d.ts +0 -2
- package/src/adapters/index.js +0 -8
- package/src/adapters/node/create-app.d.ts +0 -18
- package/src/adapters/node/create-app.js +0 -149
- package/src/adapters/node/node-client-bridge.d.ts +0 -26
- package/src/adapters/node/node-client-bridge.js +0 -66
- package/src/adapters/node/node-hmr-manager.d.ts +0 -133
- package/src/adapters/node/node-hmr-manager.js +0 -311
- package/src/adapters/node/server-adapter.d.ts +0 -161
- package/src/adapters/node/server-adapter.js +0 -359
- package/src/adapters/node/static-content-server.d.ts +0 -60
- package/src/adapters/node/static-content-server.js +0 -194
- package/src/adapters/shared/api-response.d.ts +0 -52
- package/src/adapters/shared/api-response.js +0 -96
- package/src/adapters/shared/application-adapter.d.ts +0 -18
- package/src/adapters/shared/application-adapter.js +0 -90
- package/src/adapters/shared/define-api-handler.d.ts +0 -25
- package/src/adapters/shared/define-api-handler.js +0 -15
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
- package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
- package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
- package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
- package/src/adapters/shared/fs-server-response-factory.js +0 -97
- package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
- package/src/adapters/shared/fs-server-response-matcher.js +0 -147
- package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
- package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
- package/src/adapters/shared/hmr-html-response.d.ts +0 -22
- package/src/adapters/shared/hmr-html-response.js +0 -32
- package/src/adapters/shared/render-context.d.ts +0 -15
- package/src/adapters/shared/render-context.js +0 -72
- package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
- package/src/adapters/shared/runtime-bootstrap.js +0 -43
- package/src/adapters/shared/server-adapter.d.ts +0 -97
- package/src/adapters/shared/server-adapter.js +0 -390
- package/src/adapters/shared/server-route-handler.d.ts +0 -89
- package/src/adapters/shared/server-route-handler.js +0 -111
- package/src/adapters/shared/server-static-builder.d.ts +0 -70
- package/src/adapters/shared/server-static-builder.js +0 -100
- package/src/build/build-adapter.d.ts +0 -239
- package/src/build/build-adapter.js +0 -642
- package/src/build/build-manifest.d.ts +0 -27
- package/src/build/build-manifest.js +0 -30
- package/src/build/build-types.d.ts +0 -57
- package/src/build/build-types.js +0 -0
- package/src/build/dev-build-coordinator.d.ts +0 -72
- package/src/build/dev-build-coordinator.js +0 -154
- package/src/build/esbuild-build-adapter.d.ts +0 -78
- package/src/build/esbuild-build-adapter.js +0 -505
- package/src/build/runtime-build-executor.d.ts +0 -14
- package/src/build/runtime-build-executor.js +0 -22
- package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
- package/src/build/runtime-specifier-alias-plugin.js +0 -35
- package/src/build/runtime-specifier-aliases.d.ts +0 -5
- package/src/build/runtime-specifier-aliases.js +0 -95
- package/src/config/config-builder.d.ts +0 -252
- package/src/config/config-builder.js +0 -603
- package/src/config/constants.js +0 -25
- package/src/dev/sc-server.d.ts +0 -30
- package/src/dev/sc-server.js +0 -111
- package/src/eco/eco.browser.d.ts +0 -2
- package/src/eco/eco.browser.js +0 -83
- package/src/eco/eco.d.ts +0 -9
- package/src/eco/eco.js +0 -85
- package/src/eco/eco.types.d.ts +0 -178
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.utils.d.ts +0 -1
- package/src/eco/eco.utils.js +0 -10
- package/src/eco/global-injector-map.d.ts +0 -16
- package/src/eco/global-injector-map.js +0 -80
- package/src/eco/lazy-injector-map.d.ts +0 -8
- package/src/eco/lazy-injector-map.js +0 -70
- package/src/eco/module-dependencies.d.ts +0 -18
- package/src/eco/module-dependencies.js +0 -49
- package/src/errors/http-error.d.ts +0 -31
- package/src/errors/http-error.js +0 -50
- package/src/errors/index.js +0 -4
- package/src/errors/locals-access-error.d.ts +0 -4
- package/src/errors/locals-access-error.js +0 -9
- package/src/global/app-logger.d.ts +0 -2
- package/src/global/app-logger.js +0 -6
- package/src/hmr/client/hmr-runtime.d.ts +0 -5
- package/src/hmr/client/hmr-runtime.js +0 -109
- package/src/hmr/hmr-strategy.d.ts +0 -162
- package/src/hmr/hmr-strategy.js +0 -44
- package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.postcss.test.e2e.js +0 -31
- package/src/hmr/hmr.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.test.e2e.js +0 -43
- package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
- package/src/hmr/strategies/default-hmr-strategy.js +0 -34
- package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
- package/src/hmr/strategies/js-hmr-strategy.js +0 -178
- package/src/index.browser.d.ts +0 -3
- package/src/index.browser.js +0 -4
- package/src/index.d.ts +0 -6
- package/src/index.js +0 -21
- package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
- package/src/integrations/ghtml/ghtml-renderer.js +0 -63
- package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
- package/src/integrations/ghtml/ghtml.constants.js +0 -4
- package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
- package/src/integrations/ghtml/ghtml.plugin.js +0 -20
- package/src/plugins/alias-resolver-plugin.d.ts +0 -2
- package/src/plugins/alias-resolver-plugin.js +0 -53
- package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
- package/src/plugins/eco-component-meta-plugin.js +0 -163
- package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
- package/src/plugins/foreign-jsx-override-plugin.js +0 -35
- package/src/plugins/integration-plugin.d.ts +0 -219
- package/src/plugins/integration-plugin.js +0 -196
- package/src/plugins/processor.d.ts +0 -95
- package/src/plugins/processor.js +0 -136
- package/src/plugins/runtime-capability.js +0 -0
- package/src/plugins/source-transform.d.ts +0 -46
- package/src/plugins/source-transform.js +0 -71
- package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
- package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
- package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
- package/src/route-renderer/orchestration/component-render-context.js +0 -147
- package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -554
- package/src/route-renderer/orchestration/integration-renderer.js +0 -957
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
- package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
- package/src/route-renderer/orchestration/render-execution.service.js +0 -106
- package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -46
- package/src/route-renderer/orchestration/render-output.utils.js +0 -65
- package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
- package/src/route-renderer/orchestration/render-preparation.service.js +0 -341
- package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
- package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
- package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
- package/src/route-renderer/orchestration/template-serialization.js +0 -45
- package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
- package/src/route-renderer/page-loading/dependency-resolver.js +0 -444
- package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
- package/src/route-renderer/page-loading/page-module-loader.js +0 -127
- package/src/route-renderer/route-renderer.d.ts +0 -67
- package/src/route-renderer/route-renderer.js +0 -103
- package/src/router/client/link-intent.js +0 -34
- package/src/router/client/link-intent.test.browser.d.ts +0 -1
- package/src/router/client/link-intent.test.browser.js +0 -43
- package/src/router/client/navigation-coordinator.d.ts +0 -149
- package/src/router/client/navigation-coordinator.js +0 -215
- package/src/router/server/fs-router-scanner.d.ts +0 -41
- package/src/router/server/fs-router-scanner.js +0 -161
- package/src/router/server/fs-router.d.ts +0 -26
- package/src/router/server/fs-router.js +0 -100
- package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -120
- package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -331
- package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
- package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
- package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -89
- 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 +0 -55
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -48
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
- package/src/services/assets/asset-processing-service/index.d.ts +0 -5
- package/src/services/assets/asset-processing-service/index.js +0 -5
- package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
- package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
- package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
- package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
- package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -64
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -17
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -72
- package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
- package/src/services/assets/asset-processing-service/processors/index.js +0 -5
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -5
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -57
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -88
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -75
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -5
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -66
- package/src/services/assets/browser-bundle.service.d.ts +0 -32
- package/src/services/assets/browser-bundle.service.js +0 -33
- package/src/services/cache/cache.types.d.ts +0 -107
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/index.d.ts +0 -7
- package/src/services/cache/index.js +0 -7
- package/src/services/cache/memory-cache-store.d.ts +0 -42
- package/src/services/cache/memory-cache-store.js +0 -98
- package/src/services/cache/page-cache-service.d.ts +0 -70
- package/src/services/cache/page-cache-service.js +0 -152
- package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
- package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
- package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
- package/src/services/html/html-rewriter-provider.service.js +0 -68
- package/src/services/html/html-transformer.service.d.ts +0 -77
- package/src/services/html/html-transformer.service.js +0 -215
- package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
- package/src/services/invalidation/development-invalidation.service.js +0 -190
- package/src/services/module-loading/app-module-loader.service.d.ts +0 -28
- package/src/services/module-loading/app-module-loader.service.js +0 -35
- package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
- package/src/services/module-loading/app-server-module-transpiler.service.js +0 -109
- package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
- package/src/services/module-loading/host-module-loader-registry.js +0 -15
- package/src/services/module-loading/module-loading-types.js +0 -0
- package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
- package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
- package/src/services/module-loading/page-module-import.service.d.ts +0 -76
- package/src/services/module-loading/page-module-import.service.js +0 -173
- package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -72
- package/src/services/module-loading/server-module-transpiler.service.js +0 -64
- package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
- package/src/services/runtime-state/dev-graph.service.js +0 -162
- package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
- package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
- package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
- package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
- package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
- package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
- package/src/services/validation/schema-validation-service.d.ts +0 -122
- package/src/services/validation/schema-validation-service.js +0 -101
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/static-site-generator/static-site-generator.d.ts +0 -104
- package/src/static-site-generator/static-site-generator.js +0 -338
- package/src/types/internal-types.d.ts +0 -231
- package/src/types/internal-types.js +0 -0
- package/src/types/public-types.d.ts +0 -1219
- package/src/types/public-types.js +0 -0
- package/src/utils/deep-merge.d.ts +0 -14
- package/src/utils/deep-merge.js +0 -32
- package/src/utils/hash.d.ts +0 -1
- package/src/utils/hash.js +0 -7
- package/src/utils/html-escaping.d.ts +0 -7
- package/src/utils/html-escaping.js +0 -6
- package/src/utils/html.js +0 -4
- package/src/utils/invariant.d.ts +0 -5
- package/src/utils/invariant.js +0 -11
- package/src/utils/locals-utils.d.ts +0 -15
- package/src/utils/locals-utils.js +0 -24
- package/src/utils/parse-cli-args.d.ts +0 -27
- package/src/utils/parse-cli-args.js +0 -62
- package/src/utils/path-utils.module.d.ts +0 -5
- package/src/utils/path-utils.module.js +0 -14
- package/src/utils/resolve-work-dir.d.ts +0 -11
- package/src/utils/resolve-work-dir.js +0 -31
- package/src/utils/runtime.d.ts +0 -11
- package/src/utils/runtime.js +0 -40
- package/src/utils/server-utils.module.d.ts +0 -19
- package/src/utils/server-utils.module.js +0 -56
- package/src/watchers/project-watcher.d.ts +0 -136
- package/src/watchers/project-watcher.js +0 -275
- package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
- package/src/watchers/project-watcher.test-helpers.js +0 -52
- /package/src/utils/{html.d.ts → html.ts} +0 -0
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Ecopages HMR Runtime
|
|
3
|
+
* Injected into the browser to handle Hot Module Replacement updates.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { getEcoNavigationRuntime } from '../../router/client/navigation-coordinator.ts';
|
|
7
|
+
|
|
8
|
+
interface HMRPayload {
|
|
9
|
+
type: 'reload' | 'error' | 'update' | 'css-update' | 'layout-update';
|
|
10
|
+
path?: string;
|
|
11
|
+
message?: string;
|
|
12
|
+
timestamp?: number;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
(function () {
|
|
16
|
+
const WS_URL = 'ws://' + location.host + '/_hmr';
|
|
17
|
+
let socket: WebSocket;
|
|
18
|
+
let reconnectAttempts = 0;
|
|
19
|
+
|
|
20
|
+
function connect() {
|
|
21
|
+
socket = new WebSocket(WS_URL);
|
|
22
|
+
|
|
23
|
+
socket.addEventListener('open', () => {
|
|
24
|
+
console.log('[ecopages] HMR Connected');
|
|
25
|
+
reconnectAttempts = 0;
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
socket.addEventListener('message', async (event) => {
|
|
29
|
+
try {
|
|
30
|
+
const payload: HMRPayload = JSON.parse(event.data);
|
|
31
|
+
handleMessage(payload);
|
|
32
|
+
} catch (e) {
|
|
33
|
+
console.error('[ecopages] Invalid HMR message:', e);
|
|
34
|
+
}
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
socket.addEventListener('close', () => {
|
|
38
|
+
if (reconnectAttempts < 10) {
|
|
39
|
+
setTimeout(connect, 1000 * 2 ** reconnectAttempts);
|
|
40
|
+
reconnectAttempts++;
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
async function handleMessage(payload: HMRPayload) {
|
|
46
|
+
const navigationRuntime = getEcoNavigationRuntime(window);
|
|
47
|
+
|
|
48
|
+
switch (payload.type) {
|
|
49
|
+
case 'reload':
|
|
50
|
+
await waitForNavigationToSettle(navigationRuntime);
|
|
51
|
+
location.reload();
|
|
52
|
+
break;
|
|
53
|
+
case 'layout-update': {
|
|
54
|
+
await waitForNavigationToSettle(navigationRuntime);
|
|
55
|
+
if (await navigationRuntime.reloadCurrentPage({ clearCache: true, moduleUrl: getActiveHmrModuleUrl() })) {
|
|
56
|
+
} else {
|
|
57
|
+
location.reload();
|
|
58
|
+
}
|
|
59
|
+
break;
|
|
60
|
+
}
|
|
61
|
+
case 'error':
|
|
62
|
+
console.error('[ecopages] HMR Error:', payload.message);
|
|
63
|
+
break;
|
|
64
|
+
case 'update':
|
|
65
|
+
if (payload.path) {
|
|
66
|
+
await applyUpdate(payload.path, payload.timestamp);
|
|
67
|
+
}
|
|
68
|
+
break;
|
|
69
|
+
case 'css-update':
|
|
70
|
+
if (payload.path) {
|
|
71
|
+
refreshStylesheet(payload.path);
|
|
72
|
+
}
|
|
73
|
+
break;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Applies a module update by calling registered HMR handlers or re-importing the module.
|
|
79
|
+
* @param path - The module path to update
|
|
80
|
+
* @param timestamp - Optional timestamp for cache busting
|
|
81
|
+
*/
|
|
82
|
+
async function applyUpdate(path: string, timestamp?: number) {
|
|
83
|
+
try {
|
|
84
|
+
const url = path + '?t=' + (timestamp || Date.now());
|
|
85
|
+
const handlers = window.__ECO_PAGES__?.hmrHandlers;
|
|
86
|
+
const navigationRuntime = getEcoNavigationRuntime(window);
|
|
87
|
+
await waitForNavigationToSettle(navigationRuntime);
|
|
88
|
+
|
|
89
|
+
if (handlers?.[path]) {
|
|
90
|
+
await handlers[path](url);
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
await import(url);
|
|
95
|
+
|
|
96
|
+
if (await navigationRuntime.reloadCurrentPage({ clearCache: false })) {
|
|
97
|
+
}
|
|
98
|
+
} catch (e) {
|
|
99
|
+
console.error('[ecopages] Failed to apply HMR update:', e);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
function getActiveHmrModuleUrl(): string | undefined {
|
|
104
|
+
const pageModule = window.__ECO_PAGES__?.page?.module;
|
|
105
|
+
if (pageModule && pageModule.includes('/assets/_hmr/')) {
|
|
106
|
+
return pageModule.split('?')[0];
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
const handlerPaths = Object.keys(window.__ECO_PAGES__?.hmrHandlers ?? {});
|
|
110
|
+
return handlerPaths.at(-1);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
async function waitForNavigationToSettle(navigationRuntime: ReturnType<typeof getEcoNavigationRuntime>) {
|
|
114
|
+
if (!navigationRuntime.hasPendingNavigationTransaction()) {
|
|
115
|
+
return;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
await new Promise<void>((resolve) => {
|
|
119
|
+
const startedAt = performance.now();
|
|
120
|
+
const timeoutMs = 2000;
|
|
121
|
+
|
|
122
|
+
const poll = () => {
|
|
123
|
+
if (!navigationRuntime.hasPendingNavigationTransaction()) {
|
|
124
|
+
resolve();
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
if (performance.now() - startedAt >= timeoutMs) {
|
|
129
|
+
resolve();
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
requestAnimationFrame(poll);
|
|
134
|
+
};
|
|
135
|
+
|
|
136
|
+
requestAnimationFrame(poll);
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/**
|
|
141
|
+
* Hot-reload CSS by updating stylesheet link href with cache-busting query param.
|
|
142
|
+
* This causes the browser to re-fetch the stylesheet without a full page reload.
|
|
143
|
+
*/
|
|
144
|
+
function refreshStylesheet(path: string) {
|
|
145
|
+
const filename = path.split('/').pop() || '';
|
|
146
|
+
const links = document.querySelectorAll('link[rel="stylesheet"]');
|
|
147
|
+
|
|
148
|
+
links.forEach((link) => {
|
|
149
|
+
const href = (link as HTMLLinkElement).href;
|
|
150
|
+
if (href.includes(filename) || href.includes(path.replace(/^.*\/src\//, '/assets/'))) {
|
|
151
|
+
const url = new URL(href, location.origin);
|
|
152
|
+
url.searchParams.set('t', Date.now().toString());
|
|
153
|
+
(link as HTMLLinkElement).href = url.toString();
|
|
154
|
+
console.log('[ecopages] CSS updated:', filename);
|
|
155
|
+
}
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
connect();
|
|
160
|
+
})();
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import { HmrStrategy, HmrStrategyType, type HmrAction } from './hmr-strategy';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Mock strategy for testing the base class behavior
|
|
6
|
+
*/
|
|
7
|
+
class MockStrategy extends HmrStrategy {
|
|
8
|
+
readonly type: HmrStrategyType;
|
|
9
|
+
override readonly priorityOffset: number;
|
|
10
|
+
private matchPattern: string;
|
|
11
|
+
|
|
12
|
+
constructor(options: { type: HmrStrategyType; priorityOffset?: number; matchPattern?: string }) {
|
|
13
|
+
super();
|
|
14
|
+
this.type = options.type;
|
|
15
|
+
this.priorityOffset = options.priorityOffset ?? 0;
|
|
16
|
+
this.matchPattern = options.matchPattern ?? '.';
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
matches(filePath: string): boolean {
|
|
20
|
+
return filePath.endsWith(this.matchPattern);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async process(filePath: string): Promise<HmrAction> {
|
|
24
|
+
return {
|
|
25
|
+
type: 'broadcast',
|
|
26
|
+
events: [{ type: 'update', path: filePath, timestamp: Date.now() }],
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
describe('HmrStrategyType', () => {
|
|
32
|
+
it('INTEGRATION has value 100', () => {
|
|
33
|
+
expect(HmrStrategyType.INTEGRATION).toBe(100);
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
it('ASSET has value 50', () => {
|
|
37
|
+
expect(HmrStrategyType.ASSET).toBe(50);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('SCRIPT has value 25', () => {
|
|
41
|
+
expect(HmrStrategyType.SCRIPT).toBe(25);
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
it('FALLBACK has value 0', () => {
|
|
45
|
+
expect(HmrStrategyType.FALLBACK).toBe(0);
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it('maintains priority order: INTEGRATION > ASSET > SCRIPT > FALLBACK', () => {
|
|
49
|
+
expect(HmrStrategyType.INTEGRATION).toBeGreaterThan(HmrStrategyType.ASSET);
|
|
50
|
+
expect(HmrStrategyType.ASSET).toBeGreaterThan(HmrStrategyType.SCRIPT);
|
|
51
|
+
expect(HmrStrategyType.SCRIPT).toBeGreaterThan(HmrStrategyType.FALLBACK);
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
describe('HmrStrategy', () => {
|
|
56
|
+
describe('priority', () => {
|
|
57
|
+
it('returns type value when priorityOffset is 0', () => {
|
|
58
|
+
const strategy = new MockStrategy({ type: HmrStrategyType.INTEGRATION });
|
|
59
|
+
expect(strategy.priority).toBe(100);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('returns type + priorityOffset', () => {
|
|
63
|
+
const strategy = new MockStrategy({ type: HmrStrategyType.INTEGRATION, priorityOffset: 5 });
|
|
64
|
+
expect(strategy.priority).toBe(105);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
it('allows negative priorityOffset', () => {
|
|
68
|
+
const strategy = new MockStrategy({ type: HmrStrategyType.ASSET, priorityOffset: -10 });
|
|
69
|
+
expect(strategy.priority).toBe(40);
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
describe('strategy selection by priority', () => {
|
|
74
|
+
it('higher priority strategies are selected first', () => {
|
|
75
|
+
const strategies = [
|
|
76
|
+
new MockStrategy({ type: HmrStrategyType.FALLBACK, matchPattern: '.test' }),
|
|
77
|
+
new MockStrategy({ type: HmrStrategyType.INTEGRATION, matchPattern: '.test' }),
|
|
78
|
+
new MockStrategy({ type: HmrStrategyType.SCRIPT, matchPattern: '.test' }),
|
|
79
|
+
new MockStrategy({ type: HmrStrategyType.ASSET, matchPattern: '.test' }),
|
|
80
|
+
];
|
|
81
|
+
|
|
82
|
+
const sorted = [...strategies].sort((a, b) => b.priority - a.priority);
|
|
83
|
+
|
|
84
|
+
expect(sorted[0].type).toBe(HmrStrategyType.INTEGRATION);
|
|
85
|
+
expect(sorted[1].type).toBe(HmrStrategyType.ASSET);
|
|
86
|
+
expect(sorted[2].type).toBe(HmrStrategyType.SCRIPT);
|
|
87
|
+
expect(sorted[3].type).toBe(HmrStrategyType.FALLBACK);
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('priorityOffset can override default ordering', () => {
|
|
91
|
+
const assetWithHighOffset = new MockStrategy({
|
|
92
|
+
type: HmrStrategyType.ASSET,
|
|
93
|
+
priorityOffset: 60,
|
|
94
|
+
matchPattern: '.test',
|
|
95
|
+
});
|
|
96
|
+
const integration = new MockStrategy({
|
|
97
|
+
type: HmrStrategyType.INTEGRATION,
|
|
98
|
+
matchPattern: '.test',
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
expect(assetWithHighOffset.priority).toBeGreaterThan(integration.priority);
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
describe('matches', () => {
|
|
106
|
+
it('should be implemented by subclasses', () => {
|
|
107
|
+
const strategy = new MockStrategy({ type: HmrStrategyType.ASSET, matchPattern: '.css' });
|
|
108
|
+
expect(strategy.matches('styles.css')).toBe(true);
|
|
109
|
+
expect(strategy.matches('script.js')).toBe(false);
|
|
110
|
+
});
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
describe('process', () => {
|
|
114
|
+
it('should return an HmrAction', async () => {
|
|
115
|
+
const strategy = new MockStrategy({ type: HmrStrategyType.ASSET });
|
|
116
|
+
const action = await strategy.process('/path/to/file.');
|
|
117
|
+
|
|
118
|
+
expect(action.type).toBe('broadcast');
|
|
119
|
+
expect(action.events).toBeDefined();
|
|
120
|
+
expect(action.events?.[0].type).toBe('update');
|
|
121
|
+
expect(action.events?.[0].path).toBe('/path/to/file.');
|
|
122
|
+
});
|
|
123
|
+
});
|
|
124
|
+
});
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
import type { ClientBridgeEvent } from '../types/public-types.ts';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* HMR Strategy Pattern
|
|
5
|
+
*
|
|
6
|
+
* This module defines the base classes and types for Hot Module Replacement strategies.
|
|
7
|
+
* Each strategy handles a specific type of file change and determines how to process it.
|
|
8
|
+
*
|
|
9
|
+
* @module
|
|
10
|
+
* @example
|
|
11
|
+
* ```typescript
|
|
12
|
+
* class CustomHmrStrategy extends HmrStrategy {
|
|
13
|
+
* readonly type = HmrStrategyType.INTEGRATION;
|
|
14
|
+
*
|
|
15
|
+
* matches(filePath: string): boolean {
|
|
16
|
+
* return filePath.endsWith('.custom');
|
|
17
|
+
* }
|
|
18
|
+
*
|
|
19
|
+
* async process(filePath: string): Promise<HmrAction> {
|
|
20
|
+
* return {
|
|
21
|
+
* type: 'broadcast',
|
|
22
|
+
* events: [{ type: 'update', path: filePath, timestamp: Date.now() }],
|
|
23
|
+
* };
|
|
24
|
+
* }
|
|
25
|
+
* }
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Defines the category of an HMR strategy, which determines its execution priority.
|
|
31
|
+
* Strategies are evaluated in descending order: INTEGRATION → ASSET → SCRIPT → FALLBACK.
|
|
32
|
+
*
|
|
33
|
+
* @remarks
|
|
34
|
+
* The numeric values represent base priorities. Strategies can fine-tune their priority
|
|
35
|
+
* using the `priorityOffset` property.
|
|
36
|
+
*/
|
|
37
|
+
export const HmrStrategyType = {
|
|
38
|
+
/**
|
|
39
|
+
* Integration-specific strategies (React, Lit, etc.)
|
|
40
|
+
* Highest priority to allow framework-specific HMR handling.
|
|
41
|
+
*/
|
|
42
|
+
INTEGRATION: 100,
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Asset processing strategies (CSS, images, etc.)
|
|
46
|
+
* High priority for specialized asset handling.
|
|
47
|
+
*/
|
|
48
|
+
ASSET: 50,
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* Generic script bundling strategies (JS/TS)
|
|
52
|
+
* Medium priority for standard script processing.
|
|
53
|
+
*/
|
|
54
|
+
SCRIPT: 25,
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Fallback strategy for unhandled file types.
|
|
58
|
+
* Lowest priority, triggers full page reload.
|
|
59
|
+
*/
|
|
60
|
+
FALLBACK: 0,
|
|
61
|
+
} as const;
|
|
62
|
+
|
|
63
|
+
export type HmrStrategyType = (typeof HmrStrategyType)[keyof typeof HmrStrategyType];
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Represents an action to be taken after processing a file change.
|
|
67
|
+
*/
|
|
68
|
+
export interface HmrAction {
|
|
69
|
+
/**
|
|
70
|
+
* Whether to broadcast an HMR event to connected clients.
|
|
71
|
+
*/
|
|
72
|
+
type: 'broadcast' | 'none';
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* The HMR events to broadcast, if type is 'broadcast'.
|
|
76
|
+
* Multiple events may be broadcast in one action.
|
|
77
|
+
*/
|
|
78
|
+
events?: ClientBridgeEvent[];
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Base class for HMR strategies.
|
|
83
|
+
*
|
|
84
|
+
* Each strategy handles a specific type of file change and determines how to process it.
|
|
85
|
+
* Strategies are selected based on their priority (higher values are evaluated first) and
|
|
86
|
+
* whether they match the changed file path.
|
|
87
|
+
*
|
|
88
|
+
* @remarks
|
|
89
|
+
* Strategies are expected to be stateless and idempotent. The same file change
|
|
90
|
+
* should produce the same result when processed by the same strategy.
|
|
91
|
+
*
|
|
92
|
+
* @example
|
|
93
|
+
* ```typescript
|
|
94
|
+
* class MyAssetStrategy extends HmrStrategy {
|
|
95
|
+
* readonly type = HmrStrategyType.ASSET;
|
|
96
|
+
* readonly priorityOffset = 5;
|
|
97
|
+
*
|
|
98
|
+
* matches(filePath: string): boolean {
|
|
99
|
+
* return /\.(png|jpg|svg)$/.test(filePath);
|
|
100
|
+
* }
|
|
101
|
+
*
|
|
102
|
+
* async process(filePath: string): Promise<HmrAction> {
|
|
103
|
+
* await this.optimizeImage(filePath);
|
|
104
|
+
* return {
|
|
105
|
+
* type: 'broadcast',
|
|
106
|
+
* events: [{ type: 'update', path: filePath, timestamp: Date.now() }]
|
|
107
|
+
* };
|
|
108
|
+
* }
|
|
109
|
+
* }
|
|
110
|
+
* ```
|
|
111
|
+
*/
|
|
112
|
+
export abstract class HmrStrategy {
|
|
113
|
+
/**
|
|
114
|
+
* The category of this strategy, determining its base priority.
|
|
115
|
+
*/
|
|
116
|
+
abstract readonly type: HmrStrategyType;
|
|
117
|
+
|
|
118
|
+
/**
|
|
119
|
+
* Optional offset to fine-tune priority within the same category.
|
|
120
|
+
* Useful when multiple strategies share the same type.
|
|
121
|
+
*
|
|
122
|
+
* @defaultValue 0
|
|
123
|
+
*/
|
|
124
|
+
readonly priorityOffset: number = 0;
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Computed priority for strategy selection.
|
|
128
|
+
* Higher values are evaluated first.
|
|
129
|
+
*
|
|
130
|
+
* @returns The sum of the strategy type and priority offset
|
|
131
|
+
*/
|
|
132
|
+
get priority(): number {
|
|
133
|
+
return this.type + this.priorityOffset;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/**
|
|
137
|
+
* Determines if this strategy can handle the given file path.
|
|
138
|
+
*
|
|
139
|
+
* @param filePath - Absolute path to the changed file
|
|
140
|
+
* @returns `true` when this strategy should process the file
|
|
141
|
+
*
|
|
142
|
+
* @example
|
|
143
|
+
* ```typescript
|
|
144
|
+
* matches(filePath: string): boolean {
|
|
145
|
+
* return filePath.endsWith('.css');
|
|
146
|
+
* }
|
|
147
|
+
* ```
|
|
148
|
+
*/
|
|
149
|
+
abstract matches(filePath: string): boolean;
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Processes a file change and returns the action to take.
|
|
153
|
+
*
|
|
154
|
+
* This method may perform side effects such as:
|
|
155
|
+
* - Rebuilding files
|
|
156
|
+
* - Writing to disk
|
|
157
|
+
* - Transforming code
|
|
158
|
+
* - Updating caches
|
|
159
|
+
*
|
|
160
|
+
* @param filePath - Absolute path to the changed file
|
|
161
|
+
* @returns Action to take (broadcast event or none)
|
|
162
|
+
*
|
|
163
|
+
* @example
|
|
164
|
+
* ```typescript
|
|
165
|
+
* async process(filePath: string): Promise<HmrAction> {
|
|
166
|
+
* const processed = await this.transform(filePath);
|
|
167
|
+
* await Bun.write(outputPath, processed);
|
|
168
|
+
*
|
|
169
|
+
* return {
|
|
170
|
+
* type: 'broadcast',
|
|
171
|
+
* events: [{ type: 'update', path: outputPath, timestamp: Date.now() }]
|
|
172
|
+
* };
|
|
173
|
+
* }
|
|
174
|
+
* ```
|
|
175
|
+
*/
|
|
176
|
+
abstract process(filePath: string): Promise<HmrAction>;
|
|
177
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { test, expect } from '@playwright/test';
|
|
2
|
+
import { readFileSync } from 'node:fs';
|
|
3
|
+
import { writeFile } from 'node:fs/promises';
|
|
4
|
+
import { resolve } from 'node:path';
|
|
5
|
+
|
|
6
|
+
const FIXTURE_DIR = resolve(process.cwd(), 'packages/core/__fixtures__/app');
|
|
7
|
+
const TEST_CSS_FILE = resolve(FIXTURE_DIR, 'src/pages/postcss-hmr.css');
|
|
8
|
+
|
|
9
|
+
test.describe('HMR E2E PostCSS', () => {
|
|
10
|
+
test('should hot-update processor-owned CSS without a full page reload', async ({ page }) => {
|
|
11
|
+
const originalCss = readFileSync(TEST_CSS_FILE, 'utf-8');
|
|
12
|
+
const title = page.locator('.postcss-title').first();
|
|
13
|
+
|
|
14
|
+
try {
|
|
15
|
+
await page.goto('/postcss-hmr', { waitUntil: 'networkidle' });
|
|
16
|
+
await expect(title).toBeVisible();
|
|
17
|
+
|
|
18
|
+
const initialColor = await title.evaluate((el) => getComputedStyle(el).color);
|
|
19
|
+
expect(initialColor).toBeTruthy();
|
|
20
|
+
await page.evaluate(() => {
|
|
21
|
+
(window as typeof window & { __postcssReloadProbe?: string }).__postcssReloadProbe = 'before-change';
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
const modifiedCss = originalCss.replace('.postcss-title {', '.postcss-title {\n\tcolor: rgb(255, 0, 0);');
|
|
25
|
+
|
|
26
|
+
await writeFile(TEST_CSS_FILE, modifiedCss, { flush: true });
|
|
27
|
+
|
|
28
|
+
await expect
|
|
29
|
+
.poll(async () => title.evaluate((el) => getComputedStyle(el).color), { timeout: 10000 })
|
|
30
|
+
.toBe('rgb(255, 0, 0)');
|
|
31
|
+
|
|
32
|
+
await expect(
|
|
33
|
+
page.evaluate(
|
|
34
|
+
() => (window as typeof window & { __postcssReloadProbe?: string }).__postcssReloadProbe ?? null,
|
|
35
|
+
),
|
|
36
|
+
).resolves.toBe('before-change');
|
|
37
|
+
} finally {
|
|
38
|
+
await writeFile(TEST_CSS_FILE, originalCss, { flush: true });
|
|
39
|
+
}
|
|
40
|
+
});
|
|
41
|
+
});
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { test, expect } from '@playwright/test';
|
|
2
|
+
import { readFileSync } from 'node:fs';
|
|
3
|
+
import { writeFile } from 'node:fs/promises';
|
|
4
|
+
import { resolve } from 'node:path';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* HMR E2E Tests
|
|
8
|
+
*
|
|
9
|
+
* True end-to-end tests for Hot Module Replacement using Playwright.
|
|
10
|
+
* These tests modify actual source files and verify the browser updates.
|
|
11
|
+
*
|
|
12
|
+
* Run with: bunx playwright test packages/core/src/hmr/hmr.e2e.test.ts
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
const FIXTURE_DIR = resolve(process.cwd(), 'packages/core/__fixtures__/app');
|
|
16
|
+
const TEST_CSS_FILE = resolve(FIXTURE_DIR, 'src/pages/index.css');
|
|
17
|
+
|
|
18
|
+
test.describe('HMR E2E', () => {
|
|
19
|
+
test('should load page with .main-title element', async ({ page }) => {
|
|
20
|
+
await page.goto('/', { waitUntil: 'networkidle' });
|
|
21
|
+
const title = page.locator('.main-title').first();
|
|
22
|
+
await expect(title).toBeVisible();
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
test('should connect to HMR WebSocket', async ({ page }) => {
|
|
26
|
+
const socketPromise = page.waitForEvent('websocket', {
|
|
27
|
+
predicate: (socket) => socket.url().endsWith('/_hmr'),
|
|
28
|
+
timeout: 10000,
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
await page.goto('/', { waitUntil: 'networkidle' });
|
|
32
|
+
const socket = await socketPromise;
|
|
33
|
+
|
|
34
|
+
expect(socket.url()).toMatch(/\/_hmr$/);
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test('should fall back to a full page reload when raw CSS file changes', async ({ page }) => {
|
|
38
|
+
const originalCss = readFileSync(TEST_CSS_FILE, 'utf-8');
|
|
39
|
+
const title = page.locator('.main-title').first();
|
|
40
|
+
|
|
41
|
+
try {
|
|
42
|
+
await page.goto('/', { waitUntil: 'networkidle' });
|
|
43
|
+
await expect(title).toBeVisible();
|
|
44
|
+
|
|
45
|
+
const initialColor = await title.evaluate((el) => getComputedStyle(el).color);
|
|
46
|
+
expect(initialColor).toBeTruthy();
|
|
47
|
+
|
|
48
|
+
const modifiedCss = originalCss.replace('.main-title {', '.main-title {\n\tcolor: rgb(255, 0, 0);');
|
|
49
|
+
const reloadPromise = page.waitForEvent('framenavigated', {
|
|
50
|
+
predicate: (frame) => frame === page.mainFrame(),
|
|
51
|
+
timeout: 10000,
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
await writeFile(TEST_CSS_FILE, modifiedCss, { flush: true });
|
|
55
|
+
|
|
56
|
+
await reloadPromise;
|
|
57
|
+
await page.waitForLoadState('networkidle');
|
|
58
|
+
|
|
59
|
+
await expect
|
|
60
|
+
.poll(async () => title.evaluate((el) => getComputedStyle(el).color), { timeout: 10000 })
|
|
61
|
+
.toBe('rgb(255, 0, 0)');
|
|
62
|
+
} finally {
|
|
63
|
+
await writeFile(TEST_CSS_FILE, originalCss, { flush: true });
|
|
64
|
+
}
|
|
65
|
+
});
|
|
66
|
+
});
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default HMR Strategy
|
|
3
|
+
*
|
|
4
|
+
* Fallback strategy for file types that don't have specialized handling.
|
|
5
|
+
* Triggers a full page reload to ensure changes are reflected.
|
|
6
|
+
*
|
|
7
|
+
* @module
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { HmrStrategy, HmrStrategyType, type HmrAction } from '../hmr-strategy.ts';
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Default fallback strategy for unhandled file types.
|
|
14
|
+
*
|
|
15
|
+
* This strategy matches all files and triggers a full page reload.
|
|
16
|
+
* It has the lowest priority (FALLBACK) and acts as a catch-all when
|
|
17
|
+
* no other strategy matches the changed file.
|
|
18
|
+
*
|
|
19
|
+
* @remarks
|
|
20
|
+
* This strategy ensures that all file changes result in some action,
|
|
21
|
+
* even if we don't have specialized handling for that file type.
|
|
22
|
+
*
|
|
23
|
+
* @example
|
|
24
|
+
* ```typescript
|
|
25
|
+
* const strategy = new DefaultHmrStrategy();
|
|
26
|
+
* const action = await strategy.process('/path/to/unknown.file');
|
|
27
|
+
* ```
|
|
28
|
+
*/
|
|
29
|
+
export class DefaultHmrStrategy extends HmrStrategy {
|
|
30
|
+
readonly type = HmrStrategyType.FALLBACK;
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Matches all file paths.
|
|
34
|
+
*
|
|
35
|
+
* @param _filePath - Absolute path to the changed file (unused)
|
|
36
|
+
* @returns Always returns true as this is a catch-all strategy
|
|
37
|
+
*/
|
|
38
|
+
matches(_filePath: string): boolean {
|
|
39
|
+
return true;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Processes a file change by triggering a full page reload.
|
|
44
|
+
*
|
|
45
|
+
* @param filePath - Absolute path to the changed file
|
|
46
|
+
* @returns Action to broadcast a reload event
|
|
47
|
+
*/
|
|
48
|
+
async process(filePath: string): Promise<HmrAction> {
|
|
49
|
+
return {
|
|
50
|
+
type: 'broadcast',
|
|
51
|
+
events: [
|
|
52
|
+
{
|
|
53
|
+
type: 'reload',
|
|
54
|
+
path: filePath,
|
|
55
|
+
message: 'fallback-strategy',
|
|
56
|
+
},
|
|
57
|
+
],
|
|
58
|
+
};
|
|
59
|
+
}
|
|
60
|
+
}
|