@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,118 @@
|
|
|
1
|
+
import { createRequire } from '../../utils/locals-utils.ts';
|
|
2
|
+
import type { FileRouteMiddleware, FileRouteMiddlewareContext, RequestLocals } from '../../types/public-types.ts';
|
|
3
|
+
import type { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
4
|
+
import { ApiResponseBuilder } from './api-response.ts';
|
|
5
|
+
import { LocalsAccessError } from '../../errors/locals-access-error.ts';
|
|
6
|
+
|
|
7
|
+
export const FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS = {
|
|
8
|
+
middlewareRequiresDynamic: (filePath: string) =>
|
|
9
|
+
`[ecopages] Page middleware requires cache: 'dynamic'. Page: ${filePath}`,
|
|
10
|
+
} as const;
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Executes middleware for file-based page routes.
|
|
14
|
+
*
|
|
15
|
+
* This pipeline owns the middleware-specific rules that are distinct from route
|
|
16
|
+
* matching and rendering, including request-local storage, the middleware
|
|
17
|
+
* execution context, and the invariant that page middleware only runs for
|
|
18
|
+
* request-time dynamic rendering.
|
|
19
|
+
*/
|
|
20
|
+
export class FileRouteMiddlewarePipeline {
|
|
21
|
+
private cacheService: PageCacheService | null;
|
|
22
|
+
|
|
23
|
+
constructor(cacheService: PageCacheService | null) {
|
|
24
|
+
this.cacheService = cacheService;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Enforces the current middleware contract for file-routed pages.
|
|
29
|
+
*
|
|
30
|
+
* Middleware depends on request-scoped locals and therefore must not run when
|
|
31
|
+
* the page is treated as statically cacheable.
|
|
32
|
+
*
|
|
33
|
+
* @param input Middleware list, effective cache strategy, and page path.
|
|
34
|
+
* @throws LocalsAccessError When middleware is configured for a non-dynamic page.
|
|
35
|
+
*/
|
|
36
|
+
assertValidConfiguration(input: {
|
|
37
|
+
middleware: FileRouteMiddleware[];
|
|
38
|
+
pageCacheStrategy: 'static' | 'dynamic' | { revalidate: number; tags?: string[] };
|
|
39
|
+
filePath: string;
|
|
40
|
+
}): void {
|
|
41
|
+
if (input.middleware.length > 0 && input.pageCacheStrategy !== 'dynamic') {
|
|
42
|
+
throw new LocalsAccessError(
|
|
43
|
+
FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS.middlewareRequiresDynamic(input.filePath),
|
|
44
|
+
);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Creates the request-scoped middleware context used by page middleware.
|
|
50
|
+
*
|
|
51
|
+
* The context intentionally omits `render()` and `renderPartial()` because
|
|
52
|
+
* rendering is owned by the page route pipeline, not by middleware stages.
|
|
53
|
+
*
|
|
54
|
+
* @param input Request details and the mutable locals store.
|
|
55
|
+
* @returns Middleware execution context.
|
|
56
|
+
*/
|
|
57
|
+
createContext(input: {
|
|
58
|
+
request: Request;
|
|
59
|
+
params: Record<string, string>;
|
|
60
|
+
locals: RequestLocals;
|
|
61
|
+
}): FileRouteMiddlewareContext {
|
|
62
|
+
const context: FileRouteMiddlewareContext = {
|
|
63
|
+
request: input.request,
|
|
64
|
+
params: input.params,
|
|
65
|
+
response: new ApiResponseBuilder(),
|
|
66
|
+
server: null,
|
|
67
|
+
services: {
|
|
68
|
+
cache: this.cacheService,
|
|
69
|
+
},
|
|
70
|
+
locals: input.locals,
|
|
71
|
+
require: createRequire(() => context.locals as unknown as Record<string, unknown>),
|
|
72
|
+
json: (data, options) => {
|
|
73
|
+
const builder = new ApiResponseBuilder();
|
|
74
|
+
if (options?.status) builder.status(options.status);
|
|
75
|
+
if (options?.headers) builder.headers(options.headers);
|
|
76
|
+
return builder.json(data);
|
|
77
|
+
},
|
|
78
|
+
html: (content, options) => {
|
|
79
|
+
const builder = new ApiResponseBuilder();
|
|
80
|
+
if (options?.status) builder.status(options.status);
|
|
81
|
+
if (options?.headers) builder.headers(options.headers);
|
|
82
|
+
return builder.html(content);
|
|
83
|
+
},
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
return context;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Runs the middleware chain and eventually delegates to the render callback.
|
|
91
|
+
*
|
|
92
|
+
* Middleware may short-circuit by returning a response directly. If the chain
|
|
93
|
+
* completes, the supplied `renderResponse` callback is executed exactly once.
|
|
94
|
+
*
|
|
95
|
+
* @param input Middleware context, chain, and terminal render callback.
|
|
96
|
+
* @returns Response from middleware or final render stage.
|
|
97
|
+
*/
|
|
98
|
+
async run(input: {
|
|
99
|
+
middleware: FileRouteMiddleware[];
|
|
100
|
+
context: FileRouteMiddlewareContext;
|
|
101
|
+
renderResponse: () => Promise<Response>;
|
|
102
|
+
}): Promise<Response> {
|
|
103
|
+
if (input.middleware.length === 0) {
|
|
104
|
+
return input.renderResponse();
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
let index = 0;
|
|
108
|
+
const executeNext = async (): Promise<Response> => {
|
|
109
|
+
if (index < input.middleware.length) {
|
|
110
|
+
const current = input.middleware[index++];
|
|
111
|
+
return current(input.context, executeNext);
|
|
112
|
+
}
|
|
113
|
+
return input.renderResponse();
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
return executeNext();
|
|
117
|
+
}
|
|
118
|
+
}
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
import { beforeAll, describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
3
|
+
import {
|
|
4
|
+
FIXTURE_APP_PROJECT_DIR,
|
|
5
|
+
FIXTURE_EXISTING_CSS_FILE_IN_DIST,
|
|
6
|
+
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
7
|
+
} from '../../../__fixtures__/constants.js';
|
|
8
|
+
import { appLogger } from '../../global/app-logger.ts';
|
|
9
|
+
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
10
|
+
import { STATUS_MESSAGE } from '../../config/constants.ts';
|
|
11
|
+
import { FileSystemServerResponseFactory } from './fs-server-response-factory.ts';
|
|
12
|
+
|
|
13
|
+
let appConfig: Awaited<ReturnType<ConfigBuilder['build']>>;
|
|
14
|
+
let responseFactory: FileSystemServerResponseFactory;
|
|
15
|
+
|
|
16
|
+
describe('FileSystemServerResponseFactory', () => {
|
|
17
|
+
beforeAll(async () => {
|
|
18
|
+
appConfig = await new ConfigBuilder().setRootDir(FIXTURE_APP_PROJECT_DIR).build();
|
|
19
|
+
|
|
20
|
+
for (const integration of appConfig.integrations) {
|
|
21
|
+
integration.setConfig(appConfig);
|
|
22
|
+
integration.setRuntimeOrigin(appConfig.baseUrl);
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
responseFactory = new FileSystemServerResponseFactory({
|
|
26
|
+
options: {
|
|
27
|
+
watchMode: false,
|
|
28
|
+
},
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
describe('isHtml', () => {
|
|
33
|
+
it('should return true for text/html content type', () => {
|
|
34
|
+
const result = responseFactory.isHtml('text/html');
|
|
35
|
+
expect(result).toBe(true);
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it('should return false for text/plain content type', () => {
|
|
39
|
+
const result = responseFactory.isHtml('text/plain');
|
|
40
|
+
expect(result).toBe(false);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it('should return false for other content types', () => {
|
|
44
|
+
const result = responseFactory.isHtml('application/json');
|
|
45
|
+
expect(result).toBe(false);
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
describe('shouldEnableGzip', () => {
|
|
50
|
+
it('should return false in watch mode', () => {
|
|
51
|
+
const responseFactoryWatch = new FileSystemServerResponseFactory({
|
|
52
|
+
options: {
|
|
53
|
+
watchMode: true,
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
const result = responseFactoryWatch.shouldEnableGzip('text/javascript');
|
|
58
|
+
expect(result).toBe(false);
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
it('should return true for text/javascript content type', () => {
|
|
62
|
+
const result = responseFactory.shouldEnableGzip('text/javascript');
|
|
63
|
+
expect(result).toBe(true);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it('should return true for text/css content type', () => {
|
|
67
|
+
const result = responseFactory.shouldEnableGzip('text/css');
|
|
68
|
+
expect(result).toBe(true);
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it('should return false for other content types', () => {
|
|
72
|
+
const result = responseFactory.shouldEnableGzip('application/json');
|
|
73
|
+
expect(result).toBe(false);
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
describe('createResponseWithBody', () => {
|
|
78
|
+
it('should create a response with the given route renderer body', async () => {
|
|
79
|
+
const routeRendererBody = '<html><body>Test</body></html>';
|
|
80
|
+
const response = await responseFactory.createResponseWithBody(routeRendererBody);
|
|
81
|
+
const body = await response.text();
|
|
82
|
+
expect(body).toBe(routeRendererBody);
|
|
83
|
+
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
describe('createDefaultNotFoundResponse', () => {
|
|
88
|
+
it('should create a response with status 404 and body "file not found"', async () => {
|
|
89
|
+
const response = await responseFactory.createDefaultNotFoundResponse();
|
|
90
|
+
expect(response.status).toBe(404);
|
|
91
|
+
expect(await response.text()).toBe(STATUS_MESSAGE[404]);
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
describe('createHtmlNotFoundResponse', () => {
|
|
96
|
+
it('should create an html 404 response from a rendered body', async () => {
|
|
97
|
+
const response = await responseFactory.createHtmlNotFoundResponse('<h1>404 - Page Not Found</h1>');
|
|
98
|
+
const body = await response.text();
|
|
99
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
100
|
+
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
101
|
+
expect(response.status).toBe(404);
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
describe('createFileResponse', () => {
|
|
106
|
+
it('should create a response with the file content and content type', async () => {
|
|
107
|
+
const readFileAsBufferSpy = vi
|
|
108
|
+
.spyOn(fileSystem, 'readFileAsBuffer')
|
|
109
|
+
.mockReturnValue(Buffer.from('<svg></svg>'));
|
|
110
|
+
|
|
111
|
+
const response = await responseFactory.createFileResponse(
|
|
112
|
+
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
113
|
+
'image/svg+xml',
|
|
114
|
+
);
|
|
115
|
+
readFileAsBufferSpy.mockRestore();
|
|
116
|
+
if (!response) {
|
|
117
|
+
throw new Error('Expected static file response');
|
|
118
|
+
}
|
|
119
|
+
expect(response.headers.get('Content-Type')).toBe('image/svg+xml');
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
it('should return null if the file does not exist', async () => {
|
|
123
|
+
const response = await responseFactory.createFileResponse('/path/to/nonexistent.txt', 'text/plain');
|
|
124
|
+
expect(response).toBeNull();
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
it('should log debug for ENOENT errors', async () => {
|
|
128
|
+
const debugSpy = vi.spyOn(appLogger, 'debug');
|
|
129
|
+
const errorSpy = vi.spyOn(appLogger, 'error');
|
|
130
|
+
|
|
131
|
+
await responseFactory.createFileResponse('/path/to/nonexistent-debug.txt', 'text/plain');
|
|
132
|
+
|
|
133
|
+
expect(debugSpy).toHaveBeenCalled();
|
|
134
|
+
expect(errorSpy).not.toHaveBeenCalled();
|
|
135
|
+
|
|
136
|
+
debugSpy.mockRestore();
|
|
137
|
+
errorSpy.mockRestore();
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
it('should serve gzip file with Content-Encoding header when gzip is enabled', async () => {
|
|
141
|
+
const cssFilePath = `${appConfig.absolutePaths.distDir}/${FIXTURE_EXISTING_CSS_FILE_IN_DIST}`;
|
|
142
|
+
const existsSpy = vi
|
|
143
|
+
.spyOn(fileSystem, 'exists')
|
|
144
|
+
.mockImplementation((filePath) => filePath === `${cssFilePath}.gz`);
|
|
145
|
+
const readFileAsBufferSpy = vi
|
|
146
|
+
.spyOn(fileSystem, 'readFileAsBuffer')
|
|
147
|
+
.mockReturnValue(Buffer.from('body{color:red}'));
|
|
148
|
+
const response = await responseFactory.createFileResponse(cssFilePath, 'text/css');
|
|
149
|
+
existsSpy.mockRestore();
|
|
150
|
+
readFileAsBufferSpy.mockRestore();
|
|
151
|
+
|
|
152
|
+
if (!response) {
|
|
153
|
+
throw new Error('Expected gzip file response');
|
|
154
|
+
}
|
|
155
|
+
expect(response.headers.get('Content-Type')).toBe('text/css');
|
|
156
|
+
expect(response.headers.get('Content-Encoding')).toBe('gzip');
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
it('should not set Content-Encoding header for non-gzip content types', async () => {
|
|
160
|
+
const readFileAsBufferSpy = vi
|
|
161
|
+
.spyOn(fileSystem, 'readFileAsBuffer')
|
|
162
|
+
.mockReturnValue(Buffer.from('<svg></svg>'));
|
|
163
|
+
const response = await responseFactory.createFileResponse(
|
|
164
|
+
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
165
|
+
'image/svg+xml',
|
|
166
|
+
);
|
|
167
|
+
readFileAsBufferSpy.mockRestore();
|
|
168
|
+
|
|
169
|
+
if (!response) {
|
|
170
|
+
throw new Error('Expected non-gzip file response');
|
|
171
|
+
}
|
|
172
|
+
expect(response.headers.get('Content-Type')).toBe('image/svg+xml');
|
|
173
|
+
expect(response.headers.get('Content-Encoding')).toBeNull();
|
|
174
|
+
});
|
|
175
|
+
});
|
|
176
|
+
});
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { STATUS_MESSAGE } from '../../config/constants.ts';
|
|
2
|
+
import { appLogger } from '../../global/app-logger.ts';
|
|
3
|
+
import type { FileSystemServerOptions } from '../../types/internal-types.ts';
|
|
4
|
+
import type { RouteRendererBody } from '../../types/public-types.ts';
|
|
5
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Builds HTTP responses for static files and shared file-system fallbacks.
|
|
9
|
+
*/
|
|
10
|
+
export class FileSystemServerResponseFactory {
|
|
11
|
+
private options: FileSystemServerOptions;
|
|
12
|
+
|
|
13
|
+
constructor({ options }: { options: FileSystemServerOptions }) {
|
|
14
|
+
this.options = options;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
isHtml(contentType: string) {
|
|
18
|
+
return contentType === 'text/html';
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
shouldEnableGzip(contentType: string) {
|
|
22
|
+
if (this.options.watchMode) return false;
|
|
23
|
+
const gzipEnabledExtensions = ['text/javascript', 'text/css'];
|
|
24
|
+
return gzipEnabledExtensions.includes(contentType);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
async createResponseWithBody(
|
|
28
|
+
body: RouteRendererBody,
|
|
29
|
+
init: ResponseInit = {
|
|
30
|
+
headers: {
|
|
31
|
+
'Content-Type': 'text/html',
|
|
32
|
+
},
|
|
33
|
+
},
|
|
34
|
+
) {
|
|
35
|
+
return new Response(body as BodyInit, init);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async createDefaultNotFoundResponse() {
|
|
39
|
+
return new Response(STATUS_MESSAGE[404], {
|
|
40
|
+
status: 404,
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Wraps already-rendered HTML in a 404 response envelope.
|
|
46
|
+
*/
|
|
47
|
+
async createHtmlNotFoundResponse(body: RouteRendererBody) {
|
|
48
|
+
return await this.createResponseWithBody(body, {
|
|
49
|
+
status: 404,
|
|
50
|
+
statusText: STATUS_MESSAGE[404],
|
|
51
|
+
headers: {
|
|
52
|
+
'Content-Type': 'text/html',
|
|
53
|
+
},
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Reads a static file response, returning `null` when the file is missing.
|
|
59
|
+
*/
|
|
60
|
+
async createFileResponse(filePath: string, contentType: string): Promise<Response | null> {
|
|
61
|
+
try {
|
|
62
|
+
let file: Buffer;
|
|
63
|
+
const contentEncodingHeader: HeadersInit = {};
|
|
64
|
+
|
|
65
|
+
if (this.shouldEnableGzip(contentType)) {
|
|
66
|
+
const gzipPath = `${filePath}.gz`;
|
|
67
|
+
if (fileSystem.exists(gzipPath)) {
|
|
68
|
+
file = fileSystem.readFileAsBuffer(gzipPath);
|
|
69
|
+
contentEncodingHeader['Content-Encoding'] = 'gzip';
|
|
70
|
+
contentEncodingHeader['Vary'] = 'Accept-Encoding';
|
|
71
|
+
} else {
|
|
72
|
+
appLogger.debug('Gzip file not found, serving uncompressed', gzipPath);
|
|
73
|
+
file = fileSystem.readFileAsBuffer(filePath);
|
|
74
|
+
}
|
|
75
|
+
} else {
|
|
76
|
+
file = fileSystem.readFileAsBuffer(filePath);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
return await this.createResponseWithBody(file as unknown as BodyInit, {
|
|
80
|
+
headers: {
|
|
81
|
+
'Content-Type': contentType,
|
|
82
|
+
...contentEncodingHeader,
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
} catch (error) {
|
|
86
|
+
const err = error as Error & { code?: string; cause?: { code?: string } };
|
|
87
|
+
const code = err.code || err.cause?.code;
|
|
88
|
+
if (code === 'ENOENT') {
|
|
89
|
+
appLogger.debug('File not found', filePath);
|
|
90
|
+
} else {
|
|
91
|
+
appLogger.error('Error reading file', filePath, err);
|
|
92
|
+
}
|
|
93
|
+
return null;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
@@ -0,0 +1,311 @@
|
|
|
1
|
+
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
3
|
+
import path from 'node:path';
|
|
4
|
+
import { APP_TEST_ROUTES, FIXTURE_APP_PROJECT_DIR, INDEX_TEMPLATE_FILE } from '../../../__fixtures__/constants.ts';
|
|
5
|
+
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
6
|
+
import type { MatchResult } from '../../types/internal-types.ts';
|
|
7
|
+
import { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
|
|
8
|
+
import { RouteRegistry } from '../../router/server/route-registry.ts';
|
|
9
|
+
import { MemoryCacheStore } from '../../services/cache/memory-cache-store.ts';
|
|
10
|
+
import { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
11
|
+
import { FileSystemServerResponseFactory } from './fs-server-response-factory.ts';
|
|
12
|
+
import { FileSystemResponseMatcher } from './fs-server-response-matcher.ts';
|
|
13
|
+
|
|
14
|
+
const appConfig = await new ConfigBuilder().setRootDir(FIXTURE_APP_PROJECT_DIR).build();
|
|
15
|
+
|
|
16
|
+
for (const integration of appConfig.integrations) {
|
|
17
|
+
integration.setConfig(appConfig);
|
|
18
|
+
integration.setRuntimeOrigin(appConfig.baseUrl);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
const router = new RouteRegistry({
|
|
22
|
+
pagesDir: path.join(appConfig.rootDir, appConfig.srcDir, appConfig.pagesDir),
|
|
23
|
+
appConfig,
|
|
24
|
+
origin: appConfig.baseUrl,
|
|
25
|
+
templatesExt: appConfig.templatesExt,
|
|
26
|
+
buildMode: false,
|
|
27
|
+
pageModuleAdapter: {
|
|
28
|
+
loadPageModule: vi.fn(async () => ({})),
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
await router.init();
|
|
33
|
+
|
|
34
|
+
const routeRendererFactory = new RouteRendererFactory({
|
|
35
|
+
appConfig,
|
|
36
|
+
runtimeOrigin: appConfig.baseUrl,
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
const fileSystemResponseFactory = new FileSystemServerResponseFactory({
|
|
40
|
+
options: {
|
|
41
|
+
watchMode: false,
|
|
42
|
+
},
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
describe('FileSystemResponseMatcher', () => {
|
|
46
|
+
describe('without cache service', () => {
|
|
47
|
+
const matcherWithoutCache = new FileSystemResponseMatcher({
|
|
48
|
+
appConfig,
|
|
49
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
50
|
+
router,
|
|
51
|
+
routeRendererFactory,
|
|
52
|
+
fileSystemResponseFactory,
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it('should return custom 404 page for unmatched request URL', async () => {
|
|
56
|
+
const requestUrl = APP_TEST_ROUTES.nonExistentFile;
|
|
57
|
+
const response = await matcherWithoutCache.handleNoMatch(requestUrl);
|
|
58
|
+
const body = await response.text();
|
|
59
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('should handle match with disabled cache headers', async () => {
|
|
63
|
+
const match: MatchResult = {
|
|
64
|
+
requestedPathname: APP_TEST_ROUTES.index,
|
|
65
|
+
templateRoute: {
|
|
66
|
+
kind: 'exact',
|
|
67
|
+
pathname: APP_TEST_ROUTES.index,
|
|
68
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
69
|
+
},
|
|
70
|
+
params: {},
|
|
71
|
+
query: {},
|
|
72
|
+
};
|
|
73
|
+
const response = await matcherWithoutCache.handleMatch(match);
|
|
74
|
+
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
75
|
+
expect(response.headers.get('X-Cache')).toBe('DISABLED');
|
|
76
|
+
expect(response.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
it('should return null for getCacheService when not configured', () => {
|
|
80
|
+
expect(matcherWithoutCache.getCacheService()).toBeNull();
|
|
81
|
+
});
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
describe('with cache service', () => {
|
|
85
|
+
const cacheService = new PageCacheService({
|
|
86
|
+
store: new MemoryCacheStore(),
|
|
87
|
+
enabled: true,
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
const matcherWithCache = new FileSystemResponseMatcher({
|
|
91
|
+
appConfig,
|
|
92
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
93
|
+
router,
|
|
94
|
+
routeRendererFactory,
|
|
95
|
+
fileSystemResponseFactory,
|
|
96
|
+
cacheService,
|
|
97
|
+
defaultCacheStrategy: 'static',
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
it('should return cache service via getCacheService', () => {
|
|
101
|
+
expect(matcherWithCache.getCacheService()).toBe(cacheService);
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
it('should return X-Cache header on first request (MISS)', async () => {
|
|
105
|
+
const match: MatchResult = {
|
|
106
|
+
requestedPathname: '/cache-test-miss',
|
|
107
|
+
templateRoute: {
|
|
108
|
+
kind: 'exact',
|
|
109
|
+
pathname: '/cache-test-miss',
|
|
110
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
111
|
+
},
|
|
112
|
+
params: {},
|
|
113
|
+
query: {},
|
|
114
|
+
};
|
|
115
|
+
const response = await matcherWithCache.handleMatch(match);
|
|
116
|
+
expect(response.headers.get('X-Cache')).toBe('MISS');
|
|
117
|
+
expect(response.headers.get('Cache-Control')).toBe('public, max-age=31536000, immutable');
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it('should return X-Cache HIT on second request to same path', async () => {
|
|
121
|
+
const uniquePath = `/cache-test-hit-${Date.now()}`;
|
|
122
|
+
const match: MatchResult = {
|
|
123
|
+
requestedPathname: uniquePath,
|
|
124
|
+
templateRoute: {
|
|
125
|
+
kind: 'exact',
|
|
126
|
+
pathname: uniquePath,
|
|
127
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
128
|
+
},
|
|
129
|
+
params: {},
|
|
130
|
+
query: {},
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
const response1 = await matcherWithCache.handleMatch(match);
|
|
134
|
+
const response2 = await matcherWithCache.handleMatch(match);
|
|
135
|
+
|
|
136
|
+
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
137
|
+
expect(response2.headers.get('X-Cache')).toBe('HIT');
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
it('should cache different paths separately', async () => {
|
|
141
|
+
const match1: MatchResult = {
|
|
142
|
+
requestedPathname: '/path-a',
|
|
143
|
+
templateRoute: {
|
|
144
|
+
kind: 'exact',
|
|
145
|
+
pathname: '/path-a',
|
|
146
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
147
|
+
},
|
|
148
|
+
params: {},
|
|
149
|
+
query: {},
|
|
150
|
+
};
|
|
151
|
+
const match2: MatchResult = {
|
|
152
|
+
requestedPathname: '/path-b',
|
|
153
|
+
templateRoute: {
|
|
154
|
+
kind: 'exact',
|
|
155
|
+
pathname: '/path-b',
|
|
156
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
157
|
+
},
|
|
158
|
+
params: {},
|
|
159
|
+
query: {},
|
|
160
|
+
};
|
|
161
|
+
|
|
162
|
+
const response1 = await matcherWithCache.handleMatch(match1);
|
|
163
|
+
const response2 = await matcherWithCache.handleMatch(match2);
|
|
164
|
+
|
|
165
|
+
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
166
|
+
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
it('should include query params in cache key', async () => {
|
|
170
|
+
const basePath = `/search-${Date.now()}`;
|
|
171
|
+
const matchWithQuery: MatchResult = {
|
|
172
|
+
requestedPathname: basePath,
|
|
173
|
+
templateRoute: {
|
|
174
|
+
kind: 'exact',
|
|
175
|
+
pathname: basePath,
|
|
176
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
177
|
+
},
|
|
178
|
+
params: {},
|
|
179
|
+
query: { q: 'test' },
|
|
180
|
+
};
|
|
181
|
+
const matchWithDifferentQuery: MatchResult = {
|
|
182
|
+
requestedPathname: basePath,
|
|
183
|
+
templateRoute: {
|
|
184
|
+
kind: 'exact',
|
|
185
|
+
pathname: basePath,
|
|
186
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
187
|
+
},
|
|
188
|
+
params: {},
|
|
189
|
+
query: { q: 'other' },
|
|
190
|
+
};
|
|
191
|
+
|
|
192
|
+
const firstCall = await matcherWithCache.handleMatch(matchWithQuery);
|
|
193
|
+
const response1 = await matcherWithCache.handleMatch(matchWithQuery);
|
|
194
|
+
const response2 = await matcherWithCache.handleMatch(matchWithDifferentQuery);
|
|
195
|
+
|
|
196
|
+
expect(firstCall.headers.get('X-Cache')).toBe('MISS');
|
|
197
|
+
expect(response1.headers.get('X-Cache')).toBe('HIT');
|
|
198
|
+
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
199
|
+
});
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
describe('with dynamic cache strategy', () => {
|
|
203
|
+
const cacheService = new PageCacheService({
|
|
204
|
+
store: new MemoryCacheStore(),
|
|
205
|
+
enabled: true,
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
const dynamicMatcher = new FileSystemResponseMatcher({
|
|
209
|
+
appConfig,
|
|
210
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
211
|
+
router,
|
|
212
|
+
routeRendererFactory,
|
|
213
|
+
fileSystemResponseFactory,
|
|
214
|
+
cacheService,
|
|
215
|
+
defaultCacheStrategy: 'dynamic',
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
it('should bypass cache entirely for dynamic strategy', async () => {
|
|
219
|
+
const match: MatchResult = {
|
|
220
|
+
requestedPathname: '/dynamic-page',
|
|
221
|
+
templateRoute: {
|
|
222
|
+
kind: 'exact',
|
|
223
|
+
pathname: '/dynamic-page',
|
|
224
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
225
|
+
},
|
|
226
|
+
params: {},
|
|
227
|
+
query: {},
|
|
228
|
+
};
|
|
229
|
+
|
|
230
|
+
const response1 = await dynamicMatcher.handleMatch(match);
|
|
231
|
+
const response2 = await dynamicMatcher.handleMatch(match);
|
|
232
|
+
|
|
233
|
+
expect(response1.headers.get('X-Cache')).toBe('DISABLED');
|
|
234
|
+
expect(response2.headers.get('X-Cache')).toBe('DISABLED');
|
|
235
|
+
expect(response1.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
236
|
+
});
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
describe('handleNoMatch content type behavior', () => {
|
|
240
|
+
const matcher = new FileSystemResponseMatcher({
|
|
241
|
+
appConfig,
|
|
242
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
243
|
+
router,
|
|
244
|
+
routeRendererFactory,
|
|
245
|
+
fileSystemResponseFactory,
|
|
246
|
+
});
|
|
247
|
+
|
|
248
|
+
it('should return custom 404 page for known extension like .html', async () => {
|
|
249
|
+
const response = await matcher.handleNoMatch('/non-existent-page.html');
|
|
250
|
+
const body = await response.text();
|
|
251
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
it('should return custom 404 page for page requests without extension', async () => {
|
|
255
|
+
const response = await matcher.handleNoMatch('/non-existent-page');
|
|
256
|
+
const body = await response.text();
|
|
257
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
it('should return custom 404 page for unknown extensions', async () => {
|
|
261
|
+
const response = await matcher.handleNoMatch('/page.xyz');
|
|
262
|
+
const body = await response.text();
|
|
263
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
it('should return custom 404 page for trailing dot', async () => {
|
|
267
|
+
const response = await matcher.handleNoMatch('/page.');
|
|
268
|
+
const body = await response.text();
|
|
269
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
it('should serve text/plain files from disk', async () => {
|
|
273
|
+
const readFileAsBuffer = vi.spyOn(fileSystem, 'readFileAsBuffer').mockReturnValue(Buffer.from('robots'));
|
|
274
|
+
const response = await matcher.handleNoMatch('/robots.txt');
|
|
275
|
+
readFileAsBuffer.mockRestore();
|
|
276
|
+
expect(response.headers.get('Content-Type')).toBe('text/plain');
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
it('should return custom 404 page for non-existent static files', async () => {
|
|
280
|
+
const response = await matcher.handleNoMatch('/non-existent.txt');
|
|
281
|
+
const body = await response.text();
|
|
282
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
283
|
+
});
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
describe('internal transpile output', () => {
|
|
287
|
+
it('should inspect page modules through the owning route renderer', async () => {
|
|
288
|
+
const matcher = new FileSystemResponseMatcher({
|
|
289
|
+
appConfig,
|
|
290
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
291
|
+
router,
|
|
292
|
+
routeRendererFactory,
|
|
293
|
+
fileSystemResponseFactory,
|
|
294
|
+
});
|
|
295
|
+
|
|
296
|
+
const loadPageModule = vi.fn(async () => ({}));
|
|
297
|
+
(matcher as any).routeRendererFactory = {
|
|
298
|
+
getPageRenderer: vi.fn(() => ({
|
|
299
|
+
loadPageModule,
|
|
300
|
+
})),
|
|
301
|
+
};
|
|
302
|
+
|
|
303
|
+
await (matcher as any).importPageModule(INDEX_TEMPLATE_FILE);
|
|
304
|
+
|
|
305
|
+
expect((matcher as any).routeRendererFactory.getPageRenderer).toHaveBeenCalledWith(INDEX_TEMPLATE_FILE);
|
|
306
|
+
expect(loadPageModule).toHaveBeenCalledWith(INDEX_TEMPLATE_FILE, {
|
|
307
|
+
cacheScope: 'request-metadata',
|
|
308
|
+
});
|
|
309
|
+
});
|
|
310
|
+
});
|
|
311
|
+
});
|