@ecopages/core 0.2.0-alpha.22 → 0.2.0-alpha.24
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/package.json +93 -226
- 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 +180 -0
- package/src/adapters/bun/hmr-manager.test.ts +267 -0
- package/src/adapters/bun/hmr-manager.ts +406 -0
- package/src/adapters/bun/index.ts +2 -0
- package/src/adapters/bun/server-adapter.ts +500 -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 +322 -0
- package/src/adapters/node/node-hmr-manager.ts +378 -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-route-matcher.test.ts +381 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +140 -0
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +90 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +127 -0
- package/src/adapters/shared/fs-server-response-factory.test.ts +187 -0
- package/src/adapters/shared/fs-server-response-factory.ts +118 -0
- package/src/adapters/shared/fs-server-response-matcher.test.ts +285 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +189 -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 +232 -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 +77 -0
- package/src/adapters/shared/server-adapter.ts +493 -0
- package/src/adapters/shared/server-route-handler.test.ts +110 -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/config-builder.test.ts +443 -0
- package/src/config/config-builder.ts +742 -0
- package/src/config/config-builder.typecheck.test.ts +96 -0
- package/src/config/{constants.d.ts → constants.ts} +22 -13
- 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.d.ts → index.ts} +2 -2
- 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 +160 -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 +335 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +320 -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 +97 -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 +495 -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 +156 -0
- package/src/plugins/integration-plugin.ts +311 -0
- package/src/plugins/processor.test.ts +148 -0
- package/src/plugins/processor.ts +240 -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/orchestration/boundary-planning.service.ts +146 -0
- package/src/route-renderer/orchestration/component-render-context.ts +318 -0
- package/src/route-renderer/orchestration/integration-renderer.test.ts +2088 -0
- package/src/route-renderer/orchestration/integration-renderer.ts +1285 -0
- package/src/route-renderer/orchestration/page-packaging.service.test.ts +76 -0
- package/src/route-renderer/orchestration/page-packaging.service.ts +85 -0
- package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.test.ts +319 -0
- package/src/route-renderer/orchestration/queued-boundary-runtime.service.ts +289 -0
- package/src/route-renderer/orchestration/render-execution.service.test.ts +196 -0
- package/src/route-renderer/orchestration/render-execution.service.ts +182 -0
- package/src/route-renderer/orchestration/render-output.utils.ts +302 -0
- package/src/route-renderer/orchestration/render-preparation.service.test.ts +569 -0
- package/src/route-renderer/orchestration/render-preparation.service.ts +508 -0
- package/src/route-renderer/orchestration/route-shell-composer.service.ts +162 -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 +196 -0
- package/src/route-renderer/page-loading/declared-asset-collection.ts +156 -0
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +665 -0
- package/src/route-renderer/page-loading/dependency-resolver.ts +150 -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 +205 -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 +136 -0
- package/src/router/client/link-intent.test.browser.ts +51 -0
- package/src/router/client/link-intent.ts +92 -0
- package/src/router/client/navigation-coordinator.test.ts +237 -0
- package/src/router/client/navigation-coordinator.ts +453 -0
- package/src/router/server/fs-router-scanner.test.ts +83 -0
- package/src/router/server/fs-router-scanner.ts +224 -0
- package/src/router/server/fs-router.test.ts +214 -0
- package/src/router/server/fs-router.ts +122 -0
- 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 +476 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.ts +345 -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 +125 -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 +5 -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 +174 -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 +192 -0
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +134 -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 +110 -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 +87 -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 +71 -0
- package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +81 -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 +479 -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 +143 -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 +297 -0
- package/src/services/module-loading/page-module-import.service.test.ts +504 -0
- package/src/services/module-loading/page-module-import.service.ts +252 -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/runtime-specifier-registry.service.ts +96 -0
- package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
- package/src/services/validation/schema-validation-service.test.ts +223 -0
- package/src/services/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 +316 -0
- package/src/static-site-generator/static-site-generator.ts +462 -0
- package/src/types/internal-types.ts +242 -0
- package/src/types/public-types.ts +1443 -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 +41 -0
- package/src/watchers/project-watcher.test.ts +768 -0
- package/src/watchers/project-watcher.ts +357 -0
- package/CHANGELOG.md +0 -51
- 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 -161
- package/src/adapters/node/server-adapter.js +0 -359
- 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 -70
- 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.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 -109
- 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 -554
- package/src/route-renderer/orchestration/integration-renderer.js +0 -957
- 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 -46
- package/src/route-renderer/orchestration/render-output.utils.js +0 -65
- package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
- package/src/route-renderer/orchestration/render-preparation.service.js +0 -341
- 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/dependency-resolver.d.ts +0 -35
- package/src/route-renderer/page-loading/dependency-resolver.js +0 -444
- 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 -149
- 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-processing.service.d.ts +0 -120
- package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -331
- 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 -89
- 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 -48
- 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/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 -64
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -17
- package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -72
- 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 -5
- package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -57
- 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 -88
- 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 -75
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -5
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
- 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 -66
- package/src/services/assets/browser-bundle.service.d.ts +0 -32
- package/src/services/assets/browser-bundle.service.js +0 -33
- 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 -77
- package/src/services/html/html-transformer.service.js +0 -215
- 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 -28
- package/src/services/module-loading/app-module-loader.service.js +0 -35
- 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 -109
- 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 -173
- package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -72
- package/src/services/module-loading/server-module-transpiler.service.js +0 -64
- 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 -104
- package/src/static-site-generator/static-site-generator.js +0 -338
- 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 -1219
- 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,285 @@
|
|
|
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 '../../types/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 { 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 scanner = new FSRouterScanner({
|
|
22
|
+
dir: path.join(appConfig.rootDir, appConfig.srcDir, appConfig.pagesDir),
|
|
23
|
+
appConfig,
|
|
24
|
+
origin: appConfig.baseUrl,
|
|
25
|
+
templatesExt: appConfig.templatesExt,
|
|
26
|
+
options: {
|
|
27
|
+
buildMode: false,
|
|
28
|
+
},
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
const router = new FSRouter({
|
|
32
|
+
origin: appConfig.baseUrl,
|
|
33
|
+
assetPrefix: path.join(appConfig.rootDir, appConfig.distDir),
|
|
34
|
+
scanner,
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
const routeRendererFactory = new RouteRendererFactory({
|
|
38
|
+
appConfig,
|
|
39
|
+
runtimeOrigin: appConfig.baseUrl,
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
const fileSystemResponseFactory = new FileSystemServerResponseFactory({
|
|
43
|
+
appConfig,
|
|
44
|
+
routeRendererFactory,
|
|
45
|
+
options: {
|
|
46
|
+
watchMode: false,
|
|
47
|
+
},
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
describe('FileSystemResponseMatcher', () => {
|
|
51
|
+
describe('without cache service', () => {
|
|
52
|
+
const matcherWithoutCache = new FileSystemResponseMatcher({
|
|
53
|
+
appConfig,
|
|
54
|
+
router,
|
|
55
|
+
routeRendererFactory,
|
|
56
|
+
fileSystemResponseFactory,
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it('should return custom 404 page for unmatched request URL', async () => {
|
|
60
|
+
const requestUrl = APP_TEST_ROUTES.nonExistentFile;
|
|
61
|
+
const response = await matcherWithoutCache.handleNoMatch(requestUrl);
|
|
62
|
+
const body = await response.text();
|
|
63
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it('should handle match with disabled cache headers', async () => {
|
|
67
|
+
const match: MatchResult = {
|
|
68
|
+
kind: 'exact',
|
|
69
|
+
pathname: APP_TEST_ROUTES.index,
|
|
70
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
71
|
+
params: {},
|
|
72
|
+
query: {},
|
|
73
|
+
};
|
|
74
|
+
const response = await matcherWithoutCache.handleMatch(match);
|
|
75
|
+
expect(response.headers.get('Content-Type')).toBe('text/html');
|
|
76
|
+
expect(response.headers.get('X-Cache')).toBe('DISABLED');
|
|
77
|
+
expect(response.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('should return null for getCacheService when not configured', () => {
|
|
81
|
+
expect(matcherWithoutCache.getCacheService()).toBeNull();
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
describe('with cache service', () => {
|
|
86
|
+
const cacheService = new PageCacheService({
|
|
87
|
+
store: new MemoryCacheStore(),
|
|
88
|
+
enabled: true,
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
const matcherWithCache = new FileSystemResponseMatcher({
|
|
92
|
+
appConfig,
|
|
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
|
+
kind: 'exact',
|
|
107
|
+
pathname: '/cache-test-miss',
|
|
108
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
109
|
+
params: {},
|
|
110
|
+
query: {},
|
|
111
|
+
};
|
|
112
|
+
const response = await matcherWithCache.handleMatch(match);
|
|
113
|
+
expect(response.headers.get('X-Cache')).toBe('MISS');
|
|
114
|
+
expect(response.headers.get('Cache-Control')).toBe('public, max-age=31536000, immutable');
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
it('should return X-Cache HIT on second request to same path', async () => {
|
|
118
|
+
const uniquePath = `/cache-test-hit-${Date.now()}`;
|
|
119
|
+
const match: MatchResult = {
|
|
120
|
+
kind: 'exact',
|
|
121
|
+
pathname: uniquePath,
|
|
122
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
123
|
+
params: {},
|
|
124
|
+
query: {},
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
const response1 = await matcherWithCache.handleMatch(match);
|
|
128
|
+
const response2 = await matcherWithCache.handleMatch(match);
|
|
129
|
+
|
|
130
|
+
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
131
|
+
expect(response2.headers.get('X-Cache')).toBe('HIT');
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it('should cache different paths separately', async () => {
|
|
135
|
+
const match1: MatchResult = {
|
|
136
|
+
kind: 'exact',
|
|
137
|
+
pathname: '/path-a',
|
|
138
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
139
|
+
params: {},
|
|
140
|
+
query: {},
|
|
141
|
+
};
|
|
142
|
+
const match2: MatchResult = {
|
|
143
|
+
kind: 'exact',
|
|
144
|
+
pathname: '/path-b',
|
|
145
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
146
|
+
params: {},
|
|
147
|
+
query: {},
|
|
148
|
+
};
|
|
149
|
+
|
|
150
|
+
const response1 = await matcherWithCache.handleMatch(match1);
|
|
151
|
+
const response2 = await matcherWithCache.handleMatch(match2);
|
|
152
|
+
|
|
153
|
+
expect(response1.headers.get('X-Cache')).toBe('MISS');
|
|
154
|
+
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
155
|
+
});
|
|
156
|
+
|
|
157
|
+
it('should include query params in cache key', async () => {
|
|
158
|
+
const basePath = `/search-${Date.now()}`;
|
|
159
|
+
const matchWithQuery: MatchResult = {
|
|
160
|
+
kind: 'exact',
|
|
161
|
+
pathname: basePath,
|
|
162
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
163
|
+
params: {},
|
|
164
|
+
query: { q: 'test' },
|
|
165
|
+
};
|
|
166
|
+
const matchWithDifferentQuery: MatchResult = {
|
|
167
|
+
kind: 'exact',
|
|
168
|
+
pathname: basePath,
|
|
169
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
170
|
+
params: {},
|
|
171
|
+
query: { q: 'other' },
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
const firstCall = await matcherWithCache.handleMatch(matchWithQuery);
|
|
175
|
+
const response1 = await matcherWithCache.handleMatch(matchWithQuery);
|
|
176
|
+
const response2 = await matcherWithCache.handleMatch(matchWithDifferentQuery);
|
|
177
|
+
|
|
178
|
+
expect(firstCall.headers.get('X-Cache')).toBe('MISS');
|
|
179
|
+
expect(response1.headers.get('X-Cache')).toBe('HIT');
|
|
180
|
+
expect(response2.headers.get('X-Cache')).toBe('MISS');
|
|
181
|
+
});
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
describe('with dynamic cache strategy', () => {
|
|
185
|
+
const cacheService = new PageCacheService({
|
|
186
|
+
store: new MemoryCacheStore(),
|
|
187
|
+
enabled: true,
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
const dynamicMatcher = new FileSystemResponseMatcher({
|
|
191
|
+
appConfig,
|
|
192
|
+
router,
|
|
193
|
+
routeRendererFactory,
|
|
194
|
+
fileSystemResponseFactory,
|
|
195
|
+
cacheService,
|
|
196
|
+
defaultCacheStrategy: 'dynamic',
|
|
197
|
+
});
|
|
198
|
+
|
|
199
|
+
it('should bypass cache entirely for dynamic strategy', async () => {
|
|
200
|
+
const match: MatchResult = {
|
|
201
|
+
kind: 'exact',
|
|
202
|
+
pathname: '/dynamic-page',
|
|
203
|
+
filePath: INDEX_TEMPLATE_FILE,
|
|
204
|
+
params: {},
|
|
205
|
+
query: {},
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
const response1 = await dynamicMatcher.handleMatch(match);
|
|
209
|
+
const response2 = await dynamicMatcher.handleMatch(match);
|
|
210
|
+
|
|
211
|
+
expect(response1.headers.get('X-Cache')).toBe('DISABLED');
|
|
212
|
+
expect(response2.headers.get('X-Cache')).toBe('DISABLED');
|
|
213
|
+
expect(response1.headers.get('Cache-Control')).toBe('no-store, must-revalidate');
|
|
214
|
+
});
|
|
215
|
+
});
|
|
216
|
+
|
|
217
|
+
describe('handleNoMatch content type behavior', () => {
|
|
218
|
+
const matcher = new FileSystemResponseMatcher({
|
|
219
|
+
appConfig,
|
|
220
|
+
router,
|
|
221
|
+
routeRendererFactory,
|
|
222
|
+
fileSystemResponseFactory,
|
|
223
|
+
});
|
|
224
|
+
|
|
225
|
+
it('should return custom 404 page for known extension like .html', async () => {
|
|
226
|
+
const response = await matcher.handleNoMatch('/non-existent-page.html');
|
|
227
|
+
const body = await response.text();
|
|
228
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
it('should return custom 404 page for page requests without extension', async () => {
|
|
232
|
+
const response = await matcher.handleNoMatch('/non-existent-page');
|
|
233
|
+
const body = await response.text();
|
|
234
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
it('should return custom 404 page for unknown extensions', async () => {
|
|
238
|
+
const response = await matcher.handleNoMatch('/page.xyz');
|
|
239
|
+
const body = await response.text();
|
|
240
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
it('should return custom 404 page for trailing dot', async () => {
|
|
244
|
+
const response = await matcher.handleNoMatch('/page.');
|
|
245
|
+
const body = await response.text();
|
|
246
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
247
|
+
});
|
|
248
|
+
|
|
249
|
+
it('should serve text/plain files from disk', async () => {
|
|
250
|
+
const response = await matcher.handleNoMatch('/robots.txt');
|
|
251
|
+
expect(response.headers.get('Content-Type')).toBe('text/plain');
|
|
252
|
+
});
|
|
253
|
+
|
|
254
|
+
it('should return custom 404 page for non-existent static files', async () => {
|
|
255
|
+
const response = await matcher.handleNoMatch('/non-existent.txt');
|
|
256
|
+
const body = await response.text();
|
|
257
|
+
expect(body).toContain('<h1>404 - Page Not Found</h1>');
|
|
258
|
+
});
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
describe('internal transpile output', () => {
|
|
262
|
+
it('should inspect page modules through the owning route renderer', async () => {
|
|
263
|
+
const matcher = new FileSystemResponseMatcher({
|
|
264
|
+
appConfig,
|
|
265
|
+
router,
|
|
266
|
+
routeRendererFactory,
|
|
267
|
+
fileSystemResponseFactory,
|
|
268
|
+
});
|
|
269
|
+
|
|
270
|
+
const loadPageModule = vi.fn(async () => ({}));
|
|
271
|
+
(matcher as any).routeRendererFactory = {
|
|
272
|
+
createRenderer: vi.fn(() => ({
|
|
273
|
+
loadPageModule,
|
|
274
|
+
})),
|
|
275
|
+
};
|
|
276
|
+
|
|
277
|
+
await (matcher as any).importPageModule(INDEX_TEMPLATE_FILE);
|
|
278
|
+
|
|
279
|
+
expect((matcher as any).routeRendererFactory.createRenderer).toHaveBeenCalledWith(INDEX_TEMPLATE_FILE);
|
|
280
|
+
expect(loadPageModule).toHaveBeenCalledWith(INDEX_TEMPLATE_FILE, {
|
|
281
|
+
cacheScope: 'request-metadata',
|
|
282
|
+
});
|
|
283
|
+
});
|
|
284
|
+
});
|
|
285
|
+
});
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { appLogger } from '../../global/app-logger.ts';
|
|
3
|
+
import type { EcoPagesAppConfig, MatchResult } from '../../types/internal-types.ts';
|
|
4
|
+
import type { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
|
|
5
|
+
import type { FSRouter } from '../../router/server/fs-router.ts';
|
|
6
|
+
import type { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
7
|
+
import type { CacheStrategy, RenderResult } from '../../services/cache/cache.types.ts';
|
|
8
|
+
import { PageRequestCacheCoordinator } from '../../services/cache/page-request-cache-coordinator.service.ts';
|
|
9
|
+
import { ServerUtils } from '../../utils/server-utils.module.ts';
|
|
10
|
+
import type { Middleware, RequestLocals } from '../../types/public-types.ts';
|
|
11
|
+
import { FileRouteMiddlewarePipeline } from './file-route-middleware-pipeline.ts';
|
|
12
|
+
import { LocalsAccessError } from '../../errors/locals-access-error.ts';
|
|
13
|
+
import { isDevelopmentRuntime } from '../../utils/runtime.ts';
|
|
14
|
+
import type { FileSystemServerResponseFactory } from './fs-server-response-factory.ts';
|
|
15
|
+
|
|
16
|
+
export interface FileSystemResponseMatcherOptions {
|
|
17
|
+
appConfig: EcoPagesAppConfig;
|
|
18
|
+
router: FSRouter;
|
|
19
|
+
routeRendererFactory: RouteRendererFactory;
|
|
20
|
+
fileSystemResponseFactory: FileSystemServerResponseFactory;
|
|
21
|
+
/** Optional cache service. When null, caching is disabled. */
|
|
22
|
+
cacheService?: PageCacheService | null;
|
|
23
|
+
/** Default cache strategy when caching is enabled. @default 'static' */
|
|
24
|
+
defaultCacheStrategy?: CacheStrategy;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Matches file-system routes to rendered HTML responses.
|
|
29
|
+
*
|
|
30
|
+
* render pipeline. It coordinates page module inspection, request-local policy,
|
|
31
|
+
* renderer invocation, middleware execution, cache integration, and fallback
|
|
32
|
+
* error translation.
|
|
33
|
+
*/
|
|
34
|
+
export class FileSystemResponseMatcher {
|
|
35
|
+
private appConfig: EcoPagesAppConfig;
|
|
36
|
+
private router: FSRouter;
|
|
37
|
+
private routeRendererFactory: RouteRendererFactory;
|
|
38
|
+
private fileSystemResponseFactory: FileSystemServerResponseFactory;
|
|
39
|
+
private pageRequestCacheCoordinator: PageRequestCacheCoordinator;
|
|
40
|
+
private fileRouteMiddlewarePipeline: FileRouteMiddlewarePipeline;
|
|
41
|
+
|
|
42
|
+
constructor({
|
|
43
|
+
appConfig,
|
|
44
|
+
router,
|
|
45
|
+
routeRendererFactory,
|
|
46
|
+
fileSystemResponseFactory,
|
|
47
|
+
cacheService = null,
|
|
48
|
+
defaultCacheStrategy = 'static',
|
|
49
|
+
}: FileSystemResponseMatcherOptions) {
|
|
50
|
+
this.appConfig = appConfig;
|
|
51
|
+
this.router = router;
|
|
52
|
+
this.routeRendererFactory = routeRendererFactory;
|
|
53
|
+
this.fileSystemResponseFactory = fileSystemResponseFactory;
|
|
54
|
+
this.pageRequestCacheCoordinator = new PageRequestCacheCoordinator(cacheService, defaultCacheStrategy);
|
|
55
|
+
this.fileRouteMiddlewarePipeline = new FileRouteMiddlewarePipeline(cacheService);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Resolves unmatched paths either as static asset requests or as the custom
|
|
60
|
+
* not-found page.
|
|
61
|
+
* @param requestUrl Incoming pathname.
|
|
62
|
+
* @returns Static file response or rendered 404 response.
|
|
63
|
+
*/
|
|
64
|
+
async handleNoMatch(requestUrl: string): Promise<Response> {
|
|
65
|
+
const isStaticFileRequest = ServerUtils.hasKnownExtension(requestUrl);
|
|
66
|
+
|
|
67
|
+
if (!isStaticFileRequest) {
|
|
68
|
+
return this.fileSystemResponseFactory.createCustomNotFoundResponse();
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
const relativeUrl = requestUrl.startsWith('/') ? requestUrl.slice(1) : requestUrl;
|
|
72
|
+
const filePath = path.join(this.router.assetPrefix, relativeUrl);
|
|
73
|
+
const contentType = ServerUtils.getContentType(filePath);
|
|
74
|
+
|
|
75
|
+
return this.fileSystemResponseFactory.createFileResponse(filePath, contentType);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Handles a matched file-system page route.
|
|
80
|
+
*
|
|
81
|
+
* The method inspects page metadata needed for request-time execution,
|
|
82
|
+
* prepares the renderer invocation, validates middleware/cache constraints,
|
|
83
|
+
* and delegates caching plus middleware execution to dedicated collaborators.
|
|
84
|
+
*
|
|
85
|
+
* @param match Router match result.
|
|
86
|
+
* @param request Optional incoming request. A synthetic GET request is created when omitted.
|
|
87
|
+
* @returns Final response for the matched route.
|
|
88
|
+
*/
|
|
89
|
+
async handleMatch(match: MatchResult, request?: Request): Promise<Response> {
|
|
90
|
+
const cacheKey = this.pageRequestCacheCoordinator.buildCacheKey(match);
|
|
91
|
+
|
|
92
|
+
try {
|
|
93
|
+
const resolvedRequest =
|
|
94
|
+
request ??
|
|
95
|
+
new Request(new URL(cacheKey, this.router.origin).toString(), {
|
|
96
|
+
method: 'GET',
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
const localsStore: RequestLocals = {};
|
|
100
|
+
const pageModule = await this.importPageModule(match.filePath);
|
|
101
|
+
const Page = (pageModule as any)?.default;
|
|
102
|
+
const pageMiddleware = (Page?.middleware ?? []) as Middleware[];
|
|
103
|
+
const pageCacheStrategy =
|
|
104
|
+
(Page?.cache as CacheStrategy | undefined) ??
|
|
105
|
+
this.pageRequestCacheCoordinator.getDefaultCacheStrategy();
|
|
106
|
+
const localsForRender: RequestLocals | undefined =
|
|
107
|
+
pageCacheStrategy === 'dynamic' ? localsStore : undefined;
|
|
108
|
+
|
|
109
|
+
this.fileRouteMiddlewarePipeline.assertValidConfiguration({
|
|
110
|
+
middleware: pageMiddleware,
|
|
111
|
+
pageCacheStrategy,
|
|
112
|
+
filePath: match.filePath,
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
const routeRenderer = this.routeRendererFactory.createRenderer(match.filePath);
|
|
116
|
+
const middlewareContext = this.fileRouteMiddlewarePipeline.createContext({
|
|
117
|
+
request: resolvedRequest,
|
|
118
|
+
params: match.params as Record<string, string>,
|
|
119
|
+
locals: localsStore,
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
const renderFn = async (): Promise<RenderResult> => {
|
|
123
|
+
const result = await routeRenderer.createRoute({
|
|
124
|
+
file: match.filePath,
|
|
125
|
+
params: match.params,
|
|
126
|
+
query: match.query,
|
|
127
|
+
locals: localsForRender,
|
|
128
|
+
});
|
|
129
|
+
const html = await this.pageRequestCacheCoordinator.bodyToString(result.body);
|
|
130
|
+
const strategy = result.cacheStrategy ?? this.pageRequestCacheCoordinator.getDefaultCacheStrategy();
|
|
131
|
+
return { html, strategy };
|
|
132
|
+
};
|
|
133
|
+
const renderResponse = async (): Promise<Response> => {
|
|
134
|
+
return this.pageRequestCacheCoordinator.render({
|
|
135
|
+
cacheKey,
|
|
136
|
+
pageCacheStrategy,
|
|
137
|
+
renderFn,
|
|
138
|
+
});
|
|
139
|
+
};
|
|
140
|
+
|
|
141
|
+
return await this.fileRouteMiddlewarePipeline.run({
|
|
142
|
+
middleware: pageMiddleware,
|
|
143
|
+
context: middlewareContext,
|
|
144
|
+
renderResponse,
|
|
145
|
+
});
|
|
146
|
+
} catch (error) {
|
|
147
|
+
if (error instanceof Response) {
|
|
148
|
+
return error;
|
|
149
|
+
}
|
|
150
|
+
if (error instanceof LocalsAccessError) {
|
|
151
|
+
return new Response(error.message, {
|
|
152
|
+
status: 500,
|
|
153
|
+
headers: { 'Content-Type': 'text/plain; charset=utf-8' },
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
if (error instanceof Error) {
|
|
157
|
+
if (isDevelopmentRuntime() || appLogger.isDebugEnabled()) {
|
|
158
|
+
appLogger.error(`[FileSystemResponseMatcher] ${error.message} at ${match.pathname}`);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return this.fileSystemResponseFactory.createCustomNotFoundResponse();
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
/**
|
|
166
|
+
* Loads the matched page module for request-time inspection.
|
|
167
|
+
*
|
|
168
|
+
* The matcher needs access to page-level metadata such as `cache` and
|
|
169
|
+
* `middleware` before full rendering starts, so it asks the owning route
|
|
170
|
+
* renderer to load the page module. That preserves integration-specific page
|
|
171
|
+
* import setup for request-time inspection as well as for full rendering.
|
|
172
|
+
*
|
|
173
|
+
* @param filePath Absolute page module path.
|
|
174
|
+
* @returns Imported page module.
|
|
175
|
+
*/
|
|
176
|
+
private async importPageModule(filePath: string): Promise<unknown> {
|
|
177
|
+
const routeRenderer = this.routeRendererFactory.createRenderer(filePath);
|
|
178
|
+
return routeRenderer.loadPageModule(filePath, {
|
|
179
|
+
cacheScope: 'request-metadata',
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
/**
|
|
184
|
+
* Get the underlying cache service for external invalidation.
|
|
185
|
+
*/
|
|
186
|
+
getCacheService(): PageCacheService | null {
|
|
187
|
+
return this.pageRequestCacheCoordinator.getCacheService();
|
|
188
|
+
}
|
|
189
|
+
}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import path from 'node:path';
|
|
2
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
3
|
+
import { appLogger } from '../../global/app-logger.ts';
|
|
4
|
+
import { RESOLVED_ASSETS_DIR } from '../../config/constants.ts';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Shared runtime state used while registering HMR-owned entrypoints.
|
|
8
|
+
*/
|
|
9
|
+
export interface HmrEntrypointRegistrarOptions {
|
|
10
|
+
/** Absolute source directory used to derive the emitted HMR path. */
|
|
11
|
+
srcDir: string;
|
|
12
|
+
/** Absolute distribution directory where HMR outputs are written. */
|
|
13
|
+
distDir: string;
|
|
14
|
+
/** In-flight registrations keyed by normalized absolute entrypoint path. */
|
|
15
|
+
entrypointRegistrations: Map<string, Promise<string>>;
|
|
16
|
+
/** Stable entrypoint-to-output mapping retained once an entrypoint is registered. */
|
|
17
|
+
watchedFiles: Map<string, string>;
|
|
18
|
+
/** Runtime-specific cleanup invoked when an entrypoint registration fails. */
|
|
19
|
+
clearFailedRegistration?: (entrypointPath: string) => void;
|
|
20
|
+
/** Development-only guardrail for integrations that never finish producing output. */
|
|
21
|
+
registrationTimeoutMs: number;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Runtime-specific hooks required to materialize a single HMR entrypoint.
|
|
26
|
+
*/
|
|
27
|
+
export interface HmrEntrypointRegistrationOptions {
|
|
28
|
+
/**
|
|
29
|
+
* Emits the browser-consumable HMR artifact for an entrypoint.
|
|
30
|
+
*/
|
|
31
|
+
emit(entrypointPath: string, outputPath: string): Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* Creates the runtime-specific error raised when the emit hook completes without producing output.
|
|
34
|
+
*/
|
|
35
|
+
getMissingOutputError(entrypointPath: string, outputPath: string): Error;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Coordinates the shared HMR entrypoint registration lifecycle for both Node and Bun managers.
|
|
40
|
+
*
|
|
41
|
+
* The registrar owns the cross-runtime policy: normalize entrypoint identities, dedupe concurrent
|
|
42
|
+
* registrations, derive the emitted `_hmr` output path, clear stale output before rebuilding, and
|
|
43
|
+
* apply the development timeout that prevents unresolved registrations from hanging navigation.
|
|
44
|
+
* Runtime-specific managers remain responsible for the actual emit step and any cleanup outside
|
|
45
|
+
* this shared registration flow.
|
|
46
|
+
*/
|
|
47
|
+
export class HmrEntrypointRegistrar {
|
|
48
|
+
private readonly options: HmrEntrypointRegistrarOptions;
|
|
49
|
+
|
|
50
|
+
constructor(options: HmrEntrypointRegistrarOptions) {
|
|
51
|
+
this.options = options;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Registers a single source entrypoint and returns the browser URL for its emitted HMR module.
|
|
56
|
+
*
|
|
57
|
+
* Concurrent requests for the same normalized entrypoint share the same in-flight promise so the
|
|
58
|
+
* integration only builds once per registration cycle.
|
|
59
|
+
*/
|
|
60
|
+
async registerEntrypoint(
|
|
61
|
+
entrypointPath: string,
|
|
62
|
+
registrationOptions: HmrEntrypointRegistrationOptions,
|
|
63
|
+
): Promise<string> {
|
|
64
|
+
const normalizedEntrypoint = path.resolve(entrypointPath);
|
|
65
|
+
const existingRegistration = this.options.entrypointRegistrations.get(normalizedEntrypoint);
|
|
66
|
+
if (existingRegistration) {
|
|
67
|
+
return await this.awaitEntrypointRegistration(existingRegistration, normalizedEntrypoint);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const registration = this.registerEntrypointInternal(normalizedEntrypoint, registrationOptions);
|
|
71
|
+
this.options.entrypointRegistrations.set(normalizedEntrypoint, registration);
|
|
72
|
+
|
|
73
|
+
try {
|
|
74
|
+
return await this.awaitEntrypointRegistration(registration, normalizedEntrypoint);
|
|
75
|
+
} catch (error) {
|
|
76
|
+
this.options.clearFailedRegistration?.(normalizedEntrypoint);
|
|
77
|
+
throw error;
|
|
78
|
+
} finally {
|
|
79
|
+
this.options.entrypointRegistrations.delete(normalizedEntrypoint);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
private async registerEntrypointInternal(
|
|
84
|
+
entrypointPath: string,
|
|
85
|
+
registrationOptions: HmrEntrypointRegistrationOptions,
|
|
86
|
+
): Promise<string> {
|
|
87
|
+
if (this.options.watchedFiles.has(entrypointPath)) {
|
|
88
|
+
return this.options.watchedFiles.get(entrypointPath)!;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const { outputPath, outputUrl } = this.getEntrypointOutput(entrypointPath);
|
|
92
|
+
|
|
93
|
+
this.options.watchedFiles.set(entrypointPath, outputUrl);
|
|
94
|
+
this.removeStaleEntrypointOutput(outputPath);
|
|
95
|
+
|
|
96
|
+
await registrationOptions.emit(entrypointPath, outputPath);
|
|
97
|
+
|
|
98
|
+
if (!fileSystem.exists(outputPath)) {
|
|
99
|
+
throw registrationOptions.getMissingOutputError(entrypointPath, outputPath);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return outputUrl;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
private async awaitEntrypointRegistration(registration: Promise<string>, entrypointPath: string): Promise<string> {
|
|
106
|
+
if (process.env.NODE_ENV !== 'development') {
|
|
107
|
+
return await registration;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return await Promise.race([
|
|
111
|
+
registration,
|
|
112
|
+
new Promise<string>((_, reject) => {
|
|
113
|
+
setTimeout(() => {
|
|
114
|
+
reject(new Error(`[HMR] Timed out registering entrypoint: ${entrypointPath}`));
|
|
115
|
+
}, this.options.registrationTimeoutMs);
|
|
116
|
+
}),
|
|
117
|
+
]);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
private getEntrypointOutput(entrypointPath: string): { outputPath: string; outputUrl: string } {
|
|
121
|
+
const relativePath = path.relative(this.options.srcDir, entrypointPath);
|
|
122
|
+
const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
|
|
123
|
+
const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
|
|
124
|
+
const urlPath = encodedPathJs.split(path.sep).join('/');
|
|
125
|
+
|
|
126
|
+
return {
|
|
127
|
+
outputUrl: `/${path.join(RESOLVED_ASSETS_DIR, '_hmr', urlPath)}`,
|
|
128
|
+
outputPath: path.join(this.options.distDir, urlPath),
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
private removeStaleEntrypointOutput(outputPath: string): void {
|
|
133
|
+
if (!fileSystem.exists(outputPath)) {
|
|
134
|
+
return;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
try {
|
|
138
|
+
fileSystem.remove(outputPath);
|
|
139
|
+
} catch (error) {
|
|
140
|
+
appLogger.warn(
|
|
141
|
+
`[HMR] Failed to remove stale entrypoint output ${outputPath}: ${error instanceof Error ? error.message : String(error)}`,
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
private encodeDynamicSegments(filepath: string): string {
|
|
147
|
+
return filepath.replace(/\[([^\]]+)\]/g, '_$1_');
|
|
148
|
+
}
|
|
149
|
+
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import type { IHmrManager } from '../../types/public-types.ts';
|
|
2
|
+
|
|
3
|
+
const HMR_RUNTIME_IMPORT = "import '/_hmr_runtime.js'";
|
|
4
|
+
const HMR_RUNTIME_SCRIPT = `<script type="module">${HMR_RUNTIME_IMPORT};</script>`;
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Returns whether a response is HTML and therefore eligible for development HMR
|
|
8
|
+
* runtime injection.
|
|
9
|
+
*/
|
|
10
|
+
export function isHtmlResponse(response: Response): boolean {
|
|
11
|
+
const contentType = response.headers.get('Content-Type');
|
|
12
|
+
return contentType !== null && contentType.startsWith('text/html');
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Returns whether HTML responses should receive the HMR runtime bootstrap.
|
|
17
|
+
*
|
|
18
|
+
* This is shared because filesystem page responses and adapter-level HTML
|
|
19
|
+
* responses flow through different layers, but both need identical injection
|
|
20
|
+
* behavior in watch mode.
|
|
21
|
+
*/
|
|
22
|
+
export function shouldInjectHmrHtmlResponse(watch: boolean, hmrManager?: Pick<IHmrManager, 'isEnabled'>): boolean {
|
|
23
|
+
return watch && hmrManager?.isEnabled() === true;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Injects the development HMR runtime script into an HTML response if it is not
|
|
28
|
+
* already present.
|
|
29
|
+
*
|
|
30
|
+
* The check is intentionally idempotent because an HTML response can pass
|
|
31
|
+
* through more than one development-layer wrapper before reaching the client.
|
|
32
|
+
*/
|
|
33
|
+
export async function injectHmrRuntimeIntoHtmlResponse(response: Response): Promise<Response> {
|
|
34
|
+
const html = await response.text();
|
|
35
|
+
if (html.includes(HMR_RUNTIME_IMPORT)) {
|
|
36
|
+
return new Response(html, {
|
|
37
|
+
status: response.status,
|
|
38
|
+
statusText: response.statusText,
|
|
39
|
+
headers: response.headers,
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
const updatedHtml = html.replace(/<\/html>/i, `${HMR_RUNTIME_SCRIPT}</html>`);
|
|
44
|
+
const headers = new Headers(response.headers);
|
|
45
|
+
headers.delete('Content-Length');
|
|
46
|
+
|
|
47
|
+
return new Response(updatedHtml, {
|
|
48
|
+
status: response.status,
|
|
49
|
+
statusText: response.statusText,
|
|
50
|
+
headers,
|
|
51
|
+
});
|
|
52
|
+
}
|