@ecopages/core 0.2.0-alpha.4 → 0.2.0-alpha.6
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 +213 -12
- package/package.json +100 -188
- package/src/adapters/README.md +39 -0
- package/src/adapters/bun/hmr-manager.test.ts +267 -0
- package/src/adapters/bun/hmr-manager.ts +181 -68
- package/src/adapters/bun/index.ts +1 -2
- package/src/adapters/bun/server-adapter.ts +41 -34
- package/src/adapters/bun/server-lifecycle.ts +40 -70
- package/src/adapters/index.ts +1 -1
- package/src/adapters/node/bootstrap-dependency-resolver.test.ts +282 -0
- package/src/adapters/node/bootstrap-dependency-resolver.ts +301 -0
- package/src/adapters/node/index.ts +7 -0
- package/src/adapters/node/node-client-bridge.test.ts +198 -0
- package/src/adapters/node/node-hmr-manager.test.ts +322 -0
- package/src/adapters/node/node-hmr-manager.ts +208 -116
- package/src/adapters/node/runtime-adapter.test.ts +868 -0
- package/src/adapters/node/runtime-adapter.ts +439 -0
- package/src/adapters/node/server-adapter.ts +31 -104
- package/src/adapters/node/static-content-server.test.ts +60 -0
- package/src/adapters/node/static-content-server.ts +36 -0
- package/src/adapters/node/write-runtime-manifest.ts +38 -0
- package/src/adapters/shared/api-response.test.ts +97 -0
- package/src/{define-api-handler.ts → adapters/shared/define-api-handler.ts} +1 -1
- package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +7 -1
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +90 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +6 -2
- package/src/adapters/shared/fs-server-response-factory.test.ts +187 -0
- package/src/adapters/shared/fs-server-response-matcher.test.ts +286 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +17 -10
- 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 +196 -0
- package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
- package/src/adapters/shared/render-context.test.ts +146 -0
- package/src/adapters/shared/render-context.ts +21 -6
- 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 +51 -4
- package/src/adapters/shared/server-route-handler.test.ts +110 -0
- package/src/adapters/shared/server-route-handler.ts +5 -18
- package/src/adapters/shared/server-static-builder.test.ts +316 -0
- package/src/adapters/shared/server-static-builder.ts +92 -8
- package/src/build/README.md +101 -0
- package/src/build/build-adapter-serialization.test.ts +268 -0
- package/src/build/build-adapter.test.ts +815 -0
- package/src/build/build-adapter.ts +235 -6
- package/src/build/build-manifest.ts +54 -0
- package/src/build/dev-build-coordinator.ts +221 -0
- package/src/build/esbuild-build-adapter.ts +132 -83
- package/src/build/runtime-build-executor.ts +34 -0
- package/src/build/runtime-specifier-alias-plugin.test.ts +43 -0
- package/src/build/runtime-specifier-alias-plugin.ts +58 -0
- package/src/config/README.md +33 -0
- package/src/config/config-builder.test.ts +410 -0
- package/src/config/config-builder.ts +281 -49
- package/src/constants.ts +15 -0
- package/src/declarations.d.ts +18 -13
- package/src/eco/README.md +70 -16
- package/src/eco/component-render-context.ts +39 -17
- package/src/eco/eco.test.ts +678 -0
- package/src/eco/eco.ts +29 -8
- package/src/eco/eco.types.ts +20 -1
- package/src/eco/eco.utils.test.ts +124 -0
- package/src/eco/global-injector-map.test.ts +42 -0
- package/src/eco/lazy-injector-map.test.ts +66 -0
- package/src/eco/module-dependencies.test.ts +30 -0
- package/src/errors/http-error.test.ts +134 -0
- package/src/global/utils.test.ts +12 -0
- package/src/hmr/README.md +26 -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 +38 -7
- package/src/hmr/hmr-strategy.test.ts +124 -0
- package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
- package/src/hmr/hmr.test.e2e.ts +29 -38
- package/src/hmr/strategies/js-hmr-strategy.test.ts +335 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +115 -115
- package/src/index.ts +1 -1
- package/src/integrations/ghtml/ghtml-renderer.test.ts +63 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +4 -1
- package/src/internal-types.ts +39 -19
- package/src/plugins/README.md +34 -0
- package/src/plugins/alias-resolver-plugin.test.ts +41 -0
- package/src/plugins/alias-resolver-plugin.ts +21 -3
- package/src/plugins/eco-component-meta-plugin.test.ts +380 -0
- package/src/plugins/eco-component-meta-plugin.ts +10 -3
- package/src/plugins/integration-plugin.test.ts +111 -0
- package/src/plugins/integration-plugin.ts +45 -3
- package/src/plugins/processor.test.ts +148 -0
- package/src/plugins/processor.ts +22 -2
- package/src/plugins/runtime-capability.ts +14 -0
- package/src/public-types.ts +73 -16
- package/src/route-renderer/GRAPH.md +16 -20
- package/src/route-renderer/README.md +8 -21
- package/src/route-renderer/component-graph/component-graph-executor.test.ts +41 -0
- package/src/route-renderer/component-graph/component-graph.test.ts +63 -0
- package/src/route-renderer/component-graph/component-marker.test.ts +73 -0
- package/src/route-renderer/component-graph/component-reference.ts +29 -0
- package/src/route-renderer/component-graph/marker-graph-resolver.test.ts +135 -0
- package/src/route-renderer/{marker-graph-resolver.ts → component-graph/marker-graph-resolver.ts} +11 -9
- package/src/route-renderer/orchestration/integration-renderer.test.ts +936 -0
- package/src/route-renderer/{integration-renderer.ts → orchestration/integration-renderer.ts} +113 -19
- package/src/route-renderer/orchestration/render-execution.service.test.ts +97 -0
- package/src/route-renderer/{render-execution.service.ts → orchestration/render-execution.service.ts} +109 -37
- package/src/route-renderer/orchestration/render-preparation.service.test.ts +235 -0
- package/src/route-renderer/{render-preparation.service.ts → orchestration/render-preparation.service.ts} +127 -9
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +345 -0
- package/src/route-renderer/{dependency-resolver.ts → page-loading/dependency-resolver.ts} +28 -12
- package/src/route-renderer/page-loading/page-module-loader.test.ts +96 -0
- package/src/route-renderer/{page-module-loader.ts → page-loading/page-module-loader.ts} +49 -21
- package/src/route-renderer/route-renderer.ts +36 -1
- package/src/router/README.md +26 -0
- package/src/router/client/link-intent.d.ts +53 -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 +433 -0
- package/src/router/server/fs-router-scanner.test.ts +83 -0
- package/src/router/{fs-router-scanner.ts → server/fs-router-scanner.ts} +12 -10
- package/src/router/server/fs-router.test.ts +214 -0
- package/src/router/{fs-router.ts → server/fs-router.ts} +2 -2
- package/src/services/README.md +29 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +385 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.ts +101 -6
- package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.ts +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.ts +2 -1
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +72 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +95 -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/{asset-processing-service → assets/asset-processing-service}/index.ts +2 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.ts +1 -1
- package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.ts +11 -5
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-script-processor.ts +17 -27
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +286 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.ts +3 -3
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.ts +5 -4
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +199 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.ts +4 -1
- package/src/services/assets/browser-bundle.service.test.ts +36 -0
- package/src/services/assets/browser-bundle.service.ts +53 -0
- package/src/services/cache/index.ts +3 -3
- package/src/services/cache/memory-cache-store.test.ts +225 -0
- package/src/services/cache/memory-cache-store.ts +1 -1
- package/src/services/cache/page-cache-service.test.ts +175 -0
- package/src/services/cache/page-cache-service.ts +3 -3
- package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
- package/src/services/{page-request-cache-coordinator.service.ts → cache/page-request-cache-coordinator.service.ts} +9 -6
- package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
- package/src/services/html/html-rewriter-provider.service.ts +103 -0
- package/src/services/html/html-transformer.service.test.ts +378 -0
- package/src/services/html/html-transformer.service.ts +279 -0
- package/src/services/invalidation/development-invalidation.service.test.ts +77 -0
- package/src/services/invalidation/development-invalidation.service.ts +261 -0
- package/src/services/module-loading/app-server-module-transpiler.service.ts +52 -0
- package/src/services/module-loading/page-module-import.service.test.ts +253 -0
- package/src/services/module-loading/page-module-import.service.ts +200 -0
- package/src/services/module-loading/server-loader.service.test.ts +161 -0
- package/src/services/module-loading/server-loader.service.ts +130 -0
- package/src/services/module-loading/server-module-transpiler.service.test.ts +115 -0
- package/src/services/module-loading/server-module-transpiler.service.ts +105 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.test.ts +95 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.ts +101 -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/{schema-validation-service.ts → validation/schema-validation-service.ts} +1 -1
- package/src/static-site-generator/README.md +26 -0
- package/src/static-site-generator/static-site-generator.test.ts +307 -0
- package/src/static-site-generator/static-site-generator.ts +109 -6
- package/src/utils/deep-merge.test.ts +114 -0
- package/src/utils/invariant.test.ts +22 -0
- package/src/utils/path-utils.test.ts +15 -0
- package/src/utils/resolve-work-dir.ts +45 -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 +1 -1
- package/src/watchers/project-watcher.test.ts +678 -0
- package/src/watchers/project-watcher.ts +130 -111
- package/CHANGELOG.md +0 -91
- package/src/adapters/abstract/application-adapter.d.ts +0 -168
- package/src/adapters/abstract/application-adapter.js +0 -109
- 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 -60
- package/src/adapters/bun/create-app.js +0 -117
- package/src/adapters/bun/define-api-handler.d.ts +0 -61
- package/src/adapters/bun/define-api-handler.js +0 -15
- package/src/adapters/bun/define-api-handler.ts +0 -114
- package/src/adapters/bun/hmr-manager.d.ts +0 -85
- package/src/adapters/bun/hmr-manager.js +0 -240
- package/src/adapters/bun/index.d.ts +0 -3
- 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 -368
- package/src/adapters/bun/server-lifecycle.d.ts +0 -52
- package/src/adapters/bun/server-lifecycle.js +0 -120
- package/src/adapters/index.d.ts +0 -6
- package/src/adapters/index.js +0 -14
- package/src/adapters/node/create-app.d.ts +0 -21
- package/src/adapters/node/create-app.js +0 -143
- package/src/adapters/node/index.d.ts +0 -4
- package/src/adapters/node/index.js +0 -8
- 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 -63
- package/src/adapters/node/node-hmr-manager.js +0 -237
- package/src/adapters/node/server-adapter.d.ts +0 -190
- package/src/adapters/node/server-adapter.js +0 -420
- package/src/adapters/node/static-content-server.d.ts +0 -24
- package/src/adapters/node/static-content-server.js +0 -166
- 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/explicit-static-route-matcher.d.ts +0 -38
- package/src/adapters/shared/explicit-static-route-matcher.js +0 -100
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
- package/src/adapters/shared/file-route-middleware-pipeline.js +0 -98
- 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 -71
- package/src/adapters/shared/fs-server-response-matcher.js +0 -155
- package/src/adapters/shared/render-context.d.ts +0 -14
- package/src/adapters/shared/render-context.js +0 -69
- package/src/adapters/shared/server-adapter.d.ts +0 -87
- package/src/adapters/shared/server-adapter.js +0 -353
- package/src/adapters/shared/server-route-handler.d.ts +0 -89
- package/src/adapters/shared/server-route-handler.js +0 -120
- package/src/adapters/shared/server-static-builder.d.ts +0 -38
- package/src/adapters/shared/server-static-builder.js +0 -46
- package/src/build/build-adapter.d.ts +0 -74
- package/src/build/build-adapter.js +0 -54
- package/src/build/build-types.d.ts +0 -57
- package/src/build/build-types.js +0 -0
- package/src/build/esbuild-build-adapter.d.ts +0 -69
- package/src/build/esbuild-build-adapter.js +0 -390
- package/src/config/config-builder.d.ts +0 -227
- package/src/config/config-builder.js +0 -392
- package/src/constants.d.ts +0 -32
- package/src/constants.js +0 -21
- package/src/create-app.d.ts +0 -17
- package/src/create-app.js +0 -66
- package/src/define-api-handler.d.ts +0 -25
- package/src/define-api-handler.js +0 -15
- package/src/dev/sc-server.d.ts +0 -30
- package/src/dev/sc-server.js +0 -111
- package/src/eco/component-render-context.d.ts +0 -105
- package/src/eco/component-render-context.js +0 -77
- package/src/eco/eco.d.ts +0 -9
- package/src/eco/eco.js +0 -110
- package/src/eco/eco.types.d.ts +0 -170
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.utils.d.ts +0 -40
- package/src/eco/eco.utils.js +0 -40
- 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.d.ts +0 -2
- 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 -10
- package/src/hmr/client/hmr-runtime.js +0 -86
- package/src/hmr/hmr-strategy.d.ts +0 -159
- package/src/hmr/hmr-strategy.js +0 -29
- package/src/hmr/hmr.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.test.e2e.js +0 -50
- 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 -136
- package/src/hmr/strategies/js-hmr-strategy.js +0 -188
- package/src/index.browser.d.ts +0 -3
- package/src/index.browser.js +0 -4
- package/src/index.d.ts +0 -5
- package/src/index.js +0 -10
- package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -15
- package/src/integrations/ghtml/ghtml-renderer.js +0 -60
- package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -20
- package/src/integrations/ghtml/ghtml.plugin.js +0 -21
- package/src/internal-types.d.ts +0 -200
- package/src/internal-types.js +0 -0
- package/src/plugins/alias-resolver-plugin.d.ts +0 -2
- package/src/plugins/alias-resolver-plugin.js +0 -39
- package/src/plugins/eco-component-meta-plugin.d.ts +0 -95
- package/src/plugins/eco-component-meta-plugin.js +0 -157
- package/src/plugins/integration-plugin.d.ts +0 -102
- package/src/plugins/integration-plugin.js +0 -100
- package/src/plugins/processor.d.ts +0 -82
- package/src/plugins/processor.js +0 -122
- package/src/public-types.d.ts +0 -1098
- package/src/public-types.js +0 -0
- 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.d.ts +0 -42
- package/src/route-renderer/component-graph.js +0 -72
- package/src/route-renderer/component-marker.d.ts +0 -52
- package/src/route-renderer/component-marker.js +0 -46
- package/src/route-renderer/dependency-resolver.d.ts +0 -24
- package/src/route-renderer/dependency-resolver.js +0 -428
- 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/marker-graph-resolver.d.ts +0 -76
- package/src/route-renderer/marker-graph-resolver.js +0 -93
- package/src/route-renderer/page-module-loader.d.ts +0 -61
- package/src/route-renderer/page-module-loader.js +0 -102
- 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-preparation.service.d.ts +0 -112
- package/src/route-renderer/render-preparation.service.js +0 -243
- package/src/route-renderer/route-renderer.d.ts +0 -26
- package/src/route-renderer/route-renderer.js +0 -68
- package/src/router/fs-router-scanner.d.ts +0 -41
- package/src/router/fs-router-scanner.js +0 -155
- package/src/router/fs-router.d.ts +0 -26
- package/src/router/fs-router.js +0 -100
- package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
- package/src/services/asset-processing-service/asset-processing.service.js +0 -250
- package/src/services/asset-processing-service/asset.factory.d.ts +0 -17
- package/src/services/asset-processing-service/asset.factory.js +0 -82
- package/src/services/asset-processing-service/assets.types.d.ts +0 -88
- package/src/services/asset-processing-service/assets.types.js +0 -0
- 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/processor.interface.d.ts +0 -22
- package/src/services/asset-processing-service/processor.interface.js +0 -6
- package/src/services/asset-processing-service/processor.registry.d.ts +0 -8
- package/src/services/asset-processing-service/processor.registry.js +0 -15
- package/src/services/asset-processing-service/processors/base/base-processor.d.ts +0 -24
- package/src/services/asset-processing-service/processors/base/base-processor.js +0 -59
- 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/index.d.ts +0 -5
- package/src/services/asset-processing-service/processors/index.js +0 -5
- package/src/services/asset-processing-service/processors/script/content-script.processor.d.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/file-script.processor.d.ts +0 -8
- package/src/services/asset-processing-service/processors/script/file-script.processor.js +0 -76
- 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/stylesheet/content-stylesheet.processor.d.ts +0 -5
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -63
- 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/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.d.ts +0 -75
- package/src/services/page-request-cache-coordinator.service.js +0 -107
- package/src/services/schema-validation-service.d.ts +0 -122
- package/src/services/schema-validation-service.js +0 -101
- package/src/services/validation/standard-schema.types.d.ts +0 -65
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/static-site-generator/static-site-generator.d.ts +0 -57
- package/src/static-site-generator/static-site-generator.js +0 -272
- package/src/utils/css.d.ts +0 -1
- package/src/utils/css.js +0 -7
- 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.d.ts +0 -1
- 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 -24
- package/src/utils/parse-cli-args.js +0 -47
- package/src/utils/path-utils.module.d.ts +0 -5
- package/src/utils/path-utils.module.js +0 -14
- 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 -125
- package/src/watchers/project-watcher.js +0 -265
- package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
- package/src/watchers/project-watcher.test-helpers.js +0 -52
- /package/src/route-renderer/{component-graph-executor.ts → component-graph/component-graph-executor.ts} +0 -0
- /package/src/route-renderer/{component-graph.ts → component-graph/component-graph.ts} +0 -0
- /package/src/route-renderer/{component-marker.ts → component-graph/component-marker.ts} +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/index.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.ts +0 -0
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
3
|
+
import { appLogger } from '../../global/app-logger.ts';
|
|
4
|
+
import { RESOLVED_ASSETS_DIR } from '../../constants.ts';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Shared runtime state used while registering HMR-owned entrypoints.
|
|
8
|
+
*/
|
|
9
|
+
export interface HmrEntrypointRegistrarOptions {
|
|
10
|
+
/** Absolute source directory used to derive the emitted HMR path. */
|
|
11
|
+
srcDir: string;
|
|
12
|
+
/** Absolute distribution directory where HMR outputs are written. */
|
|
13
|
+
distDir: string;
|
|
14
|
+
/** In-flight registrations keyed by normalized absolute entrypoint path. */
|
|
15
|
+
entrypointRegistrations: Map<string, Promise<string>>;
|
|
16
|
+
/** Stable entrypoint-to-output mapping retained once an entrypoint is registered. */
|
|
17
|
+
watchedFiles: Map<string, string>;
|
|
18
|
+
/** Runtime-specific cleanup invoked when an entrypoint registration fails. */
|
|
19
|
+
clearFailedRegistration?: (entrypointPath: string) => void;
|
|
20
|
+
/** Development-only guardrail for integrations that never finish producing output. */
|
|
21
|
+
registrationTimeoutMs: number;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Runtime-specific hooks required to materialize a single HMR entrypoint.
|
|
26
|
+
*/
|
|
27
|
+
export interface HmrEntrypointRegistrationOptions {
|
|
28
|
+
/**
|
|
29
|
+
* Emits the browser-consumable HMR artifact for an entrypoint.
|
|
30
|
+
*/
|
|
31
|
+
emit(entrypointPath: string, outputPath: string): Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* Creates the runtime-specific error raised when the emit hook completes without producing output.
|
|
34
|
+
*/
|
|
35
|
+
getMissingOutputError(entrypointPath: string, outputPath: string): Error;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Coordinates the shared HMR entrypoint registration lifecycle for both Node and Bun managers.
|
|
40
|
+
*
|
|
41
|
+
* The registrar owns the cross-runtime policy: normalize entrypoint identities, dedupe concurrent
|
|
42
|
+
* registrations, derive the emitted `_hmr` output path, clear stale output before rebuilding, and
|
|
43
|
+
* apply the development timeout that prevents unresolved registrations from hanging navigation.
|
|
44
|
+
* Runtime-specific managers remain responsible for the actual emit step and any cleanup outside
|
|
45
|
+
* this shared registration flow.
|
|
46
|
+
*/
|
|
47
|
+
export class HmrEntrypointRegistrar {
|
|
48
|
+
private readonly options: HmrEntrypointRegistrarOptions;
|
|
49
|
+
|
|
50
|
+
constructor(options: HmrEntrypointRegistrarOptions) {
|
|
51
|
+
this.options = options;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Registers a single source entrypoint and returns the browser URL for its emitted HMR module.
|
|
56
|
+
*
|
|
57
|
+
* Concurrent requests for the same normalized entrypoint share the same in-flight promise so the
|
|
58
|
+
* integration only builds once per registration cycle.
|
|
59
|
+
*/
|
|
60
|
+
async registerEntrypoint(
|
|
61
|
+
entrypointPath: string,
|
|
62
|
+
registrationOptions: HmrEntrypointRegistrationOptions,
|
|
63
|
+
): Promise<string> {
|
|
64
|
+
const normalizedEntrypoint = path.resolve(entrypointPath);
|
|
65
|
+
const existingRegistration = this.options.entrypointRegistrations.get(normalizedEntrypoint);
|
|
66
|
+
if (existingRegistration) {
|
|
67
|
+
return await this.awaitEntrypointRegistration(existingRegistration, normalizedEntrypoint);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const registration = this.registerEntrypointInternal(normalizedEntrypoint, registrationOptions);
|
|
71
|
+
this.options.entrypointRegistrations.set(normalizedEntrypoint, registration);
|
|
72
|
+
|
|
73
|
+
try {
|
|
74
|
+
return await this.awaitEntrypointRegistration(registration, normalizedEntrypoint);
|
|
75
|
+
} catch (error) {
|
|
76
|
+
this.options.clearFailedRegistration?.(normalizedEntrypoint);
|
|
77
|
+
throw error;
|
|
78
|
+
} finally {
|
|
79
|
+
this.options.entrypointRegistrations.delete(normalizedEntrypoint);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
private async registerEntrypointInternal(
|
|
84
|
+
entrypointPath: string,
|
|
85
|
+
registrationOptions: HmrEntrypointRegistrationOptions,
|
|
86
|
+
): Promise<string> {
|
|
87
|
+
if (this.options.watchedFiles.has(entrypointPath)) {
|
|
88
|
+
return this.options.watchedFiles.get(entrypointPath)!;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const { outputPath, outputUrl } = this.getEntrypointOutput(entrypointPath);
|
|
92
|
+
|
|
93
|
+
this.options.watchedFiles.set(entrypointPath, outputUrl);
|
|
94
|
+
this.removeStaleEntrypointOutput(outputPath);
|
|
95
|
+
|
|
96
|
+
await registrationOptions.emit(entrypointPath, outputPath);
|
|
97
|
+
|
|
98
|
+
if (!fileSystem.exists(outputPath)) {
|
|
99
|
+
throw registrationOptions.getMissingOutputError(entrypointPath, outputPath);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return outputUrl;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
private async awaitEntrypointRegistration(registration: Promise<string>, entrypointPath: string): Promise<string> {
|
|
106
|
+
if (process.env.NODE_ENV !== 'development') {
|
|
107
|
+
return await registration;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return await Promise.race([
|
|
111
|
+
registration,
|
|
112
|
+
new Promise<string>((_, reject) => {
|
|
113
|
+
setTimeout(() => {
|
|
114
|
+
reject(new Error(`[HMR] Timed out registering entrypoint: ${entrypointPath}`));
|
|
115
|
+
}, this.options.registrationTimeoutMs);
|
|
116
|
+
}),
|
|
117
|
+
]);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
private getEntrypointOutput(entrypointPath: string): { outputPath: string; outputUrl: string } {
|
|
121
|
+
const relativePath = path.relative(this.options.srcDir, entrypointPath);
|
|
122
|
+
const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
|
|
123
|
+
const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
|
|
124
|
+
const urlPath = encodedPathJs.split(path.sep).join('/');
|
|
125
|
+
|
|
126
|
+
return {
|
|
127
|
+
outputUrl: `/${path.join(RESOLVED_ASSETS_DIR, '_hmr', urlPath)}`,
|
|
128
|
+
outputPath: path.join(this.options.distDir, urlPath),
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
private removeStaleEntrypointOutput(outputPath: string): void {
|
|
133
|
+
if (!fileSystem.exists(outputPath)) {
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
try {
|
|
138
|
+
fileSystem.remove(outputPath);
|
|
139
|
+
} catch (error) {
|
|
140
|
+
appLogger.warn(
|
|
141
|
+
`[HMR] Failed to remove stale entrypoint output ${outputPath}: ${error instanceof Error ? error.message : String(error)}`,
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
private encodeDynamicSegments(filepath: string): string {
|
|
147
|
+
return filepath.replace(/\[([^\]]+)\]/g, '_$1_');
|
|
148
|
+
}
|
|
149
|
+
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import type { IHmrManager } from '../../public-types';
|
|
2
|
+
|
|
3
|
+
const HMR_RUNTIME_IMPORT = "import '/_hmr_runtime.js'";
|
|
4
|
+
const HMR_RUNTIME_SCRIPT = `<script type="module">${HMR_RUNTIME_IMPORT};</script>`;
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Returns whether a response is HTML and therefore eligible for development HMR
|
|
8
|
+
* runtime injection.
|
|
9
|
+
*/
|
|
10
|
+
export function isHtmlResponse(response: Response): boolean {
|
|
11
|
+
const contentType = response.headers.get('Content-Type');
|
|
12
|
+
return contentType !== null && contentType.startsWith('text/html');
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Returns whether HTML responses should receive the HMR runtime bootstrap.
|
|
17
|
+
*
|
|
18
|
+
* This is shared because filesystem page responses and adapter-level HTML
|
|
19
|
+
* responses flow through different layers, but both need identical injection
|
|
20
|
+
* behavior in watch mode.
|
|
21
|
+
*/
|
|
22
|
+
export function shouldInjectHmrHtmlResponse(watch: boolean, hmrManager?: Pick<IHmrManager, 'isEnabled'>): boolean {
|
|
23
|
+
return watch && hmrManager?.isEnabled() === true;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Injects the development HMR runtime script into an HTML response if it is not
|
|
28
|
+
* already present.
|
|
29
|
+
*
|
|
30
|
+
* The check is intentionally idempotent because an HTML response can pass
|
|
31
|
+
* through more than one development-layer wrapper before reaching the client.
|
|
32
|
+
*/
|
|
33
|
+
export async function injectHmrRuntimeIntoHtmlResponse(response: Response): Promise<Response> {
|
|
34
|
+
const html = await response.text();
|
|
35
|
+
if (html.includes(HMR_RUNTIME_IMPORT)) {
|
|
36
|
+
return new Response(html, {
|
|
37
|
+
status: response.status,
|
|
38
|
+
statusText: response.statusText,
|
|
39
|
+
headers: response.headers,
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
const updatedHtml = html.replace(/<\/html>/i, `${HMR_RUNTIME_SCRIPT}</html>`);
|
|
44
|
+
const headers = new Headers(response.headers);
|
|
45
|
+
headers.delete('Content-Length');
|
|
46
|
+
|
|
47
|
+
return new Response(updatedHtml, {
|
|
48
|
+
status: response.status,
|
|
49
|
+
statusText: response.statusText,
|
|
50
|
+
headers,
|
|
51
|
+
});
|
|
52
|
+
}
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
import os from 'node:os';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
import { afterEach, describe, test, vi } from 'vitest';
|
|
6
|
+
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
7
|
+
import { resolveInternalWorkDir } from '../../utils/resolve-work-dir.ts';
|
|
8
|
+
import { NodeHmrManager } from '../node/node-hmr-manager.ts';
|
|
9
|
+
import { HmrManager as BunHmrManager } from '../bun/hmr-manager.ts';
|
|
10
|
+
|
|
11
|
+
type SharedHmrManager = {
|
|
12
|
+
handleFileChange(filePath: string, options?: { broadcast?: boolean }): Promise<void>;
|
|
13
|
+
registerEntrypoint(entrypointPath: string): Promise<string>;
|
|
14
|
+
registerScriptEntrypoint(entrypointPath: string): Promise<string>;
|
|
15
|
+
registerSpecifierMap(map: Record<string, string>): void;
|
|
16
|
+
getWatchedFiles(): Map<string, string>;
|
|
17
|
+
getSpecifierMap(): Map<string, string>;
|
|
18
|
+
stop(): void;
|
|
19
|
+
appConfig: Awaited<ReturnType<ConfigBuilder['build']>>;
|
|
20
|
+
readonly runtimeName: string;
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
function withRuntimeName<T extends Omit<SharedHmrManager, 'runtimeName'>>(
|
|
24
|
+
manager: T,
|
|
25
|
+
runtimeName: SharedHmrManager['runtimeName'],
|
|
26
|
+
): T & Pick<SharedHmrManager, 'runtimeName'> {
|
|
27
|
+
return Object.assign(manager, { runtimeName });
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const tempRoots: string[] = [];
|
|
31
|
+
|
|
32
|
+
function createTempRoot(prefix: string): string {
|
|
33
|
+
const root = fs.mkdtempSync(path.join(os.tmpdir(), `${prefix}-`));
|
|
34
|
+
tempRoots.push(root);
|
|
35
|
+
return root;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
afterEach(() => {
|
|
39
|
+
for (const root of tempRoots.splice(0)) {
|
|
40
|
+
fs.rmSync(root, { recursive: true, force: true });
|
|
41
|
+
}
|
|
42
|
+
vi.restoreAllMocks();
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
const runtimes = [
|
|
46
|
+
{
|
|
47
|
+
name: 'node',
|
|
48
|
+
async create(rootDir: string): Promise<SharedHmrManager> {
|
|
49
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
50
|
+
return withRuntimeName(
|
|
51
|
+
new NodeHmrManager({
|
|
52
|
+
appConfig: config,
|
|
53
|
+
bridge: {
|
|
54
|
+
subscriberCount: 0,
|
|
55
|
+
broadcast: () => {},
|
|
56
|
+
} as any,
|
|
57
|
+
}),
|
|
58
|
+
'node',
|
|
59
|
+
);
|
|
60
|
+
},
|
|
61
|
+
},
|
|
62
|
+
{
|
|
63
|
+
name: 'bun',
|
|
64
|
+
async create(rootDir: string): Promise<SharedHmrManager> {
|
|
65
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
66
|
+
return withRuntimeName(
|
|
67
|
+
new BunHmrManager({
|
|
68
|
+
appConfig: config,
|
|
69
|
+
bridge: {
|
|
70
|
+
subscriberCount: 0,
|
|
71
|
+
broadcast: () => {},
|
|
72
|
+
subscribe: () => {},
|
|
73
|
+
unsubscribe: () => {},
|
|
74
|
+
} as any,
|
|
75
|
+
}),
|
|
76
|
+
'bun',
|
|
77
|
+
);
|
|
78
|
+
},
|
|
79
|
+
},
|
|
80
|
+
] as const;
|
|
81
|
+
|
|
82
|
+
function getEntrypointOutputPath(manager: SharedHmrManager, entrypointPath: string): string {
|
|
83
|
+
const relativePathJs = path
|
|
84
|
+
.relative(manager.appConfig.absolutePaths.srcDir, entrypointPath)
|
|
85
|
+
.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
|
|
86
|
+
const encodedPathJs = relativePathJs.replace(/\[([^\]]+)\]/g, '_$1_');
|
|
87
|
+
return path.join(resolveInternalWorkDir(manager.appConfig), 'assets', '_hmr', encodedPathJs);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
describe.each(runtimes)('shared HMR manager contract: $name', ({ create }) => {
|
|
91
|
+
test('shares one in-flight entrypoint registration across concurrent callers', async () => {
|
|
92
|
+
const rootDir = createTempRoot('ecopages-hmr-contract-register');
|
|
93
|
+
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
94
|
+
fs.mkdirSync(pagesDir, { recursive: true });
|
|
95
|
+
|
|
96
|
+
const entrypointPath = path.join(pagesDir, 'react-lab.tsx');
|
|
97
|
+
fs.writeFileSync(entrypointPath, 'export default function Page() { return null; }', 'utf8');
|
|
98
|
+
|
|
99
|
+
const manager = await create(rootDir);
|
|
100
|
+
const outputPath = getEntrypointOutputPath(manager, entrypointPath);
|
|
101
|
+
|
|
102
|
+
const handleFileChange = vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {
|
|
103
|
+
await new Promise((resolve) => setTimeout(resolve, 25));
|
|
104
|
+
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
105
|
+
fs.writeFileSync(outputPath, 'export default 1;', 'utf8');
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
const [firstUrl, secondUrl] = await Promise.all([
|
|
109
|
+
manager.registerEntrypoint(entrypointPath),
|
|
110
|
+
manager.registerEntrypoint(entrypointPath),
|
|
111
|
+
]);
|
|
112
|
+
|
|
113
|
+
assert.equal(firstUrl, '/assets/_hmr/pages/react-lab.js');
|
|
114
|
+
assert.equal(secondUrl, '/assets/_hmr/pages/react-lab.js');
|
|
115
|
+
assert.equal(handleFileChange.mock.calls.length, 1);
|
|
116
|
+
|
|
117
|
+
manager.stop();
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
test('fails strict page registration when no integration emits output', async () => {
|
|
121
|
+
const rootDir = createTempRoot('ecopages-hmr-contract-strict-fail');
|
|
122
|
+
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
123
|
+
fs.mkdirSync(pagesDir, { recursive: true });
|
|
124
|
+
|
|
125
|
+
const entrypointPath = path.join(pagesDir, 'react-content.mdx');
|
|
126
|
+
fs.writeFileSync(entrypointPath, '# Hello', 'utf8');
|
|
127
|
+
|
|
128
|
+
const manager = await create(rootDir);
|
|
129
|
+
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {});
|
|
130
|
+
|
|
131
|
+
await assert.rejects(() => manager.registerEntrypoint(entrypointPath), /Integration failed to emit entrypoint/);
|
|
132
|
+
assert.equal(manager.getWatchedFiles().has(path.resolve(entrypointPath)), false);
|
|
133
|
+
|
|
134
|
+
manager.stop();
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
test('uses the generic build path for explicit script entrypoints', async () => {
|
|
138
|
+
const rootDir = createTempRoot('ecopages-hmr-contract-script-register');
|
|
139
|
+
const srcDir = path.join(rootDir, 'src');
|
|
140
|
+
fs.mkdirSync(srcDir, { recursive: true });
|
|
141
|
+
|
|
142
|
+
const entrypointPath = path.join(srcDir, 'script.ts');
|
|
143
|
+
fs.writeFileSync(entrypointPath, 'console.log("hello");', 'utf8');
|
|
144
|
+
|
|
145
|
+
const manager = await create(rootDir);
|
|
146
|
+
const outputPath = path.join(resolveInternalWorkDir(manager.appConfig), 'assets', '_hmr', 'script.js');
|
|
147
|
+
const buildCalls: string[] = [];
|
|
148
|
+
manager.appConfig.runtime!.buildExecutor = {
|
|
149
|
+
build: vi.fn(async (options) => {
|
|
150
|
+
buildCalls.push(options.entrypoints[0] as string);
|
|
151
|
+
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
152
|
+
fs.writeFileSync(outputPath, 'fresh-output', 'utf8');
|
|
153
|
+
return {
|
|
154
|
+
success: true,
|
|
155
|
+
logs: [],
|
|
156
|
+
outputs: [{ path: outputPath }],
|
|
157
|
+
};
|
|
158
|
+
}),
|
|
159
|
+
};
|
|
160
|
+
|
|
161
|
+
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {});
|
|
162
|
+
|
|
163
|
+
const outputUrl = await manager.registerScriptEntrypoint(entrypointPath);
|
|
164
|
+
|
|
165
|
+
assert.equal(outputUrl, '/assets/_hmr/script.js');
|
|
166
|
+
assert.deepEqual(buildCalls, [entrypointPath]);
|
|
167
|
+
assert.equal(fs.readFileSync(outputPath, 'utf8'), 'fresh-output');
|
|
168
|
+
|
|
169
|
+
manager.stop();
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
test('stop clears retained registration state', async () => {
|
|
173
|
+
const rootDir = createTempRoot('ecopages-hmr-contract-stop');
|
|
174
|
+
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
175
|
+
fs.mkdirSync(pagesDir, { recursive: true });
|
|
176
|
+
|
|
177
|
+
const entrypointPath = path.join(pagesDir, 'react-content.tsx');
|
|
178
|
+
fs.writeFileSync(entrypointPath, 'export default function Page() { return null; }', 'utf8');
|
|
179
|
+
|
|
180
|
+
const manager = await create(rootDir);
|
|
181
|
+
const outputPath = getEntrypointOutputPath(manager, entrypointPath);
|
|
182
|
+
|
|
183
|
+
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {
|
|
184
|
+
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
185
|
+
fs.writeFileSync(outputPath, 'export default 1;', 'utf8');
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
manager.registerSpecifierMap({ react: '/assets/vendors/react.js' });
|
|
189
|
+
await manager.registerEntrypoint(entrypointPath);
|
|
190
|
+
|
|
191
|
+
manager.stop();
|
|
192
|
+
|
|
193
|
+
assert.equal(manager.getWatchedFiles().size, 0);
|
|
194
|
+
assert.equal(manager.getSpecifierMap().size, 0);
|
|
195
|
+
});
|
|
196
|
+
});
|
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
import os from 'node:os';
|
|
4
|
+
import path from 'node:path';
|
|
5
|
+
import { afterEach, describe, test } from 'vitest';
|
|
6
|
+
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
7
|
+
import { HmrStrategy, HmrStrategyType, type HmrAction } from '../../hmr/hmr-strategy.ts';
|
|
8
|
+
import type { ClientBridgeEvent } from '../../public-types.ts';
|
|
9
|
+
import { HmrManager as BunHmrManager } from '../bun/hmr-manager.ts';
|
|
10
|
+
import { NodeHmrManager } from '../node/node-hmr-manager.ts';
|
|
11
|
+
|
|
12
|
+
class FakeHmrStrategy extends HmrStrategy {
|
|
13
|
+
readonly type: HmrStrategyType;
|
|
14
|
+
private readonly _matches: (filePath: string) => boolean;
|
|
15
|
+
private readonly _action: HmrAction;
|
|
16
|
+
|
|
17
|
+
constructor(type: HmrStrategyType, matchFn: (f: string) => boolean, action: HmrAction) {
|
|
18
|
+
super();
|
|
19
|
+
this.type = type;
|
|
20
|
+
this._matches = matchFn;
|
|
21
|
+
this._action = action;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
matches(filePath: string): boolean {
|
|
25
|
+
return this._matches(filePath);
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
async process(_filePath: string): Promise<HmrAction> {
|
|
29
|
+
return this._action;
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
type BridgeSpy = {
|
|
34
|
+
broadcasts: ClientBridgeEvent[];
|
|
35
|
+
bridge: {
|
|
36
|
+
subscriberCount: number;
|
|
37
|
+
broadcast(event: ClientBridgeEvent): void;
|
|
38
|
+
subscribe(): void;
|
|
39
|
+
unsubscribe(): void;
|
|
40
|
+
};
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
function createBridgeSpy(): BridgeSpy {
|
|
44
|
+
const broadcasts: ClientBridgeEvent[] = [];
|
|
45
|
+
const bridge = {
|
|
46
|
+
subscriberCount: 0,
|
|
47
|
+
broadcast(event: ClientBridgeEvent) {
|
|
48
|
+
broadcasts.push(event);
|
|
49
|
+
},
|
|
50
|
+
subscribe() {},
|
|
51
|
+
unsubscribe() {},
|
|
52
|
+
};
|
|
53
|
+
return { broadcasts, bridge };
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const tempRoots: string[] = [];
|
|
57
|
+
|
|
58
|
+
function createTempRoot(prefix: string): string {
|
|
59
|
+
const root = fs.mkdtempSync(path.join(os.tmpdir(), `${prefix}-`));
|
|
60
|
+
tempRoots.push(root);
|
|
61
|
+
return root;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
afterEach(() => {
|
|
65
|
+
for (const root of tempRoots.splice(0)) {
|
|
66
|
+
fs.rmSync(root, { recursive: true, force: true });
|
|
67
|
+
}
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
const runtimes = [
|
|
71
|
+
{
|
|
72
|
+
name: 'node',
|
|
73
|
+
async create(rootDir: string, bridgeSpy: BridgeSpy) {
|
|
74
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
75
|
+
return new NodeHmrManager({ appConfig: config, bridge: bridgeSpy.bridge as any });
|
|
76
|
+
},
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
name: 'bun',
|
|
80
|
+
async create(rootDir: string, bridgeSpy: BridgeSpy) {
|
|
81
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
82
|
+
return new BunHmrManager({ appConfig: config, bridge: bridgeSpy.bridge as any });
|
|
83
|
+
},
|
|
84
|
+
},
|
|
85
|
+
] as const;
|
|
86
|
+
|
|
87
|
+
describe.each(runtimes)('handleFileChange dispatch: $name', ({ create }) => {
|
|
88
|
+
test('CSS file change routes to DefaultHmrStrategy and broadcasts reload', async () => {
|
|
89
|
+
const rootDir = createTempRoot('ecopages-dispatch-css');
|
|
90
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
91
|
+
const spy = createBridgeSpy();
|
|
92
|
+
const manager = await create(rootDir, spy);
|
|
93
|
+
|
|
94
|
+
const cssFile = path.join(rootDir, 'src', 'styles', 'main.css');
|
|
95
|
+
await manager.handleFileChange(cssFile);
|
|
96
|
+
|
|
97
|
+
assert.equal(spy.broadcasts.length, 1);
|
|
98
|
+
assert.equal(spy.broadcasts[0].type, 'reload');
|
|
99
|
+
assert.equal(spy.broadcasts[0].path, cssFile);
|
|
100
|
+
|
|
101
|
+
manager.stop();
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
test('HTML file change routes to DefaultHmrStrategy and broadcasts reload', async () => {
|
|
105
|
+
const rootDir = createTempRoot('ecopages-dispatch-html');
|
|
106
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
107
|
+
const spy = createBridgeSpy();
|
|
108
|
+
const manager = await create(rootDir, spy);
|
|
109
|
+
|
|
110
|
+
const htmlFile = path.join(rootDir, 'src', 'pages', 'index.html');
|
|
111
|
+
await manager.handleFileChange(htmlFile);
|
|
112
|
+
|
|
113
|
+
assert.equal(spy.broadcasts.length, 1);
|
|
114
|
+
assert.equal(spy.broadcasts[0].type, 'reload');
|
|
115
|
+
assert.equal(spy.broadcasts[0].path, htmlFile);
|
|
116
|
+
|
|
117
|
+
manager.stop();
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
test('TS file with no registered entrypoints falls back to DefaultHmrStrategy reload', async () => {
|
|
121
|
+
const rootDir = createTempRoot('ecopages-dispatch-ts-fallback');
|
|
122
|
+
const srcDir = path.join(rootDir, 'src');
|
|
123
|
+
fs.mkdirSync(srcDir, { recursive: true });
|
|
124
|
+
const spy = createBridgeSpy();
|
|
125
|
+
const manager = await create(rootDir, spy);
|
|
126
|
+
|
|
127
|
+
const tsFile = path.join(srcDir, 'component.ts');
|
|
128
|
+
await manager.handleFileChange(tsFile);
|
|
129
|
+
|
|
130
|
+
assert.equal(spy.broadcasts.length, 1);
|
|
131
|
+
assert.equal(spy.broadcasts[0].type, 'reload');
|
|
132
|
+
|
|
133
|
+
manager.stop();
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
test('broadcast:false suppresses events even when the strategy returns a broadcast action', async () => {
|
|
137
|
+
const rootDir = createTempRoot('ecopages-dispatch-no-broadcast');
|
|
138
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
139
|
+
const spy = createBridgeSpy();
|
|
140
|
+
const manager = await create(rootDir, spy);
|
|
141
|
+
|
|
142
|
+
const cssFile = path.join(rootDir, 'src', 'main.css');
|
|
143
|
+
await manager.handleFileChange(cssFile, { broadcast: false });
|
|
144
|
+
|
|
145
|
+
assert.equal(spy.broadcasts.length, 0);
|
|
146
|
+
|
|
147
|
+
manager.stop();
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
test('INTEGRATION strategy wins over DefaultHmrStrategy for matched files', async () => {
|
|
151
|
+
const rootDir = createTempRoot('ecopages-dispatch-integration-priority');
|
|
152
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
153
|
+
const spy = createBridgeSpy();
|
|
154
|
+
const manager = await create(rootDir, spy);
|
|
155
|
+
|
|
156
|
+
const customFile = path.join(rootDir, 'src', 'component.jsx');
|
|
157
|
+
const integrationEvent: ClientBridgeEvent = {
|
|
158
|
+
type: 'update',
|
|
159
|
+
path: '/assets/_hmr/component.js',
|
|
160
|
+
timestamp: 1,
|
|
161
|
+
};
|
|
162
|
+
manager.registerStrategy(
|
|
163
|
+
new FakeHmrStrategy(HmrStrategyType.INTEGRATION, (f) => f === customFile, {
|
|
164
|
+
type: 'broadcast',
|
|
165
|
+
events: [integrationEvent],
|
|
166
|
+
}),
|
|
167
|
+
);
|
|
168
|
+
|
|
169
|
+
await manager.handleFileChange(customFile);
|
|
170
|
+
|
|
171
|
+
assert.equal(spy.broadcasts.length, 1);
|
|
172
|
+
assert.deepEqual(spy.broadcasts[0], integrationEvent);
|
|
173
|
+
|
|
174
|
+
manager.stop();
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
test('action.type none suppresses broadcast even when shouldBroadcast is true', async () => {
|
|
178
|
+
const rootDir = createTempRoot('ecopages-dispatch-none-action');
|
|
179
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
180
|
+
const spy = createBridgeSpy();
|
|
181
|
+
const manager = await create(rootDir, spy);
|
|
182
|
+
|
|
183
|
+
const customFile = path.join(rootDir, 'src', 'silent.ts');
|
|
184
|
+
manager.registerStrategy(
|
|
185
|
+
new FakeHmrStrategy(HmrStrategyType.INTEGRATION, (f) => f === customFile, { type: 'none' }),
|
|
186
|
+
);
|
|
187
|
+
|
|
188
|
+
await manager.handleFileChange(customFile);
|
|
189
|
+
|
|
190
|
+
assert.equal(spy.broadcasts.length, 0);
|
|
191
|
+
|
|
192
|
+
manager.stop();
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
test('all events returned by a strategy are each broadcast individually', async () => {
|
|
196
|
+
const rootDir = createTempRoot('ecopages-dispatch-multi-event');
|
|
197
|
+
fs.mkdirSync(path.join(rootDir, 'src'), { recursive: true });
|
|
198
|
+
const spy = createBridgeSpy();
|
|
199
|
+
const manager = await create(rootDir, spy);
|
|
200
|
+
|
|
201
|
+
const customFile = path.join(rootDir, 'src', 'multi.ts');
|
|
202
|
+
const events: ClientBridgeEvent[] = [
|
|
203
|
+
{ type: 'update', path: '/assets/_hmr/a.js', timestamp: 1 },
|
|
204
|
+
{ type: 'update', path: '/assets/_hmr/b.js', timestamp: 2 },
|
|
205
|
+
];
|
|
206
|
+
manager.registerStrategy(
|
|
207
|
+
new FakeHmrStrategy(HmrStrategyType.INTEGRATION, (f) => f === customFile, {
|
|
208
|
+
type: 'broadcast',
|
|
209
|
+
events,
|
|
210
|
+
}),
|
|
211
|
+
);
|
|
212
|
+
|
|
213
|
+
await manager.handleFileChange(customFile);
|
|
214
|
+
|
|
215
|
+
assert.equal(spy.broadcasts.length, 2);
|
|
216
|
+
assert.deepEqual(spy.broadcasts, events);
|
|
217
|
+
|
|
218
|
+
manager.stop();
|
|
219
|
+
});
|
|
220
|
+
});
|