@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,146 @@
|
|
|
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 '../../public-types.ts';
|
|
6
|
+
|
|
7
|
+
describe('createRenderContext', () => {
|
|
8
|
+
const RenderToResponse = vi.fn(() => Promise.resolve(new Response('rendered')));
|
|
9
|
+
|
|
10
|
+
const Renderer = {
|
|
11
|
+
name: '-renderer',
|
|
12
|
+
renderToResponse: RenderToResponse,
|
|
13
|
+
} as unknown as IntegrationRenderer;
|
|
14
|
+
|
|
15
|
+
const ExplicitRenderer = {
|
|
16
|
+
name: 'explicit-renderer',
|
|
17
|
+
renderToResponse: RenderToResponse,
|
|
18
|
+
} as unknown as IntegrationRenderer;
|
|
19
|
+
|
|
20
|
+
const InitializeRenderer = vi.fn(() => Renderer);
|
|
21
|
+
|
|
22
|
+
const Plugin = {
|
|
23
|
+
name: '-integration',
|
|
24
|
+
initializeRenderer: InitializeRenderer,
|
|
25
|
+
} as unknown as IntegrationPlugin;
|
|
26
|
+
|
|
27
|
+
const ExplicitInitializeRenderer = vi.fn(() => ExplicitRenderer);
|
|
28
|
+
|
|
29
|
+
const ExplicitPlugin = {
|
|
30
|
+
name: 'explicit-renderer',
|
|
31
|
+
initializeRenderer: ExplicitInitializeRenderer,
|
|
32
|
+
} as unknown as IntegrationPlugin;
|
|
33
|
+
|
|
34
|
+
const ViewFn = ((props: { foo: string }) => `<div>${props.foo}</div>`) as EcoFunctionComponent<
|
|
35
|
+
{ foo: string },
|
|
36
|
+
string
|
|
37
|
+
>;
|
|
38
|
+
ViewFn.config = {
|
|
39
|
+
__eco: {
|
|
40
|
+
id: 'test',
|
|
41
|
+
file: '/some/dir/-view.ts',
|
|
42
|
+
integration: '-integration',
|
|
43
|
+
},
|
|
44
|
+
};
|
|
45
|
+
|
|
46
|
+
const renderContext = createRenderContext({
|
|
47
|
+
integrations: [Plugin, ExplicitPlugin],
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('should create a render context with methods', () => {
|
|
51
|
+
expect(renderContext.render).toBeDefined();
|
|
52
|
+
expect(renderContext.renderPartial).toBeDefined();
|
|
53
|
+
expect(renderContext.json).toBeDefined();
|
|
54
|
+
expect(renderContext.html).toBeDefined();
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
describe('render', () => {
|
|
58
|
+
it('should call renderer.renderToResponse with partial: false', async () => {
|
|
59
|
+
RenderToResponse.mockClear();
|
|
60
|
+
InitializeRenderer.mockClear();
|
|
61
|
+
ExplicitInitializeRenderer.mockClear();
|
|
62
|
+
const props = { foo: 'bar' };
|
|
63
|
+
const options = { status: 201, headers: { 'X-Custom': '1' } };
|
|
64
|
+
|
|
65
|
+
const response = await renderContext.render(ViewFn, props, options);
|
|
66
|
+
|
|
67
|
+
expect(InitializeRenderer).toHaveBeenCalled();
|
|
68
|
+
expect(RenderToResponse).toHaveBeenCalledWith(ViewFn, props, {
|
|
69
|
+
partial: false,
|
|
70
|
+
status: 201,
|
|
71
|
+
headers: { 'X-Custom': '1' },
|
|
72
|
+
});
|
|
73
|
+
expect(response instanceof Response).toBe(true);
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it('should throw if view integration is unknown', async () => {
|
|
77
|
+
const badViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
78
|
+
badViewFn.config = { __eco: { id: 'test', file: '/bad-view.ts', integration: 'unknown' } };
|
|
79
|
+
await expect(renderContext.render(badViewFn, {})).rejects.toThrow('No integration found for: unknown');
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it('should throw if view integration is missing', async () => {
|
|
83
|
+
const badViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
84
|
+
badViewFn.config = { __eco: undefined };
|
|
85
|
+
await expect(renderContext.render(badViewFn, {})).rejects.toThrow('Cannot determine integration for view');
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
it('should prefer explicit config.integration over injected metadata', async () => {
|
|
89
|
+
RenderToResponse.mockClear();
|
|
90
|
+
InitializeRenderer.mockClear();
|
|
91
|
+
ExplicitInitializeRenderer.mockClear();
|
|
92
|
+
|
|
93
|
+
const explicitViewFn = (() => '<div></div>') as EcoFunctionComponent<{}, string>;
|
|
94
|
+
explicitViewFn.config = {
|
|
95
|
+
integration: 'explicit-renderer',
|
|
96
|
+
__eco: { id: 'test', file: '/some/file.tsx', integration: '-integration' },
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
await renderContext.render(explicitViewFn, {});
|
|
100
|
+
|
|
101
|
+
expect(ExplicitInitializeRenderer).toHaveBeenCalled();
|
|
102
|
+
expect(InitializeRenderer).not.toHaveBeenCalled();
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
describe('renderPartial', () => {
|
|
107
|
+
it('should call renderer.renderToResponse with partial: true', async () => {
|
|
108
|
+
RenderToResponse.mockClear();
|
|
109
|
+
const props = { foo: 'bar' };
|
|
110
|
+
const options = { status: 200 };
|
|
111
|
+
|
|
112
|
+
await renderContext.renderPartial(ViewFn, props, options);
|
|
113
|
+
|
|
114
|
+
expect(InitializeRenderer).toHaveBeenCalled();
|
|
115
|
+
expect(RenderToResponse).toHaveBeenCalledWith(ViewFn, props, {
|
|
116
|
+
partial: true,
|
|
117
|
+
status: 200,
|
|
118
|
+
headers: undefined,
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
describe('json', () => {
|
|
124
|
+
it('should return a JSON response', async () => {
|
|
125
|
+
const data = { hello: 'world' };
|
|
126
|
+
const response = renderContext.json(data);
|
|
127
|
+
expect(response.headers.get('Content-Type')).toContain('application/json');
|
|
128
|
+
expect(await response.json()).toEqual(data);
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
it('should support custom status and headers', async () => {
|
|
132
|
+
const response = renderContext.json({}, { status: 201, headers: { 'X-Test': 'true' } });
|
|
133
|
+
expect(response.status).toBe(201);
|
|
134
|
+
expect(response.headers.get('X-Test')).toBe('true');
|
|
135
|
+
});
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
describe('html', () => {
|
|
139
|
+
it('should return an HTML response', async () => {
|
|
140
|
+
const html = '<div>hello</div>';
|
|
141
|
+
const response = renderContext.html(html);
|
|
142
|
+
expect(response.headers.get('Content-Type')).toContain('text/html');
|
|
143
|
+
expect(await response.text()).toBe(html);
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
});
|
|
@@ -1,5 +1,8 @@
|
|
|
1
1
|
import type { EcoComponent, RenderContext, RenderOptions, ResponseOptions } from '../../public-types.ts';
|
|
2
|
-
import type {
|
|
2
|
+
import type {
|
|
3
|
+
IntegrationRenderer,
|
|
4
|
+
RenderToResponseContext,
|
|
5
|
+
} from '../../route-renderer/orchestration/integration-renderer.ts';
|
|
3
6
|
import type { IntegrationPlugin } from '../../plugins/integration-plugin.ts';
|
|
4
7
|
import { invariant } from '../../utils/invariant.ts';
|
|
5
8
|
|
|
@@ -51,10 +54,15 @@ export function createRenderContext(options: CreateRenderContextOptions): Render
|
|
|
51
54
|
return integration.initializeRenderer();
|
|
52
55
|
};
|
|
53
56
|
|
|
54
|
-
|
|
55
|
-
async render<P>(
|
|
57
|
+
const renderContext: RenderContext = {
|
|
58
|
+
async render<P>(
|
|
59
|
+
this: { locals?: Record<string, unknown> } | undefined,
|
|
60
|
+
view: EcoComponent<P>,
|
|
61
|
+
props?: P,
|
|
62
|
+
renderOptions?: RenderOptions,
|
|
63
|
+
): Promise<Response> {
|
|
56
64
|
const locals = (this as { locals?: Record<string, unknown> } | undefined)?.locals;
|
|
57
|
-
const mergedProps = mergePropsWithLocals(props, locals);
|
|
65
|
+
const mergedProps = mergePropsWithLocals<P>(props ?? ({} as P), locals);
|
|
58
66
|
|
|
59
67
|
const renderer = getRendererForView(view);
|
|
60
68
|
const ctx: RenderToResponseContext = {
|
|
@@ -65,9 +73,14 @@ export function createRenderContext(options: CreateRenderContextOptions): Render
|
|
|
65
73
|
return renderer.renderToResponse(view, mergedProps, ctx);
|
|
66
74
|
},
|
|
67
75
|
|
|
68
|
-
async renderPartial<P>(
|
|
76
|
+
async renderPartial<P>(
|
|
77
|
+
this: { locals?: Record<string, unknown> } | undefined,
|
|
78
|
+
view: EcoComponent<P>,
|
|
79
|
+
props: P,
|
|
80
|
+
renderOptions?: RenderOptions,
|
|
81
|
+
): Promise<Response> {
|
|
69
82
|
const locals = (this as { locals?: Record<string, unknown> } | undefined)?.locals;
|
|
70
|
-
const mergedProps = mergePropsWithLocals(props, locals);
|
|
83
|
+
const mergedProps = mergePropsWithLocals<P>(props, locals);
|
|
71
84
|
|
|
72
85
|
const renderer = getRendererForView(view);
|
|
73
86
|
const ctx: RenderToResponseContext = {
|
|
@@ -102,4 +115,6 @@ export function createRenderContext(options: CreateRenderContextOptions): Render
|
|
|
102
115
|
});
|
|
103
116
|
},
|
|
104
117
|
};
|
|
118
|
+
|
|
119
|
+
return renderContext;
|
|
105
120
|
}
|
|
@@ -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 '../../constants.ts';
|
|
7
|
+
import type { EcoPagesAppConfig, IClientBridge, IHmrManager } from '../../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,77 @@
|
|
|
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 { SharedServerAdapter } from './server-adapter.ts';
|
|
9
|
+
|
|
10
|
+
const tempRoots: string[] = [];
|
|
11
|
+
|
|
12
|
+
function createTempRoot(prefix: string): string {
|
|
13
|
+
const root = fs.mkdtempSync(path.join(os.tmpdir(), `${prefix}-`));
|
|
14
|
+
tempRoots.push(root);
|
|
15
|
+
return root;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
afterEach(() => {
|
|
19
|
+
for (const root of tempRoots.splice(0)) {
|
|
20
|
+
fs.rmSync(root, { recursive: true, force: true });
|
|
21
|
+
}
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
class TestSharedServerAdapter extends SharedServerAdapter<any, ServerAdapterResult> {
|
|
25
|
+
public async initialize(): Promise<void> {}
|
|
26
|
+
public getServerOptions(): Record<string, never> {
|
|
27
|
+
return {};
|
|
28
|
+
}
|
|
29
|
+
public async buildStatic(): Promise<void> {}
|
|
30
|
+
public async createAdapter(): Promise<ServerAdapterResult> {
|
|
31
|
+
return {
|
|
32
|
+
getServerOptions: () => ({}),
|
|
33
|
+
buildStatic: async () => {},
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
public async handleRequest(request: Request): Promise<Response> {
|
|
37
|
+
return await this.handleSharedRequest(request, {
|
|
38
|
+
apiHandlers: [],
|
|
39
|
+
hmrManager: {
|
|
40
|
+
getRuntimePath: () => '',
|
|
41
|
+
getDistDir: () => this.hmrDir,
|
|
42
|
+
},
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
constructor(
|
|
47
|
+
private readonly hmrDir: string,
|
|
48
|
+
rootDir: string,
|
|
49
|
+
) {
|
|
50
|
+
super({
|
|
51
|
+
appConfig: { rootDir } as any,
|
|
52
|
+
runtimeOrigin: 'http://localhost:3000',
|
|
53
|
+
serveOptions: {},
|
|
54
|
+
options: {},
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
test('SharedServerAdapter serves /assets/_hmr files from the HMR manager directory', async () => {
|
|
60
|
+
const rootDir = createTempRoot('ecopages-shared-server-hmr-assets');
|
|
61
|
+
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
62
|
+
const hmrDir = path.join(config.absolutePaths.workDir, 'assets', '_hmr');
|
|
63
|
+
const assetPath = path.join(hmrDir, 'pages', 'react-content.js');
|
|
64
|
+
fs.mkdirSync(path.dirname(assetPath), { recursive: true });
|
|
65
|
+
fs.writeFileSync(assetPath, 'export default 1;', 'utf8');
|
|
66
|
+
|
|
67
|
+
const adapter = new TestSharedServerAdapter(hmrDir, rootDir);
|
|
68
|
+
const response = await adapter.handleRequest(new Request('http://localhost/assets/_hmr/pages/react-content.js'));
|
|
69
|
+
|
|
70
|
+
assert.equal(response.status, 200);
|
|
71
|
+
assert.equal(response.headers.get('Content-Type'), 'application/javascript');
|
|
72
|
+
assert.equal(await response.text(), 'export default 1;');
|
|
73
|
+
assert.equal(
|
|
74
|
+
fs.existsSync(path.join(config.absolutePaths.distDir, 'assets', '_hmr', 'pages', 'react-content.js')),
|
|
75
|
+
false,
|
|
76
|
+
);
|
|
77
|
+
});
|
|
@@ -2,11 +2,11 @@ import path from 'node:path';
|
|
|
2
2
|
import { AbstractServerAdapter } from '../abstract/server-adapter.ts';
|
|
3
3
|
import type { ServerAdapterOptions, ServerAdapterResult } from '../abstract/server-adapter.ts';
|
|
4
4
|
import { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
|
|
5
|
-
import { FSRouter } from '../../router/fs-router.ts';
|
|
6
|
-
import { FSRouterScanner } from '../../router/fs-router-scanner.ts';
|
|
5
|
+
import { FSRouter } from '../../router/server/fs-router.ts';
|
|
6
|
+
import { FSRouterScanner } from '../../router/server/fs-router-scanner.ts';
|
|
7
7
|
import { MemoryCacheStore } from '../../services/cache/memory-cache-store.ts';
|
|
8
8
|
import { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
9
|
-
import { SchemaValidationService } from '../../services/schema-validation-service.ts';
|
|
9
|
+
import { SchemaValidationService } from '../../services/validation/schema-validation-service.ts';
|
|
10
10
|
import { StaticSiteGenerator } from '../../static-site-generator/static-site-generator.ts';
|
|
11
11
|
import { ServerStaticBuilder } from './server-static-builder.ts';
|
|
12
12
|
import { ExplicitStaticRouteMatcher } from './explicit-static-route-matcher.ts';
|
|
@@ -18,6 +18,7 @@ import { createRequire } from '../../utils/locals-utils.ts';
|
|
|
18
18
|
import { HttpError } from '../../errors/http-error.ts';
|
|
19
19
|
import { ApiResponseBuilder } from './api-response.ts';
|
|
20
20
|
import { appLogger } from '../../global/app-logger.ts';
|
|
21
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
21
22
|
import type {
|
|
22
23
|
ApiHandler,
|
|
23
24
|
ApiHandlerContext,
|
|
@@ -39,6 +40,41 @@ export abstract class SharedServerAdapter<
|
|
|
39
40
|
protected staticBuilder!: ServerStaticBuilder;
|
|
40
41
|
protected readonly schemaValidator = new SchemaValidationService();
|
|
41
42
|
|
|
43
|
+
protected async initializeSharedRouteHandling(options: {
|
|
44
|
+
staticRoutes: StaticRoute[];
|
|
45
|
+
hmrManager?: any;
|
|
46
|
+
}): Promise<void> {
|
|
47
|
+
await this.initSharedRouter();
|
|
48
|
+
this.configureSharedResponseHandlers(options.staticRoutes, options.hmrManager);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
protected createSharedWatchRefreshCallback(options: {
|
|
52
|
+
staticRoutes: StaticRoute[];
|
|
53
|
+
hmrManager?: any;
|
|
54
|
+
onRoutesReady?: () => Promise<void> | void;
|
|
55
|
+
onError?: (error: Error) => Promise<void> | void;
|
|
56
|
+
}): () => Promise<void> {
|
|
57
|
+
return async () => {
|
|
58
|
+
try {
|
|
59
|
+
await this.initializeSharedRouteHandling({
|
|
60
|
+
staticRoutes: options.staticRoutes,
|
|
61
|
+
hmrManager: options.hmrManager,
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
if (options.onRoutesReady) {
|
|
65
|
+
await options.onRoutesReady();
|
|
66
|
+
}
|
|
67
|
+
} catch (error) {
|
|
68
|
+
if (options.onError) {
|
|
69
|
+
await options.onError(error instanceof Error ? error : new Error(String(error)));
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
throw error;
|
|
74
|
+
}
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
|
|
42
78
|
/**
|
|
43
79
|
* Scans the filesystem and dynamically constructs the universal router map.
|
|
44
80
|
*
|
|
@@ -108,6 +144,7 @@ export abstract class SharedServerAdapter<
|
|
|
108
144
|
}
|
|
109
145
|
|
|
110
146
|
this.fileSystemResponseMatcher = new FileSystemResponseMatcher({
|
|
147
|
+
appConfig: this.appConfig,
|
|
111
148
|
router: this.router,
|
|
112
149
|
routeRendererFactory: this.routeRendererFactory,
|
|
113
150
|
fileSystemResponseFactory,
|
|
@@ -418,7 +455,6 @@ export abstract class SharedServerAdapter<
|
|
|
418
455
|
|
|
419
456
|
if (url.pathname === '/_hmr_runtime.js' && context.hmrManager) {
|
|
420
457
|
const runtimePath = context.hmrManager.getRuntimePath();
|
|
421
|
-
const fileSystem = (await import('@ecopages/file-system')).fileSystem;
|
|
422
458
|
if (fileSystem.exists(runtimePath)) {
|
|
423
459
|
return new Response(fileSystem.readFileAsBuffer(runtimePath) as BodyInit, {
|
|
424
460
|
headers: { 'Content-Type': 'application/javascript' },
|
|
@@ -426,6 +462,17 @@ export abstract class SharedServerAdapter<
|
|
|
426
462
|
}
|
|
427
463
|
}
|
|
428
464
|
|
|
465
|
+
if (url.pathname.startsWith('/assets/_hmr/') && context.hmrManager) {
|
|
466
|
+
const relativePath = url.pathname.slice('/assets/_hmr/'.length);
|
|
467
|
+
const assetPath = path.join(context.hmrManager.getDistDir(), relativePath);
|
|
468
|
+
|
|
469
|
+
if (fileSystem.exists(assetPath)) {
|
|
470
|
+
return new Response(fileSystem.readFileAsBuffer(assetPath) as BodyInit, {
|
|
471
|
+
headers: { 'Content-Type': 'application/javascript' },
|
|
472
|
+
});
|
|
473
|
+
}
|
|
474
|
+
}
|
|
475
|
+
|
|
429
476
|
const apiMatch = this.matchApiHandler(request, context.apiHandlers);
|
|
430
477
|
if (apiMatch) {
|
|
431
478
|
return this.executeApiHandler(
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { ServerRouteHandler } from './server-route-handler';
|
|
3
|
+
import type { FSRouter } from '../../router/server/fs-router';
|
|
4
|
+
import type { FileSystemResponseMatcher } from './fs-server-response-matcher';
|
|
5
|
+
import type { IHmrManager } from '../../public-types';
|
|
6
|
+
|
|
7
|
+
function createMockDependencies() {
|
|
8
|
+
const Router = {
|
|
9
|
+
match: vi.fn(() => null),
|
|
10
|
+
} as unknown as FSRouter;
|
|
11
|
+
|
|
12
|
+
const FileSystemResponseMatcher = {
|
|
13
|
+
handleMatch: vi.fn(() => Promise.resolve(new Response('Matched Content'))),
|
|
14
|
+
handleNoMatch: vi.fn(() => Promise.resolve(new Response('Not Found', { status: 404 }))),
|
|
15
|
+
} as unknown as FileSystemResponseMatcher;
|
|
16
|
+
|
|
17
|
+
const HmrManager = {
|
|
18
|
+
isEnabled: vi.fn(() => true),
|
|
19
|
+
broadcast: vi.fn(),
|
|
20
|
+
} as unknown as IHmrManager;
|
|
21
|
+
|
|
22
|
+
return {
|
|
23
|
+
Router,
|
|
24
|
+
FileSystemResponseMatcher,
|
|
25
|
+
HmrManager,
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
describe('ServerRouteHandler', () => {
|
|
30
|
+
describe('handleResponse', () => {
|
|
31
|
+
it('should delegate to fileSystemResponseMatcher when route matches', async () => {
|
|
32
|
+
const { Router, FileSystemResponseMatcher } = createMockDependencies();
|
|
33
|
+
const handler = new ServerRouteHandler({
|
|
34
|
+
router: Router,
|
|
35
|
+
fileSystemResponseMatcher: FileSystemResponseMatcher,
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
Router.match = vi.fn(
|
|
39
|
+
() =>
|
|
40
|
+
({
|
|
41
|
+
/* match */
|
|
42
|
+
}) as any,
|
|
43
|
+
);
|
|
44
|
+
const request = new Request('http://localhost/test');
|
|
45
|
+
const response = await handler.handleResponse(request);
|
|
46
|
+
|
|
47
|
+
expect(FileSystemResponseMatcher.handleMatch).toHaveBeenCalled();
|
|
48
|
+
expect(response.status).toBe(200);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
it('should delegate to handleNoMatch when route does not match', async () => {
|
|
52
|
+
const { Router, FileSystemResponseMatcher } = createMockDependencies();
|
|
53
|
+
const handler = new ServerRouteHandler({
|
|
54
|
+
router: Router,
|
|
55
|
+
fileSystemResponseMatcher: FileSystemResponseMatcher,
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
const request = new Request('http://localhost/unknown');
|
|
59
|
+
const response = await handler.handleResponse(request);
|
|
60
|
+
|
|
61
|
+
expect(FileSystemResponseMatcher.handleNoMatch).toHaveBeenCalled();
|
|
62
|
+
expect(response.status).toBe(404);
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
it('should inject HMR script in watch mode for HTML responses', async () => {
|
|
66
|
+
const { Router, FileSystemResponseMatcher, HmrManager } = createMockDependencies();
|
|
67
|
+
const handler = new ServerRouteHandler({
|
|
68
|
+
router: Router,
|
|
69
|
+
fileSystemResponseMatcher: FileSystemResponseMatcher,
|
|
70
|
+
watch: true,
|
|
71
|
+
hmrManager: HmrManager,
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
Router.match = vi.fn(() => ({}) as any);
|
|
75
|
+
FileSystemResponseMatcher.handleMatch = vi.fn(() =>
|
|
76
|
+
Promise.resolve(
|
|
77
|
+
new Response('<html><body></body></html>', { headers: { 'Content-Type': 'text/html' } }),
|
|
78
|
+
),
|
|
79
|
+
);
|
|
80
|
+
|
|
81
|
+
const request = new Request('http://localhost/test');
|
|
82
|
+
const response = await handler.handleResponse(request);
|
|
83
|
+
const text = await response.text();
|
|
84
|
+
|
|
85
|
+
expect(text).toContain("import '/_hmr_runtime.js'");
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
describe('handleNoMatch', () => {
|
|
90
|
+
it('should broadcast error if handleNoMatch throws', async () => {
|
|
91
|
+
const { Router, FileSystemResponseMatcher, HmrManager } = createMockDependencies();
|
|
92
|
+
const handler = new ServerRouteHandler({
|
|
93
|
+
router: Router,
|
|
94
|
+
fileSystemResponseMatcher: FileSystemResponseMatcher,
|
|
95
|
+
watch: true,
|
|
96
|
+
hmrManager: HmrManager,
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
FileSystemResponseMatcher.handleNoMatch = vi.fn(() => {
|
|
100
|
+
throw new Error('Test Error');
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
const request = new Request('http://localhost/unknown');
|
|
104
|
+
const response = await handler.handleNoMatch(request);
|
|
105
|
+
|
|
106
|
+
expect(response.status).toBe(500);
|
|
107
|
+
expect(HmrManager.broadcast).toHaveBeenCalledWith({ type: 'error', message: 'Test Error' });
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
});
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
import type { IHmrManager } from '../../public-types';
|
|
2
|
-
import type { FSRouter } from '../../router/fs-router';
|
|
2
|
+
import type { FSRouter } from '../../router/server/fs-router';
|
|
3
3
|
import type { ExplicitStaticRouteMatcher } from './explicit-static-route-matcher';
|
|
4
4
|
import type { FileSystemResponseMatcher } from './fs-server-response-matcher';
|
|
5
5
|
import { appLogger } from '../../global/app-logger';
|
|
6
6
|
import { HttpError } from '../../errors/http-error';
|
|
7
|
+
import { injectHmrRuntimeIntoHtmlResponse, isHtmlResponse, shouldInjectHmrHtmlResponse } from './hmr-html-response';
|
|
7
8
|
|
|
8
9
|
/**
|
|
9
10
|
* Configuration parameters for ServerRouteHandler.
|
|
@@ -63,7 +64,7 @@ export class ServerRouteHandler {
|
|
|
63
64
|
* @returns true if in watch mode and HMR manager is enabled
|
|
64
65
|
*/
|
|
65
66
|
shouldInjectHmrScript(): boolean {
|
|
66
|
-
return this.watch
|
|
67
|
+
return shouldInjectHmrHtmlResponse(this.watch, this.hmrManager);
|
|
67
68
|
}
|
|
68
69
|
|
|
69
70
|
/**
|
|
@@ -73,8 +74,7 @@ export class ServerRouteHandler {
|
|
|
73
74
|
* @returns true if Content-Type header starts with 'text/html'
|
|
74
75
|
*/
|
|
75
76
|
isHtmlResponse(response: Response): boolean {
|
|
76
|
-
|
|
77
|
-
return contentType !== null && contentType.startsWith('text/html');
|
|
77
|
+
return isHtmlResponse(response);
|
|
78
78
|
}
|
|
79
79
|
|
|
80
80
|
/**
|
|
@@ -118,20 +118,7 @@ export class ServerRouteHandler {
|
|
|
118
118
|
*/
|
|
119
119
|
private async maybeInjectHmrScript(response: Response): Promise<Response> {
|
|
120
120
|
if (this.shouldInjectHmrScript() && this.isHtmlResponse(response)) {
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
const hmrScript = `<script type="module">import '/_hmr_runtime.js';</script>`;
|
|
124
|
-
|
|
125
|
-
const updatedHtml = html.replace(/<\/html>/i, `${hmrScript}</html>`);
|
|
126
|
-
|
|
127
|
-
const headers = new Headers(response.headers);
|
|
128
|
-
headers.delete('Content-Length');
|
|
129
|
-
|
|
130
|
-
return new Response(updatedHtml, {
|
|
131
|
-
status: response.status,
|
|
132
|
-
statusText: response.statusText,
|
|
133
|
-
headers,
|
|
134
|
-
});
|
|
121
|
+
return injectHmrRuntimeIntoHtmlResponse(response);
|
|
135
122
|
}
|
|
136
123
|
|
|
137
124
|
return response;
|