@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,301 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { existsSync, mkdirSync, readFileSync, symlinkSync } from 'node:fs';
|
|
3
|
+
import { createRequire } from 'node:module';
|
|
4
|
+
import type { EcoBuildOnResolveArgs, EcoBuildOnResolveResult, EcoBuildPlugin } from '../../build/build-types.ts';
|
|
5
|
+
import type { EcoPagesAppConfig } from '../../internal-types.ts';
|
|
6
|
+
import type { NodeRuntimeManifest } from '../../services/runtime-manifest/node-runtime-manifest.service.ts';
|
|
7
|
+
import { resolveInternalExecutionDir } from '../../utils/resolve-work-dir.ts';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Returns the runtime-local node_modules directory used by the Node thin-host
|
|
11
|
+
* bootstrap output.
|
|
12
|
+
*/
|
|
13
|
+
export function getNodeRuntimeNodeModulesDir(manifest: NodeRuntimeManifest): string {
|
|
14
|
+
return path.join(
|
|
15
|
+
resolveInternalExecutionDir({
|
|
16
|
+
rootDir: manifest.appRootDir,
|
|
17
|
+
workDir: manifest.workDir,
|
|
18
|
+
absolutePaths: {
|
|
19
|
+
workDir: manifest.workDir,
|
|
20
|
+
distDir: manifest.distDir,
|
|
21
|
+
},
|
|
22
|
+
}),
|
|
23
|
+
'node_modules',
|
|
24
|
+
);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Returns the app-local node_modules directory used by framework-owned Node
|
|
29
|
+
* bootstrap loads outside the thin-host manifest path.
|
|
30
|
+
*/
|
|
31
|
+
export function getAppRuntimeNodeModulesDir(
|
|
32
|
+
appConfig: Pick<EcoPagesAppConfig, 'rootDir' | 'workDir' | 'absolutePaths'>,
|
|
33
|
+
): string {
|
|
34
|
+
return path.join(resolveInternalExecutionDir(appConfig), 'node_modules');
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Derives the package root segment from a bare specifier.
|
|
39
|
+
*/
|
|
40
|
+
function getPackageNameFromSpecifier(specifier: string): string {
|
|
41
|
+
if (specifier.startsWith('@')) {
|
|
42
|
+
const [scope, name] = specifier.split('/');
|
|
43
|
+
return `${scope}/${name}`;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
return specifier.split('/')[0] ?? specifier;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
/**
|
|
50
|
+
* Walks upward from a resolved file until it finds the owning package root.
|
|
51
|
+
*/
|
|
52
|
+
function findPackageRoot(resolvedPath: string): string {
|
|
53
|
+
let currentPath = path.dirname(resolvedPath);
|
|
54
|
+
|
|
55
|
+
while (true) {
|
|
56
|
+
const packageJsonPath = path.join(currentPath, 'package.json');
|
|
57
|
+
if (existsSync(packageJsonPath)) {
|
|
58
|
+
return currentPath;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
const parentPath = path.dirname(currentPath);
|
|
62
|
+
if (parentPath === currentPath) {
|
|
63
|
+
throw new Error(`Could not find package root for resolved dependency path: ${resolvedPath}`);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
currentPath = parentPath;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Creates a runtime-local symlink for one external dependency package.
|
|
72
|
+
*
|
|
73
|
+
* @remarks
|
|
74
|
+
* Node thin-host bootstrap bundles externalize third-party packages but still
|
|
75
|
+
* need those packages to resolve from a deterministic runtime-local
|
|
76
|
+
* `node_modules` directory. Symlinking the package root preserves that lookup
|
|
77
|
+
* without copying package contents into the app cache.
|
|
78
|
+
*/
|
|
79
|
+
function ensureRuntimePackageLink(nodeModulesDir: string, specifier: string, resolvedPath: string): void {
|
|
80
|
+
const packageName = getPackageNameFromSpecifier(specifier);
|
|
81
|
+
const packageRoot = findPackageRoot(resolvedPath);
|
|
82
|
+
const linkPath = path.join(nodeModulesDir, packageName);
|
|
83
|
+
|
|
84
|
+
if (existsSync(linkPath)) {
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
mkdirSync(path.dirname(linkPath), { recursive: true });
|
|
89
|
+
symlinkSync(packageRoot, linkPath, 'dir');
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
export interface NodeBootstrapResolutionOptions {
|
|
93
|
+
projectDir: string;
|
|
94
|
+
runtimeNodeModulesDir: string;
|
|
95
|
+
preserveImportMetaPaths?: string[];
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Builds the user-facing error for Bun-native imports that cannot run on the
|
|
100
|
+
* Node thin-host bootstrap path.
|
|
101
|
+
*/
|
|
102
|
+
export function getNodeUnsupportedBuiltinError(specifier: string, importer?: string): string {
|
|
103
|
+
return `Node thin-host bootstrap does not support Bun builtin specifier ${JSON.stringify(specifier)}${importer ? ` imported from ${importer}` : ''}.`;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Returns whether a dependency should be resolved relative to the importing
|
|
108
|
+
* package instead of the app root.
|
|
109
|
+
*/
|
|
110
|
+
function shouldResolveFromImporter(importer: string | undefined): importer is string {
|
|
111
|
+
return Boolean(importer && importer.includes(`${path.sep}node_modules${path.sep}`));
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Selects the build loader used when bootstrap-time source rewriting emits a
|
|
116
|
+
* synthetic in-memory file.
|
|
117
|
+
*/
|
|
118
|
+
function getBootstrapBuildLoaderForPath(filePath: string): 'js' | 'jsx' | 'json' | 'ts' | 'tsx' {
|
|
119
|
+
switch (path.extname(filePath).toLowerCase()) {
|
|
120
|
+
case '.ts':
|
|
121
|
+
case '.mts':
|
|
122
|
+
case '.cts':
|
|
123
|
+
return 'ts';
|
|
124
|
+
case '.tsx':
|
|
125
|
+
return 'tsx';
|
|
126
|
+
case '.jsx':
|
|
127
|
+
return 'jsx';
|
|
128
|
+
case '.json':
|
|
129
|
+
return 'json';
|
|
130
|
+
default:
|
|
131
|
+
return 'js';
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
const REEXPORT_FROM_STATEMENT_PATTERN =
|
|
136
|
+
/^\s*export\s+(?!type\b)(?:\*|\{[\s\S]*?\})\s+from\s+(['"][^'"]+['"])\s*;?\s*$/gm;
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Prepends side-effect imports for re-export barrels during bootstrap bundling.
|
|
140
|
+
*
|
|
141
|
+
* @remarks
|
|
142
|
+
* This keeps async module initialization observable even when the bootstrap
|
|
143
|
+
* bundle only referenced the file through re-export syntax.
|
|
144
|
+
*/
|
|
145
|
+
function injectBootstrapReexportImports(source: string): string {
|
|
146
|
+
const sideEffectImports: string[] = [];
|
|
147
|
+
const importedSpecifiers = new Set<string>();
|
|
148
|
+
let importIndex = 0;
|
|
149
|
+
|
|
150
|
+
const rewrittenSource = source.replace(REEXPORT_FROM_STATEMENT_PATTERN, (statement, specifierLiteral: string) => {
|
|
151
|
+
if (!importedSpecifiers.has(specifierLiteral)) {
|
|
152
|
+
importedSpecifiers.add(specifierLiteral);
|
|
153
|
+
const importBinding = `__eco_bootstrap_reexport_${importIndex++}`;
|
|
154
|
+
sideEffectImports.push(`import * as ${importBinding} from ${specifierLiteral};`);
|
|
155
|
+
sideEffectImports.push(`void ${importBinding};`);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
return statement;
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
if (sideEffectImports.length === 0) {
|
|
162
|
+
return source;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return `${sideEffectImports.join('\n')}\n${rewrittenSource}`;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Returns whether bootstrap source rewriting is allowed for the given file.
|
|
170
|
+
*
|
|
171
|
+
* @remarks
|
|
172
|
+
* Re-export and `import.meta` rewrites are limited to project-owned sources so
|
|
173
|
+
* third-party packages keep their original semantics.
|
|
174
|
+
*/
|
|
175
|
+
function shouldRewriteBootstrapSource(filePath: string, projectDir: string): boolean {
|
|
176
|
+
const normalizedPath = path.resolve(filePath);
|
|
177
|
+
const normalizedProjectDir = path.resolve(projectDir);
|
|
178
|
+
|
|
179
|
+
return (
|
|
180
|
+
normalizedPath.startsWith(`${normalizedProjectDir}${path.sep}`) &&
|
|
181
|
+
!normalizedPath.includes(`${path.sep}node_modules${path.sep}`)
|
|
182
|
+
);
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Resolves one bare specifier encountered while bundling Node thin-host
|
|
187
|
+
* bootstrap modules.
|
|
188
|
+
*
|
|
189
|
+
* Workspace-owned `@ecopages/*` packages stay in the bundle graph so Node does
|
|
190
|
+
* not execute their source files directly. Third-party packages stay external,
|
|
191
|
+
* but are linked into the runtime-local `node_modules` tree so the generated
|
|
192
|
+
* bootstrap output resolves them from a deterministic location.
|
|
193
|
+
*/
|
|
194
|
+
export function resolveNodeBootstrapDependency(
|
|
195
|
+
args: Pick<EcoBuildOnResolveArgs, 'path' | 'importer'>,
|
|
196
|
+
options: NodeBootstrapResolutionOptions,
|
|
197
|
+
): EcoBuildOnResolveResult | undefined {
|
|
198
|
+
if (
|
|
199
|
+
args.path.startsWith('./') ||
|
|
200
|
+
args.path.startsWith('../') ||
|
|
201
|
+
args.path.startsWith('@/') ||
|
|
202
|
+
args.path.startsWith('/') ||
|
|
203
|
+
args.path.startsWith('node:')
|
|
204
|
+
) {
|
|
205
|
+
return undefined;
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
const projectRequire = createRequire(path.join(options.projectDir, 'package.json'));
|
|
209
|
+
const requireFromImporter =
|
|
210
|
+
args.importer && path.isAbsolute(args.importer) && shouldResolveFromImporter(args.importer)
|
|
211
|
+
? createRequire(args.importer)
|
|
212
|
+
: projectRequire;
|
|
213
|
+
|
|
214
|
+
if (args.path.startsWith('@ecopages/')) {
|
|
215
|
+
return {
|
|
216
|
+
path: requireFromImporter.resolve(args.path),
|
|
217
|
+
};
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
const resolvedPath = requireFromImporter.resolve(args.path);
|
|
221
|
+
|
|
222
|
+
ensureRuntimePackageLink(options.runtimeNodeModulesDir, args.path, resolvedPath);
|
|
223
|
+
|
|
224
|
+
return {
|
|
225
|
+
path: args.path,
|
|
226
|
+
external: true,
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
/**
|
|
231
|
+
* Creates the bootstrap-time resolver plugin used by the Node thin-host
|
|
232
|
+
* adapter for config and entry module loading.
|
|
233
|
+
*/
|
|
234
|
+
export function createNodeBootstrapPlugin(options: NodeBootstrapResolutionOptions): EcoBuildPlugin {
|
|
235
|
+
const projectDir = path.resolve(options.projectDir);
|
|
236
|
+
const importMetaRewritePaths = new Set(
|
|
237
|
+
(options.preserveImportMetaPaths ?? []).map((filePath) => path.resolve(filePath)),
|
|
238
|
+
);
|
|
239
|
+
|
|
240
|
+
return {
|
|
241
|
+
name: 'node-thin-host-bundle-workspace-packages',
|
|
242
|
+
setup(build) {
|
|
243
|
+
build.onResolve({ filter: /^bun:/ }, (args) => {
|
|
244
|
+
throw new Error(getNodeUnsupportedBuiltinError(args.path, args.importer));
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
build.onLoad({ filter: /\.[cm]?[jt]sx?$/ }, async (args) => {
|
|
248
|
+
const absolutePath = path.resolve(args.path);
|
|
249
|
+
const shouldPreserveImportMeta = importMetaRewritePaths.has(absolutePath);
|
|
250
|
+
const shouldRewriteReexports = shouldRewriteBootstrapSource(absolutePath, projectDir);
|
|
251
|
+
|
|
252
|
+
if (!shouldPreserveImportMeta && !shouldRewriteReexports) {
|
|
253
|
+
return undefined;
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
const originalContents = readFileSync(args.path, 'utf8');
|
|
257
|
+
let contents = originalContents;
|
|
258
|
+
|
|
259
|
+
if (shouldPreserveImportMeta) {
|
|
260
|
+
contents = contents
|
|
261
|
+
.replaceAll('import.meta.dirname', JSON.stringify(path.dirname(args.path)))
|
|
262
|
+
.replaceAll('import.meta.filename', JSON.stringify(args.path));
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
if (shouldRewriteReexports) {
|
|
266
|
+
contents = injectBootstrapReexportImports(contents);
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
if (contents === originalContents) {
|
|
270
|
+
return undefined;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
return {
|
|
274
|
+
contents,
|
|
275
|
+
loader: getBootstrapBuildLoaderForPath(args.path),
|
|
276
|
+
resolveDir: path.dirname(args.path),
|
|
277
|
+
};
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
build.onResolve({ filter: /^[@A-Za-z0-9][^:]*$/ }, (args) => {
|
|
281
|
+
return resolveNodeBootstrapDependency(args, options);
|
|
282
|
+
});
|
|
283
|
+
},
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
/**
|
|
288
|
+
* Creates the standard Node bootstrap plugin for one finalized app config.
|
|
289
|
+
*/
|
|
290
|
+
export function createAppNodeBootstrapPlugin(
|
|
291
|
+
appConfig: Pick<EcoPagesAppConfig, 'rootDir' | 'workDir' | 'absolutePaths'>,
|
|
292
|
+
options?: {
|
|
293
|
+
preserveImportMetaPaths?: string[];
|
|
294
|
+
},
|
|
295
|
+
): EcoBuildPlugin {
|
|
296
|
+
return createNodeBootstrapPlugin({
|
|
297
|
+
projectDir: appConfig.rootDir,
|
|
298
|
+
runtimeNodeModulesDir: getAppRuntimeNodeModulesDir(appConfig),
|
|
299
|
+
preserveImportMetaPaths: options?.preserveImportMetaPaths,
|
|
300
|
+
});
|
|
301
|
+
}
|
|
@@ -1,6 +1,13 @@
|
|
|
1
1
|
export { NodeServerAdapter, createNodeServerAdapter } from './server-adapter.ts';
|
|
2
2
|
export { EcopagesApp, createNodeApp } from './create-app.ts';
|
|
3
|
+
export { assertNodeRuntimeManifest, createNodeRuntimeAdapter } from './runtime-adapter.ts';
|
|
3
4
|
export type { EcopagesAppOptions } from './create-app.ts';
|
|
5
|
+
export type {
|
|
6
|
+
LoadedAppRuntime,
|
|
7
|
+
NodeRuntimeAdapter,
|
|
8
|
+
NodeRuntimeSession,
|
|
9
|
+
NodeRuntimeStartOptions,
|
|
10
|
+
} from './runtime-adapter.ts';
|
|
4
11
|
export type {
|
|
5
12
|
NodeServerAdapterParams,
|
|
6
13
|
NodeServerAdapterResult,
|
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { NodeClientBridge } from './node-client-bridge.ts';
|
|
3
|
+
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
// Minimal WebSocket mock — only the surface that NodeClientBridge touches
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
function createMockWs(readyState = 1 /* OPEN */) {
|
|
8
|
+
return {
|
|
9
|
+
readyState,
|
|
10
|
+
send: vi.fn(),
|
|
11
|
+
ping: vi.fn(),
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
// ---------------------------------------------------------------------------
|
|
16
|
+
// Tests
|
|
17
|
+
// ---------------------------------------------------------------------------
|
|
18
|
+
describe('NodeClientBridge', () => {
|
|
19
|
+
let bridge: NodeClientBridge;
|
|
20
|
+
|
|
21
|
+
beforeEach(() => {
|
|
22
|
+
vi.useFakeTimers();
|
|
23
|
+
bridge = new NodeClientBridge();
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
afterEach(() => {
|
|
27
|
+
bridge.destroy();
|
|
28
|
+
vi.useRealTimers();
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
// ── subscribe / unsubscribe ─────────────────────────────────────────────
|
|
32
|
+
|
|
33
|
+
describe('subscribe / unsubscribe', () => {
|
|
34
|
+
it('increments subscriberCount when a socket is added', () => {
|
|
35
|
+
const ws = createMockWs();
|
|
36
|
+
bridge.subscribe(ws as never);
|
|
37
|
+
expect(bridge.subscriberCount).toBe(1);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('decrements subscriberCount when a socket is removed', () => {
|
|
41
|
+
const ws = createMockWs();
|
|
42
|
+
bridge.subscribe(ws as never);
|
|
43
|
+
bridge.unsubscribe(ws as never);
|
|
44
|
+
expect(bridge.subscriberCount).toBe(0);
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it('handles unsubscribing a socket that was never added', () => {
|
|
48
|
+
const ws = createMockWs();
|
|
49
|
+
expect(() => bridge.unsubscribe(ws as never)).not.toThrow();
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
// ── broadcast ──────────────────────────────────────────────────────────
|
|
54
|
+
|
|
55
|
+
describe('broadcast', () => {
|
|
56
|
+
it('sends the serialised event to all OPEN subscribers', () => {
|
|
57
|
+
const ws1 = createMockWs(1);
|
|
58
|
+
const ws2 = createMockWs(1);
|
|
59
|
+
bridge.subscribe(ws1 as never);
|
|
60
|
+
bridge.subscribe(ws2 as never);
|
|
61
|
+
|
|
62
|
+
bridge.broadcast({ type: 'reload' });
|
|
63
|
+
|
|
64
|
+
expect(ws1.send).toHaveBeenCalledWith(JSON.stringify({ type: 'reload' }));
|
|
65
|
+
expect(ws2.send).toHaveBeenCalledWith(JSON.stringify({ type: 'reload' }));
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
it('does NOT send to a socket that is not OPEN (readyState !== 1)', () => {
|
|
69
|
+
const closedWs = createMockWs(3 /* CLOSED */);
|
|
70
|
+
bridge.subscribe(closedWs as never);
|
|
71
|
+
|
|
72
|
+
bridge.broadcast({ type: 'reload' });
|
|
73
|
+
|
|
74
|
+
expect(closedWs.send).not.toHaveBeenCalled();
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
// ── convenience helpers ────────────────────────────────────────────────
|
|
79
|
+
|
|
80
|
+
describe('convenience broadcast helpers', () => {
|
|
81
|
+
it('reload() broadcasts a reload event', () => {
|
|
82
|
+
const ws = createMockWs();
|
|
83
|
+
bridge.subscribe(ws as never);
|
|
84
|
+
|
|
85
|
+
bridge.reload();
|
|
86
|
+
|
|
87
|
+
expect(ws.send).toHaveBeenCalledWith(JSON.stringify({ type: 'reload' }));
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('cssUpdate() broadcasts a css-update event with path and timestamp', () => {
|
|
91
|
+
const ws = createMockWs();
|
|
92
|
+
bridge.subscribe(ws as never);
|
|
93
|
+
|
|
94
|
+
bridge.cssUpdate('/styles/main.css');
|
|
95
|
+
|
|
96
|
+
const payload = JSON.parse((ws.send as ReturnType<typeof vi.fn>).mock.calls[0][0]);
|
|
97
|
+
expect(payload.type).toBe('css-update');
|
|
98
|
+
expect(payload.path).toBe('/styles/main.css');
|
|
99
|
+
expect(typeof payload.timestamp).toBe('number');
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
it('update() broadcasts an update event with path and timestamp', () => {
|
|
103
|
+
const ws = createMockWs();
|
|
104
|
+
bridge.subscribe(ws as never);
|
|
105
|
+
|
|
106
|
+
bridge.update('/pages/index.ts');
|
|
107
|
+
|
|
108
|
+
const payload = JSON.parse((ws.send as ReturnType<typeof vi.fn>).mock.calls[0][0]);
|
|
109
|
+
expect(payload.type).toBe('update');
|
|
110
|
+
expect(payload.path).toBe('/pages/index.ts');
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it('error() broadcasts an error event with the message', () => {
|
|
114
|
+
const ws = createMockWs();
|
|
115
|
+
bridge.subscribe(ws as never);
|
|
116
|
+
|
|
117
|
+
bridge.error('build failed');
|
|
118
|
+
|
|
119
|
+
expect(ws.send).toHaveBeenCalledWith(JSON.stringify({ type: 'error', message: 'build failed' }));
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
// ── heartbeat sweep ────────────────────────────────────────────────────
|
|
124
|
+
|
|
125
|
+
describe('heartbeat sweep (every 30 s)', () => {
|
|
126
|
+
it('pings OPEN subscribers on each sweep', () => {
|
|
127
|
+
const ws = createMockWs(1);
|
|
128
|
+
bridge.subscribe(ws as never);
|
|
129
|
+
|
|
130
|
+
vi.advanceTimersByTime(30_000);
|
|
131
|
+
|
|
132
|
+
expect(ws.ping).toHaveBeenCalledTimes(1);
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('removes zombie sockets (non-OPEN) from the subscriber set', () => {
|
|
136
|
+
const liveWs = createMockWs(1 /* OPEN */);
|
|
137
|
+
const deadWs = createMockWs(3 /* CLOSED */);
|
|
138
|
+
bridge.subscribe(liveWs as never);
|
|
139
|
+
bridge.subscribe(deadWs as never);
|
|
140
|
+
|
|
141
|
+
expect(bridge.subscriberCount).toBe(2);
|
|
142
|
+
|
|
143
|
+
vi.advanceTimersByTime(30_000);
|
|
144
|
+
|
|
145
|
+
expect(bridge.subscriberCount).toBe(1);
|
|
146
|
+
expect(liveWs.ping).toHaveBeenCalledTimes(1);
|
|
147
|
+
expect(deadWs.ping).not.toHaveBeenCalled();
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
it('accumulates cleanup across multiple sweeps', () => {
|
|
151
|
+
const ws1 = createMockWs(1);
|
|
152
|
+
bridge.subscribe(ws1 as never);
|
|
153
|
+
|
|
154
|
+
// First sweep – still alive
|
|
155
|
+
vi.advanceTimersByTime(30_000);
|
|
156
|
+
expect(bridge.subscriberCount).toBe(1);
|
|
157
|
+
|
|
158
|
+
// Socket goes away
|
|
159
|
+
ws1.readyState = 3;
|
|
160
|
+
|
|
161
|
+
// Second sweep – removed
|
|
162
|
+
vi.advanceTimersByTime(30_000);
|
|
163
|
+
expect(bridge.subscriberCount).toBe(0);
|
|
164
|
+
});
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
// ── destroy ────────────────────────────────────────────────────────────
|
|
168
|
+
|
|
169
|
+
describe('destroy()', () => {
|
|
170
|
+
it('stops the heartbeat timer so no further pings are sent', () => {
|
|
171
|
+
const ws = createMockWs(1);
|
|
172
|
+
bridge.subscribe(ws as never);
|
|
173
|
+
|
|
174
|
+
bridge.destroy();
|
|
175
|
+
|
|
176
|
+
// Advancing time should NOT trigger any sweep
|
|
177
|
+
vi.advanceTimersByTime(30_000 * 5);
|
|
178
|
+
|
|
179
|
+
expect(ws.ping).not.toHaveBeenCalled();
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
it('clears all subscribers', () => {
|
|
183
|
+
bridge.subscribe(createMockWs() as never);
|
|
184
|
+
bridge.subscribe(createMockWs() as never);
|
|
185
|
+
|
|
186
|
+
bridge.destroy();
|
|
187
|
+
|
|
188
|
+
expect(bridge.subscriberCount).toBe(0);
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
it('is safe to call destroy() multiple times', () => {
|
|
192
|
+
expect(() => {
|
|
193
|
+
bridge.destroy();
|
|
194
|
+
bridge.destroy();
|
|
195
|
+
}).not.toThrow();
|
|
196
|
+
});
|
|
197
|
+
});
|
|
198
|
+
});
|