@ecopages/core 0.2.0-alpha.4 → 0.2.0-alpha.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +213 -12
- package/package.json +100 -188
- package/src/adapters/README.md +39 -0
- package/src/adapters/bun/hmr-manager.test.ts +267 -0
- package/src/adapters/bun/hmr-manager.ts +181 -68
- package/src/adapters/bun/index.ts +1 -2
- package/src/adapters/bun/server-adapter.ts +41 -34
- package/src/adapters/bun/server-lifecycle.ts +40 -70
- package/src/adapters/index.ts +1 -1
- package/src/adapters/node/bootstrap-dependency-resolver.test.ts +282 -0
- package/src/adapters/node/bootstrap-dependency-resolver.ts +301 -0
- package/src/adapters/node/index.ts +7 -0
- package/src/adapters/node/node-client-bridge.test.ts +198 -0
- package/src/adapters/node/node-hmr-manager.test.ts +322 -0
- package/src/adapters/node/node-hmr-manager.ts +208 -116
- package/src/adapters/node/runtime-adapter.test.ts +868 -0
- package/src/adapters/node/runtime-adapter.ts +439 -0
- package/src/adapters/node/server-adapter.ts +31 -104
- package/src/adapters/node/static-content-server.test.ts +60 -0
- package/src/adapters/node/static-content-server.ts +36 -0
- package/src/adapters/node/write-runtime-manifest.ts +38 -0
- package/src/adapters/shared/api-response.test.ts +97 -0
- package/src/{define-api-handler.ts → adapters/shared/define-api-handler.ts} +1 -1
- package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +7 -1
- package/src/adapters/shared/file-route-middleware-pipeline.test.ts +90 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +6 -2
- package/src/adapters/shared/fs-server-response-factory.test.ts +187 -0
- package/src/adapters/shared/fs-server-response-matcher.test.ts +286 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +17 -10
- package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
- package/src/adapters/shared/hmr-html-response.ts +52 -0
- package/src/adapters/shared/hmr-manager.contract.test.ts +196 -0
- package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
- package/src/adapters/shared/render-context.test.ts +146 -0
- package/src/adapters/shared/render-context.ts +21 -6
- package/src/adapters/shared/runtime-bootstrap.ts +79 -0
- package/src/adapters/shared/server-adapter.test.ts +77 -0
- package/src/adapters/shared/server-adapter.ts +51 -4
- package/src/adapters/shared/server-route-handler.test.ts +110 -0
- package/src/adapters/shared/server-route-handler.ts +5 -18
- package/src/adapters/shared/server-static-builder.test.ts +316 -0
- package/src/adapters/shared/server-static-builder.ts +92 -8
- package/src/build/README.md +101 -0
- package/src/build/build-adapter-serialization.test.ts +268 -0
- package/src/build/build-adapter.test.ts +815 -0
- package/src/build/build-adapter.ts +235 -6
- package/src/build/build-manifest.ts +54 -0
- package/src/build/dev-build-coordinator.ts +221 -0
- package/src/build/esbuild-build-adapter.ts +132 -83
- package/src/build/runtime-build-executor.ts +34 -0
- package/src/build/runtime-specifier-alias-plugin.test.ts +43 -0
- package/src/build/runtime-specifier-alias-plugin.ts +58 -0
- package/src/config/README.md +33 -0
- package/src/config/config-builder.test.ts +410 -0
- package/src/config/config-builder.ts +281 -49
- package/src/constants.ts +15 -0
- package/src/declarations.d.ts +18 -13
- package/src/eco/README.md +70 -16
- package/src/eco/component-render-context.ts +39 -17
- package/src/eco/eco.test.ts +678 -0
- package/src/eco/eco.ts +29 -8
- package/src/eco/eco.types.ts +20 -1
- package/src/eco/eco.utils.test.ts +124 -0
- package/src/eco/global-injector-map.test.ts +42 -0
- package/src/eco/lazy-injector-map.test.ts +66 -0
- package/src/eco/module-dependencies.test.ts +30 -0
- package/src/errors/http-error.test.ts +134 -0
- package/src/global/utils.test.ts +12 -0
- package/src/hmr/README.md +26 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
- package/src/hmr/client/hmr-runtime.ts +38 -7
- package/src/hmr/hmr-strategy.test.ts +124 -0
- package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
- package/src/hmr/hmr.test.e2e.ts +29 -38
- package/src/hmr/strategies/js-hmr-strategy.test.ts +335 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +115 -115
- package/src/index.ts +1 -1
- package/src/integrations/ghtml/ghtml-renderer.test.ts +63 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +4 -1
- package/src/internal-types.ts +39 -19
- package/src/plugins/README.md +34 -0
- package/src/plugins/alias-resolver-plugin.test.ts +41 -0
- package/src/plugins/alias-resolver-plugin.ts +21 -3
- package/src/plugins/eco-component-meta-plugin.test.ts +380 -0
- package/src/plugins/eco-component-meta-plugin.ts +10 -3
- package/src/plugins/integration-plugin.test.ts +111 -0
- package/src/plugins/integration-plugin.ts +45 -3
- package/src/plugins/processor.test.ts +148 -0
- package/src/plugins/processor.ts +22 -2
- package/src/plugins/runtime-capability.ts +14 -0
- package/src/public-types.ts +73 -16
- package/src/route-renderer/GRAPH.md +16 -20
- package/src/route-renderer/README.md +8 -21
- package/src/route-renderer/component-graph/component-graph-executor.test.ts +41 -0
- package/src/route-renderer/component-graph/component-graph.test.ts +63 -0
- package/src/route-renderer/component-graph/component-marker.test.ts +73 -0
- package/src/route-renderer/component-graph/component-reference.ts +29 -0
- package/src/route-renderer/component-graph/marker-graph-resolver.test.ts +135 -0
- package/src/route-renderer/{marker-graph-resolver.ts → component-graph/marker-graph-resolver.ts} +11 -9
- package/src/route-renderer/orchestration/integration-renderer.test.ts +936 -0
- package/src/route-renderer/{integration-renderer.ts → orchestration/integration-renderer.ts} +113 -19
- package/src/route-renderer/orchestration/render-execution.service.test.ts +97 -0
- package/src/route-renderer/{render-execution.service.ts → orchestration/render-execution.service.ts} +109 -37
- package/src/route-renderer/orchestration/render-preparation.service.test.ts +235 -0
- package/src/route-renderer/{render-preparation.service.ts → orchestration/render-preparation.service.ts} +127 -9
- package/src/route-renderer/page-loading/dependency-resolver.test.ts +345 -0
- package/src/route-renderer/{dependency-resolver.ts → page-loading/dependency-resolver.ts} +28 -12
- package/src/route-renderer/page-loading/page-module-loader.test.ts +96 -0
- package/src/route-renderer/{page-module-loader.ts → page-loading/page-module-loader.ts} +49 -21
- package/src/route-renderer/route-renderer.ts +36 -1
- package/src/router/README.md +26 -0
- package/src/router/client/link-intent.d.ts +53 -0
- package/src/router/client/link-intent.test.browser.ts +51 -0
- package/src/router/client/link-intent.ts +92 -0
- package/src/router/client/navigation-coordinator.test.ts +237 -0
- package/src/router/client/navigation-coordinator.ts +433 -0
- package/src/router/server/fs-router-scanner.test.ts +83 -0
- package/src/router/{fs-router-scanner.ts → server/fs-router-scanner.ts} +12 -10
- package/src/router/server/fs-router.test.ts +214 -0
- package/src/router/{fs-router.ts → server/fs-router.ts} +2 -2
- package/src/services/README.md +29 -0
- package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +385 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.ts +101 -6
- package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.ts +2 -2
- package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.ts +2 -1
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +72 -0
- package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +95 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
- package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/index.ts +2 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.ts +1 -1
- package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.ts +11 -5
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-script-processor.ts +17 -27
- package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +286 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.ts +3 -3
- package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.ts +5 -4
- package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +199 -0
- package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.ts +4 -1
- package/src/services/assets/browser-bundle.service.test.ts +36 -0
- package/src/services/assets/browser-bundle.service.ts +53 -0
- package/src/services/cache/index.ts +3 -3
- package/src/services/cache/memory-cache-store.test.ts +225 -0
- package/src/services/cache/memory-cache-store.ts +1 -1
- package/src/services/cache/page-cache-service.test.ts +175 -0
- package/src/services/cache/page-cache-service.ts +3 -3
- package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
- package/src/services/{page-request-cache-coordinator.service.ts → cache/page-request-cache-coordinator.service.ts} +9 -6
- package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
- package/src/services/html/html-rewriter-provider.service.ts +103 -0
- package/src/services/html/html-transformer.service.test.ts +378 -0
- package/src/services/html/html-transformer.service.ts +279 -0
- package/src/services/invalidation/development-invalidation.service.test.ts +77 -0
- package/src/services/invalidation/development-invalidation.service.ts +261 -0
- package/src/services/module-loading/app-server-module-transpiler.service.ts +52 -0
- package/src/services/module-loading/page-module-import.service.test.ts +253 -0
- package/src/services/module-loading/page-module-import.service.ts +200 -0
- package/src/services/module-loading/server-loader.service.test.ts +161 -0
- package/src/services/module-loading/server-loader.service.ts +130 -0
- package/src/services/module-loading/server-module-transpiler.service.test.ts +115 -0
- package/src/services/module-loading/server-module-transpiler.service.ts +105 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.test.ts +95 -0
- package/src/services/runtime-manifest/node-runtime-manifest.service.ts +101 -0
- package/src/services/runtime-state/dev-graph.service.ts +217 -0
- package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
- package/src/services/runtime-state/runtime-specifier-registry.service.ts +96 -0
- package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
- package/src/services/validation/schema-validation-service.test.ts +223 -0
- package/src/services/{schema-validation-service.ts → validation/schema-validation-service.ts} +1 -1
- package/src/static-site-generator/README.md +26 -0
- package/src/static-site-generator/static-site-generator.test.ts +307 -0
- package/src/static-site-generator/static-site-generator.ts +109 -6
- package/src/utils/deep-merge.test.ts +114 -0
- package/src/utils/invariant.test.ts +22 -0
- package/src/utils/path-utils.test.ts +15 -0
- package/src/utils/resolve-work-dir.ts +45 -0
- package/src/utils/server-utils.test.ts +38 -0
- package/src/watchers/project-watcher.integration.test.ts +337 -0
- package/src/watchers/project-watcher.test-helpers.ts +1 -1
- package/src/watchers/project-watcher.test.ts +678 -0
- package/src/watchers/project-watcher.ts +130 -111
- package/CHANGELOG.md +0 -91
- package/src/adapters/abstract/application-adapter.d.ts +0 -168
- package/src/adapters/abstract/application-adapter.js +0 -109
- package/src/adapters/abstract/router-adapter.d.ts +0 -26
- package/src/adapters/abstract/router-adapter.js +0 -5
- package/src/adapters/abstract/server-adapter.d.ts +0 -69
- package/src/adapters/abstract/server-adapter.js +0 -15
- package/src/adapters/bun/client-bridge.d.ts +0 -34
- package/src/adapters/bun/client-bridge.js +0 -48
- package/src/adapters/bun/create-app.d.ts +0 -60
- package/src/adapters/bun/create-app.js +0 -117
- package/src/adapters/bun/define-api-handler.d.ts +0 -61
- package/src/adapters/bun/define-api-handler.js +0 -15
- package/src/adapters/bun/define-api-handler.ts +0 -114
- package/src/adapters/bun/hmr-manager.d.ts +0 -85
- package/src/adapters/bun/hmr-manager.js +0 -240
- package/src/adapters/bun/index.d.ts +0 -3
- package/src/adapters/bun/index.js +0 -8
- package/src/adapters/bun/server-adapter.d.ts +0 -155
- package/src/adapters/bun/server-adapter.js +0 -368
- package/src/adapters/bun/server-lifecycle.d.ts +0 -52
- package/src/adapters/bun/server-lifecycle.js +0 -120
- package/src/adapters/index.d.ts +0 -6
- package/src/adapters/index.js +0 -14
- package/src/adapters/node/create-app.d.ts +0 -21
- package/src/adapters/node/create-app.js +0 -143
- package/src/adapters/node/index.d.ts +0 -4
- package/src/adapters/node/index.js +0 -8
- package/src/adapters/node/node-client-bridge.d.ts +0 -26
- package/src/adapters/node/node-client-bridge.js +0 -66
- package/src/adapters/node/node-hmr-manager.d.ts +0 -63
- package/src/adapters/node/node-hmr-manager.js +0 -237
- package/src/adapters/node/server-adapter.d.ts +0 -190
- package/src/adapters/node/server-adapter.js +0 -420
- package/src/adapters/node/static-content-server.d.ts +0 -24
- package/src/adapters/node/static-content-server.js +0 -166
- package/src/adapters/shared/api-response.d.ts +0 -52
- package/src/adapters/shared/api-response.js +0 -96
- package/src/adapters/shared/application-adapter.d.ts +0 -18
- package/src/adapters/shared/application-adapter.js +0 -90
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
- package/src/adapters/shared/explicit-static-route-matcher.js +0 -100
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
- package/src/adapters/shared/file-route-middleware-pipeline.js +0 -98
- package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
- package/src/adapters/shared/fs-server-response-factory.js +0 -97
- package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -71
- package/src/adapters/shared/fs-server-response-matcher.js +0 -155
- package/src/adapters/shared/render-context.d.ts +0 -14
- package/src/adapters/shared/render-context.js +0 -69
- package/src/adapters/shared/server-adapter.d.ts +0 -87
- package/src/adapters/shared/server-adapter.js +0 -353
- package/src/adapters/shared/server-route-handler.d.ts +0 -89
- package/src/adapters/shared/server-route-handler.js +0 -120
- package/src/adapters/shared/server-static-builder.d.ts +0 -38
- package/src/adapters/shared/server-static-builder.js +0 -46
- package/src/build/build-adapter.d.ts +0 -74
- package/src/build/build-adapter.js +0 -54
- package/src/build/build-types.d.ts +0 -57
- package/src/build/build-types.js +0 -0
- package/src/build/esbuild-build-adapter.d.ts +0 -69
- package/src/build/esbuild-build-adapter.js +0 -390
- package/src/config/config-builder.d.ts +0 -227
- package/src/config/config-builder.js +0 -392
- package/src/constants.d.ts +0 -32
- package/src/constants.js +0 -21
- package/src/create-app.d.ts +0 -17
- package/src/create-app.js +0 -66
- package/src/define-api-handler.d.ts +0 -25
- package/src/define-api-handler.js +0 -15
- package/src/dev/sc-server.d.ts +0 -30
- package/src/dev/sc-server.js +0 -111
- package/src/eco/component-render-context.d.ts +0 -105
- package/src/eco/component-render-context.js +0 -77
- package/src/eco/eco.d.ts +0 -9
- package/src/eco/eco.js +0 -110
- package/src/eco/eco.types.d.ts +0 -170
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.utils.d.ts +0 -40
- package/src/eco/eco.utils.js +0 -40
- package/src/eco/global-injector-map.d.ts +0 -16
- package/src/eco/global-injector-map.js +0 -80
- package/src/eco/lazy-injector-map.d.ts +0 -8
- package/src/eco/lazy-injector-map.js +0 -70
- package/src/eco/module-dependencies.d.ts +0 -18
- package/src/eco/module-dependencies.js +0 -49
- package/src/errors/http-error.d.ts +0 -31
- package/src/errors/http-error.js +0 -50
- package/src/errors/index.d.ts +0 -2
- package/src/errors/index.js +0 -4
- package/src/errors/locals-access-error.d.ts +0 -4
- package/src/errors/locals-access-error.js +0 -9
- package/src/global/app-logger.d.ts +0 -2
- package/src/global/app-logger.js +0 -6
- package/src/hmr/client/hmr-runtime.d.ts +0 -10
- package/src/hmr/client/hmr-runtime.js +0 -86
- package/src/hmr/hmr-strategy.d.ts +0 -159
- package/src/hmr/hmr-strategy.js +0 -29
- package/src/hmr/hmr.test.e2e.d.ts +0 -1
- package/src/hmr/hmr.test.e2e.js +0 -50
- package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
- package/src/hmr/strategies/default-hmr-strategy.js +0 -34
- package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -136
- package/src/hmr/strategies/js-hmr-strategy.js +0 -188
- package/src/index.browser.d.ts +0 -3
- package/src/index.browser.js +0 -4
- package/src/index.d.ts +0 -5
- package/src/index.js +0 -10
- package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -15
- package/src/integrations/ghtml/ghtml-renderer.js +0 -60
- package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -20
- package/src/integrations/ghtml/ghtml.plugin.js +0 -21
- package/src/internal-types.d.ts +0 -200
- package/src/internal-types.js +0 -0
- package/src/plugins/alias-resolver-plugin.d.ts +0 -2
- package/src/plugins/alias-resolver-plugin.js +0 -39
- package/src/plugins/eco-component-meta-plugin.d.ts +0 -95
- package/src/plugins/eco-component-meta-plugin.js +0 -157
- package/src/plugins/integration-plugin.d.ts +0 -102
- package/src/plugins/integration-plugin.js +0 -100
- package/src/plugins/processor.d.ts +0 -82
- package/src/plugins/processor.js +0 -122
- package/src/public-types.d.ts +0 -1098
- package/src/public-types.js +0 -0
- package/src/route-renderer/component-graph-executor.d.ts +0 -32
- package/src/route-renderer/component-graph-executor.js +0 -31
- package/src/route-renderer/component-graph.d.ts +0 -42
- package/src/route-renderer/component-graph.js +0 -72
- package/src/route-renderer/component-marker.d.ts +0 -52
- package/src/route-renderer/component-marker.js +0 -46
- package/src/route-renderer/dependency-resolver.d.ts +0 -24
- package/src/route-renderer/dependency-resolver.js +0 -428
- package/src/route-renderer/html-post-processing.service.d.ts +0 -40
- package/src/route-renderer/html-post-processing.service.js +0 -86
- package/src/route-renderer/html-post-processing.service.ts +0 -103
- package/src/route-renderer/integration-renderer.d.ts +0 -339
- package/src/route-renderer/integration-renderer.js +0 -526
- package/src/route-renderer/marker-graph-resolver.d.ts +0 -76
- package/src/route-renderer/marker-graph-resolver.js +0 -93
- package/src/route-renderer/page-module-loader.d.ts +0 -61
- package/src/route-renderer/page-module-loader.js +0 -102
- package/src/route-renderer/render-execution.service.d.ts +0 -69
- package/src/route-renderer/render-execution.service.js +0 -91
- package/src/route-renderer/render-preparation.service.d.ts +0 -112
- package/src/route-renderer/render-preparation.service.js +0 -243
- package/src/route-renderer/route-renderer.d.ts +0 -26
- package/src/route-renderer/route-renderer.js +0 -68
- package/src/router/fs-router-scanner.d.ts +0 -41
- package/src/router/fs-router-scanner.js +0 -155
- package/src/router/fs-router.d.ts +0 -26
- package/src/router/fs-router.js +0 -100
- package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
- package/src/services/asset-processing-service/asset-processing.service.js +0 -250
- package/src/services/asset-processing-service/asset.factory.d.ts +0 -17
- package/src/services/asset-processing-service/asset.factory.js +0 -82
- package/src/services/asset-processing-service/assets.types.d.ts +0 -88
- package/src/services/asset-processing-service/assets.types.js +0 -0
- package/src/services/asset-processing-service/index.d.ts +0 -3
- package/src/services/asset-processing-service/index.js +0 -3
- package/src/services/asset-processing-service/processor.interface.d.ts +0 -22
- package/src/services/asset-processing-service/processor.interface.js +0 -6
- package/src/services/asset-processing-service/processor.registry.d.ts +0 -8
- package/src/services/asset-processing-service/processor.registry.js +0 -15
- package/src/services/asset-processing-service/processors/base/base-processor.d.ts +0 -24
- package/src/services/asset-processing-service/processors/base/base-processor.js +0 -59
- package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +0 -16
- package/src/services/asset-processing-service/processors/base/base-script-processor.js +0 -80
- package/src/services/asset-processing-service/processors/index.d.ts +0 -5
- package/src/services/asset-processing-service/processors/index.js +0 -5
- package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +0 -5
- package/src/services/asset-processing-service/processors/script/content-script.processor.js +0 -57
- package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +0 -8
- package/src/services/asset-processing-service/processors/script/file-script.processor.js +0 -76
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +0 -74
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -5
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -63
- package/src/services/cache/cache.types.d.ts +0 -107
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/index.d.ts +0 -7
- package/src/services/cache/index.js +0 -7
- package/src/services/cache/memory-cache-store.d.ts +0 -42
- package/src/services/cache/memory-cache-store.js +0 -98
- package/src/services/cache/page-cache-service.d.ts +0 -70
- package/src/services/cache/page-cache-service.js +0 -152
- package/src/services/html-transformer.service.d.ts +0 -50
- package/src/services/html-transformer.service.js +0 -163
- package/src/services/html-transformer.service.ts +0 -217
- package/src/services/page-module-import.service.d.ts +0 -37
- package/src/services/page-module-import.service.js +0 -88
- package/src/services/page-module-import.service.ts +0 -129
- package/src/services/page-request-cache-coordinator.service.d.ts +0 -75
- package/src/services/page-request-cache-coordinator.service.js +0 -107
- package/src/services/schema-validation-service.d.ts +0 -122
- package/src/services/schema-validation-service.js +0 -101
- package/src/services/validation/standard-schema.types.d.ts +0 -65
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/static-site-generator/static-site-generator.d.ts +0 -57
- package/src/static-site-generator/static-site-generator.js +0 -272
- package/src/utils/css.d.ts +0 -1
- package/src/utils/css.js +0 -7
- package/src/utils/deep-merge.d.ts +0 -14
- package/src/utils/deep-merge.js +0 -32
- package/src/utils/hash.d.ts +0 -1
- package/src/utils/hash.js +0 -7
- package/src/utils/html.d.ts +0 -1
- package/src/utils/html.js +0 -4
- package/src/utils/invariant.d.ts +0 -5
- package/src/utils/invariant.js +0 -11
- package/src/utils/locals-utils.d.ts +0 -15
- package/src/utils/locals-utils.js +0 -24
- package/src/utils/parse-cli-args.d.ts +0 -24
- package/src/utils/parse-cli-args.js +0 -47
- package/src/utils/path-utils.module.d.ts +0 -5
- package/src/utils/path-utils.module.js +0 -14
- package/src/utils/runtime.d.ts +0 -11
- package/src/utils/runtime.js +0 -40
- package/src/utils/server-utils.module.d.ts +0 -19
- package/src/utils/server-utils.module.js +0 -56
- package/src/watchers/project-watcher.d.ts +0 -125
- package/src/watchers/project-watcher.js +0 -265
- package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
- package/src/watchers/project-watcher.test-helpers.js +0 -52
- /package/src/route-renderer/{component-graph-executor.ts → component-graph/component-graph-executor.ts} +0 -0
- /package/src/route-renderer/{component-graph.ts → component-graph/component-graph.ts} +0 -0
- /package/src/route-renderer/{component-marker.ts → component-graph/component-marker.ts} +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/index.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.ts +0 -0
- /package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.ts +0 -0
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import { describe, expect, it } from 'vitest';
|
|
2
|
+
import type { Middleware } from '../../public-types.ts';
|
|
3
|
+
import { LocalsAccessError } from '../../errors/locals-access-error.ts';
|
|
4
|
+
import {
|
|
5
|
+
FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS,
|
|
6
|
+
FileRouteMiddlewarePipeline,
|
|
7
|
+
} from './file-route-middleware-pipeline.ts';
|
|
8
|
+
|
|
9
|
+
declare module '../../public-types.ts' {
|
|
10
|
+
interface RequestLocals {
|
|
11
|
+
user?: string;
|
|
12
|
+
first?: boolean;
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
describe('FileRouteMiddlewarePipeline', () => {
|
|
17
|
+
it('should reject middleware on non-dynamic pages', () => {
|
|
18
|
+
const service = new FileRouteMiddlewarePipeline(null);
|
|
19
|
+
|
|
20
|
+
expect(() =>
|
|
21
|
+
service.assertValidConfiguration({
|
|
22
|
+
middleware: [async (_ctx, next) => next()],
|
|
23
|
+
pageCacheStrategy: 'static',
|
|
24
|
+
filePath: '/app/pages/index.tsx',
|
|
25
|
+
}),
|
|
26
|
+
).toThrowError(
|
|
27
|
+
new LocalsAccessError(
|
|
28
|
+
FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS.middlewareRequiresDynamic('/app/pages/index.tsx'),
|
|
29
|
+
),
|
|
30
|
+
);
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it('should create middleware context with render methods disabled', async () => {
|
|
34
|
+
const service = new FileRouteMiddlewarePipeline(null);
|
|
35
|
+
const DummyView = (() => '<div>dummy</div>') as never;
|
|
36
|
+
const context = service.createContext({
|
|
37
|
+
request: new Request('http://localhost:3000/hello'),
|
|
38
|
+
params: { slug: 'hello' },
|
|
39
|
+
locals: { user: 'andee' },
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
await expect(context.render(DummyView, {})).rejects.toThrow(
|
|
43
|
+
FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS.CTX_RENDER_UNAVAILABLE,
|
|
44
|
+
);
|
|
45
|
+
await expect(context.renderPartial(DummyView, {})).rejects.toThrow(
|
|
46
|
+
FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS.CTX_RENDER_PARTIAL_UNAVAILABLE,
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
expect(context.require('user', () => new Response('missing', { status: 500 }))).toBe('andee');
|
|
50
|
+
expect(await context.html('<p>ok</p>').text()).toContain('<p>ok</p>');
|
|
51
|
+
expect(await context.json({ ok: true }).text()).toContain('{"ok":true}');
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
it('should execute middleware in order and eventually render the response', async () => {
|
|
55
|
+
const service = new FileRouteMiddlewarePipeline(null);
|
|
56
|
+
const context = service.createContext({
|
|
57
|
+
request: new Request('http://localhost:3000/hello'),
|
|
58
|
+
params: {},
|
|
59
|
+
locals: {},
|
|
60
|
+
});
|
|
61
|
+
const events: string[] = [];
|
|
62
|
+
|
|
63
|
+
const middleware: Middleware[] = [
|
|
64
|
+
async (ctx, next) => {
|
|
65
|
+
events.push('first:before');
|
|
66
|
+
ctx.locals.first = true;
|
|
67
|
+
const response = await next();
|
|
68
|
+
events.push('first:after');
|
|
69
|
+
return response;
|
|
70
|
+
},
|
|
71
|
+
async (ctx, next) => {
|
|
72
|
+
events.push('second');
|
|
73
|
+
expect(ctx.locals.first).toBe(true);
|
|
74
|
+
return next();
|
|
75
|
+
},
|
|
76
|
+
];
|
|
77
|
+
|
|
78
|
+
const response = await service.run({
|
|
79
|
+
middleware,
|
|
80
|
+
context,
|
|
81
|
+
renderResponse: async () => {
|
|
82
|
+
events.push('render');
|
|
83
|
+
return new Response('done');
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
expect(await response.text()).toBe('done');
|
|
88
|
+
expect(events).toEqual(['first:before', 'second', 'render', 'first:after']);
|
|
89
|
+
});
|
|
90
|
+
});
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { createRequire } from '../../utils/locals-utils.ts';
|
|
2
2
|
import type { Middleware, ApiHandlerContext, RequestLocals } from '../../public-types.ts';
|
|
3
|
-
import type { PageCacheService } from '../../services/cache/page-cache-service.
|
|
3
|
+
import type { PageCacheService } from '../../services/cache/page-cache-service.js';
|
|
4
4
|
import { ApiResponseBuilder } from './api-response.js';
|
|
5
5
|
import { LocalsAccessError } from '../../errors/locals-access-error.ts';
|
|
6
6
|
|
|
@@ -20,7 +20,11 @@ export const FILE_ROUTE_MIDDLEWARE_PIPELINE_ERRORS = {
|
|
|
20
20
|
* request-time dynamic rendering.
|
|
21
21
|
*/
|
|
22
22
|
export class FileRouteMiddlewarePipeline {
|
|
23
|
-
|
|
23
|
+
private cacheService: PageCacheService | null;
|
|
24
|
+
|
|
25
|
+
constructor(cacheService: PageCacheService | null) {
|
|
26
|
+
this.cacheService = cacheService;
|
|
27
|
+
}
|
|
24
28
|
|
|
25
29
|
/**
|
|
26
30
|
* Enforces the current middleware contract for file-routed pages.
|
|
@@ -0,0 +1,187 @@
|
|
|
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
|
+
});
|
|
@@ -0,0 +1,286 @@
|
|
|
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,12 +1,14 @@
|
|
|
1
1
|
import path from 'node:path';
|
|
2
2
|
import { appLogger } from '../../global/app-logger.ts';
|
|
3
|
-
import type { MatchResult } from '../../internal-types.ts';
|
|
3
|
+
import type { EcoPagesAppConfig, MatchResult } from '../../internal-types.ts';
|
|
4
4
|
import type { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
|
|
5
|
-
import type { FSRouter } from '../../router/fs-router.ts';
|
|
5
|
+
import type { FSRouter } from '../../router/server/fs-router.ts';
|
|
6
6
|
import type { PageCacheService } from '../../services/cache/page-cache-service.ts';
|
|
7
7
|
import type { CacheStrategy, RenderResult } from '../../services/cache/cache.types.ts';
|
|
8
|
-
import {
|
|
9
|
-
import {
|
|
8
|
+
import { PageRequestCacheCoordinator } from '../../services/cache/page-request-cache-coordinator.service.ts';
|
|
9
|
+
import { getAppServerModuleTranspiler } from '../../services/module-loading/app-server-module-transpiler.service.ts';
|
|
10
|
+
import type { ServerModuleTranspiler } from '../../services/module-loading/server-module-transpiler.service.ts';
|
|
11
|
+
import { resolveInternalExecutionDir } from '../../utils/resolve-work-dir.ts';
|
|
10
12
|
import { ServerUtils } from '../../utils/server-utils.module.ts';
|
|
11
13
|
import type { Middleware, RequestLocals } from '../../public-types.ts';
|
|
12
14
|
import { FileRouteMiddlewarePipeline } from './file-route-middleware-pipeline.ts';
|
|
@@ -21,6 +23,7 @@ export const FILE_SYSTEM_RESPONSE_MATCHER_ERRORS = {
|
|
|
21
23
|
} as const;
|
|
22
24
|
|
|
23
25
|
export interface FileSystemResponseMatcherOptions {
|
|
26
|
+
appConfig: EcoPagesAppConfig;
|
|
24
27
|
router: FSRouter;
|
|
25
28
|
routeRendererFactory: RouteRendererFactory;
|
|
26
29
|
fileSystemResponseFactory: FileSystemServerResponseFactory;
|
|
@@ -39,24 +42,27 @@ export interface FileSystemResponseMatcherOptions {
|
|
|
39
42
|
* error translation.
|
|
40
43
|
*/
|
|
41
44
|
export class FileSystemResponseMatcher {
|
|
45
|
+
private appConfig: EcoPagesAppConfig;
|
|
42
46
|
private router: FSRouter;
|
|
43
47
|
private routeRendererFactory: RouteRendererFactory;
|
|
44
48
|
private fileSystemResponseFactory: FileSystemServerResponseFactory;
|
|
45
|
-
private
|
|
49
|
+
private serverModuleTranspiler: ServerModuleTranspiler;
|
|
46
50
|
private pageRequestCacheCoordinator: PageRequestCacheCoordinator;
|
|
47
51
|
private fileRouteMiddlewarePipeline: FileRouteMiddlewarePipeline;
|
|
48
52
|
|
|
49
53
|
constructor({
|
|
54
|
+
appConfig,
|
|
50
55
|
router,
|
|
51
56
|
routeRendererFactory,
|
|
52
57
|
fileSystemResponseFactory,
|
|
53
58
|
cacheService = null,
|
|
54
59
|
defaultCacheStrategy = 'static',
|
|
55
60
|
}: FileSystemResponseMatcherOptions) {
|
|
61
|
+
this.appConfig = appConfig;
|
|
56
62
|
this.router = router;
|
|
57
63
|
this.routeRendererFactory = routeRendererFactory;
|
|
58
64
|
this.fileSystemResponseFactory = fileSystemResponseFactory;
|
|
59
|
-
this.
|
|
65
|
+
this.serverModuleTranspiler = getAppServerModuleTranspiler(appConfig);
|
|
60
66
|
this.pageRequestCacheCoordinator = new PageRequestCacheCoordinator(cacheService, defaultCacheStrategy);
|
|
61
67
|
this.fileRouteMiddlewarePipeline = new FileRouteMiddlewarePipeline(cacheService);
|
|
62
68
|
}
|
|
@@ -174,16 +180,17 @@ export class FileSystemResponseMatcher {
|
|
|
174
180
|
*
|
|
175
181
|
* The matcher needs access to page-level metadata such as `cache` and
|
|
176
182
|
* `middleware` before full rendering starts, so it uses the shared module
|
|
177
|
-
* import service directly rather than going through route rendering.
|
|
183
|
+
* import service directly rather than going through route rendering. The
|
|
184
|
+
* app config is injected explicitly so build ownership stays at the adapter
|
|
185
|
+
* boundary instead of leaking through nested router collaborators.
|
|
178
186
|
*
|
|
179
187
|
* @param filePath Absolute page module path.
|
|
180
188
|
* @returns Imported page module.
|
|
181
189
|
*/
|
|
182
190
|
private async importPageModule(filePath: string): Promise<unknown> {
|
|
183
|
-
return this.
|
|
191
|
+
return this.serverModuleTranspiler.importModule({
|
|
184
192
|
filePath,
|
|
185
|
-
|
|
186
|
-
outdir: path.join(this.router.assetPrefix, '.server-modules-meta'),
|
|
193
|
+
outdir: path.join(resolveInternalExecutionDir(this.appConfig), '.server-modules-meta'),
|
|
187
194
|
transpileErrorMessage: FILE_SYSTEM_RESPONSE_MATCHER_ERRORS.transpilePageModuleFailed,
|
|
188
195
|
noOutputMessage: FILE_SYSTEM_RESPONSE_MATCHER_ERRORS.noTranspiledOutputForPageModule,
|
|
189
196
|
});
|