@ecopages/core 0.2.0-alpha.7 → 0.2.0-alpha.8
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/CHANGELOG.md +31 -0
- package/package.json +212 -92
- package/src/adapters/abstract/application-adapter.d.ts +168 -0
- package/src/adapters/abstract/application-adapter.js +109 -0
- package/src/adapters/abstract/router-adapter.d.ts +26 -0
- package/src/adapters/abstract/router-adapter.js +5 -0
- package/src/adapters/abstract/server-adapter.d.ts +69 -0
- package/src/adapters/abstract/server-adapter.js +15 -0
- package/src/adapters/bun/client-bridge.d.ts +34 -0
- package/src/adapters/bun/client-bridge.js +48 -0
- package/src/adapters/bun/create-app.d.ts +60 -0
- package/src/adapters/bun/create-app.js +117 -0
- package/src/adapters/bun/hmr-manager.d.ts +143 -0
- package/src/adapters/bun/hmr-manager.js +334 -0
- package/src/adapters/bun/index.d.ts +2 -0
- package/src/adapters/bun/index.js +8 -0
- package/src/adapters/bun/server-adapter.d.ts +155 -0
- package/src/adapters/bun/server-adapter.js +373 -0
- package/src/adapters/bun/server-lifecycle.d.ts +63 -0
- package/src/adapters/bun/server-lifecycle.js +92 -0
- package/src/adapters/index.d.ts +6 -0
- package/src/adapters/index.js +14 -0
- package/src/adapters/node/bootstrap-dependency-resolver.d.ts +44 -0
- package/src/adapters/node/bootstrap-dependency-resolver.js +172 -0
- package/src/adapters/node/create-app.d.ts +21 -0
- package/src/adapters/node/create-app.js +143 -0
- package/src/adapters/node/index.d.ts +6 -0
- package/src/adapters/node/index.js +11 -0
- package/src/adapters/node/node-client-bridge.d.ts +26 -0
- package/src/adapters/node/node-client-bridge.js +66 -0
- package/src/adapters/node/node-hmr-manager.d.ts +133 -0
- package/src/adapters/node/node-hmr-manager.js +312 -0
- package/src/adapters/node/runtime-adapter.d.ts +46 -0
- package/src/adapters/node/runtime-adapter.js +306 -0
- package/src/adapters/node/server-adapter.d.ts +161 -0
- package/src/adapters/node/server-adapter.js +358 -0
- package/src/adapters/node/static-content-server.d.ts +60 -0
- package/src/adapters/node/static-content-server.js +194 -0
- package/src/adapters/node/write-runtime-manifest.d.ts +26 -0
- package/src/adapters/node/write-runtime-manifest.js +12 -0
- package/src/adapters/shared/api-response.d.ts +52 -0
- package/src/adapters/shared/api-response.js +96 -0
- package/src/adapters/shared/application-adapter.d.ts +18 -0
- package/src/adapters/shared/application-adapter.js +90 -0
- package/src/adapters/shared/define-api-handler.d.ts +25 -0
- package/src/adapters/shared/define-api-handler.js +15 -0
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
- package/src/adapters/shared/explicit-static-route-matcher.js +103 -0
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
- package/src/adapters/shared/file-route-middleware-pipeline.js +99 -0
- package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
- package/src/adapters/shared/fs-server-response-factory.js +97 -0
- package/src/adapters/shared/fs-server-response-matcher.d.ts +75 -0
- package/src/adapters/shared/fs-server-response-matcher.js +160 -0
- package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
- package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
- package/src/adapters/shared/hmr-html-response.d.ts +22 -0
- package/src/adapters/shared/hmr-html-response.js +32 -0
- package/src/adapters/shared/render-context.d.ts +14 -0
- package/src/adapters/shared/render-context.js +70 -0
- package/src/adapters/shared/runtime-bootstrap.d.ts +38 -0
- package/src/adapters/shared/runtime-bootstrap.js +43 -0
- package/src/adapters/shared/server-adapter.d.ts +97 -0
- package/src/adapters/shared/server-adapter.js +386 -0
- package/src/adapters/shared/server-route-handler.d.ts +89 -0
- package/src/adapters/shared/server-route-handler.js +111 -0
- package/src/adapters/shared/server-static-builder.d.ts +70 -0
- package/src/adapters/shared/server-static-builder.js +99 -0
- package/src/build/build-adapter.d.ts +186 -0
- package/src/build/build-adapter.js +168 -0
- package/src/build/build-manifest.d.ts +27 -0
- package/src/build/build-manifest.js +30 -0
- package/src/build/build-types.d.ts +57 -0
- package/src/build/build-types.js +0 -0
- package/src/build/dev-build-coordinator.d.ts +74 -0
- package/src/build/dev-build-coordinator.js +161 -0
- package/src/build/esbuild-build-adapter.d.ts +72 -0
- package/src/build/esbuild-build-adapter.js +422 -0
- package/src/build/runtime-build-executor.d.ts +13 -0
- package/src/build/runtime-build-executor.js +20 -0
- package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
- package/src/build/runtime-specifier-alias-plugin.js +31 -0
- package/src/config/config-builder.d.ts +238 -0
- package/src/config/config-builder.js +565 -0
- package/src/constants.d.ts +45 -0
- package/src/constants.js +25 -0
- package/src/create-app.d.ts +17 -0
- package/src/create-app.js +66 -0
- package/src/dev/sc-server.d.ts +30 -0
- package/src/dev/sc-server.js +111 -0
- package/src/eco/component-render-context.d.ts +105 -0
- package/src/eco/component-render-context.js +87 -0
- package/src/eco/eco.d.ts +9 -0
- package/src/eco/eco.js +114 -0
- package/src/eco/eco.types.d.ts +178 -0
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.utils.d.ts +40 -0
- package/src/eco/eco.utils.js +40 -0
- package/src/eco/global-injector-map.d.ts +16 -0
- package/src/eco/global-injector-map.js +80 -0
- package/src/eco/lazy-injector-map.d.ts +8 -0
- package/src/eco/lazy-injector-map.js +70 -0
- package/src/eco/module-dependencies.d.ts +18 -0
- package/src/eco/module-dependencies.js +49 -0
- package/src/errors/http-error.d.ts +31 -0
- package/src/errors/http-error.js +50 -0
- package/src/errors/index.d.ts +2 -0
- package/src/errors/index.js +4 -0
- package/src/errors/locals-access-error.d.ts +4 -0
- package/src/errors/locals-access-error.js +9 -0
- package/src/global/app-logger.d.ts +2 -0
- package/src/global/app-logger.js +6 -0
- package/src/hmr/client/hmr-runtime.d.ts +5 -0
- package/src/hmr/client/hmr-runtime.js +109 -0
- package/src/hmr/hmr-strategy.d.ts +159 -0
- package/src/hmr/hmr-strategy.js +29 -0
- package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.postcss.test.e2e.js +31 -0
- package/src/hmr/hmr.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.test.e2e.js +43 -0
- package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
- package/src/hmr/strategies/default-hmr-strategy.js +34 -0
- package/src/hmr/strategies/js-hmr-strategy.d.ts +139 -0
- package/src/hmr/strategies/js-hmr-strategy.js +178 -0
- package/src/index.browser.d.ts +3 -0
- package/src/index.browser.js +4 -0
- package/src/index.d.ts +5 -0
- package/src/index.js +10 -0
- package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.js +62 -0
- package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
- package/src/integrations/ghtml/ghtml.plugin.js +21 -0
- package/src/internal-types.d.ts +221 -0
- package/src/internal-types.js +0 -0
- package/src/plugins/alias-resolver-plugin.d.ts +2 -0
- package/src/plugins/alias-resolver-plugin.js +53 -0
- package/src/plugins/eco-component-meta-plugin.d.ts +97 -0
- package/src/plugins/eco-component-meta-plugin.js +157 -0
- package/src/plugins/integration-plugin.d.ts +136 -0
- package/src/plugins/integration-plugin.js +133 -0
- package/src/plugins/processor.d.ts +95 -0
- package/src/plugins/processor.js +136 -0
- package/src/plugins/runtime-capability.d.ts +9 -0
- package/src/plugins/runtime-capability.js +0 -0
- package/src/public-types.d.ts +1149 -0
- package/src/public-types.js +0 -0
- package/src/route-renderer/component-graph/component-graph-executor.d.ts +32 -0
- package/src/route-renderer/component-graph/component-graph-executor.js +31 -0
- package/src/route-renderer/component-graph/component-graph.d.ts +42 -0
- package/src/route-renderer/component-graph/component-graph.js +72 -0
- package/src/route-renderer/component-graph/component-marker.d.ts +52 -0
- package/src/route-renderer/component-graph/component-marker.js +46 -0
- package/src/route-renderer/component-graph/component-reference.d.ts +10 -0
- package/src/route-renderer/component-graph/component-reference.js +19 -0
- package/src/route-renderer/component-graph/marker-graph-resolver.d.ts +77 -0
- package/src/route-renderer/component-graph/marker-graph-resolver.js +95 -0
- package/src/route-renderer/orchestration/integration-renderer.d.ts +372 -0
- package/src/route-renderer/orchestration/integration-renderer.js +589 -0
- package/src/route-renderer/orchestration/render-execution.service.d.ts +103 -0
- package/src/route-renderer/orchestration/render-execution.service.js +121 -0
- package/src/route-renderer/orchestration/render-preparation.service.d.ts +121 -0
- package/src/route-renderer/orchestration/render-preparation.service.js +332 -0
- package/src/route-renderer/page-loading/dependency-resolver.d.ts +35 -0
- package/src/route-renderer/page-loading/dependency-resolver.js +442 -0
- package/src/route-renderer/page-loading/page-module-loader.d.ts +87 -0
- package/src/route-renderer/page-loading/page-module-loader.js +124 -0
- package/src/route-renderer/route-renderer.d.ts +61 -0
- package/src/route-renderer/route-renderer.js +87 -0
- package/src/router/client/link-intent.js +34 -0
- package/src/router/client/link-intent.test.browser.d.ts +1 -0
- package/src/router/client/link-intent.test.browser.js +43 -0
- package/src/router/client/navigation-coordinator.d.ts +149 -0
- package/src/router/client/navigation-coordinator.js +215 -0
- package/src/router/server/fs-router-scanner.d.ts +41 -0
- package/src/router/server/fs-router-scanner.js +156 -0
- package/src/router/server/fs-router.d.ts +26 -0
- package/src/router/server/fs-router.js +100 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +120 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.js +331 -0
- package/src/services/assets/asset-processing-service/asset.factory.d.ts +17 -0
- package/src/services/assets/asset-processing-service/asset.factory.js +82 -0
- package/src/services/assets/asset-processing-service/assets.types.d.ts +89 -0
- 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 +55 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +48 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
- package/src/services/assets/asset-processing-service/index.d.ts +5 -0
- package/src/services/assets/asset-processing-service/index.js +5 -0
- package/src/services/assets/asset-processing-service/processor.interface.d.ts +22 -0
- package/src/services/assets/asset-processing-service/processor.interface.js +6 -0
- package/src/services/assets/asset-processing-service/processor.registry.d.ts +8 -0
- package/src/services/assets/asset-processing-service/processor.registry.js +15 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +24 -0
- package/src/services/assets/asset-processing-service/processors/base/base-processor.js +64 -0
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +17 -0
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +72 -0
- package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/index.js +5 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +57 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +76 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +75 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +66 -0
- package/src/services/assets/browser-bundle.service.d.ts +32 -0
- package/src/services/assets/browser-bundle.service.js +33 -0
- package/src/services/cache/cache.types.d.ts +107 -0
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/index.d.ts +7 -0
- package/src/services/cache/index.js +7 -0
- package/src/services/cache/memory-cache-store.d.ts +42 -0
- package/src/services/cache/memory-cache-store.js +98 -0
- package/src/services/cache/page-cache-service.d.ts +70 -0
- package/src/services/cache/page-cache-service.js +152 -0
- package/src/services/cache/page-request-cache-coordinator.service.d.ts +75 -0
- package/src/services/cache/page-request-cache-coordinator.service.js +109 -0
- package/src/services/html/html-rewriter-provider.service.d.ts +37 -0
- package/src/services/html/html-rewriter-provider.service.js +65 -0
- package/src/services/html/html-transformer.service.d.ts +77 -0
- package/src/services/html/html-transformer.service.js +221 -0
- package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
- package/src/services/invalidation/development-invalidation.service.js +189 -0
- package/src/services/module-loading/app-server-module-transpiler.service.d.ts +16 -0
- package/src/services/module-loading/app-server-module-transpiler.service.js +34 -0
- package/src/services/module-loading/page-module-import.service.d.ts +71 -0
- package/src/services/module-loading/page-module-import.service.js +132 -0
- package/src/services/module-loading/server-loader.service.d.ts +96 -0
- package/src/services/module-loading/server-loader.service.js +32 -0
- package/src/services/module-loading/server-module-transpiler.service.d.ts +69 -0
- package/src/services/module-loading/server-module-transpiler.service.js +61 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.d.ts +35 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.js +60 -0
- package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
- package/src/services/runtime-state/dev-graph.service.js +162 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
- package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +69 -0
- package/src/services/runtime-state/runtime-specifier-registry.service.js +37 -0
- package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
- package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
- package/src/services/validation/schema-validation-service.d.ts +122 -0
- package/src/services/validation/schema-validation-service.js +101 -0
- package/src/services/validation/standard-schema.types.d.ts +65 -0
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/static-site-generator/static-site-generator.d.ts +109 -0
- package/src/static-site-generator/static-site-generator.js +353 -0
- package/src/utils/css.d.ts +1 -0
- package/src/utils/css.js +7 -0
- package/src/utils/deep-merge.d.ts +14 -0
- package/src/utils/deep-merge.js +32 -0
- package/src/utils/hash.d.ts +1 -0
- package/src/utils/hash.js +7 -0
- package/src/utils/html.d.ts +1 -0
- package/src/utils/html.js +4 -0
- package/src/utils/invariant.d.ts +5 -0
- package/src/utils/invariant.js +11 -0
- package/src/utils/locals-utils.d.ts +15 -0
- package/src/utils/locals-utils.js +24 -0
- package/src/utils/parse-cli-args.d.ts +24 -0
- package/src/utils/parse-cli-args.js +47 -0
- package/src/utils/path-utils.module.d.ts +5 -0
- package/src/utils/path-utils.module.js +14 -0
- package/src/utils/resolve-work-dir.d.ts +11 -0
- package/src/utils/resolve-work-dir.js +31 -0
- package/src/utils/runtime.d.ts +11 -0
- package/src/utils/runtime.js +40 -0
- package/src/utils/server-utils.module.d.ts +19 -0
- package/src/utils/server-utils.module.js +56 -0
- package/src/watchers/project-watcher.d.ts +136 -0
- package/src/watchers/project-watcher.js +281 -0
- package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
- package/src/watchers/project-watcher.test-helpers.js +52 -0
- package/src/adapters/bun/hmr-manager.test.ts +0 -267
- package/src/adapters/node/bootstrap-dependency-resolver.test.ts +0 -282
- package/src/adapters/node/node-client-bridge.test.ts +0 -198
- package/src/adapters/node/node-hmr-manager.test.ts +0 -322
- package/src/adapters/node/runtime-adapter.test.ts +0 -868
- package/src/adapters/node/static-content-server.test.ts +0 -60
- package/src/adapters/shared/api-response.test.ts +0 -97
- package/src/adapters/shared/explicit-static-route-matcher.test.ts +0 -381
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +0 -90
- package/src/adapters/shared/fs-server-response-factory.test.ts +0 -187
- package/src/adapters/shared/fs-server-response-matcher.test.ts +0 -286
- package/src/adapters/shared/hmr-manager.contract.test.ts +0 -196
- package/src/adapters/shared/hmr-manager.dispatch.test.ts +0 -220
- package/src/adapters/shared/render-context.test.ts +0 -146
- package/src/adapters/shared/server-adapter.test.ts +0 -77
- package/src/adapters/shared/server-route-handler.test.ts +0 -110
- package/src/adapters/shared/server-static-builder.test.ts +0 -316
- package/src/build/build-adapter-serialization.test.ts +0 -268
- package/src/build/build-adapter.test.ts +0 -815
- package/src/build/runtime-specifier-alias-plugin.test.ts +0 -43
- package/src/config/config-builder.test.ts +0 -410
- package/src/eco/eco.test.ts +0 -678
- package/src/eco/eco.utils.test.ts +0 -124
- package/src/eco/global-injector-map.test.ts +0 -42
- package/src/eco/lazy-injector-map.test.ts +0 -66
- package/src/eco/module-dependencies.test.ts +0 -30
- package/src/errors/http-error.test.ts +0 -134
- package/src/global/utils.test.ts +0 -12
- 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/hmr-strategy.test.ts +0 -124
- package/src/hmr/strategies/js-hmr-strategy.test.ts +0 -335
- package/src/integrations/ghtml/ghtml-renderer.test.ts +0 -63
- package/src/plugins/alias-resolver-plugin.test.ts +0 -41
- package/src/plugins/eco-component-meta-plugin.test.ts +0 -380
- package/src/plugins/integration-plugin.test.ts +0 -111
- package/src/plugins/processor.test.ts +0 -148
- package/src/route-renderer/component-graph/component-graph-executor.test.ts +0 -41
- package/src/route-renderer/component-graph/component-graph.test.ts +0 -63
- package/src/route-renderer/component-graph/component-marker.test.ts +0 -73
- package/src/route-renderer/component-graph/marker-graph-resolver.test.ts +0 -135
- package/src/route-renderer/orchestration/integration-renderer.test.ts +0 -936
- package/src/route-renderer/orchestration/render-execution.service.test.ts +0 -97
- package/src/route-renderer/orchestration/render-preparation.service.test.ts +0 -235
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +0 -345
- package/src/route-renderer/page-loading/page-module-loader.test.ts +0 -96
- package/src/router/client/navigation-coordinator.test.ts +0 -237
- package/src/router/server/fs-router-scanner.test.ts +0 -83
- package/src/router/server/fs-router.test.ts +0 -214
- package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +0 -385
- package/src/services/assets/asset-processing-service/asset.factory.test.ts +0 -63
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +0 -72
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +0 -67
- package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +0 -59
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +0 -286
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +0 -227
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +0 -199
- package/src/services/assets/browser-bundle.service.test.ts +0 -36
- package/src/services/cache/memory-cache-store.test.ts +0 -225
- package/src/services/cache/page-cache-service.test.ts +0 -175
- package/src/services/cache/page-request-cache-coordinator.service.test.ts +0 -79
- package/src/services/html/html-rewriter-provider.service.test.ts +0 -183
- package/src/services/html/html-transformer.service.test.ts +0 -378
- package/src/services/invalidation/development-invalidation.service.test.ts +0 -77
- package/src/services/module-loading/page-module-import.service.test.ts +0 -253
- package/src/services/module-loading/server-loader.service.test.ts +0 -161
- package/src/services/module-loading/server-module-transpiler.service.test.ts +0 -115
- package/src/services/runtime-manifest/node-runtime-manifest.service.test.ts +0 -95
- package/src/services/validation/schema-validation-service.test.ts +0 -223
- package/src/static-site-generator/static-site-generator.test.ts +0 -307
- package/src/utils/deep-merge.test.ts +0 -114
- package/src/utils/invariant.test.ts +0 -22
- package/src/utils/path-utils.test.ts +0 -15
- package/src/utils/server-utils.test.ts +0 -38
- package/src/watchers/project-watcher.integration.test.ts +0 -337
- package/src/watchers/project-watcher.test.ts +0 -678
|
@@ -1,187 +0,0 @@
|
|
|
1
|
-
import { beforeAll, describe, expect, it, vi } from 'vitest';
|
|
2
|
-
import {
|
|
3
|
-
FIXTURE_APP_PROJECT_DIR,
|
|
4
|
-
FIXTURE_EXISTING_CSS_FILE_IN_DIST,
|
|
5
|
-
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
6
|
-
} from '../../../__fixtures__/constants.js';
|
|
7
|
-
import { appLogger } from '../../global/app-logger.ts';
|
|
8
|
-
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
9
|
-
import { STATUS_MESSAGE } from '../../constants.ts';
|
|
10
|
-
import { RouteRendererFactory } from '../../route-renderer/route-renderer.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
|
-
appConfig,
|
|
27
|
-
routeRendererFactory: new RouteRendererFactory({
|
|
28
|
-
appConfig,
|
|
29
|
-
runtimeOrigin: appConfig.baseUrl,
|
|
30
|
-
}),
|
|
31
|
-
options: {
|
|
32
|
-
watchMode: false,
|
|
33
|
-
},
|
|
34
|
-
});
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
describe('isHtml', () => {
|
|
38
|
-
it('should return true for text/html content type', () => {
|
|
39
|
-
const result = responseFactory.isHtml('text/html');
|
|
40
|
-
expect(result).toBe(true);
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
it('should return false for text/plain content type', () => {
|
|
44
|
-
const result = responseFactory.isHtml('text/plain');
|
|
45
|
-
expect(result).toBe(false);
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
it('should return false for other content types', () => {
|
|
49
|
-
const result = responseFactory.isHtml('application/json');
|
|
50
|
-
expect(result).toBe(false);
|
|
51
|
-
});
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
describe('shouldEnableGzip', () => {
|
|
55
|
-
it('should return false in watch mode', () => {
|
|
56
|
-
const responseFactoryWatch = new FileSystemServerResponseFactory({
|
|
57
|
-
appConfig,
|
|
58
|
-
routeRendererFactory: new RouteRendererFactory({
|
|
59
|
-
appConfig,
|
|
60
|
-
runtimeOrigin: appConfig.baseUrl,
|
|
61
|
-
}),
|
|
62
|
-
options: {
|
|
63
|
-
watchMode: true,
|
|
64
|
-
},
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
const result = responseFactoryWatch.shouldEnableGzip('text/javascript');
|
|
68
|
-
expect(result).toBe(false);
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
it('should return true for text/javascript content type', () => {
|
|
72
|
-
const result = responseFactory.shouldEnableGzip('text/javascript');
|
|
73
|
-
expect(result).toBe(true);
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
it('should return true for text/css content type', () => {
|
|
77
|
-
const result = responseFactory.shouldEnableGzip('text/css');
|
|
78
|
-
expect(result).toBe(true);
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
it('should return false for other content types', () => {
|
|
82
|
-
const result = responseFactory.shouldEnableGzip('application/json');
|
|
83
|
-
expect(result).toBe(false);
|
|
84
|
-
});
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
describe('createResponseWithBody', () => {
|
|
88
|
-
it('should create a response with the given route renderer body', async () => {
|
|
89
|
-
const routeRendererBody = '<html><body>Test</body></html>';
|
|
90
|
-
const response = await responseFactory.createResponseWithBody(routeRendererBody);
|
|
91
|
-
const body = await response.text();
|
|
92
|
-
expect(body).toBe(routeRendererBody);
|
|
93
|
-
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
94
|
-
});
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
describe('createDefaultNotFoundResponse', () => {
|
|
98
|
-
it('should create a response with status 404 and body "file not found"', async () => {
|
|
99
|
-
const response = await responseFactory.createDefaultNotFoundResponse();
|
|
100
|
-
expect(response.status).toBe(404);
|
|
101
|
-
expect(await response.text()).toBe(STATUS_MESSAGE[404]);
|
|
102
|
-
});
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
describe('createCustomNotFoundResponse', () => {
|
|
106
|
-
it('should create a response with status 404 if error404 template file does not exist', async () => {
|
|
107
|
-
const customAppConfig = {
|
|
108
|
-
...appConfig,
|
|
109
|
-
absolutePaths: {
|
|
110
|
-
...appConfig.absolutePaths,
|
|
111
|
-
error404TemplatePath: 'non-existent-file',
|
|
112
|
-
},
|
|
113
|
-
};
|
|
114
|
-
|
|
115
|
-
const responseFactoryNo404Template = new FileSystemServerResponseFactory({
|
|
116
|
-
appConfig: customAppConfig,
|
|
117
|
-
routeRendererFactory: new RouteRendererFactory({
|
|
118
|
-
appConfig: customAppConfig,
|
|
119
|
-
runtimeOrigin: customAppConfig.baseUrl,
|
|
120
|
-
}),
|
|
121
|
-
options: {
|
|
122
|
-
watchMode: false,
|
|
123
|
-
},
|
|
124
|
-
});
|
|
125
|
-
|
|
126
|
-
const response = await responseFactoryNo404Template.createCustomNotFoundResponse();
|
|
127
|
-
expect(response.status).toBe(404);
|
|
128
|
-
expect(await response.text()).toBe(STATUS_MESSAGE[404]);
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
it('should create a response with the route renderer body if error404 template file exists', async () => {
|
|
132
|
-
const response = await responseFactory.createCustomNotFoundResponse();
|
|
133
|
-
const body = await response.text();
|
|
134
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
135
|
-
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
136
|
-
expect(response.status).toBe(404);
|
|
137
|
-
});
|
|
138
|
-
});
|
|
139
|
-
|
|
140
|
-
describe('createFileResponse', () => {
|
|
141
|
-
it('should create a response with the file content and content type', async () => {
|
|
142
|
-
const response = await responseFactory.createFileResponse(
|
|
143
|
-
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
144
|
-
'image/svg+xml',
|
|
145
|
-
);
|
|
146
|
-
expect(response.headers.get('Content-Type')).toBe('image/svg+xml');
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
it('should return custom 404 page if the file does not exist', async () => {
|
|
150
|
-
const response = await responseFactory.createFileResponse('/path/to/nonexistent.txt', 'text/plain');
|
|
151
|
-
const body = await response.text();
|
|
152
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
153
|
-
expect(response.status).toBe(404);
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
it('should log debug for ENOENT errors', async () => {
|
|
157
|
-
const debugSpy = vi.spyOn(appLogger, 'debug');
|
|
158
|
-
const errorSpy = vi.spyOn(appLogger, 'error');
|
|
159
|
-
|
|
160
|
-
await responseFactory.createFileResponse('/path/to/nonexistent-debug.txt', 'text/plain');
|
|
161
|
-
|
|
162
|
-
expect(debugSpy).toHaveBeenCalled();
|
|
163
|
-
expect(errorSpy).not.toHaveBeenCalled();
|
|
164
|
-
|
|
165
|
-
debugSpy.mockRestore();
|
|
166
|
-
errorSpy.mockRestore();
|
|
167
|
-
});
|
|
168
|
-
|
|
169
|
-
it('should serve gzip file with Content-Encoding header when gzip is enabled', async () => {
|
|
170
|
-
const cssFilePath = `${appConfig.absolutePaths.distDir}/${FIXTURE_EXISTING_CSS_FILE_IN_DIST}`;
|
|
171
|
-
const response = await responseFactory.createFileResponse(cssFilePath, 'text/css');
|
|
172
|
-
|
|
173
|
-
expect(response.headers.get('Content-Type')).toBe('text/css');
|
|
174
|
-
expect(response.headers.get('Content-Encoding')).toBe('gzip');
|
|
175
|
-
});
|
|
176
|
-
|
|
177
|
-
it('should not set Content-Encoding header for non-gzip content types', async () => {
|
|
178
|
-
const response = await responseFactory.createFileResponse(
|
|
179
|
-
FIXTURE_EXISTING_SVG_FILE_IN_DIST_PATH,
|
|
180
|
-
'image/svg+xml',
|
|
181
|
-
);
|
|
182
|
-
|
|
183
|
-
expect(response.headers.get('Content-Type')).toBe('image/svg+xml');
|
|
184
|
-
expect(response.headers.get('Content-Encoding')).toBeNull();
|
|
185
|
-
});
|
|
186
|
-
});
|
|
187
|
-
});
|
|
@@ -1,286 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
-
import path from 'node:path';
|
|
3
|
-
import { APP_TEST_ROUTES, FIXTURE_APP_PROJECT_DIR, INDEX_TEMPLATE_FILE } from '../../../__fixtures__/constants.ts';
|
|
4
|
-
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
5
|
-
import type { MatchResult } from '../../internal-types.ts';
|
|
6
|
-
import { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
|
|
7
|
-
import { FSRouter } from '../../router/server/fs-router.ts';
|
|
8
|
-
import { FSRouterScanner } from '../../router/server/fs-router-scanner.ts';
|
|
9
|
-
import { MemoryCacheStore } from '../../services/cache/memory-cache-store.ts';
|
|
10
|
-
import { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
11
|
-
import { resolveInternalExecutionDir } from '../../utils/resolve-work-dir.ts';
|
|
12
|
-
import { FileSystemServerResponseFactory } from './fs-server-response-factory.ts';
|
|
13
|
-
import { FileSystemResponseMatcher } from './fs-server-response-matcher.ts';
|
|
14
|
-
|
|
15
|
-
const appConfig = await new ConfigBuilder().setRootDir(FIXTURE_APP_PROJECT_DIR).build();
|
|
16
|
-
|
|
17
|
-
for (const integration of appConfig.integrations) {
|
|
18
|
-
integration.setConfig(appConfig);
|
|
19
|
-
integration.setRuntimeOrigin(appConfig.baseUrl);
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
const scanner = new FSRouterScanner({
|
|
23
|
-
dir: path.join(appConfig.rootDir, appConfig.srcDir, appConfig.pagesDir),
|
|
24
|
-
appConfig,
|
|
25
|
-
origin: appConfig.baseUrl,
|
|
26
|
-
templatesExt: appConfig.templatesExt,
|
|
27
|
-
options: {
|
|
28
|
-
buildMode: false,
|
|
29
|
-
},
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
const router = new FSRouter({
|
|
33
|
-
origin: appConfig.baseUrl,
|
|
34
|
-
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
35
|
-
scanner,
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
const routeRendererFactory = new RouteRendererFactory({
|
|
39
|
-
appConfig,
|
|
40
|
-
runtimeOrigin: appConfig.baseUrl,
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
const fileSystemResponseFactory = new FileSystemServerResponseFactory({
|
|
44
|
-
appConfig,
|
|
45
|
-
routeRendererFactory,
|
|
46
|
-
options: {
|
|
47
|
-
watchMode: false,
|
|
48
|
-
},
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
describe('FileSystemResponseMatcher', () => {
|
|
52
|
-
describe('without cache service', () => {
|
|
53
|
-
const matcherWithoutCache = new FileSystemResponseMatcher({
|
|
54
|
-
appConfig,
|
|
55
|
-
router,
|
|
56
|
-
routeRendererFactory,
|
|
57
|
-
fileSystemResponseFactory,
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it('should return custom 404 page for unmatched request URL', async () => {
|
|
61
|
-
const requestUrl = APP_TEST_ROUTES.nonExistentFile;
|
|
62
|
-
const response = await matcherWithoutCache.handleNoMatch(requestUrl);
|
|
63
|
-
const body = await response.text();
|
|
64
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
it('should handle match with disabled cache headers', async () => {
|
|
68
|
-
const match: MatchResult = {
|
|
69
|
-
kind: 'exact',
|
|
70
|
-
pathname: APP_TEST_ROUTES.index,
|
|
71
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
72
|
-
params: {},
|
|
73
|
-
query: {},
|
|
74
|
-
};
|
|
75
|
-
const response = await matcherWithoutCache.handleMatch(match);
|
|
76
|
-
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
77
|
-
expect(response.headers.get('X-Cache')).toBe('DISABLED');
|
|
78
|
-
expect(response.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
it('should return null for getCacheService when not configured', () => {
|
|
82
|
-
expect(matcherWithoutCache.getCacheService()).toBeNull();
|
|
83
|
-
});
|
|
84
|
-
});
|
|
85
|
-
|
|
86
|
-
describe('with cache service', () => {
|
|
87
|
-
const cacheService = new PageCacheService({
|
|
88
|
-
store: new MemoryCacheStore(),
|
|
89
|
-
enabled: true,
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
const matcherWithCache = new FileSystemResponseMatcher({
|
|
93
|
-
appConfig,
|
|
94
|
-
router,
|
|
95
|
-
routeRendererFactory,
|
|
96
|
-
fileSystemResponseFactory,
|
|
97
|
-
cacheService,
|
|
98
|
-
defaultCacheStrategy: 'static',
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
it('should return cache service via getCacheService', () => {
|
|
102
|
-
expect(matcherWithCache.getCacheService()).toBe(cacheService);
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
it('should return X-Cache header on first request (MISS)', async () => {
|
|
106
|
-
const match: MatchResult = {
|
|
107
|
-
kind: 'exact',
|
|
108
|
-
pathname: '/cache-test-miss',
|
|
109
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
110
|
-
params: {},
|
|
111
|
-
query: {},
|
|
112
|
-
};
|
|
113
|
-
const response = await matcherWithCache.handleMatch(match);
|
|
114
|
-
expect(response.headers.get('X-Cache')).toBe('MISS');
|
|
115
|
-
expect(response.headers.get('Cache-Control')).toBe('public, max-age=31536000, immutable');
|
|
116
|
-
});
|
|
117
|
-
|
|
118
|
-
it('should return X-Cache HIT on second request to same path', async () => {
|
|
119
|
-
const uniquePath = `/cache-test-hit-${Date.now()}`;
|
|
120
|
-
const match: MatchResult = {
|
|
121
|
-
kind: 'exact',
|
|
122
|
-
pathname: uniquePath,
|
|
123
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
124
|
-
params: {},
|
|
125
|
-
query: {},
|
|
126
|
-
};
|
|
127
|
-
|
|
128
|
-
const response1 = await matcherWithCache.handleMatch(match);
|
|
129
|
-
const response2 = await matcherWithCache.handleMatch(match);
|
|
130
|
-
|
|
131
|
-
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
132
|
-
expect(response2.headers.get('X-Cache')).toBe('HIT');
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
it('should cache different paths separately', async () => {
|
|
136
|
-
const match1: MatchResult = {
|
|
137
|
-
kind: 'exact',
|
|
138
|
-
pathname: '/path-a',
|
|
139
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
140
|
-
params: {},
|
|
141
|
-
query: {},
|
|
142
|
-
};
|
|
143
|
-
const match2: MatchResult = {
|
|
144
|
-
kind: 'exact',
|
|
145
|
-
pathname: '/path-b',
|
|
146
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
147
|
-
params: {},
|
|
148
|
-
query: {},
|
|
149
|
-
};
|
|
150
|
-
|
|
151
|
-
const response1 = await matcherWithCache.handleMatch(match1);
|
|
152
|
-
const response2 = await matcherWithCache.handleMatch(match2);
|
|
153
|
-
|
|
154
|
-
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
155
|
-
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
156
|
-
});
|
|
157
|
-
|
|
158
|
-
it('should include query params in cache key', async () => {
|
|
159
|
-
const basePath = `/search-${Date.now()}`;
|
|
160
|
-
const matchWithQuery: MatchResult = {
|
|
161
|
-
kind: 'exact',
|
|
162
|
-
pathname: basePath,
|
|
163
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
164
|
-
params: {},
|
|
165
|
-
query: { q: 'test' },
|
|
166
|
-
};
|
|
167
|
-
const matchWithDifferentQuery: MatchResult = {
|
|
168
|
-
kind: 'exact',
|
|
169
|
-
pathname: basePath,
|
|
170
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
171
|
-
params: {},
|
|
172
|
-
query: { q: 'other' },
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
const firstCall = await matcherWithCache.handleMatch(matchWithQuery);
|
|
176
|
-
const response1 = await matcherWithCache.handleMatch(matchWithQuery);
|
|
177
|
-
const response2 = await matcherWithCache.handleMatch(matchWithDifferentQuery);
|
|
178
|
-
|
|
179
|
-
expect(firstCall.headers.get('X-Cache')).toBe('MISS');
|
|
180
|
-
expect(response1.headers.get('X-Cache')).toBe('HIT');
|
|
181
|
-
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
182
|
-
});
|
|
183
|
-
});
|
|
184
|
-
|
|
185
|
-
describe('with dynamic cache strategy', () => {
|
|
186
|
-
const cacheService = new PageCacheService({
|
|
187
|
-
store: new MemoryCacheStore(),
|
|
188
|
-
enabled: true,
|
|
189
|
-
});
|
|
190
|
-
|
|
191
|
-
const dynamicMatcher = new FileSystemResponseMatcher({
|
|
192
|
-
appConfig,
|
|
193
|
-
router,
|
|
194
|
-
routeRendererFactory,
|
|
195
|
-
fileSystemResponseFactory,
|
|
196
|
-
cacheService,
|
|
197
|
-
defaultCacheStrategy: 'dynamic',
|
|
198
|
-
});
|
|
199
|
-
|
|
200
|
-
it('should bypass cache entirely for dynamic strategy', async () => {
|
|
201
|
-
const match: MatchResult = {
|
|
202
|
-
kind: 'exact',
|
|
203
|
-
pathname: '/dynamic-page',
|
|
204
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
205
|
-
params: {},
|
|
206
|
-
query: {},
|
|
207
|
-
};
|
|
208
|
-
|
|
209
|
-
const response1 = await dynamicMatcher.handleMatch(match);
|
|
210
|
-
const response2 = await dynamicMatcher.handleMatch(match);
|
|
211
|
-
|
|
212
|
-
expect(response1.headers.get('X-Cache')).toBe('DISABLED');
|
|
213
|
-
expect(response2.headers.get('X-Cache')).toBe('DISABLED');
|
|
214
|
-
expect(response1.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
215
|
-
});
|
|
216
|
-
});
|
|
217
|
-
|
|
218
|
-
describe('handleNoMatch content type behavior', () => {
|
|
219
|
-
const matcher = new FileSystemResponseMatcher({
|
|
220
|
-
appConfig,
|
|
221
|
-
router,
|
|
222
|
-
routeRendererFactory,
|
|
223
|
-
fileSystemResponseFactory,
|
|
224
|
-
});
|
|
225
|
-
|
|
226
|
-
it('should return custom 404 page for known extension like .html', async () => {
|
|
227
|
-
const response = await matcher.handleNoMatch('/non-existent-page.html');
|
|
228
|
-
const body = await response.text();
|
|
229
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
230
|
-
});
|
|
231
|
-
|
|
232
|
-
it('should return custom 404 page for page requests without extension', async () => {
|
|
233
|
-
const response = await matcher.handleNoMatch('/non-existent-page');
|
|
234
|
-
const body = await response.text();
|
|
235
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
236
|
-
});
|
|
237
|
-
|
|
238
|
-
it('should return custom 404 page for unknown extensions', async () => {
|
|
239
|
-
const response = await matcher.handleNoMatch('/page.xyz');
|
|
240
|
-
const body = await response.text();
|
|
241
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
242
|
-
});
|
|
243
|
-
|
|
244
|
-
it('should return custom 404 page for trailing dot', async () => {
|
|
245
|
-
const response = await matcher.handleNoMatch('/page.');
|
|
246
|
-
const body = await response.text();
|
|
247
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
248
|
-
});
|
|
249
|
-
|
|
250
|
-
it('should serve text/plain files from disk', async () => {
|
|
251
|
-
const response = await matcher.handleNoMatch('/robots.txt');
|
|
252
|
-
expect(response.headers.get('Content-Type')).toBe('text/plain');
|
|
253
|
-
});
|
|
254
|
-
|
|
255
|
-
it('should return custom 404 page for non-existent static files', async () => {
|
|
256
|
-
const response = await matcher.handleNoMatch('/non-existent.txt');
|
|
257
|
-
const body = await response.text();
|
|
258
|
-
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
259
|
-
});
|
|
260
|
-
});
|
|
261
|
-
|
|
262
|
-
describe('internal transpile output', () => {
|
|
263
|
-
it('should write page metadata modules to the execution directory', async () => {
|
|
264
|
-
const matcher = new FileSystemResponseMatcher({
|
|
265
|
-
appConfig,
|
|
266
|
-
router,
|
|
267
|
-
routeRendererFactory,
|
|
268
|
-
fileSystemResponseFactory,
|
|
269
|
-
});
|
|
270
|
-
|
|
271
|
-
const importModule = vi.fn(async () => ({}));
|
|
272
|
-
(matcher as any).serverModuleTranspiler = {
|
|
273
|
-
importModule,
|
|
274
|
-
};
|
|
275
|
-
|
|
276
|
-
await (matcher as any).importPageModule(INDEX_TEMPLATE_FILE);
|
|
277
|
-
|
|
278
|
-
expect(importModule).toHaveBeenCalledWith(
|
|
279
|
-
expect.objectContaining({
|
|
280
|
-
filePath: INDEX_TEMPLATE_FILE,
|
|
281
|
-
outdir: path.join(resolveInternalExecutionDir(appConfig), '.server-modules-meta'),
|
|
282
|
-
}),
|
|
283
|
-
);
|
|
284
|
-
});
|
|
285
|
-
});
|
|
286
|
-
});
|
|
@@ -1,196 +0,0 @@
|
|
|
1
|
-
import assert from 'node:assert/strict';
|
|
2
|
-
import fs from 'node:fs';
|
|
3
|
-
import os from 'node:os';
|
|
4
|
-
import path from 'node:path';
|
|
5
|
-
import { afterEach, describe, test, vi } from 'vitest';
|
|
6
|
-
import { ConfigBuilder } from '../../config/config-builder.ts';
|
|
7
|
-
import { resolveInternalWorkDir } from '../../utils/resolve-work-dir.ts';
|
|
8
|
-
import { NodeHmrManager } from '../node/node-hmr-manager.ts';
|
|
9
|
-
import { HmrManager as BunHmrManager } from '../bun/hmr-manager.ts';
|
|
10
|
-
|
|
11
|
-
type SharedHmrManager = {
|
|
12
|
-
handleFileChange(filePath: string, options?: { broadcast?: boolean }): Promise<void>;
|
|
13
|
-
registerEntrypoint(entrypointPath: string): Promise<string>;
|
|
14
|
-
registerScriptEntrypoint(entrypointPath: string): Promise<string>;
|
|
15
|
-
registerSpecifierMap(map: Record<string, string>): void;
|
|
16
|
-
getWatchedFiles(): Map<string, string>;
|
|
17
|
-
getSpecifierMap(): Map<string, string>;
|
|
18
|
-
stop(): void;
|
|
19
|
-
appConfig: Awaited<ReturnType<ConfigBuilder['build']>>;
|
|
20
|
-
readonly runtimeName: string;
|
|
21
|
-
};
|
|
22
|
-
|
|
23
|
-
function withRuntimeName<T extends Omit<SharedHmrManager, 'runtimeName'>>(
|
|
24
|
-
manager: T,
|
|
25
|
-
runtimeName: SharedHmrManager['runtimeName'],
|
|
26
|
-
): T & Pick<SharedHmrManager, 'runtimeName'> {
|
|
27
|
-
return Object.assign(manager, { runtimeName });
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
const tempRoots: string[] = [];
|
|
31
|
-
|
|
32
|
-
function createTempRoot(prefix: string): string {
|
|
33
|
-
const root = fs.mkdtempSync(path.join(os.tmpdir(), `${prefix}-`));
|
|
34
|
-
tempRoots.push(root);
|
|
35
|
-
return root;
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
afterEach(() => {
|
|
39
|
-
for (const root of tempRoots.splice(0)) {
|
|
40
|
-
fs.rmSync(root, { recursive: true, force: true });
|
|
41
|
-
}
|
|
42
|
-
vi.restoreAllMocks();
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
const runtimes = [
|
|
46
|
-
{
|
|
47
|
-
name: 'node',
|
|
48
|
-
async create(rootDir: string): Promise<SharedHmrManager> {
|
|
49
|
-
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
50
|
-
return withRuntimeName(
|
|
51
|
-
new NodeHmrManager({
|
|
52
|
-
appConfig: config,
|
|
53
|
-
bridge: {
|
|
54
|
-
subscriberCount: 0,
|
|
55
|
-
broadcast: () => {},
|
|
56
|
-
} as any,
|
|
57
|
-
}),
|
|
58
|
-
'node',
|
|
59
|
-
);
|
|
60
|
-
},
|
|
61
|
-
},
|
|
62
|
-
{
|
|
63
|
-
name: 'bun',
|
|
64
|
-
async create(rootDir: string): Promise<SharedHmrManager> {
|
|
65
|
-
const config = await new ConfigBuilder().setRootDir(rootDir).build();
|
|
66
|
-
return withRuntimeName(
|
|
67
|
-
new BunHmrManager({
|
|
68
|
-
appConfig: config,
|
|
69
|
-
bridge: {
|
|
70
|
-
subscriberCount: 0,
|
|
71
|
-
broadcast: () => {},
|
|
72
|
-
subscribe: () => {},
|
|
73
|
-
unsubscribe: () => {},
|
|
74
|
-
} as any,
|
|
75
|
-
}),
|
|
76
|
-
'bun',
|
|
77
|
-
);
|
|
78
|
-
},
|
|
79
|
-
},
|
|
80
|
-
] as const;
|
|
81
|
-
|
|
82
|
-
function getEntrypointOutputPath(manager: SharedHmrManager, entrypointPath: string): string {
|
|
83
|
-
const relativePathJs = path
|
|
84
|
-
.relative(manager.appConfig.absolutePaths.srcDir, entrypointPath)
|
|
85
|
-
.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
|
|
86
|
-
const encodedPathJs = relativePathJs.replace(/\[([^\]]+)\]/g, '_$1_');
|
|
87
|
-
return path.join(resolveInternalWorkDir(manager.appConfig), 'assets', '_hmr', encodedPathJs);
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
describe.each(runtimes)('shared HMR manager contract: $name', ({ create }) => {
|
|
91
|
-
test('shares one in-flight entrypoint registration across concurrent callers', async () => {
|
|
92
|
-
const rootDir = createTempRoot('ecopages-hmr-contract-register');
|
|
93
|
-
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
94
|
-
fs.mkdirSync(pagesDir, { recursive: true });
|
|
95
|
-
|
|
96
|
-
const entrypointPath = path.join(pagesDir, 'react-lab.tsx');
|
|
97
|
-
fs.writeFileSync(entrypointPath, 'export default function Page() { return null; }', 'utf8');
|
|
98
|
-
|
|
99
|
-
const manager = await create(rootDir);
|
|
100
|
-
const outputPath = getEntrypointOutputPath(manager, entrypointPath);
|
|
101
|
-
|
|
102
|
-
const handleFileChange = vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {
|
|
103
|
-
await new Promise((resolve) => setTimeout(resolve, 25));
|
|
104
|
-
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
105
|
-
fs.writeFileSync(outputPath, 'export default 1;', 'utf8');
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
const [firstUrl, secondUrl] = await Promise.all([
|
|
109
|
-
manager.registerEntrypoint(entrypointPath),
|
|
110
|
-
manager.registerEntrypoint(entrypointPath),
|
|
111
|
-
]);
|
|
112
|
-
|
|
113
|
-
assert.equal(firstUrl, '/assets/_hmr/pages/react-lab.js');
|
|
114
|
-
assert.equal(secondUrl, '/assets/_hmr/pages/react-lab.js');
|
|
115
|
-
assert.equal(handleFileChange.mock.calls.length, 1);
|
|
116
|
-
|
|
117
|
-
manager.stop();
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
test('fails strict page registration when no integration emits output', async () => {
|
|
121
|
-
const rootDir = createTempRoot('ecopages-hmr-contract-strict-fail');
|
|
122
|
-
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
123
|
-
fs.mkdirSync(pagesDir, { recursive: true });
|
|
124
|
-
|
|
125
|
-
const entrypointPath = path.join(pagesDir, 'react-content.mdx');
|
|
126
|
-
fs.writeFileSync(entrypointPath, '# Hello', 'utf8');
|
|
127
|
-
|
|
128
|
-
const manager = await create(rootDir);
|
|
129
|
-
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {});
|
|
130
|
-
|
|
131
|
-
await assert.rejects(() => manager.registerEntrypoint(entrypointPath), /Integration failed to emit entrypoint/);
|
|
132
|
-
assert.equal(manager.getWatchedFiles().has(path.resolve(entrypointPath)), false);
|
|
133
|
-
|
|
134
|
-
manager.stop();
|
|
135
|
-
});
|
|
136
|
-
|
|
137
|
-
test('uses the generic build path for explicit script entrypoints', async () => {
|
|
138
|
-
const rootDir = createTempRoot('ecopages-hmr-contract-script-register');
|
|
139
|
-
const srcDir = path.join(rootDir, 'src');
|
|
140
|
-
fs.mkdirSync(srcDir, { recursive: true });
|
|
141
|
-
|
|
142
|
-
const entrypointPath = path.join(srcDir, 'script.ts');
|
|
143
|
-
fs.writeFileSync(entrypointPath, 'console.log("hello");', 'utf8');
|
|
144
|
-
|
|
145
|
-
const manager = await create(rootDir);
|
|
146
|
-
const outputPath = path.join(resolveInternalWorkDir(manager.appConfig), 'assets', '_hmr', 'script.js');
|
|
147
|
-
const buildCalls: string[] = [];
|
|
148
|
-
manager.appConfig.runtime!.buildExecutor = {
|
|
149
|
-
build: vi.fn(async (options) => {
|
|
150
|
-
buildCalls.push(options.entrypoints[0] as string);
|
|
151
|
-
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
152
|
-
fs.writeFileSync(outputPath, 'fresh-output', 'utf8');
|
|
153
|
-
return {
|
|
154
|
-
success: true,
|
|
155
|
-
logs: [],
|
|
156
|
-
outputs: [{ path: outputPath }],
|
|
157
|
-
};
|
|
158
|
-
}),
|
|
159
|
-
};
|
|
160
|
-
|
|
161
|
-
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {});
|
|
162
|
-
|
|
163
|
-
const outputUrl = await manager.registerScriptEntrypoint(entrypointPath);
|
|
164
|
-
|
|
165
|
-
assert.equal(outputUrl, '/assets/_hmr/script.js');
|
|
166
|
-
assert.deepEqual(buildCalls, [entrypointPath]);
|
|
167
|
-
assert.equal(fs.readFileSync(outputPath, 'utf8'), 'fresh-output');
|
|
168
|
-
|
|
169
|
-
manager.stop();
|
|
170
|
-
});
|
|
171
|
-
|
|
172
|
-
test('stop clears retained registration state', async () => {
|
|
173
|
-
const rootDir = createTempRoot('ecopages-hmr-contract-stop');
|
|
174
|
-
const pagesDir = path.join(rootDir, 'src', 'pages');
|
|
175
|
-
fs.mkdirSync(pagesDir, { recursive: true });
|
|
176
|
-
|
|
177
|
-
const entrypointPath = path.join(pagesDir, 'react-content.tsx');
|
|
178
|
-
fs.writeFileSync(entrypointPath, 'export default function Page() { return null; }', 'utf8');
|
|
179
|
-
|
|
180
|
-
const manager = await create(rootDir);
|
|
181
|
-
const outputPath = getEntrypointOutputPath(manager, entrypointPath);
|
|
182
|
-
|
|
183
|
-
vi.spyOn(manager, 'handleFileChange').mockImplementation(async () => {
|
|
184
|
-
fs.mkdirSync(path.dirname(outputPath), { recursive: true });
|
|
185
|
-
fs.writeFileSync(outputPath, 'export default 1;', 'utf8');
|
|
186
|
-
});
|
|
187
|
-
|
|
188
|
-
manager.registerSpecifierMap({ react: '/assets/vendors/react.js' });
|
|
189
|
-
await manager.registerEntrypoint(entrypointPath);
|
|
190
|
-
|
|
191
|
-
manager.stop();
|
|
192
|
-
|
|
193
|
-
assert.equal(manager.getWatchedFiles().size, 0);
|
|
194
|
-
assert.equal(manager.getSpecifierMap().size, 0);
|
|
195
|
-
});
|
|
196
|
-
});
|