@ecopages/core 0.2.0-alpha.26 → 0.2.0-alpha.27
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 +63 -7
- package/package.json +73 -249
- package/src/adapters/abstract/application-adapter.test.ts +172 -0
- package/src/adapters/abstract/application-adapter.ts +379 -0
- package/src/adapters/abstract/router-adapter.ts +30 -0
- package/src/adapters/abstract/server-adapter.ts +79 -0
- package/src/adapters/bun/client-bridge.ts +62 -0
- package/src/adapters/bun/create-app.ts +232 -0
- package/src/adapters/bun/hmr-manager.test.ts +265 -0
- package/src/adapters/bun/hmr-manager.ts +383 -0
- package/src/adapters/bun/index.ts +2 -0
- package/src/adapters/bun/server-adapter.ts +526 -0
- package/src/adapters/bun/server-lifecycle.ts +124 -0
- package/src/adapters/create-app.test.ts +10 -0
- package/src/adapters/create-app.ts +91 -0
- package/src/adapters/index.ts +2 -0
- package/src/adapters/node/create-app.test.ts +53 -0
- package/src/adapters/node/create-app.ts +183 -0
- package/src/adapters/node/node-client-bridge.test.ts +198 -0
- package/src/adapters/node/node-client-bridge.ts +79 -0
- package/src/adapters/node/node-hmr-manager.test.ts +320 -0
- package/src/adapters/node/node-hmr-manager.ts +355 -0
- package/src/adapters/node/server-adapter.ts +502 -0
- package/src/adapters/node/static-content-server.test.ts +60 -0
- package/src/adapters/node/static-content-server.ts +239 -0
- package/src/adapters/shared/api-response.test.ts +97 -0
- package/src/adapters/shared/api-response.ts +104 -0
- package/src/adapters/shared/application-adapter.ts +199 -0
- package/src/adapters/shared/define-api-handler.ts +66 -0
- package/src/adapters/shared/explicit-static-render-preparation.ts +58 -0
- package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +131 -0
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +85 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +118 -0
- package/src/adapters/shared/fs-server-response-factory.test.ts +176 -0
- package/src/adapters/shared/fs-server-response-factory.ts +96 -0
- package/src/adapters/shared/fs-server-response-matcher.test.ts +311 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +240 -0
- package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
- package/src/adapters/shared/hmr-html-response.ts +52 -0
- package/src/adapters/shared/hmr-manager.contract.test.ts +228 -0
- package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
- package/src/adapters/shared/render-context.test.ts +150 -0
- package/src/adapters/shared/render-context.ts +123 -0
- package/src/adapters/shared/runtime-bootstrap.ts +79 -0
- package/src/adapters/shared/server-adapter.test.ts +130 -0
- package/src/adapters/shared/server-adapter.ts +562 -0
- package/src/adapters/shared/server-route-handler.test.ts +111 -0
- package/src/adapters/shared/server-route-handler.ts +153 -0
- package/src/adapters/shared/server-static-builder.test.ts +338 -0
- package/src/adapters/shared/server-static-builder.ts +170 -0
- package/src/build/build-adapter-serialization.test.ts +281 -0
- package/src/build/build-adapter.test.ts +1240 -0
- package/src/build/build-adapter.ts +1012 -0
- package/src/build/build-manifest.ts +54 -0
- package/src/build/build-types.ts +83 -0
- package/src/build/dev-build-coordinator.ts +220 -0
- package/src/build/esbuild-build-adapter.ts +660 -0
- package/src/build/runtime-build-executor.test.ts +81 -0
- package/src/build/runtime-build-executor.ts +40 -0
- package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
- package/src/build/runtime-specifier-alias-plugin.ts +62 -0
- package/src/build/runtime-specifier-aliases.ts +135 -0
- package/src/config/README.md +1 -1
- package/src/config/config-builder.test.ts +442 -0
- package/src/config/config-builder.ts +737 -0
- package/src/config/config-builder.typecheck.test.ts +96 -0
- package/src/config/{constants.d.ts → constants.ts} +22 -13
- package/src/dev/host-runtime.ts +34 -0
- package/src/dev/sc-server.ts +143 -0
- package/src/eco/eco.browser.test.ts +43 -0
- package/src/eco/eco.browser.ts +118 -0
- package/src/eco/eco.test.ts +654 -0
- package/src/eco/eco.ts +205 -0
- package/src/eco/eco.types.ts +221 -0
- package/src/eco/eco.utils.test.ts +219 -0
- package/src/eco/eco.utils.ts +5 -0
- package/src/eco/global-injector-map.test.ts +42 -0
- package/src/eco/global-injector-map.ts +112 -0
- package/src/eco/lazy-injector-map.test.ts +66 -0
- package/src/eco/lazy-injector-map.ts +120 -0
- package/src/eco/module-dependencies.test.ts +30 -0
- package/src/eco/module-dependencies.ts +75 -0
- package/src/errors/http-error.test.ts +134 -0
- package/src/errors/http-error.ts +72 -0
- package/src/errors/index.ts +3 -0
- package/src/errors/locals-access-error.ts +7 -0
- package/src/global/app-logger.ts +4 -0
- package/src/global/utils.test.ts +12 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
- package/src/hmr/client/hmr-runtime.ts +162 -0
- package/src/hmr/hmr-strategy.test.ts +124 -0
- package/src/hmr/hmr-strategy.ts +177 -0
- package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
- package/src/hmr/hmr.test.e2e.ts +66 -0
- package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
- package/src/hmr/strategies/js-hmr-strategy.test.ts +334 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +314 -0
- package/src/index.browser.ts +3 -0
- package/src/index.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +87 -0
- package/src/integrations/ghtml/ghtml.constants.ts +1 -0
- package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
- package/src/plugins/alias-resolver-plugin.test.ts +41 -0
- package/src/plugins/alias-resolver-plugin.ts +63 -0
- package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
- package/src/plugins/eco-component-meta-plugin.ts +494 -0
- package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
- package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
- package/src/plugins/integration-plugin.test.ts +151 -0
- package/src/plugins/integration-plugin.ts +323 -0
- package/src/plugins/processor.test.ts +148 -0
- package/src/plugins/processor.ts +257 -0
- package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
- package/src/plugins/source-transform.test.ts +82 -0
- package/src/plugins/source-transform.ts +123 -0
- package/src/route-renderer/GRAPH.md +81 -289
- package/src/route-renderer/README.md +67 -105
- package/src/route-renderer/orchestration/component-render-context.ts +325 -0
- package/src/route-renderer/orchestration/declared-ownership-graph.ts +62 -0
- package/src/route-renderer/orchestration/foreign-subtree-execution.service.ts +383 -0
- package/src/route-renderer/orchestration/integration-renderer.test.ts +2085 -0
- package/src/route-renderer/orchestration/integration-renderer.ts +1244 -0
- package/src/route-renderer/orchestration/ownership-planning.service.ts +97 -0
- package/src/route-renderer/orchestration/ownership-validation.service.ts +76 -0
- package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
- package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.test.ts +324 -0
- package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.ts +294 -0
- package/src/route-renderer/orchestration/render-output.utils.ts +310 -0
- package/src/route-renderer/orchestration/route-render-orchestrator.prepare-render-options.test.ts +644 -0
- package/src/route-renderer/orchestration/route-render-orchestrator.test.ts +265 -0
- package/src/route-renderer/orchestration/route-render-orchestrator.ts +592 -0
- package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
- package/src/route-renderer/orchestration/template-serialization.ts +117 -0
- package/src/route-renderer/page-loading/component-dependency-collection.ts +202 -0
- package/src/route-renderer/page-loading/declared-asset-collection.ts +153 -0
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +761 -0
- package/src/route-renderer/page-loading/dependency-resolver.ts +144 -0
- package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
- package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
- package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
- package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
- package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
- package/src/route-renderer/page-loading/page-dependency-bundling.ts +244 -0
- package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
- package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
- package/src/route-renderer/route-renderer.ts +133 -0
- package/src/router/README.md +16 -19
- package/src/router/client/link-intent.test.browser.ts +51 -0
- package/src/router/client/link-intent.ts +92 -0
- package/src/router/client/navigation-coordinator.test.ts +237 -0
- package/src/router/client/navigation-coordinator.ts +453 -0
- package/src/router/server/route-registry.test.ts +176 -0
- package/src/router/server/route-registry.ts +382 -0
- package/src/services/README.md +1 -2
- package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +473 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.ts +344 -0
- package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
- package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
- package/src/services/assets/asset-processing-service/assets.types.ts +128 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
- package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
- package/src/services/assets/asset-processing-service/index.ts +6 -0
- package/src/services/assets/asset-processing-service/page-package.test.ts +100 -0
- package/src/services/assets/asset-processing-service/page-package.ts +93 -0
- package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
- package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +173 -0
- package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +195 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +137 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +116 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +89 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +72 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +83 -0
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
- package/src/services/assets/browser-bundle.service.test.ts +66 -0
- package/src/services/assets/browser-bundle.service.ts +109 -0
- package/src/services/cache/cache.types.ts +126 -0
- package/src/services/cache/index.ts +18 -0
- package/src/services/cache/memory-cache-store.test.ts +225 -0
- package/src/services/cache/memory-cache-store.ts +130 -0
- package/src/services/cache/page-cache-service.test.ts +175 -0
- package/src/services/cache/page-cache-service.ts +202 -0
- package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
- package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
- package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
- package/src/services/html/html-rewriter-provider.service.ts +104 -0
- package/src/services/html/html-transformer.service.test.ts +476 -0
- package/src/services/html/html-transformer.service.ts +275 -0
- package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
- package/src/services/invalidation/development-invalidation.service.ts +262 -0
- package/src/services/module-loading/app-module-loader.service.ts +9 -0
- package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
- package/src/services/module-loading/app-server-module-transpiler.service.ts +141 -0
- package/src/services/module-loading/host-module-loader-registry.ts +15 -0
- package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
- package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
- package/src/services/module-loading/node-bootstrap-plugin.ts +311 -0
- package/src/services/module-loading/page-module-import.service.test.ts +504 -0
- package/src/services/module-loading/page-module-import.service.ts +251 -0
- package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
- package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
- package/src/services/module-loading/source-module-support.ts +19 -0
- package/src/services/runtime-state/dev-graph.service.ts +217 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
- package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
- package/src/services/validation/schema-validation-service.test.ts +223 -0
- package/src/services/validation/schema-validation-service.ts +204 -0
- package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
- package/src/static-site-generator/static-site-generator.test.ts +408 -0
- package/src/static-site-generator/static-site-generator.ts +445 -0
- package/src/types/internal-types.ts +243 -0
- package/src/types/public-types.ts +1459 -0
- package/src/utils/deep-merge.test.ts +114 -0
- package/src/utils/deep-merge.ts +47 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/html-escaping.ts +9 -0
- package/src/utils/invariant.test.ts +22 -0
- package/src/utils/invariant.ts +15 -0
- package/src/utils/locals-utils.ts +37 -0
- package/src/utils/parse-cli-args.test.ts +69 -0
- package/src/utils/parse-cli-args.ts +105 -0
- package/src/utils/path-utils.module.ts +14 -0
- package/src/utils/path-utils.test.ts +15 -0
- package/src/utils/resolve-work-dir.ts +45 -0
- package/src/utils/runtime.ts +44 -0
- package/src/utils/server-utils.module.ts +67 -0
- package/src/utils/server-utils.test.ts +38 -0
- package/src/watchers/project-watcher.integration.test.ts +337 -0
- package/src/watchers/project-watcher.test-helpers.ts +42 -0
- package/src/watchers/project-watcher.test.ts +768 -0
- package/src/watchers/project-watcher.ts +357 -0
- package/CHANGELOG.md +0 -66
- package/src/adapters/abstract/application-adapter.d.ts +0 -194
- package/src/adapters/abstract/application-adapter.js +0 -121
- package/src/adapters/abstract/router-adapter.d.ts +0 -26
- package/src/adapters/abstract/router-adapter.js +0 -5
- package/src/adapters/abstract/server-adapter.d.ts +0 -69
- package/src/adapters/abstract/server-adapter.js +0 -15
- package/src/adapters/bun/client-bridge.d.ts +0 -34
- package/src/adapters/bun/client-bridge.js +0 -48
- package/src/adapters/bun/create-app.d.ts +0 -52
- package/src/adapters/bun/create-app.js +0 -116
- package/src/adapters/bun/hmr-manager.d.ts +0 -143
- package/src/adapters/bun/hmr-manager.js +0 -333
- package/src/adapters/bun/index.d.ts +0 -2
- package/src/adapters/bun/index.js +0 -8
- package/src/adapters/bun/server-adapter.d.ts +0 -155
- package/src/adapters/bun/server-adapter.js +0 -374
- package/src/adapters/bun/server-lifecycle.d.ts +0 -63
- package/src/adapters/bun/server-lifecycle.js +0 -92
- package/src/adapters/create-app.d.ts +0 -20
- package/src/adapters/create-app.js +0 -66
- package/src/adapters/index.d.ts +0 -2
- package/src/adapters/index.js +0 -8
- package/src/adapters/node/create-app.d.ts +0 -18
- package/src/adapters/node/create-app.js +0 -149
- package/src/adapters/node/node-client-bridge.d.ts +0 -26
- package/src/adapters/node/node-client-bridge.js +0 -66
- package/src/adapters/node/node-hmr-manager.d.ts +0 -133
- package/src/adapters/node/node-hmr-manager.js +0 -311
- package/src/adapters/node/server-adapter.d.ts +0 -162
- package/src/adapters/node/server-adapter.js +0 -368
- package/src/adapters/node/static-content-server.d.ts +0 -60
- package/src/adapters/node/static-content-server.js +0 -194
- package/src/adapters/shared/api-response.d.ts +0 -52
- package/src/adapters/shared/api-response.js +0 -96
- package/src/adapters/shared/application-adapter.d.ts +0 -18
- package/src/adapters/shared/application-adapter.js +0 -90
- package/src/adapters/shared/define-api-handler.d.ts +0 -25
- package/src/adapters/shared/define-api-handler.js +0 -15
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
- package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
- package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
- package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
- package/src/adapters/shared/fs-server-response-factory.js +0 -97
- package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
- package/src/adapters/shared/fs-server-response-matcher.js +0 -147
- package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
- package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
- package/src/adapters/shared/hmr-html-response.d.ts +0 -22
- package/src/adapters/shared/hmr-html-response.js +0 -32
- package/src/adapters/shared/render-context.d.ts +0 -15
- package/src/adapters/shared/render-context.js +0 -72
- package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
- package/src/adapters/shared/runtime-bootstrap.js +0 -43
- package/src/adapters/shared/server-adapter.d.ts +0 -97
- package/src/adapters/shared/server-adapter.js +0 -390
- package/src/adapters/shared/server-route-handler.d.ts +0 -89
- package/src/adapters/shared/server-route-handler.js +0 -111
- package/src/adapters/shared/server-static-builder.d.ts +0 -71
- package/src/adapters/shared/server-static-builder.js +0 -100
- package/src/build/build-adapter.d.ts +0 -239
- package/src/build/build-adapter.js +0 -642
- package/src/build/build-manifest.d.ts +0 -27
- package/src/build/build-manifest.js +0 -30
- package/src/build/build-types.d.ts +0 -57
- package/src/build/build-types.js +0 -0
- package/src/build/dev-build-coordinator.d.ts +0 -72
- package/src/build/dev-build-coordinator.js +0 -154
- package/src/build/esbuild-build-adapter.d.ts +0 -78
- package/src/build/esbuild-build-adapter.js +0 -505
- package/src/build/runtime-build-executor.d.ts +0 -14
- package/src/build/runtime-build-executor.js +0 -22
- package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
- package/src/build/runtime-specifier-alias-plugin.js +0 -35
- package/src/build/runtime-specifier-aliases.d.ts +0 -5
- package/src/build/runtime-specifier-aliases.js +0 -95
- package/src/config/config-builder.d.ts +0 -252
- package/src/config/config-builder.js +0 -603
- package/src/config/constants.js +0 -25
- package/src/dev/sc-server.d.ts +0 -30
- package/src/dev/sc-server.js +0 -111
- package/src/eco/eco.browser.d.ts +0 -2
- package/src/eco/eco.browser.js +0 -83
- package/src/eco/eco.d.ts +0 -9
- package/src/eco/eco.js +0 -85
- package/src/eco/eco.types.d.ts +0 -178
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.utils.d.ts +0 -1
- package/src/eco/eco.utils.js +0 -10
- package/src/eco/global-injector-map.d.ts +0 -16
- package/src/eco/global-injector-map.js +0 -80
- package/src/eco/lazy-injector-map.d.ts +0 -8
- package/src/eco/lazy-injector-map.js +0 -70
- package/src/eco/module-dependencies.d.ts +0 -18
- package/src/eco/module-dependencies.js +0 -49
- package/src/errors/http-error.d.ts +0 -31
- package/src/errors/http-error.js +0 -50
- package/src/errors/index.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 -5
- package/src/hmr/client/hmr-runtime.js +0 -117
- package/src/hmr/hmr-strategy.d.ts +0 -162
- package/src/hmr/hmr-strategy.js +0 -44
- package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.postcss.test.e2e.js +0 -31
- package/src/hmr/hmr.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.test.e2e.js +0 -43
- package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
- package/src/hmr/strategies/default-hmr-strategy.js +0 -34
- package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
- package/src/hmr/strategies/js-hmr-strategy.js +0 -178
- package/src/index.browser.d.ts +0 -3
- package/src/index.browser.js +0 -4
- package/src/index.d.ts +0 -6
- package/src/index.js +0 -21
- package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
- package/src/integrations/ghtml/ghtml-renderer.js +0 -63
- package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
- package/src/integrations/ghtml/ghtml.constants.js +0 -4
- package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
- package/src/integrations/ghtml/ghtml.plugin.js +0 -20
- package/src/plugins/alias-resolver-plugin.d.ts +0 -2
- package/src/plugins/alias-resolver-plugin.js +0 -53
- package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
- package/src/plugins/eco-component-meta-plugin.js +0 -163
- package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
- package/src/plugins/foreign-jsx-override-plugin.js +0 -35
- package/src/plugins/integration-plugin.d.ts +0 -219
- package/src/plugins/integration-plugin.js +0 -196
- package/src/plugins/processor.d.ts +0 -95
- package/src/plugins/processor.js +0 -136
- package/src/plugins/runtime-capability.js +0 -0
- package/src/plugins/source-transform.d.ts +0 -46
- package/src/plugins/source-transform.js +0 -71
- package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
- package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
- package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
- package/src/route-renderer/orchestration/component-render-context.js +0 -147
- package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -556
- package/src/route-renderer/orchestration/integration-renderer.js +0 -932
- package/src/route-renderer/orchestration/page-packaging.service.d.ts +0 -16
- package/src/route-renderer/orchestration/page-packaging.service.js +0 -66
- package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +0 -2
- package/src/route-renderer/orchestration/processed-asset-dedupe.js +0 -23
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
- package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
- package/src/route-renderer/orchestration/render-execution.service.js +0 -106
- package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -66
- package/src/route-renderer/orchestration/render-output.utils.js +0 -171
- package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
- package/src/route-renderer/orchestration/render-preparation.service.js +0 -364
- package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
- package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
- package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
- package/src/route-renderer/orchestration/template-serialization.js +0 -45
- package/src/route-renderer/page-loading/component-dependency-collection.d.ts +0 -37
- package/src/route-renderer/page-loading/component-dependency-collection.js +0 -125
- package/src/route-renderer/page-loading/declared-asset-collection.d.ts +0 -24
- package/src/route-renderer/page-loading/declared-asset-collection.js +0 -106
- package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
- package/src/route-renderer/page-loading/dependency-resolver.js +0 -117
- package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +0 -11
- package/src/route-renderer/page-loading/ecopages-virtual-imports.js +0 -57
- package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +0 -45
- package/src/route-renderer/page-loading/lazy-entry-collection.js +0 -105
- package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +0 -19
- package/src/route-renderer/page-loading/lazy-trigger-planning.js +0 -40
- package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +0 -5
- package/src/route-renderer/page-loading/module-declaration-aggregation.js +0 -33
- package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +0 -3
- package/src/route-renderer/page-loading/module-declaration-scripts.js +0 -18
- package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +0 -13
- package/src/route-renderer/page-loading/page-dependency-bundling.js +0 -115
- package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
- package/src/route-renderer/page-loading/page-module-loader.js +0 -127
- package/src/route-renderer/route-renderer.d.ts +0 -67
- package/src/route-renderer/route-renderer.js +0 -103
- package/src/router/client/link-intent.js +0 -34
- package/src/router/client/link-intent.test.browser.d.ts +0 -1
- package/src/router/client/link-intent.test.browser.js +0 -43
- package/src/router/client/navigation-coordinator.d.ts +0 -169
- package/src/router/client/navigation-coordinator.js +0 -215
- package/src/router/server/fs-router-scanner.d.ts +0 -41
- package/src/router/server/fs-router-scanner.js +0 -161
- package/src/router/server/fs-router.d.ts +0 -26
- package/src/router/server/fs-router.js +0 -100
- package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +0 -3
- package/src/services/assets/asset-processing-service/asset-dependency-keys.js +0 -56
- package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -103
- package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -285
- package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
- package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
- package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -100
- package/src/services/assets/asset-processing-service/assets.types.js +0 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +0 -55
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -49
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
- package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +0 -30
- package/src/services/assets/asset-processing-service/grouped-content-bundles.js +0 -65
- package/src/services/assets/asset-processing-service/index.d.ts +0 -5
- package/src/services/assets/asset-processing-service/index.js +0 -5
- package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
- package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
- package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
- package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
- package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -65
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -22
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -136
- package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
- package/src/services/assets/asset-processing-service/processors/index.js +0 -5
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -6
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -116
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -91
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -77
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -8
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -58
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -67
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +0 -18
- package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +0 -45
- package/src/services/assets/browser-bundle.service.d.ts +0 -73
- package/src/services/assets/browser-bundle.service.js +0 -41
- package/src/services/cache/cache.types.d.ts +0 -107
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/index.d.ts +0 -7
- package/src/services/cache/index.js +0 -7
- package/src/services/cache/memory-cache-store.d.ts +0 -42
- package/src/services/cache/memory-cache-store.js +0 -98
- package/src/services/cache/page-cache-service.d.ts +0 -70
- package/src/services/cache/page-cache-service.js +0 -152
- package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
- package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
- package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
- package/src/services/html/html-rewriter-provider.service.js +0 -68
- package/src/services/html/html-transformer.service.d.ts +0 -87
- package/src/services/html/html-transformer.service.js +0 -216
- package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
- package/src/services/invalidation/development-invalidation.service.js +0 -190
- package/src/services/module-loading/app-module-loader.service.d.ts +0 -7
- package/src/services/module-loading/app-module-loader.service.js +0 -0
- package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
- package/src/services/module-loading/app-server-module-transpiler.service.js +0 -115
- package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
- package/src/services/module-loading/host-module-loader-registry.js +0 -15
- package/src/services/module-loading/module-loading-types.js +0 -0
- package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
- package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
- package/src/services/module-loading/page-module-import.service.d.ts +0 -76
- package/src/services/module-loading/page-module-import.service.js +0 -170
- package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -63
- package/src/services/module-loading/server-module-transpiler.service.js +0 -64
- package/src/services/module-loading/source-module-support.d.ts +0 -5
- package/src/services/module-loading/source-module-support.js +0 -8
- package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
- package/src/services/runtime-state/dev-graph.service.js +0 -162
- package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
- package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
- package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
- package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
- package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
- package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
- package/src/services/validation/schema-validation-service.d.ts +0 -122
- package/src/services/validation/schema-validation-service.js +0 -101
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/static-site-generator/static-site-generator.d.ts +0 -105
- package/src/static-site-generator/static-site-generator.js +0 -349
- package/src/types/internal-types.d.ts +0 -231
- package/src/types/internal-types.js +0 -0
- package/src/types/public-types.d.ts +0 -1257
- package/src/types/public-types.js +0 -0
- package/src/utils/deep-merge.d.ts +0 -14
- package/src/utils/deep-merge.js +0 -32
- package/src/utils/hash.d.ts +0 -1
- package/src/utils/hash.js +0 -7
- package/src/utils/html-escaping.d.ts +0 -7
- package/src/utils/html-escaping.js +0 -6
- package/src/utils/html.js +0 -4
- package/src/utils/invariant.d.ts +0 -5
- package/src/utils/invariant.js +0 -11
- package/src/utils/locals-utils.d.ts +0 -15
- package/src/utils/locals-utils.js +0 -24
- package/src/utils/parse-cli-args.d.ts +0 -27
- package/src/utils/parse-cli-args.js +0 -62
- package/src/utils/path-utils.module.d.ts +0 -5
- package/src/utils/path-utils.module.js +0 -14
- package/src/utils/resolve-work-dir.d.ts +0 -11
- package/src/utils/resolve-work-dir.js +0 -31
- package/src/utils/runtime.d.ts +0 -11
- package/src/utils/runtime.js +0 -40
- package/src/utils/server-utils.module.d.ts +0 -19
- package/src/utils/server-utils.module.js +0 -56
- package/src/watchers/project-watcher.d.ts +0 -136
- package/src/watchers/project-watcher.js +0 -275
- package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
- package/src/watchers/project-watcher.test-helpers.js +0 -52
- /package/src/utils/{html.d.ts → html.ts} +0 -0
|
@@ -0,0 +1,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 '../../types/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
|
+
});
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from 'vitest';
|
|
2
|
+
import { createRenderContext } from './render-context.ts';
|
|
3
|
+
import type { IntegrationPlugin } from '../../plugins/integration-plugin.ts';
|
|
4
|
+
import type { IntegrationRenderer } from '../../route-renderer/orchestration/integration-renderer.ts';
|
|
5
|
+
import type { EcoFunctionComponent } from '../../types/public-types.ts';
|
|
6
|
+
|
|
7
|
+
describe('createRenderContext', () => {
|
|
8
|
+
const RenderToResponse = vi.fn(() => Promise.resolve(new Response('rendered')));
|
|
9
|
+
const rendererModules = {
|
|
10
|
+
integrationManifestModuleId: 'virtual:ecopages/integration-manifest.ts',
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
const Renderer = {
|
|
14
|
+
name: '-renderer',
|
|
15
|
+
renderToResponse: RenderToResponse,
|
|
16
|
+
} as unknown as IntegrationRenderer;
|
|
17
|
+
|
|
18
|
+
const ExplicitRenderer = {
|
|
19
|
+
name: 'explicit-renderer',
|
|
20
|
+
renderToResponse: RenderToResponse,
|
|
21
|
+
} as unknown as IntegrationRenderer;
|
|
22
|
+
|
|
23
|
+
const InitializeRenderer = vi.fn(() => Renderer);
|
|
24
|
+
|
|
25
|
+
const Plugin = {
|
|
26
|
+
name: '-integration',
|
|
27
|
+
initializeRenderer: InitializeRenderer,
|
|
28
|
+
} as unknown as IntegrationPlugin;
|
|
29
|
+
|
|
30
|
+
const ExplicitInitializeRenderer = vi.fn(() => ExplicitRenderer);
|
|
31
|
+
|
|
32
|
+
const ExplicitPlugin = {
|
|
33
|
+
name: 'explicit-renderer',
|
|
34
|
+
initializeRenderer: ExplicitInitializeRenderer,
|
|
35
|
+
} as unknown as IntegrationPlugin;
|
|
36
|
+
|
|
37
|
+
const ViewFn = ((props: { foo: string }) => `<div>${props.foo}</div>`) as EcoFunctionComponent<
|
|
38
|
+
{ foo: string },
|
|
39
|
+
string
|
|
40
|
+
>;
|
|
41
|
+
ViewFn.config = {
|
|
42
|
+
__eco: {
|
|
43
|
+
id: 'test',
|
|
44
|
+
file: '/some/dir/-view.ts',
|
|
45
|
+
integration: '-integration',
|
|
46
|
+
},
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
const renderContext = createRenderContext({
|
|
50
|
+
integrations: [Plugin, ExplicitPlugin],
|
|
51
|
+
rendererModules,
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
it('should create a render context with methods', () => {
|
|
55
|
+
expect(renderContext.render).toBeDefined();
|
|
56
|
+
expect(renderContext.renderPartial).toBeDefined();
|
|
57
|
+
expect(renderContext.json).toBeDefined();
|
|
58
|
+
expect(renderContext.html).toBeDefined();
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
describe('render', () => {
|
|
62
|
+
it('should call renderer.renderToResponse with partial: false', async () => {
|
|
63
|
+
RenderToResponse.mockClear();
|
|
64
|
+
InitializeRenderer.mockClear();
|
|
65
|
+
ExplicitInitializeRenderer.mockClear();
|
|
66
|
+
const props = { foo: 'bar' };
|
|
67
|
+
const options = { status: 201, headers: { 'X-Custom': '1' } };
|
|
68
|
+
|
|
69
|
+
const response = await renderContext.render(ViewFn, props, options);
|
|
70
|
+
|
|
71
|
+
expect(InitializeRenderer).toHaveBeenCalledWith({ rendererModules });
|
|
72
|
+
expect(RenderToResponse).toHaveBeenCalledWith(ViewFn, props, {
|
|
73
|
+
partial: false,
|
|
74
|
+
status: 201,
|
|
75
|
+
headers: { 'X-Custom': '1' },
|
|
76
|
+
});
|
|
77
|
+
expect(response instanceof Response).toBe(true);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('should throw if view integration is unknown', async () => {
|
|
81
|
+
const badViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
82
|
+
badViewFn.config = { __eco: { id: 'test', file: '/bad-view.ts', integration: 'unknown' } };
|
|
83
|
+
await expect(renderContext.render(badViewFn, {})).rejects.toThrow('No integration found for: unknown');
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
it('should throw if view integration is missing', async () => {
|
|
87
|
+
const badViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
88
|
+
badViewFn.config = { __eco: undefined };
|
|
89
|
+
await expect(renderContext.render(badViewFn, {})).rejects.toThrow('Cannot determine integration for view');
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('should prefer explicit config.integration over injected metadata', async () => {
|
|
93
|
+
RenderToResponse.mockClear();
|
|
94
|
+
InitializeRenderer.mockClear();
|
|
95
|
+
ExplicitInitializeRenderer.mockClear();
|
|
96
|
+
|
|
97
|
+
const explicitViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
98
|
+
explicitViewFn.config = {
|
|
99
|
+
integration: 'explicit-renderer',
|
|
100
|
+
__eco: { id: 'test', file: '/some/file.tsx', integration: '-integration' },
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
await renderContext.render(explicitViewFn, {});
|
|
104
|
+
|
|
105
|
+
expect(ExplicitInitializeRenderer).toHaveBeenCalled();
|
|
106
|
+
expect(InitializeRenderer).not.toHaveBeenCalled();
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
describe('renderPartial', () => {
|
|
111
|
+
it('should call renderer.renderToResponse with partial: true', async () => {
|
|
112
|
+
RenderToResponse.mockClear();
|
|
113
|
+
const props = { foo: 'bar' };
|
|
114
|
+
const options = { status: 200 };
|
|
115
|
+
|
|
116
|
+
await renderContext.renderPartial(ViewFn, props, options);
|
|
117
|
+
|
|
118
|
+
expect(InitializeRenderer).toHaveBeenCalled();
|
|
119
|
+
expect(RenderToResponse).toHaveBeenCalledWith(ViewFn, props, {
|
|
120
|
+
partial: true,
|
|
121
|
+
status: 200,
|
|
122
|
+
headers: undefined,
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
describe('json', () => {
|
|
128
|
+
it('should return a JSON response', async () => {
|
|
129
|
+
const data = { hello: 'world' };
|
|
130
|
+
const response = renderContext.json(data);
|
|
131
|
+
expect(response.headers.get('Content-Type')).toContain('application/json');
|
|
132
|
+
expect(await response.json()).toEqual(data);
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('should support custom status and headers', async () => {
|
|
136
|
+
const response = renderContext.json({}, { status: 201, headers: { 'X-Test': 'true' } });
|
|
137
|
+
expect(response.status).toBe(201);
|
|
138
|
+
expect(response.headers.get('X-Test')).toBe('true');
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
describe('html', () => {
|
|
143
|
+
it('should return an HTML response', async () => {
|
|
144
|
+
const html = '<div>hello</div>';
|
|
145
|
+
const response = renderContext.html(html);
|
|
146
|
+
expect(response.headers.get('Content-Type')).toContain('text/html');
|
|
147
|
+
expect(await response.text()).toBe(html);
|
|
148
|
+
});
|
|
149
|
+
});
|
|
150
|
+
});
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import type { EcoComponent, RenderContext, RenderOptions, ResponseOptions } from '../../types/public-types.ts';
|
|
2
|
+
import type {
|
|
3
|
+
IntegrationRenderer,
|
|
4
|
+
RenderToResponseContext,
|
|
5
|
+
} from '../../route-renderer/orchestration/integration-renderer.ts';
|
|
6
|
+
import type { AnyIntegrationPlugin } from '../../plugins/integration-plugin.ts';
|
|
7
|
+
import { invariant } from '../../utils/invariant.ts';
|
|
8
|
+
|
|
9
|
+
export interface CreateRenderContextOptions {
|
|
10
|
+
integrations: AnyIntegrationPlugin[];
|
|
11
|
+
rendererModules?: unknown;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Merges request locals into component props as a dedicated 'locals' property.
|
|
16
|
+
* This prevents naming conflicts and maintains type safety by keeping locals
|
|
17
|
+
* separate from other props.
|
|
18
|
+
*
|
|
19
|
+
* @param props - The original component props
|
|
20
|
+
* @param locals - Optional request-scoped locals from middleware
|
|
21
|
+
* @returns Props with locals merged in, or original props if no locals provided
|
|
22
|
+
*/
|
|
23
|
+
function mergePropsWithLocals<P>(props: P, locals: Record<string, unknown> | undefined): P {
|
|
24
|
+
if (!locals || typeof props !== 'object' || props === null) {
|
|
25
|
+
return props;
|
|
26
|
+
}
|
|
27
|
+
return { ...(props as Record<string, unknown>), locals } as P;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Creates a render context for route handlers.
|
|
32
|
+
* Provides render(), renderPartial(), json(), and html() methods that can be used
|
|
33
|
+
* within route handlers to generate responses.
|
|
34
|
+
*
|
|
35
|
+
* @param options - Configuration options including available integrations
|
|
36
|
+
* @returns A RenderContext object with methods for rendering views and creating responses
|
|
37
|
+
*/
|
|
38
|
+
export function createRenderContext(options: CreateRenderContextOptions): RenderContext {
|
|
39
|
+
const { integrations } = options;
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Resolves the appropriate renderer for a given view component based on its integration.
|
|
43
|
+
* Throws an error if the integration cannot be determined or is not found.
|
|
44
|
+
*/
|
|
45
|
+
const getRendererForView = <P>(view: EcoComponent<P>): IntegrationRenderer => {
|
|
46
|
+
const integrationName = view.config?.integration ?? view.config?.__eco?.integration;
|
|
47
|
+
invariant(
|
|
48
|
+
!!integrationName,
|
|
49
|
+
'Cannot determine integration for view. Set view.config.integration explicitly or ensure the view is defined with eco.page() in a file with a recognized extension.',
|
|
50
|
+
);
|
|
51
|
+
|
|
52
|
+
const integration = integrations.find((i) => i.name === integrationName);
|
|
53
|
+
invariant(!!integration, `No integration found for: ${integrationName}`);
|
|
54
|
+
|
|
55
|
+
return integration.initializeRenderer({
|
|
56
|
+
rendererModules: options.rendererModules,
|
|
57
|
+
}) as IntegrationRenderer;
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
const renderContext: RenderContext = {
|
|
61
|
+
async render<P>(
|
|
62
|
+
this: { locals?: Record<string, unknown> } | undefined,
|
|
63
|
+
view: EcoComponent<P>,
|
|
64
|
+
props?: P,
|
|
65
|
+
renderOptions?: RenderOptions,
|
|
66
|
+
): Promise<Response> {
|
|
67
|
+
const locals = (this as { locals?: Record<string, unknown> } | undefined)?.locals;
|
|
68
|
+
const mergedProps = mergePropsWithLocals<P>(props ?? ({} as P), locals);
|
|
69
|
+
|
|
70
|
+
const renderer = getRendererForView(view);
|
|
71
|
+
const ctx: RenderToResponseContext = {
|
|
72
|
+
partial: false,
|
|
73
|
+
status: renderOptions?.status,
|
|
74
|
+
headers: renderOptions?.headers,
|
|
75
|
+
};
|
|
76
|
+
return renderer.renderToResponse(view, mergedProps, ctx);
|
|
77
|
+
},
|
|
78
|
+
|
|
79
|
+
async renderPartial<P>(
|
|
80
|
+
this: { locals?: Record<string, unknown> } | undefined,
|
|
81
|
+
view: EcoComponent<P>,
|
|
82
|
+
props: P,
|
|
83
|
+
renderOptions?: RenderOptions,
|
|
84
|
+
): Promise<Response> {
|
|
85
|
+
const locals = (this as { locals?: Record<string, unknown> } | undefined)?.locals;
|
|
86
|
+
const mergedProps = mergePropsWithLocals<P>(props, locals);
|
|
87
|
+
|
|
88
|
+
const renderer = getRendererForView(view);
|
|
89
|
+
const ctx: RenderToResponseContext = {
|
|
90
|
+
partial: true,
|
|
91
|
+
status: renderOptions?.status,
|
|
92
|
+
headers: renderOptions?.headers,
|
|
93
|
+
};
|
|
94
|
+
return renderer.renderToResponse(view, mergedProps, ctx);
|
|
95
|
+
},
|
|
96
|
+
|
|
97
|
+
json(data: unknown, responseOptions?: ResponseOptions): Response {
|
|
98
|
+
const headers = new Headers({ 'Content-Type': 'application/json; charset=utf-8' });
|
|
99
|
+
if (responseOptions?.headers) {
|
|
100
|
+
const incomingHeaders = new Headers(responseOptions.headers);
|
|
101
|
+
incomingHeaders.forEach((value, key) => headers.set(key, value));
|
|
102
|
+
}
|
|
103
|
+
return new Response(JSON.stringify(data), {
|
|
104
|
+
status: responseOptions?.status ?? 200,
|
|
105
|
+
headers,
|
|
106
|
+
});
|
|
107
|
+
},
|
|
108
|
+
|
|
109
|
+
html(content: string, responseOptions?: ResponseOptions): Response {
|
|
110
|
+
const headers = new Headers({ 'Content-Type': 'text/html; charset=utf-8' });
|
|
111
|
+
if (responseOptions?.headers) {
|
|
112
|
+
const incomingHeaders = new Headers(responseOptions.headers);
|
|
113
|
+
incomingHeaders.forEach((value, key) => headers.set(key, value));
|
|
114
|
+
}
|
|
115
|
+
return new Response(content, {
|
|
116
|
+
status: responseOptions?.status ?? 200,
|
|
117
|
+
headers,
|
|
118
|
+
});
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
return renderContext;
|
|
123
|
+
}
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
3
|
+
import { getAppBrowserBuildPlugins, setupAppRuntimePlugins, type BuildExecutor } from '../../build/build-adapter.ts';
|
|
4
|
+
import type { EcoBuildPlugin } from '../../build/build-types.ts';
|
|
5
|
+
import { installAppRuntimeBuildExecutor } from '../../build/runtime-build-executor.ts';
|
|
6
|
+
import { RESOLVED_ASSETS_DIR } from '../../config/constants.ts';
|
|
7
|
+
import type { EcoPagesAppConfig, IClientBridge, IHmrManager } from '../../types/internal-types.ts';
|
|
8
|
+
import { ProjectWatcher } from '../../watchers/project-watcher.ts';
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Installs and returns the app-owned runtime build executor used by adapter
|
|
12
|
+
* startup and follow-up runtime work.
|
|
13
|
+
*/
|
|
14
|
+
export function installSharedRuntimeBuildExecutor(
|
|
15
|
+
appConfig: EcoPagesAppConfig,
|
|
16
|
+
options: { development: boolean },
|
|
17
|
+
): BuildExecutor {
|
|
18
|
+
return installAppRuntimeBuildExecutor(appConfig, options);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Copies app public assets into dist and ensures the resolved assets directory
|
|
23
|
+
* exists before request handling begins.
|
|
24
|
+
*/
|
|
25
|
+
export function prepareSharedRuntimePublicDir(appConfig: EcoPagesAppConfig): void {
|
|
26
|
+
const srcPublicDir = path.join(appConfig.rootDir, appConfig.srcDir, appConfig.publicDir);
|
|
27
|
+
|
|
28
|
+
if (fileSystem.exists(srcPublicDir)) {
|
|
29
|
+
fileSystem.copyDir(srcPublicDir, path.join(appConfig.rootDir, appConfig.distDir));
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
fileSystem.ensureDir(path.join(appConfig.absolutePaths.distDir, RESOLVED_ASSETS_DIR));
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Runs runtime plugin setup against app-owned config/runtime state and optional
|
|
37
|
+
* host plugin registration hooks.
|
|
38
|
+
*/
|
|
39
|
+
export async function initializeSharedRuntimePlugins(options: {
|
|
40
|
+
appConfig: EcoPagesAppConfig;
|
|
41
|
+
runtimeOrigin: string;
|
|
42
|
+
hmrManager?: IHmrManager;
|
|
43
|
+
onRuntimePlugin?: (plugin: unknown) => void;
|
|
44
|
+
}): Promise<void> {
|
|
45
|
+
await setupAppRuntimePlugins(options);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Starts shared project watching for runtime adapters.
|
|
50
|
+
*/
|
|
51
|
+
export async function startSharedProjectWatching(options: {
|
|
52
|
+
appConfig: EcoPagesAppConfig;
|
|
53
|
+
refreshRouterRoutesCallback: () => Promise<void>;
|
|
54
|
+
hmrManager: IHmrManager;
|
|
55
|
+
bridge: IClientBridge;
|
|
56
|
+
}): Promise<void> {
|
|
57
|
+
const watcher = new ProjectWatcher({
|
|
58
|
+
config: options.appConfig,
|
|
59
|
+
refreshRouterRoutesCallback: options.refreshRouterRoutesCallback,
|
|
60
|
+
hmrManager: options.hmrManager,
|
|
61
|
+
bridge: options.bridge,
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
await watcher.createWatcherSubscription();
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Binds a runtime HMR manager to app-owned plugin and integration state.
|
|
69
|
+
*/
|
|
70
|
+
export function bindSharedRuntimeHmrManager(appConfig: EcoPagesAppConfig, hmrManager: IHmrManager): EcoBuildPlugin[] {
|
|
71
|
+
const browserBuildPlugins = getAppBrowserBuildPlugins(appConfig);
|
|
72
|
+
hmrManager.setPlugins(browserBuildPlugins);
|
|
73
|
+
|
|
74
|
+
for (const integration of appConfig.integrations) {
|
|
75
|
+
integration.setHmrManager(hmrManager);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
return browserBuildPlugins;
|
|
79
|
+
}
|
|
@@ -0,0 +1,130 @@
|
|
|
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, test } from 'vitest';
|
|
6
|
+
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
7
|
+
import type { ServerAdapterResult } from '../abstract/server-adapter.ts';
|
|
8
|
+
import type { ApiHandler } from '../../types/public-types.ts';
|
|
9
|
+
import { SharedServerAdapter } from './server-adapter.ts';
|
|
10
|
+
|
|
11
|
+
const tempRoots: string[] = [];
|
|
12
|
+
|
|
13
|
+
function createTempRoot(prefix: string): string {
|
|
14
|
+
const root = fs.mkdtempSync(path.join(os.tmpdir(), `${prefix}-`));
|
|
15
|
+
tempRoots.push(root);
|
|
16
|
+
return root;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
afterEach(() => {
|
|
20
|
+
for (const root of tempRoots.splice(0)) {
|
|
21
|
+
fs.rmSync(root, { recursive: true, force: true });
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
class TestSharedServerAdapter extends SharedServerAdapter<any, ServerAdapterResult> {
|
|
26
|
+
public async initialize(): Promise<void> {}
|
|
27
|
+
public getServerOptions(): Record<string, never> {
|
|
28
|
+
return {};
|
|
29
|
+
}
|
|
30
|
+
public async buildStatic(): Promise<void> {}
|
|
31
|
+
public async createAdapter(): Promise<ServerAdapterResult> {
|
|
32
|
+
return {
|
|
33
|
+
getServerOptions: () => ({}),
|
|
34
|
+
buildStatic: async () => {},
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
public async handleRequest(request: Request): Promise<Response> {
|
|
38
|
+
return await this.handleSharedRequest(request, {
|
|
39
|
+
apiHandlers: [],
|
|
40
|
+
hmrManager: {
|
|
41
|
+
getRuntimePath: () => '',
|
|
42
|
+
getDistDir: () => this.hmrDir,
|
|
43
|
+
},
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
public async handleSharedRequestForTest(request: Request, apiHandlers: ApiHandler[]): Promise<Response> {
|
|
48
|
+
return await this.handleSharedRequest(request, {
|
|
49
|
+
apiHandlers,
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
public setRouteHandlerForTest(handleResponse: (request: Request) => Promise<Response>): void {
|
|
54
|
+
this.routeHandler = {
|
|
55
|
+
handleResponse,
|
|
56
|
+
} as any;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
constructor(
|
|
60
|
+
private readonly hmrDir: string,
|
|
61
|
+
rootDir: string,
|
|
62
|
+
) {
|
|
63
|
+
super({
|
|
64
|
+
appConfig: { rootDir } as any,
|
|
65
|
+
runtimeOrigin: 'http://localhost:3000',
|
|
66
|
+
serveOptions: {},
|
|
67
|
+
options: {},
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
test('SharedServerAdapter serves /assets/_hmr files from the HMR manager directory', async () => {
|
|
73
|
+
const rootDir = createTempRoot('ecopages-shared-server-hmr-assets');
|
|
74
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
75
|
+
const hmrDir = path.join(config.absolutePaths.workDir, 'assets', '_hmr');
|
|
76
|
+
const assetPath = path.join(hmrDir, 'pages', 'react-content.js');
|
|
77
|
+
fs.mkdirSync(path.dirname(assetPath), { recursive: true });
|
|
78
|
+
fs.writeFileSync(assetPath, 'export default 1;', 'utf8');
|
|
79
|
+
|
|
80
|
+
const adapter = new TestSharedServerAdapter(hmrDir, rootDir);
|
|
81
|
+
const response = await adapter.handleRequest(new Request('http://localhost/assets/_hmr/pages/react-content.js'));
|
|
82
|
+
|
|
83
|
+
assert.equal(response.status, 200);
|
|
84
|
+
assert.equal(response.headers.get('Content-Type'), 'application/javascript');
|
|
85
|
+
assert.equal(await response.text(), 'export default 1;');
|
|
86
|
+
assert.equal(
|
|
87
|
+
fs.existsSync(path.join(config.absolutePaths.distDir, 'assets', '_hmr', 'pages', 'react-content.js')),
|
|
88
|
+
false,
|
|
89
|
+
);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
test('SharedServerAdapter dispatches matching API handlers before filesystem routes', async () => {
|
|
93
|
+
const rootDir = createTempRoot('ecopages-shared-server-api-dispatch');
|
|
94
|
+
const adapter = new TestSharedServerAdapter('', rootDir);
|
|
95
|
+
adapter.setRouteHandlerForTest(async () => new Response('filesystem'));
|
|
96
|
+
|
|
97
|
+
const response = await adapter.handleSharedRequestForTest(
|
|
98
|
+
new Request('http://localhost/api/posts/123', { method: 'GET' }),
|
|
99
|
+
[
|
|
100
|
+
{
|
|
101
|
+
path: '/api/posts/[id]',
|
|
102
|
+
method: 'GET',
|
|
103
|
+
handler: ({ params }) => new Response(`api:${params.id}`),
|
|
104
|
+
},
|
|
105
|
+
],
|
|
106
|
+
);
|
|
107
|
+
|
|
108
|
+
assert.equal(response.status, 200);
|
|
109
|
+
assert.equal(await response.text(), 'api:123');
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
test('SharedServerAdapter falls back to the route handler when no API handler matches', async () => {
|
|
113
|
+
const rootDir = createTempRoot('ecopages-shared-server-route-fallback');
|
|
114
|
+
const adapter = new TestSharedServerAdapter('', rootDir);
|
|
115
|
+
adapter.setRouteHandlerForTest(async () => new Response('filesystem'));
|
|
116
|
+
|
|
117
|
+
const response = await adapter.handleSharedRequestForTest(
|
|
118
|
+
new Request('http://localhost/blog/post-1', { method: 'GET' }),
|
|
119
|
+
[
|
|
120
|
+
{
|
|
121
|
+
path: '/api/posts/[id]',
|
|
122
|
+
method: 'GET',
|
|
123
|
+
handler: () => new Response('api'),
|
|
124
|
+
},
|
|
125
|
+
],
|
|
126
|
+
);
|
|
127
|
+
|
|
128
|
+
assert.equal(response.status, 200);
|
|
129
|
+
assert.equal(await response.text(), 'filesystem');
|
|
130
|
+
});
|