@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,239 @@
|
|
|
1
|
+
import { createServer, type IncomingMessage, type Server as NodeHttpServer, type ServerResponse } from 'node:http';
|
|
2
|
+
import { extname, join, normalize, sep } from 'node:path';
|
|
3
|
+
import { DEFAULT_ECOPAGES_HOSTNAME, DEFAULT_ECOPAGES_PORT, STATUS_MESSAGE } from '../../config/constants.ts';
|
|
4
|
+
import { fileSystem } from '@ecopages/file-system';
|
|
5
|
+
import type { EcoPagesAppConfig } from '../../types/internal-types.ts';
|
|
6
|
+
import { ServerUtils } from '../../utils/server-utils.module.ts';
|
|
7
|
+
|
|
8
|
+
type NodeStaticContentServerOptions = {
|
|
9
|
+
hostname?: string;
|
|
10
|
+
port?: number;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Serves prebuilt static Ecopages output through Node's HTTP server.
|
|
15
|
+
*
|
|
16
|
+
* @remarks
|
|
17
|
+
* This server is used by the Node preview/build path once the app has already
|
|
18
|
+
* emitted its static output. It intentionally stays small: path sanitization,
|
|
19
|
+
* content-type selection, optional gzip serving, and 404 handling.
|
|
20
|
+
*/
|
|
21
|
+
export class NodeStaticContentServer {
|
|
22
|
+
private readonly appConfig: EcoPagesAppConfig;
|
|
23
|
+
private readonly options: NodeStaticContentServerOptions;
|
|
24
|
+
private server: NodeHttpServer | null = null;
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Creates the Node static-content server for one built app output directory.
|
|
28
|
+
*/
|
|
29
|
+
constructor({ appConfig, options }: { appConfig: EcoPagesAppConfig; options?: NodeStaticContentServerOptions }) {
|
|
30
|
+
this.appConfig = appConfig;
|
|
31
|
+
this.options = {
|
|
32
|
+
hostname: options?.hostname ?? DEFAULT_ECOPAGES_HOSTNAME,
|
|
33
|
+
port: options?.port ?? DEFAULT_ECOPAGES_PORT,
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Returns whether the given content type should be served from a pre-gzipped
|
|
39
|
+
* companion file when available.
|
|
40
|
+
*/
|
|
41
|
+
private shouldServeGzip(contentType: string): boolean {
|
|
42
|
+
return ['text/javascript', 'text/css'].includes(contentType);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Normalizes a request pathname and rejects directory traversal attempts.
|
|
47
|
+
*/
|
|
48
|
+
private sanitizePath(pathname: string): string | null {
|
|
49
|
+
const withoutLeadingSlash = pathname.replace(/^\/+/, '');
|
|
50
|
+
const normalizedPath = normalize(withoutLeadingSlash);
|
|
51
|
+
|
|
52
|
+
if (normalizedPath.startsWith('..') || normalizedPath.includes(`..${sep}`)) {
|
|
53
|
+
return null;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
return normalizedPath;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Writes one HTTP response with the provided headers and optional body.
|
|
61
|
+
*/
|
|
62
|
+
private sendResponse(res: ServerResponse, status: number, headers: Record<string, string>, body?: Buffer): void {
|
|
63
|
+
res.statusCode = status;
|
|
64
|
+
for (const [key, value] of Object.entries(headers)) {
|
|
65
|
+
res.setHeader(key, value);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (!body) {
|
|
69
|
+
res.end();
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
res.end(body);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Serves the generated 404 page when present, or a plain-text fallback.
|
|
78
|
+
*/
|
|
79
|
+
private sendNotFoundPage(req: IncomingMessage, res: ServerResponse): void {
|
|
80
|
+
const error404TemplatePath = join(this.appConfig.absolutePaths.distDir, '404.html');
|
|
81
|
+
const isHead = (req.method ?? 'GET').toUpperCase() === 'HEAD';
|
|
82
|
+
|
|
83
|
+
if (!fileSystem.exists(error404TemplatePath)) {
|
|
84
|
+
this.sendResponse(
|
|
85
|
+
res,
|
|
86
|
+
404,
|
|
87
|
+
{ 'Content-Type': 'text/plain' },
|
|
88
|
+
isHead ? undefined : Buffer.from(STATUS_MESSAGE[404]),
|
|
89
|
+
);
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const file = fileSystem.readFileAsBuffer(error404TemplatePath);
|
|
94
|
+
this.sendResponse(res, 404, { 'Content-Type': 'text/html' }, isHead ? undefined : file);
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* Serves one concrete file path, honoring gzip and HEAD semantics.
|
|
99
|
+
*/
|
|
100
|
+
private serveFile(req: IncomingMessage, res: ServerResponse, filePath: string, status = 200): void {
|
|
101
|
+
const contentType = ServerUtils.getContentType(extname(filePath));
|
|
102
|
+
const acceptsGzip = req.headers['accept-encoding']?.includes('gzip');
|
|
103
|
+
const isHead = (req.method ?? 'GET').toUpperCase() === 'HEAD';
|
|
104
|
+
|
|
105
|
+
if (acceptsGzip && this.shouldServeGzip(contentType)) {
|
|
106
|
+
const gzipPath = `${filePath}.gz`;
|
|
107
|
+
if (fileSystem.exists(gzipPath)) {
|
|
108
|
+
const file = fileSystem.readFileAsBuffer(gzipPath);
|
|
109
|
+
this.sendResponse(
|
|
110
|
+
res,
|
|
111
|
+
status,
|
|
112
|
+
{
|
|
113
|
+
'Content-Type': contentType,
|
|
114
|
+
'Content-Encoding': 'gzip',
|
|
115
|
+
Vary: 'Accept-Encoding',
|
|
116
|
+
},
|
|
117
|
+
isHead ? undefined : file,
|
|
118
|
+
);
|
|
119
|
+
return;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
if (!fileSystem.exists(filePath)) {
|
|
124
|
+
this.sendNotFoundPage(req, res);
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
const file = fileSystem.readFileAsBuffer(filePath);
|
|
129
|
+
this.sendResponse(res, status, { 'Content-Type': contentType }, isHead ? undefined : file);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Handles one incoming Node HTTP request against the built static output tree.
|
|
134
|
+
*/
|
|
135
|
+
private handleRequest(req: IncomingMessage, res: ServerResponse): void {
|
|
136
|
+
const method = (req.method ?? 'GET').toUpperCase();
|
|
137
|
+
const isHead = method === 'HEAD';
|
|
138
|
+
if (method !== 'GET' && method !== 'HEAD') {
|
|
139
|
+
this.sendResponse(
|
|
140
|
+
res,
|
|
141
|
+
405,
|
|
142
|
+
{ Allow: 'GET, HEAD', 'Content-Type': 'text/plain' },
|
|
143
|
+
isHead ? undefined : Buffer.from('Method Not Allowed'),
|
|
144
|
+
);
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
const url = new URL(req.url ?? '/', 'http://localhost');
|
|
149
|
+
let decodedPathname = '/';
|
|
150
|
+
try {
|
|
151
|
+
decodedPathname = decodeURIComponent(url.pathname);
|
|
152
|
+
} catch {
|
|
153
|
+
this.sendResponse(
|
|
154
|
+
res,
|
|
155
|
+
400,
|
|
156
|
+
{ 'Content-Type': 'text/plain' },
|
|
157
|
+
isHead ? undefined : Buffer.from('Invalid path'),
|
|
158
|
+
);
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
const pathname = decodedPathname === '/' ? '/index.html' : decodedPathname;
|
|
163
|
+
const relativePath = this.sanitizePath(pathname);
|
|
164
|
+
|
|
165
|
+
if (!relativePath) {
|
|
166
|
+
this.sendResponse(
|
|
167
|
+
res,
|
|
168
|
+
400,
|
|
169
|
+
{ 'Content-Type': 'text/plain' },
|
|
170
|
+
isHead ? undefined : Buffer.from('Invalid path'),
|
|
171
|
+
);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
const basePath = join(this.appConfig.absolutePaths.distDir, relativePath);
|
|
176
|
+
|
|
177
|
+
if (pathname.includes('.')) {
|
|
178
|
+
this.serveFile(req, res, basePath);
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
const htmlCandidates = [`${basePath}.html`, join(basePath, 'index.html')];
|
|
183
|
+
|
|
184
|
+
for (const candidate of htmlCandidates) {
|
|
185
|
+
if (fileSystem.exists(candidate)) {
|
|
186
|
+
this.serveFile(req, res, candidate);
|
|
187
|
+
return;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
this.sendNotFoundPage(req, res);
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
/**
|
|
195
|
+
* Starts the static preview server.
|
|
196
|
+
*/
|
|
197
|
+
public async start(): Promise<NodeHttpServer> {
|
|
198
|
+
if (this.server) {
|
|
199
|
+
return this.server;
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
this.server = createServer(this.handleRequest.bind(this));
|
|
203
|
+
const hostname = this.options.hostname ?? DEFAULT_ECOPAGES_HOSTNAME;
|
|
204
|
+
const port = this.options.port ?? DEFAULT_ECOPAGES_PORT;
|
|
205
|
+
|
|
206
|
+
await new Promise<void>((resolve) => {
|
|
207
|
+
this.server!.listen(port, hostname, () => resolve());
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
return this.server;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
/**
|
|
214
|
+
* Stops the static preview server and optionally closes active connections.
|
|
215
|
+
*/
|
|
216
|
+
public async stop(force = true): Promise<void> {
|
|
217
|
+
if (!this.server) {
|
|
218
|
+
return;
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
const activeServer = this.server;
|
|
222
|
+
this.server = null;
|
|
223
|
+
|
|
224
|
+
await new Promise<void>((resolve, reject) => {
|
|
225
|
+
activeServer.close((error) => {
|
|
226
|
+
if (error) {
|
|
227
|
+
reject(error);
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
resolve();
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
if (force) {
|
|
235
|
+
activeServer.closeAllConnections();
|
|
236
|
+
}
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { beforeEach, describe, expect, it } from 'vitest';
|
|
2
|
+
import { ApiResponseBuilder } from './api-response';
|
|
3
|
+
|
|
4
|
+
describe('ApiResponseBuilder', () => {
|
|
5
|
+
let builder: ApiResponseBuilder;
|
|
6
|
+
|
|
7
|
+
beforeEach(() => {
|
|
8
|
+
builder = new ApiResponseBuilder();
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
describe('json()', () => {
|
|
12
|
+
it('should create JSON response with correct content type and data', async () => {
|
|
13
|
+
const data = { foo: 'bar' };
|
|
14
|
+
const response = builder.json(data);
|
|
15
|
+
|
|
16
|
+
expect(response.headers.get('Content-Type')).toBe('application/json; charset=utf-8');
|
|
17
|
+
expect(await response.json()).toEqual(data);
|
|
18
|
+
expect(response.status).toBe(200);
|
|
19
|
+
});
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
describe('text()', () => {
|
|
23
|
+
it('should create text response with correct content type and data', async () => {
|
|
24
|
+
const text = 'Hello World';
|
|
25
|
+
const response = builder.text(text);
|
|
26
|
+
|
|
27
|
+
expect(response.headers.get('Content-Type')).toBe('text/plain; charset=utf-8');
|
|
28
|
+
expect(await response.text()).toBe(text);
|
|
29
|
+
expect(response.status).toBe(200);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
describe('html()', () => {
|
|
34
|
+
it('should create HTML response with correct content type and data', async () => {
|
|
35
|
+
const html = '<h1>Hello</h1>';
|
|
36
|
+
const response = builder.html(html);
|
|
37
|
+
|
|
38
|
+
expect(response.headers.get('Content-Type')).toBe('text/html; charset=utf-8');
|
|
39
|
+
expect(await response.text()).toBe(html);
|
|
40
|
+
expect(response.status).toBe(200);
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
describe('status()', () => {
|
|
45
|
+
it('should set custom status code', () => {
|
|
46
|
+
const response = builder.status(404).json({ message: 'Not Found' });
|
|
47
|
+
expect(response.status).toBe(404);
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
describe('headers()', () => {
|
|
52
|
+
it('should set custom headers', () => {
|
|
53
|
+
const response = builder.headers({ 'X-Custom': 'Value' }).json({});
|
|
54
|
+
expect(response.headers.get('X-Custom')).toBe('Value');
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
describe('redirect()', () => {
|
|
59
|
+
it('should create redirect response with location header', () => {
|
|
60
|
+
const url = 'https://example.com';
|
|
61
|
+
const response = builder.redirect(url);
|
|
62
|
+
|
|
63
|
+
expect(response.headers.get('Location')).toBe(url);
|
|
64
|
+
expect(response.status).toBe(302);
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
it('should use explicit status for redirect', () => {
|
|
68
|
+
const response = builder.redirect('https://example.com', 301);
|
|
69
|
+
expect(response.status).toBe(301);
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
describe('error()', () => {
|
|
74
|
+
it('should create JSON error response for object data', async () => {
|
|
75
|
+
const error = { code: 'ERR_1', message: 'Error occurred' };
|
|
76
|
+
const response = builder.error(error);
|
|
77
|
+
|
|
78
|
+
expect(response.headers.get('Content-Type')).toBe('application/json; charset=utf-8');
|
|
79
|
+
expect(await response.json()).toEqual({ error });
|
|
80
|
+
expect(response.status).toBe(500);
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it('should create text error response for string data', async () => {
|
|
84
|
+
const error = 'Error occurred';
|
|
85
|
+
const response = builder.error(error);
|
|
86
|
+
|
|
87
|
+
expect(response.headers.get('Content-Type')).toBe('text/plain; charset=utf-8');
|
|
88
|
+
expect(await response.text()).toBe(error);
|
|
89
|
+
expect(response.status).toBe(500);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('should use explicit status for error', () => {
|
|
93
|
+
const response = builder.error('Not Found', 404);
|
|
94
|
+
expect(response.status).toBe(404);
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
});
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* A builder class for creating Response objects fluently.
|
|
3
|
+
*/
|
|
4
|
+
export class ApiResponseBuilder {
|
|
5
|
+
private _status: Response['status'] = 200;
|
|
6
|
+
private _headers: Headers = new Headers();
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Sets the HTTP status code for the response.
|
|
10
|
+
* @param code - The HTTP status code.
|
|
11
|
+
* @returns The builder instance for chaining.
|
|
12
|
+
*/
|
|
13
|
+
status(code: Response['status']): this {
|
|
14
|
+
this._status = code;
|
|
15
|
+
return this;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Adds or merges headers for the response.
|
|
20
|
+
* @param headersInit - Headers to add or merge.
|
|
21
|
+
* @returns The builder instance for chaining.
|
|
22
|
+
*/
|
|
23
|
+
headers(headersInit: HeadersInit): this {
|
|
24
|
+
const inputHeaders = new Headers(headersInit);
|
|
25
|
+
inputHeaders.forEach((value, key) => {
|
|
26
|
+
this._headers.set(key, value);
|
|
27
|
+
});
|
|
28
|
+
return this;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Creates a JSON response using the configured status and headers.
|
|
33
|
+
* @param data - The data to serialize.
|
|
34
|
+
* @returns A Response object.
|
|
35
|
+
*/
|
|
36
|
+
json(data: any): Response {
|
|
37
|
+
if (!this._headers.has('Content-Type')) {
|
|
38
|
+
this._headers.set('Content-Type', 'application/json; charset=utf-8');
|
|
39
|
+
}
|
|
40
|
+
return new Response(JSON.stringify(data), { status: this._status, headers: this._headers });
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Creates a plain text response using the configured status and headers.
|
|
45
|
+
* @param data - The text content.
|
|
46
|
+
* @returns A Response object.
|
|
47
|
+
*/
|
|
48
|
+
text(data: string): Response {
|
|
49
|
+
if (!this._headers.has('Content-Type')) {
|
|
50
|
+
this._headers.set('Content-Type', 'text/plain; charset=utf-8');
|
|
51
|
+
}
|
|
52
|
+
return new Response(data, { status: this._status, headers: this._headers });
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Creates an HTML response using the configured status and headers.
|
|
57
|
+
* @param data - The HTML content.
|
|
58
|
+
* @returns A Response object.
|
|
59
|
+
*/
|
|
60
|
+
html(data: string): Response {
|
|
61
|
+
if (!this._headers.has('Content-Type')) {
|
|
62
|
+
this._headers.set('Content-Type', 'text/html; charset=utf-8');
|
|
63
|
+
}
|
|
64
|
+
return new Response(data, { status: this._status, headers: this._headers });
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Creates a redirect response.
|
|
69
|
+
* @param url - The URL to redirect to.
|
|
70
|
+
* @param explicitStatus - Optional explicit status code (overrides status() if provided). Defaults to 302 if not set via status().
|
|
71
|
+
* @returns A Response object.
|
|
72
|
+
*/
|
|
73
|
+
redirect(url: string, explicitStatus?: Response['status']): Response {
|
|
74
|
+
const redirectStatus = explicitStatus ?? (this._status === 200 ? 302 : this._status); // Default to 302 if status wasn't explicitly set
|
|
75
|
+
this._headers.set('Location', url);
|
|
76
|
+
return new Response(null, { status: redirectStatus, headers: this._headers });
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Creates an error response using the configured status and headers.
|
|
81
|
+
* If data is an object, it's treated as JSON, otherwise as text.
|
|
82
|
+
* @param data - The error data (string or object).
|
|
83
|
+
* @param explicitStatus - Optional explicit status code (overrides status() if provided). Defaults to 500 if not set via status().
|
|
84
|
+
* @returns A Response object.
|
|
85
|
+
*/
|
|
86
|
+
error(data: string | object, explicitStatus?: Response['status']): Response {
|
|
87
|
+
const errorStatus = explicitStatus ?? (this._status === 200 ? 500 : this._status); // Default to 500 if status wasn't explicitly set
|
|
88
|
+
this.status(errorStatus);
|
|
89
|
+
|
|
90
|
+
if (typeof data === 'object' && data !== null) {
|
|
91
|
+
if (!this._headers.has('Content-Type')) {
|
|
92
|
+
this._headers.set('Content-Type', 'application/json; charset=utf-8');
|
|
93
|
+
}
|
|
94
|
+
return new Response(JSON.stringify({ error: data }), {
|
|
95
|
+
status: this._status,
|
|
96
|
+
headers: this._headers,
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
if (!this._headers.has('Content-Type')) {
|
|
100
|
+
this._headers.set('Content-Type', 'text/plain; charset=utf-8');
|
|
101
|
+
}
|
|
102
|
+
return new Response(String(data), { status: this._status, headers: this._headers });
|
|
103
|
+
}
|
|
104
|
+
}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
ApiHandler,
|
|
3
|
+
ApiHandlerContext,
|
|
4
|
+
Middleware,
|
|
5
|
+
RouteGroupBuilder,
|
|
6
|
+
RouteOptions,
|
|
7
|
+
RouteSchema,
|
|
8
|
+
} from '../../types/public-types.ts';
|
|
9
|
+
import {
|
|
10
|
+
AbstractApplicationAdapter,
|
|
11
|
+
type ApplicationAdapterOptions,
|
|
12
|
+
type RouteGroupDefinition,
|
|
13
|
+
type RouteHandler,
|
|
14
|
+
} from '../abstract/application-adapter.ts';
|
|
15
|
+
|
|
16
|
+
export abstract class SharedApplicationAdapter<
|
|
17
|
+
TOptions extends ApplicationAdapterOptions = ApplicationAdapterOptions,
|
|
18
|
+
TServer = any,
|
|
19
|
+
TRequest extends Request = Request,
|
|
20
|
+
> extends AbstractApplicationAdapter<TOptions, TServer, TRequest> {
|
|
21
|
+
protected register<
|
|
22
|
+
P extends string,
|
|
23
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
24
|
+
>(
|
|
25
|
+
path: P,
|
|
26
|
+
method: ApiHandler['method'],
|
|
27
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
28
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
29
|
+
): this {
|
|
30
|
+
return this.addRouteHandler(path, method, handler, options?.middleware, options?.schema);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
get<P extends string, TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>>(
|
|
34
|
+
path: P,
|
|
35
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
36
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
37
|
+
): this {
|
|
38
|
+
return this.register(path, 'GET', handler, options);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
post<
|
|
42
|
+
P extends string,
|
|
43
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
44
|
+
>(
|
|
45
|
+
path: P,
|
|
46
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
47
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
48
|
+
): this {
|
|
49
|
+
return this.register(path, 'POST', handler, options);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
put<P extends string, TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>>(
|
|
53
|
+
path: P,
|
|
54
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
55
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
56
|
+
): this {
|
|
57
|
+
return this.register(path, 'PUT', handler, options);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
delete<
|
|
61
|
+
P extends string,
|
|
62
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
63
|
+
>(
|
|
64
|
+
path: P,
|
|
65
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
66
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
67
|
+
): this {
|
|
68
|
+
return this.register(path, 'DELETE', handler, options);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
patch<
|
|
72
|
+
P extends string,
|
|
73
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
74
|
+
>(
|
|
75
|
+
path: P,
|
|
76
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
77
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
78
|
+
): this {
|
|
79
|
+
return this.register(path, 'PATCH', handler, options);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
options<
|
|
83
|
+
P extends string,
|
|
84
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
85
|
+
>(
|
|
86
|
+
path: P,
|
|
87
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
88
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
89
|
+
): this {
|
|
90
|
+
return this.register(path, 'OPTIONS', handler, options);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
head<
|
|
94
|
+
P extends string,
|
|
95
|
+
TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
|
|
96
|
+
>(
|
|
97
|
+
path: P,
|
|
98
|
+
handler: RouteHandler<TRequest, TServer, TContext>,
|
|
99
|
+
options?: RouteOptions<TRequest, TServer, TContext>,
|
|
100
|
+
): this {
|
|
101
|
+
return this.register(path, 'HEAD', handler, options);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
route<P extends string>(
|
|
105
|
+
path: P,
|
|
106
|
+
method: ApiHandler['method'],
|
|
107
|
+
handler: RouteHandler<TRequest, TServer>,
|
|
108
|
+
options?: RouteOptions<TRequest, TServer>,
|
|
109
|
+
): this {
|
|
110
|
+
return this.register(path, method, handler, options);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
add(handler: ApiHandler<string, TRequest, TServer>): this {
|
|
114
|
+
return this.addRouteHandler(handler.path, handler.method, handler.handler, handler.middleware, handler.schema);
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
group<TMiddleware extends readonly Middleware<TRequest, TServer, any>[] = []>(
|
|
118
|
+
prefixOrGroup: string | RouteGroupDefinition<TRequest, TServer>,
|
|
119
|
+
callback?: (
|
|
120
|
+
builder: TMiddleware extends readonly Middleware<TRequest, TServer, infer TContext>[]
|
|
121
|
+
? RouteGroupBuilder<TRequest, TServer, TContext>
|
|
122
|
+
: RouteGroupBuilder<TRequest, TServer>,
|
|
123
|
+
) => void,
|
|
124
|
+
options?: {
|
|
125
|
+
middleware?: TMiddleware;
|
|
126
|
+
},
|
|
127
|
+
): this {
|
|
128
|
+
if (typeof prefixOrGroup === 'object') {
|
|
129
|
+
return this.registerGroup(prefixOrGroup);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
type TContext = TMiddleware extends readonly Middleware<TRequest, TServer, infer TCtx>[]
|
|
133
|
+
? TCtx
|
|
134
|
+
: ApiHandlerContext<TRequest, TServer>;
|
|
135
|
+
|
|
136
|
+
const normalizedPrefix = prefixOrGroup.endsWith('/') ? prefixOrGroup.slice(0, -1) : prefixOrGroup;
|
|
137
|
+
const groupMiddleware = (options?.middleware ?? []) as Middleware<TRequest, TServer, TContext>[];
|
|
138
|
+
|
|
139
|
+
const createHandler = (
|
|
140
|
+
method: ApiHandler['method'],
|
|
141
|
+
): RouteGroupBuilder<TRequest, TServer, TContext>[Lowercase<typeof method>] => {
|
|
142
|
+
return ((
|
|
143
|
+
path: string,
|
|
144
|
+
handler: (context: TContext) => Promise<Response> | Response,
|
|
145
|
+
routeOptions?: RouteOptions<TRequest, TServer, TContext> & { schema?: RouteSchema },
|
|
146
|
+
) => {
|
|
147
|
+
const combinedMiddleware: Middleware<TRequest, TServer, TContext>[] = [
|
|
148
|
+
...groupMiddleware,
|
|
149
|
+
...(routeOptions?.middleware ?? []),
|
|
150
|
+
];
|
|
151
|
+
const fullPath = path === '/' ? normalizedPrefix : `${normalizedPrefix}${path}`;
|
|
152
|
+
this.addRouteHandler(
|
|
153
|
+
fullPath,
|
|
154
|
+
method,
|
|
155
|
+
handler as RouteHandler<TRequest, TServer, ApiHandlerContext<TRequest, TServer>>,
|
|
156
|
+
combinedMiddleware.length > 0
|
|
157
|
+
? (combinedMiddleware as Middleware<TRequest, TServer, ApiHandlerContext<TRequest, TServer>>[])
|
|
158
|
+
: undefined,
|
|
159
|
+
routeOptions?.schema,
|
|
160
|
+
);
|
|
161
|
+
return builder;
|
|
162
|
+
}) as RouteGroupBuilder<TRequest, TServer, TContext>[Lowercase<typeof method>];
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
const builder: RouteGroupBuilder<TRequest, TServer, TContext> = {
|
|
166
|
+
get: createHandler('GET'),
|
|
167
|
+
post: createHandler('POST'),
|
|
168
|
+
put: createHandler('PUT'),
|
|
169
|
+
delete: createHandler('DELETE'),
|
|
170
|
+
patch: createHandler('PATCH'),
|
|
171
|
+
options: createHandler('OPTIONS'),
|
|
172
|
+
head: createHandler('HEAD'),
|
|
173
|
+
};
|
|
174
|
+
|
|
175
|
+
callback?.(builder as never);
|
|
176
|
+
return this;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
private registerGroup(group: RouteGroupDefinition<TRequest, TServer>): this {
|
|
180
|
+
const normalizedPrefix = group.prefix.endsWith('/') ? group.prefix.slice(0, -1) : group.prefix;
|
|
181
|
+
const groupMiddleware = group.middleware ?? [];
|
|
182
|
+
|
|
183
|
+
for (const route of group.routes) {
|
|
184
|
+
const normalizedPath = route.path.startsWith('/') ? route.path : `/${route.path}`;
|
|
185
|
+
const fullPath = route.path === '/' ? normalizedPrefix : `${normalizedPrefix}${normalizedPath}`;
|
|
186
|
+
const combinedMiddleware = [...groupMiddleware, ...(route.middleware ?? [])];
|
|
187
|
+
|
|
188
|
+
this.addRouteHandler(
|
|
189
|
+
fullPath,
|
|
190
|
+
route.method,
|
|
191
|
+
route.handler,
|
|
192
|
+
combinedMiddleware.length > 0 ? combinedMiddleware : undefined,
|
|
193
|
+
route.schema,
|
|
194
|
+
);
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
return this;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
ApiHandler,
|
|
3
|
+
ApiHandlerContext,
|
|
4
|
+
Middleware,
|
|
5
|
+
RouteSchema,
|
|
6
|
+
TypedGroupHandlerContext,
|
|
7
|
+
} from '../../types/public-types.ts';
|
|
8
|
+
|
|
9
|
+
type UniversalContext = ApiHandlerContext<Request, unknown>;
|
|
10
|
+
|
|
11
|
+
type SchemaHandlerContext<
|
|
12
|
+
TSchema extends RouteSchema | undefined,
|
|
13
|
+
TContext extends UniversalContext,
|
|
14
|
+
> = TSchema extends RouteSchema ? TypedGroupHandlerContext<TSchema, TContext> : TContext;
|
|
15
|
+
|
|
16
|
+
export function defineApiHandler<
|
|
17
|
+
TPath extends string,
|
|
18
|
+
TSchema extends RouteSchema | undefined = undefined,
|
|
19
|
+
TContext extends UniversalContext = UniversalContext,
|
|
20
|
+
>(
|
|
21
|
+
handler: Omit<ApiHandler<TPath, Request, unknown>, 'handler' | 'middleware' | 'schema'> & {
|
|
22
|
+
handler: (context: SchemaHandlerContext<TSchema, TContext>) => Promise<Response> | Response;
|
|
23
|
+
middleware?: Middleware<Request, unknown, TContext>[];
|
|
24
|
+
schema?: TSchema;
|
|
25
|
+
},
|
|
26
|
+
): ApiHandler<TPath, Request, unknown> {
|
|
27
|
+
return handler as ApiHandler<TPath, Request, unknown>;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export interface GroupHandler<TPrefix extends string = string> {
|
|
31
|
+
prefix: TPrefix;
|
|
32
|
+
middleware?: readonly Middleware<Request, unknown, any>[];
|
|
33
|
+
routes: readonly ApiHandler<string, Request, unknown>[];
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
type GroupDefineHandler<TContext extends UniversalContext> = <
|
|
37
|
+
const TPath extends string,
|
|
38
|
+
TSchema extends RouteSchema | undefined = undefined,
|
|
39
|
+
>(
|
|
40
|
+
handler: Omit<ApiHandler<TPath, Request, unknown>, 'handler' | 'middleware' | 'schema'> & {
|
|
41
|
+
path: TPath;
|
|
42
|
+
handler: (context: SchemaHandlerContext<TSchema, TContext>) => Promise<Response> | Response;
|
|
43
|
+
middleware?: Middleware<Request, unknown, TContext>[];
|
|
44
|
+
schema?: TSchema;
|
|
45
|
+
},
|
|
46
|
+
) => ApiHandler<TPath, Request, unknown>;
|
|
47
|
+
|
|
48
|
+
export function defineGroupHandler<
|
|
49
|
+
TPrefix extends string,
|
|
50
|
+
TMiddleware extends readonly Middleware<Request, unknown, any>[] = [],
|
|
51
|
+
TContext extends UniversalContext = TMiddleware extends readonly Middleware<Request, unknown, infer TGroupContext>[]
|
|
52
|
+
? TGroupContext
|
|
53
|
+
: UniversalContext,
|
|
54
|
+
>(config: {
|
|
55
|
+
prefix: TPrefix;
|
|
56
|
+
middleware?: TMiddleware;
|
|
57
|
+
routes: (define: GroupDefineHandler<TContext>) => readonly ApiHandler<string, Request, unknown>[];
|
|
58
|
+
}): GroupHandler<TPrefix> {
|
|
59
|
+
const define = ((handler) => handler) as GroupDefineHandler<TContext>;
|
|
60
|
+
|
|
61
|
+
return {
|
|
62
|
+
prefix: config.prefix,
|
|
63
|
+
middleware: config.middleware,
|
|
64
|
+
routes: config.routes(define),
|
|
65
|
+
};
|
|
66
|
+
}
|