@ecopages/core 0.2.0-alpha.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +89 -0
- package/LICENSE +21 -0
- package/README.md +32 -0
- package/package.json +279 -0
- package/src/adapters/abstract/application-adapter.d.ts +168 -0
- package/src/adapters/abstract/application-adapter.js +109 -0
- package/src/adapters/abstract/application-adapter.ts +337 -0
- package/src/adapters/abstract/router-adapter.d.ts +26 -0
- package/src/adapters/abstract/router-adapter.js +5 -0
- package/src/adapters/abstract/router-adapter.ts +30 -0
- package/src/adapters/abstract/server-adapter.d.ts +69 -0
- package/src/adapters/abstract/server-adapter.js +15 -0
- package/src/adapters/abstract/server-adapter.ts +79 -0
- package/src/adapters/bun/client-bridge.d.ts +34 -0
- package/src/adapters/bun/client-bridge.js +48 -0
- package/src/adapters/bun/client-bridge.ts +62 -0
- package/src/adapters/bun/create-app.d.ts +60 -0
- package/src/adapters/bun/create-app.js +117 -0
- package/src/adapters/bun/create-app.ts +189 -0
- package/src/adapters/bun/define-api-handler.d.ts +61 -0
- package/src/adapters/bun/define-api-handler.js +15 -0
- package/src/adapters/bun/define-api-handler.ts +114 -0
- package/src/adapters/bun/hmr-manager.d.ts +84 -0
- package/src/adapters/bun/hmr-manager.js +227 -0
- package/src/adapters/bun/hmr-manager.ts +281 -0
- package/src/adapters/bun/index.d.ts +3 -0
- package/src/adapters/bun/index.js +8 -0
- package/src/adapters/bun/index.ts +3 -0
- package/src/adapters/bun/server-adapter.d.ts +155 -0
- package/src/adapters/bun/server-adapter.js +368 -0
- package/src/adapters/bun/server-adapter.ts +492 -0
- package/src/adapters/bun/server-lifecycle.d.ts +52 -0
- package/src/adapters/bun/server-lifecycle.js +120 -0
- package/src/adapters/bun/server-lifecycle.ts +154 -0
- package/src/adapters/index.d.ts +6 -0
- package/src/adapters/index.js +14 -0
- package/src/adapters/index.ts +6 -0
- package/src/adapters/node/create-app.d.ts +21 -0
- package/src/adapters/node/create-app.js +143 -0
- package/src/adapters/node/create-app.ts +179 -0
- package/src/adapters/node/index.d.ts +4 -0
- package/src/adapters/node/index.js +8 -0
- package/src/adapters/node/index.ts +9 -0
- package/src/adapters/node/node-client-bridge.d.ts +26 -0
- package/src/adapters/node/node-client-bridge.js +66 -0
- package/src/adapters/node/node-client-bridge.ts +79 -0
- package/src/adapters/node/node-hmr-manager.d.ts +62 -0
- package/src/adapters/node/node-hmr-manager.js +221 -0
- package/src/adapters/node/node-hmr-manager.ts +271 -0
- package/src/adapters/node/server-adapter.d.ts +190 -0
- package/src/adapters/node/server-adapter.js +420 -0
- package/src/adapters/node/server-adapter.ts +561 -0
- package/src/adapters/node/static-content-server.d.ts +24 -0
- package/src/adapters/node/static-content-server.js +166 -0
- package/src/adapters/node/static-content-server.ts +203 -0
- package/src/adapters/shared/api-response.d.ts +52 -0
- package/src/adapters/shared/api-response.js +96 -0
- package/src/adapters/shared/api-response.ts +104 -0
- package/src/adapters/shared/application-adapter.d.ts +18 -0
- package/src/adapters/shared/application-adapter.js +90 -0
- package/src/adapters/shared/application-adapter.ts +199 -0
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
- package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
- package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
- package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
- package/src/adapters/shared/fs-server-response-factory.js +97 -0
- package/src/adapters/shared/fs-server-response-factory.ts +118 -0
- package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
- package/src/adapters/shared/fs-server-response-matcher.js +155 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
- package/src/adapters/shared/render-context.d.ts +14 -0
- package/src/adapters/shared/render-context.js +69 -0
- package/src/adapters/shared/render-context.ts +105 -0
- package/src/adapters/shared/server-adapter.d.ts +87 -0
- package/src/adapters/shared/server-adapter.js +353 -0
- package/src/adapters/shared/server-adapter.ts +442 -0
- package/src/adapters/shared/server-route-handler.d.ts +89 -0
- package/src/adapters/shared/server-route-handler.js +120 -0
- package/src/adapters/shared/server-route-handler.ts +166 -0
- package/src/adapters/shared/server-static-builder.d.ts +38 -0
- package/src/adapters/shared/server-static-builder.js +46 -0
- package/src/adapters/shared/server-static-builder.ts +82 -0
- package/src/build/build-adapter.d.ts +74 -0
- package/src/build/build-adapter.js +54 -0
- package/src/build/build-adapter.ts +132 -0
- package/src/build/build-types.d.ts +57 -0
- package/src/build/build-types.js +0 -0
- package/src/build/build-types.ts +83 -0
- package/src/build/esbuild-build-adapter.d.ts +69 -0
- package/src/build/esbuild-build-adapter.js +390 -0
- package/src/build/esbuild-build-adapter.ts +510 -0
- package/src/config/config-builder.d.ts +227 -0
- package/src/config/config-builder.js +392 -0
- package/src/config/config-builder.ts +474 -0
- package/src/constants.d.ts +32 -0
- package/src/constants.js +21 -0
- package/src/constants.ts +39 -0
- package/src/create-app.d.ts +17 -0
- package/src/create-app.js +66 -0
- package/src/create-app.ts +87 -0
- package/src/declarations.d.ts +26 -0
- package/src/define-api-handler.d.ts +25 -0
- package/src/define-api-handler.js +15 -0
- package/src/define-api-handler.ts +66 -0
- package/src/dev/sc-server.d.ts +30 -0
- package/src/dev/sc-server.js +111 -0
- package/src/dev/sc-server.ts +143 -0
- package/src/eco/README.md +636 -0
- package/src/eco/component-render-context.d.ts +105 -0
- package/src/eco/component-render-context.js +77 -0
- package/src/eco/component-render-context.ts +202 -0
- package/src/eco/eco.d.ts +9 -0
- package/src/eco/eco.js +110 -0
- package/src/eco/eco.ts +221 -0
- package/src/eco/eco.types.d.ts +170 -0
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.types.ts +202 -0
- package/src/eco/eco.utils.d.ts +40 -0
- package/src/eco/eco.utils.js +40 -0
- package/src/eco/eco.utils.ts +89 -0
- package/src/eco/global-injector-map.d.ts +16 -0
- package/src/eco/global-injector-map.js +80 -0
- package/src/eco/global-injector-map.ts +112 -0
- package/src/eco/lazy-injector-map.d.ts +8 -0
- package/src/eco/lazy-injector-map.js +70 -0
- package/src/eco/lazy-injector-map.ts +120 -0
- package/src/eco/module-dependencies.d.ts +18 -0
- package/src/eco/module-dependencies.js +49 -0
- package/src/eco/module-dependencies.ts +75 -0
- package/src/env.d.ts +20 -0
- package/src/errors/http-error.d.ts +31 -0
- package/src/errors/http-error.js +50 -0
- package/src/errors/http-error.ts +72 -0
- package/src/errors/index.d.ts +2 -0
- package/src/errors/index.js +4 -0
- package/src/errors/index.ts +2 -0
- package/src/errors/locals-access-error.d.ts +4 -0
- package/src/errors/locals-access-error.js +9 -0
- package/src/errors/locals-access-error.ts +7 -0
- package/src/global/app-logger.d.ts +2 -0
- package/src/global/app-logger.js +6 -0
- package/src/global/app-logger.ts +4 -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.d.ts +10 -0
- package/src/hmr/client/hmr-runtime.js +86 -0
- package/src/hmr/client/hmr-runtime.ts +121 -0
- package/src/hmr/hmr-strategy.d.ts +159 -0
- package/src/hmr/hmr-strategy.js +29 -0
- package/src/hmr/hmr-strategy.ts +172 -0
- package/src/hmr/hmr.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.test.e2e.js +50 -0
- package/src/hmr/hmr.test.e2e.ts +75 -0
- package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
- package/src/hmr/strategies/default-hmr-strategy.js +34 -0
- package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
- package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
- package/src/hmr/strategies/js-hmr-strategy.js +179 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
- package/src/index.browser.d.ts +3 -0
- package/src/index.browser.js +4 -0
- package/src/index.browser.ts +3 -0
- package/src/index.d.ts +5 -0
- package/src/index.js +10 -0
- package/src/index.ts +5 -0
- package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.js +60 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
- package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
- package/src/integrations/ghtml/ghtml.plugin.js +21 -0
- package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
- package/src/internal-types.d.ts +200 -0
- package/src/internal-types.js +0 -0
- package/src/internal-types.ts +212 -0
- package/src/plugins/alias-resolver-plugin.d.ts +2 -0
- package/src/plugins/alias-resolver-plugin.js +39 -0
- package/src/plugins/alias-resolver-plugin.ts +45 -0
- package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
- package/src/plugins/eco-component-meta-plugin.js +157 -0
- package/src/plugins/eco-component-meta-plugin.ts +474 -0
- package/src/plugins/integration-plugin.d.ts +102 -0
- package/src/plugins/integration-plugin.js +100 -0
- package/src/plugins/integration-plugin.ts +184 -0
- package/src/plugins/processor.d.ts +82 -0
- package/src/plugins/processor.js +122 -0
- package/src/plugins/processor.ts +220 -0
- package/src/public-types.d.ts +1094 -0
- package/src/public-types.js +0 -0
- package/src/public-types.ts +1255 -0
- package/src/route-renderer/GRAPH.md +387 -0
- package/src/route-renderer/README.md +135 -0
- package/src/route-renderer/component-graph-executor.d.ts +32 -0
- package/src/route-renderer/component-graph-executor.js +31 -0
- package/src/route-renderer/component-graph-executor.ts +84 -0
- package/src/route-renderer/component-graph.d.ts +42 -0
- package/src/route-renderer/component-graph.js +72 -0
- package/src/route-renderer/component-graph.ts +159 -0
- package/src/route-renderer/component-marker.d.ts +52 -0
- package/src/route-renderer/component-marker.js +46 -0
- package/src/route-renderer/component-marker.ts +117 -0
- package/src/route-renderer/dependency-resolver.d.ts +24 -0
- package/src/route-renderer/dependency-resolver.js +428 -0
- package/src/route-renderer/dependency-resolver.ts +596 -0
- package/src/route-renderer/html-post-processing.service.d.ts +40 -0
- package/src/route-renderer/html-post-processing.service.js +86 -0
- package/src/route-renderer/html-post-processing.service.ts +103 -0
- package/src/route-renderer/integration-renderer.d.ts +339 -0
- package/src/route-renderer/integration-renderer.js +526 -0
- package/src/route-renderer/integration-renderer.ts +696 -0
- package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
- package/src/route-renderer/marker-graph-resolver.js +93 -0
- package/src/route-renderer/marker-graph-resolver.ts +153 -0
- package/src/route-renderer/page-module-loader.d.ts +61 -0
- package/src/route-renderer/page-module-loader.js +102 -0
- package/src/route-renderer/page-module-loader.ts +153 -0
- package/src/route-renderer/render-execution.service.d.ts +69 -0
- package/src/route-renderer/render-execution.service.js +91 -0
- package/src/route-renderer/render-execution.service.ts +158 -0
- package/src/route-renderer/render-preparation.service.d.ts +112 -0
- package/src/route-renderer/render-preparation.service.js +243 -0
- package/src/route-renderer/render-preparation.service.ts +358 -0
- package/src/route-renderer/route-renderer.d.ts +26 -0
- package/src/route-renderer/route-renderer.js +68 -0
- package/src/route-renderer/route-renderer.ts +80 -0
- package/src/router/fs-router-scanner.d.ts +41 -0
- package/src/router/fs-router-scanner.js +155 -0
- package/src/router/fs-router-scanner.ts +217 -0
- package/src/router/fs-router.d.ts +26 -0
- package/src/router/fs-router.js +100 -0
- package/src/router/fs-router.ts +122 -0
- package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
- package/src/services/asset-processing-service/asset-processing.service.js +250 -0
- package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
- package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
- package/src/services/asset-processing-service/asset.factory.js +82 -0
- package/src/services/asset-processing-service/asset.factory.ts +105 -0
- package/src/services/asset-processing-service/assets.types.d.ts +88 -0
- package/src/services/asset-processing-service/assets.types.js +0 -0
- package/src/services/asset-processing-service/assets.types.ts +112 -0
- package/src/services/asset-processing-service/index.d.ts +3 -0
- package/src/services/asset-processing-service/index.js +3 -0
- package/src/services/asset-processing-service/index.ts +3 -0
- package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
- package/src/services/asset-processing-service/processor.interface.js +6 -0
- package/src/services/asset-processing-service/processor.interface.ts +27 -0
- package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
- package/src/services/asset-processing-service/processor.registry.js +15 -0
- package/src/services/asset-processing-service/processor.registry.ts +18 -0
- package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
- package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
- package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
- package/src/services/asset-processing-service/processors/index.d.ts +5 -0
- package/src/services/asset-processing-service/processors/index.js +5 -0
- package/src/services/asset-processing-service/processors/index.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
- package/src/services/cache/cache.types.d.ts +107 -0
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/cache.types.ts +126 -0
- package/src/services/cache/index.d.ts +7 -0
- package/src/services/cache/index.js +7 -0
- package/src/services/cache/index.ts +18 -0
- package/src/services/cache/memory-cache-store.d.ts +42 -0
- package/src/services/cache/memory-cache-store.js +98 -0
- package/src/services/cache/memory-cache-store.ts +130 -0
- package/src/services/cache/page-cache-service.d.ts +70 -0
- package/src/services/cache/page-cache-service.js +152 -0
- package/src/services/cache/page-cache-service.ts +202 -0
- package/src/services/html-transformer.service.d.ts +50 -0
- package/src/services/html-transformer.service.js +163 -0
- package/src/services/html-transformer.service.ts +217 -0
- package/src/services/page-module-import.service.d.ts +37 -0
- package/src/services/page-module-import.service.js +88 -0
- package/src/services/page-module-import.service.ts +129 -0
- package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
- package/src/services/page-request-cache-coordinator.service.js +107 -0
- package/src/services/page-request-cache-coordinator.service.ts +128 -0
- package/src/services/schema-validation-service.d.ts +122 -0
- package/src/services/schema-validation-service.js +101 -0
- package/src/services/schema-validation-service.ts +204 -0
- package/src/services/validation/standard-schema.types.d.ts +65 -0
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/services/validation/standard-schema.types.ts +68 -0
- package/src/static-site-generator/static-site-generator.d.ts +57 -0
- package/src/static-site-generator/static-site-generator.js +272 -0
- package/src/static-site-generator/static-site-generator.ts +359 -0
- package/src/utils/css.d.ts +1 -0
- package/src/utils/css.js +7 -0
- package/src/utils/css.ts +5 -0
- package/src/utils/deep-merge.d.ts +14 -0
- package/src/utils/deep-merge.js +32 -0
- package/src/utils/deep-merge.ts +47 -0
- package/src/utils/hash.d.ts +1 -0
- package/src/utils/hash.js +7 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/html.d.ts +1 -0
- package/src/utils/html.js +4 -0
- package/src/utils/html.ts +1 -0
- package/src/utils/invariant.d.ts +5 -0
- package/src/utils/invariant.js +11 -0
- package/src/utils/invariant.ts +15 -0
- package/src/utils/locals-utils.d.ts +15 -0
- package/src/utils/locals-utils.js +24 -0
- package/src/utils/locals-utils.ts +37 -0
- package/src/utils/parse-cli-args.d.ts +24 -0
- package/src/utils/parse-cli-args.js +47 -0
- package/src/utils/parse-cli-args.ts +83 -0
- package/src/utils/path-utils.module.d.ts +5 -0
- package/src/utils/path-utils.module.js +14 -0
- package/src/utils/path-utils.module.ts +14 -0
- package/src/utils/runtime.d.ts +11 -0
- package/src/utils/runtime.js +40 -0
- package/src/utils/runtime.ts +44 -0
- package/src/utils/server-utils.module.d.ts +19 -0
- package/src/utils/server-utils.module.js +56 -0
- package/src/utils/server-utils.module.ts +67 -0
- package/src/watchers/project-watcher.d.ts +120 -0
- package/src/watchers/project-watcher.js +238 -0
- package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
- package/src/watchers/project-watcher.test-helpers.js +51 -0
- package/src/watchers/project-watcher.test-helpers.ts +40 -0
- package/src/watchers/project-watcher.ts +306 -0
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { type ComponentRenderBoundaryContext } from '../eco/component-render-context.js';
|
|
2
|
+
import type { EcoComponent, IntegrationRendererRenderOptions, RouteRendererBody, RouteRendererOptions, RouteRenderResult } from '../public-types.js';
|
|
3
|
+
import type { ProcessedAsset } from '../services/asset-processing-service/index.js';
|
|
4
|
+
import type { MarkerGraphContext } from './marker-graph-resolver.js';
|
|
5
|
+
/**
|
|
6
|
+
* Serializable graph context merged from render-time captured references and
|
|
7
|
+
* optional explicit page-module graph metadata.
|
|
8
|
+
*/
|
|
9
|
+
export type RenderExecutionGraphContext = {
|
|
10
|
+
propsByRef?: Record<string, Record<string, unknown>>;
|
|
11
|
+
slotChildrenByRef?: MarkerGraphContext['slotChildrenByRef'];
|
|
12
|
+
};
|
|
13
|
+
export interface RenderExecutionCallbacks<C> {
|
|
14
|
+
prepareRenderOptions(options: RouteRendererOptions): Promise<IntegrationRendererRenderOptions<C>>;
|
|
15
|
+
render(renderOptions: IntegrationRendererRenderOptions<C>): Promise<RouteRendererBody>;
|
|
16
|
+
getComponentRenderBoundaryContext(): ComponentRenderBoundaryContext;
|
|
17
|
+
resolveMarkerGraphHtml(input: {
|
|
18
|
+
html: string;
|
|
19
|
+
componentsToResolve: EcoComponent[];
|
|
20
|
+
graphContext: RenderExecutionGraphContext;
|
|
21
|
+
}): Promise<{
|
|
22
|
+
html: string;
|
|
23
|
+
assets: ProcessedAsset[];
|
|
24
|
+
}>;
|
|
25
|
+
dedupeProcessedAssets(assets: ProcessedAsset[]): ProcessedAsset[];
|
|
26
|
+
getProcessedDependencies(): ProcessedAsset[];
|
|
27
|
+
setProcessedDependencies(dependencies: ProcessedAsset[]): void;
|
|
28
|
+
applyAttributesToFirstBodyElement(html: string, attributes: Record<string, string>): string;
|
|
29
|
+
transformHtml(html: string): Promise<RouteRendererBody>;
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Executes the main post-preparation rendering flow for integration renderers.
|
|
33
|
+
*
|
|
34
|
+
* This service owns the orchestration that happens after normalized render
|
|
35
|
+
* options have been prepared: the first render pass, graph-context capture,
|
|
36
|
+
* deferred marker resolution, root-attribute application, and final HTML
|
|
37
|
+
* transformation into a response body stream.
|
|
38
|
+
*/
|
|
39
|
+
export declare class RenderExecutionService {
|
|
40
|
+
/**
|
|
41
|
+
* Executes one integration render pass and returns the final route render
|
|
42
|
+
* result.
|
|
43
|
+
*
|
|
44
|
+
* @typeParam C Integration render output element type.
|
|
45
|
+
* @param options Route-level render options.
|
|
46
|
+
* @param currentIntegrationName Active integration name for this render pass.
|
|
47
|
+
* @param callbacks Renderer-specific hooks required during execution.
|
|
48
|
+
* @returns Final route render output with body and cache strategy.
|
|
49
|
+
*/
|
|
50
|
+
execute<C = unknown>(options: RouteRendererOptions, currentIntegrationName: string, callbacks: RenderExecutionCallbacks<C>): Promise<RouteRenderResult>;
|
|
51
|
+
/**
|
|
52
|
+
* Merges captured render-time graph references with any explicit graph context
|
|
53
|
+
* provided by the page module.
|
|
54
|
+
*
|
|
55
|
+
* @param capturedGraphContext Graph context captured from the first render pass.
|
|
56
|
+
* @param explicitGraphContext Optional page-module graph metadata.
|
|
57
|
+
* @returns Merged graph context used during marker resolution.
|
|
58
|
+
*/
|
|
59
|
+
private mergeGraphContext;
|
|
60
|
+
/**
|
|
61
|
+
* Returns the component set that participates in marker graph resolution for a
|
|
62
|
+
* render pass.
|
|
63
|
+
*
|
|
64
|
+
* @typeParam C Integration render output element type.
|
|
65
|
+
* @param renderOptions Normalized render options for the pass.
|
|
66
|
+
* @returns Ordered component list for graph registry construction.
|
|
67
|
+
*/
|
|
68
|
+
private getComponentsToResolve;
|
|
69
|
+
}
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import {
|
|
2
|
+
runWithComponentRenderContext
|
|
3
|
+
} from "../eco/component-render-context.js";
|
|
4
|
+
class RenderExecutionService {
|
|
5
|
+
/**
|
|
6
|
+
* Executes one integration render pass and returns the final route render
|
|
7
|
+
* result.
|
|
8
|
+
*
|
|
9
|
+
* @typeParam C Integration render output element type.
|
|
10
|
+
* @param options Route-level render options.
|
|
11
|
+
* @param currentIntegrationName Active integration name for this render pass.
|
|
12
|
+
* @param callbacks Renderer-specific hooks required during execution.
|
|
13
|
+
* @returns Final route render output with body and cache strategy.
|
|
14
|
+
*/
|
|
15
|
+
async execute(options, currentIntegrationName, callbacks) {
|
|
16
|
+
const renderOptions = await callbacks.prepareRenderOptions(options);
|
|
17
|
+
const shouldApplyComponentRootAttributes = renderOptions.componentRender?.canAttachAttributes && renderOptions.componentRender.rootAttributes && Object.keys(renderOptions.componentRender.rootAttributes).length > 0;
|
|
18
|
+
const renderExecution = await runWithComponentRenderContext(
|
|
19
|
+
{
|
|
20
|
+
currentIntegration: currentIntegrationName,
|
|
21
|
+
boundaryContext: callbacks.getComponentRenderBoundaryContext()
|
|
22
|
+
},
|
|
23
|
+
async () => callbacks.render(renderOptions)
|
|
24
|
+
);
|
|
25
|
+
let renderedHtml = await new Response(renderExecution.value).text();
|
|
26
|
+
const componentGraphContext = this.mergeGraphContext(
|
|
27
|
+
renderExecution.graphContext,
|
|
28
|
+
renderOptions.componentGraphContext
|
|
29
|
+
);
|
|
30
|
+
if (renderedHtml.includes("<eco-marker")) {
|
|
31
|
+
const markerResolution = await callbacks.resolveMarkerGraphHtml({
|
|
32
|
+
html: renderedHtml,
|
|
33
|
+
componentsToResolve: this.getComponentsToResolve(renderOptions),
|
|
34
|
+
graphContext: componentGraphContext
|
|
35
|
+
});
|
|
36
|
+
renderedHtml = markerResolution.html;
|
|
37
|
+
if (markerResolution.assets.length > 0) {
|
|
38
|
+
const mergedDependencies = callbacks.dedupeProcessedAssets([
|
|
39
|
+
...callbacks.getProcessedDependencies(),
|
|
40
|
+
...markerResolution.assets
|
|
41
|
+
]);
|
|
42
|
+
callbacks.setProcessedDependencies(mergedDependencies);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
if (shouldApplyComponentRootAttributes) {
|
|
46
|
+
renderedHtml = callbacks.applyAttributesToFirstBodyElement(
|
|
47
|
+
renderedHtml,
|
|
48
|
+
renderOptions.componentRender?.rootAttributes
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
const body = await callbacks.transformHtml(renderedHtml);
|
|
52
|
+
return {
|
|
53
|
+
body,
|
|
54
|
+
cacheStrategy: renderOptions.cacheStrategy
|
|
55
|
+
};
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Merges captured render-time graph references with any explicit graph context
|
|
59
|
+
* provided by the page module.
|
|
60
|
+
*
|
|
61
|
+
* @param capturedGraphContext Graph context captured from the first render pass.
|
|
62
|
+
* @param explicitGraphContext Optional page-module graph metadata.
|
|
63
|
+
* @returns Merged graph context used during marker resolution.
|
|
64
|
+
*/
|
|
65
|
+
mergeGraphContext(capturedGraphContext, explicitGraphContext) {
|
|
66
|
+
return {
|
|
67
|
+
propsByRef: {
|
|
68
|
+
...capturedGraphContext.propsByRef ?? {},
|
|
69
|
+
...explicitGraphContext?.propsByRef ?? {}
|
|
70
|
+
},
|
|
71
|
+
slotChildrenByRef: {
|
|
72
|
+
...capturedGraphContext.slotChildrenByRef ?? {},
|
|
73
|
+
...explicitGraphContext?.slotChildrenByRef ?? {}
|
|
74
|
+
}
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Returns the component set that participates in marker graph resolution for a
|
|
79
|
+
* render pass.
|
|
80
|
+
*
|
|
81
|
+
* @typeParam C Integration render output element type.
|
|
82
|
+
* @param renderOptions Normalized render options for the pass.
|
|
83
|
+
* @returns Ordered component list for graph registry construction.
|
|
84
|
+
*/
|
|
85
|
+
getComponentsToResolve(renderOptions) {
|
|
86
|
+
return renderOptions.Layout ? [renderOptions.HtmlTemplate, renderOptions.Layout, renderOptions.Page] : [renderOptions.HtmlTemplate, renderOptions.Page];
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
export {
|
|
90
|
+
RenderExecutionService
|
|
91
|
+
};
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
import {
|
|
2
|
+
runWithComponentRenderContext,
|
|
3
|
+
type ComponentRenderBoundaryContext,
|
|
4
|
+
type ComponentGraphContext as CapturedComponentGraphContext,
|
|
5
|
+
} from '../eco/component-render-context.ts';
|
|
6
|
+
import type {
|
|
7
|
+
EcoComponent,
|
|
8
|
+
IntegrationRendererRenderOptions,
|
|
9
|
+
RouteRendererBody,
|
|
10
|
+
RouteRendererOptions,
|
|
11
|
+
RouteRenderResult,
|
|
12
|
+
} from '../public-types.ts';
|
|
13
|
+
import type { ProcessedAsset } from '../services/asset-processing-service/index.ts';
|
|
14
|
+
import type { MarkerGraphContext } from './marker-graph-resolver.ts';
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Serializable graph context merged from render-time captured references and
|
|
18
|
+
* optional explicit page-module graph metadata.
|
|
19
|
+
*/
|
|
20
|
+
export type RenderExecutionGraphContext = {
|
|
21
|
+
propsByRef?: Record<string, Record<string, unknown>>;
|
|
22
|
+
slotChildrenByRef?: MarkerGraphContext['slotChildrenByRef'];
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
export interface RenderExecutionCallbacks<C> {
|
|
26
|
+
prepareRenderOptions(options: RouteRendererOptions): Promise<IntegrationRendererRenderOptions<C>>;
|
|
27
|
+
render(renderOptions: IntegrationRendererRenderOptions<C>): Promise<RouteRendererBody>;
|
|
28
|
+
getComponentRenderBoundaryContext(): ComponentRenderBoundaryContext;
|
|
29
|
+
resolveMarkerGraphHtml(input: {
|
|
30
|
+
html: string;
|
|
31
|
+
componentsToResolve: EcoComponent[];
|
|
32
|
+
graphContext: RenderExecutionGraphContext;
|
|
33
|
+
}): Promise<{ html: string; assets: ProcessedAsset[] }>;
|
|
34
|
+
dedupeProcessedAssets(assets: ProcessedAsset[]): ProcessedAsset[];
|
|
35
|
+
getProcessedDependencies(): ProcessedAsset[];
|
|
36
|
+
setProcessedDependencies(dependencies: ProcessedAsset[]): void;
|
|
37
|
+
applyAttributesToFirstBodyElement(html: string, attributes: Record<string, string>): string;
|
|
38
|
+
transformHtml(html: string): Promise<RouteRendererBody>;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Executes the main post-preparation rendering flow for integration renderers.
|
|
43
|
+
*
|
|
44
|
+
* This service owns the orchestration that happens after normalized render
|
|
45
|
+
* options have been prepared: the first render pass, graph-context capture,
|
|
46
|
+
* deferred marker resolution, root-attribute application, and final HTML
|
|
47
|
+
* transformation into a response body stream.
|
|
48
|
+
*/
|
|
49
|
+
export class RenderExecutionService {
|
|
50
|
+
/**
|
|
51
|
+
* Executes one integration render pass and returns the final route render
|
|
52
|
+
* result.
|
|
53
|
+
*
|
|
54
|
+
* @typeParam C Integration render output element type.
|
|
55
|
+
* @param options Route-level render options.
|
|
56
|
+
* @param currentIntegrationName Active integration name for this render pass.
|
|
57
|
+
* @param callbacks Renderer-specific hooks required during execution.
|
|
58
|
+
* @returns Final route render output with body and cache strategy.
|
|
59
|
+
*/
|
|
60
|
+
async execute<C = unknown>(
|
|
61
|
+
options: RouteRendererOptions,
|
|
62
|
+
currentIntegrationName: string,
|
|
63
|
+
callbacks: RenderExecutionCallbacks<C>,
|
|
64
|
+
): Promise<RouteRenderResult> {
|
|
65
|
+
const renderOptions = await callbacks.prepareRenderOptions(options);
|
|
66
|
+
const shouldApplyComponentRootAttributes =
|
|
67
|
+
renderOptions.componentRender?.canAttachAttributes &&
|
|
68
|
+
renderOptions.componentRender.rootAttributes &&
|
|
69
|
+
Object.keys(renderOptions.componentRender.rootAttributes).length > 0;
|
|
70
|
+
|
|
71
|
+
const renderExecution = await runWithComponentRenderContext(
|
|
72
|
+
{
|
|
73
|
+
currentIntegration: currentIntegrationName,
|
|
74
|
+
boundaryContext: callbacks.getComponentRenderBoundaryContext(),
|
|
75
|
+
},
|
|
76
|
+
async () => callbacks.render(renderOptions),
|
|
77
|
+
);
|
|
78
|
+
|
|
79
|
+
let renderedHtml = await new Response(renderExecution.value as BodyInit).text();
|
|
80
|
+
const componentGraphContext = this.mergeGraphContext(
|
|
81
|
+
renderExecution.graphContext,
|
|
82
|
+
(
|
|
83
|
+
renderOptions as IntegrationRendererRenderOptions<C> & {
|
|
84
|
+
componentGraphContext?: RenderExecutionGraphContext;
|
|
85
|
+
}
|
|
86
|
+
).componentGraphContext,
|
|
87
|
+
);
|
|
88
|
+
|
|
89
|
+
if (renderedHtml.includes('<eco-marker')) {
|
|
90
|
+
const markerResolution = await callbacks.resolveMarkerGraphHtml({
|
|
91
|
+
html: renderedHtml,
|
|
92
|
+
componentsToResolve: this.getComponentsToResolve(renderOptions),
|
|
93
|
+
graphContext: componentGraphContext,
|
|
94
|
+
});
|
|
95
|
+
renderedHtml = markerResolution.html;
|
|
96
|
+
|
|
97
|
+
if (markerResolution.assets.length > 0) {
|
|
98
|
+
const mergedDependencies = callbacks.dedupeProcessedAssets([
|
|
99
|
+
...callbacks.getProcessedDependencies(),
|
|
100
|
+
...markerResolution.assets,
|
|
101
|
+
]);
|
|
102
|
+
callbacks.setProcessedDependencies(mergedDependencies);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
if (shouldApplyComponentRootAttributes) {
|
|
107
|
+
renderedHtml = callbacks.applyAttributesToFirstBodyElement(
|
|
108
|
+
renderedHtml,
|
|
109
|
+
renderOptions.componentRender?.rootAttributes as Record<string, string>,
|
|
110
|
+
);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
const body = await callbacks.transformHtml(renderedHtml);
|
|
114
|
+
|
|
115
|
+
return {
|
|
116
|
+
body,
|
|
117
|
+
cacheStrategy: renderOptions.cacheStrategy,
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Merges captured render-time graph references with any explicit graph context
|
|
123
|
+
* provided by the page module.
|
|
124
|
+
*
|
|
125
|
+
* @param capturedGraphContext Graph context captured from the first render pass.
|
|
126
|
+
* @param explicitGraphContext Optional page-module graph metadata.
|
|
127
|
+
* @returns Merged graph context used during marker resolution.
|
|
128
|
+
*/
|
|
129
|
+
private mergeGraphContext(
|
|
130
|
+
capturedGraphContext: CapturedComponentGraphContext,
|
|
131
|
+
explicitGraphContext?: RenderExecutionGraphContext,
|
|
132
|
+
): RenderExecutionGraphContext {
|
|
133
|
+
return {
|
|
134
|
+
propsByRef: {
|
|
135
|
+
...(capturedGraphContext.propsByRef ?? {}),
|
|
136
|
+
...(explicitGraphContext?.propsByRef ?? {}),
|
|
137
|
+
},
|
|
138
|
+
slotChildrenByRef: {
|
|
139
|
+
...(capturedGraphContext.slotChildrenByRef ?? {}),
|
|
140
|
+
...(explicitGraphContext?.slotChildrenByRef ?? {}),
|
|
141
|
+
},
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* Returns the component set that participates in marker graph resolution for a
|
|
147
|
+
* render pass.
|
|
148
|
+
*
|
|
149
|
+
* @typeParam C Integration render output element type.
|
|
150
|
+
* @param renderOptions Normalized render options for the pass.
|
|
151
|
+
* @returns Ordered component list for graph registry construction.
|
|
152
|
+
*/
|
|
153
|
+
private getComponentsToResolve<C>(renderOptions: IntegrationRendererRenderOptions<C>): EcoComponent[] {
|
|
154
|
+
return renderOptions.Layout
|
|
155
|
+
? [renderOptions.HtmlTemplate as EcoComponent, renderOptions.Layout as EcoComponent, renderOptions.Page]
|
|
156
|
+
: [renderOptions.HtmlTemplate as EcoComponent, renderOptions.Page];
|
|
157
|
+
}
|
|
158
|
+
}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import type { EcoPagesAppConfig } from '../internal-types.js';
|
|
2
|
+
import type { ComponentRenderResult, EcoComponent, EcoPageComponent, EcoPageFile, EcoPagesElement, GetMetadata, GetStaticProps, HtmlTemplateProps, IntegrationRendererRenderOptions, PageMetadataProps, RouteRendererOptions } from '../public-types.js';
|
|
3
|
+
import { type AssetProcessingService, type ProcessedAsset } from '../services/asset-processing-service/index.js';
|
|
4
|
+
import { type ComponentRenderBoundaryContext } from '../eco/component-render-context.js';
|
|
5
|
+
type ResolvedPageModule = {
|
|
6
|
+
Page: EcoPageFile['default'] | EcoPageComponent<any>;
|
|
7
|
+
getStaticProps?: GetStaticProps<Record<string, unknown>>;
|
|
8
|
+
getMetadata?: GetMetadata;
|
|
9
|
+
integrationSpecificProps: Record<string, unknown>;
|
|
10
|
+
};
|
|
11
|
+
export interface RenderPreparationCallbacks {
|
|
12
|
+
resolvePageModule(file: string): Promise<ResolvedPageModule>;
|
|
13
|
+
getHtmlTemplate(): Promise<EcoComponent<HtmlTemplateProps>>;
|
|
14
|
+
resolvePageData(pageModule: {
|
|
15
|
+
getStaticProps?: GetStaticProps<Record<string, unknown>>;
|
|
16
|
+
getMetadata?: GetMetadata;
|
|
17
|
+
}, routeOptions: RouteRendererOptions): Promise<{
|
|
18
|
+
props: Record<string, unknown>;
|
|
19
|
+
metadata: PageMetadataProps;
|
|
20
|
+
}>;
|
|
21
|
+
resolveDependencies(components: (EcoComponent | Partial<EcoComponent>)[]): Promise<ProcessedAsset[]>;
|
|
22
|
+
buildRouteRenderAssets(file: string): Promise<ProcessedAsset[]> | undefined;
|
|
23
|
+
shouldRenderPageComponent(input: {
|
|
24
|
+
Page: EcoComponent;
|
|
25
|
+
Layout?: EcoComponent;
|
|
26
|
+
options: RouteRendererOptions;
|
|
27
|
+
}): boolean;
|
|
28
|
+
renderPageComponent(input: {
|
|
29
|
+
component: EcoComponent;
|
|
30
|
+
props: Record<string, unknown>;
|
|
31
|
+
}): Promise<ComponentRenderResult>;
|
|
32
|
+
/**
|
|
33
|
+
* Returns the boundary policy context that should be active while rendering
|
|
34
|
+
* page-root component output during preparation.
|
|
35
|
+
*/
|
|
36
|
+
getComponentRenderBoundaryContext(): ComponentRenderBoundaryContext;
|
|
37
|
+
setProcessedDependencies(dependencies: ProcessedAsset[]): void;
|
|
38
|
+
dedupeProcessedAssets(assets: ProcessedAsset[]): ProcessedAsset[];
|
|
39
|
+
createPageLocalsProxy(filePath: string): RouteRendererOptions['locals'];
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Prepares the normalized render inputs consumed by `IntegrationRenderer.execute()`.
|
|
43
|
+
*
|
|
44
|
+
* This service owns the orchestration that happens before the main HTML render:
|
|
45
|
+
* page module resolution, data loading, dependency aggregation, page-root
|
|
46
|
+
* component artifact capture, lazy trigger bootstrap generation, and request
|
|
47
|
+
* locals policy.
|
|
48
|
+
*/
|
|
49
|
+
export declare class RenderPreparationService {
|
|
50
|
+
private appConfig;
|
|
51
|
+
private assetProcessingService;
|
|
52
|
+
constructor(appConfig: EcoPagesAppConfig, assetProcessingService: AssetProcessingService);
|
|
53
|
+
/**
|
|
54
|
+
* Builds the final render options object used by the integration-specific
|
|
55
|
+
* renderer.
|
|
56
|
+
*
|
|
57
|
+
* The returned object contains normalized page data, processed dependency
|
|
58
|
+
* state, component render artifacts, and the locals contract expected by the
|
|
59
|
+
* rest of the pipeline.
|
|
60
|
+
*
|
|
61
|
+
* @typeParam C Integration render output element type.
|
|
62
|
+
* @param routeOptions Route-level render inputs.
|
|
63
|
+
* @param currentIntegrationName Active integration name for this preparation pass.
|
|
64
|
+
* @param callbacks Renderer-specific hooks used during preparation.
|
|
65
|
+
* @returns Normalized render options.
|
|
66
|
+
*/
|
|
67
|
+
prepare<C = EcoPagesElement>(routeOptions: RouteRendererOptions, currentIntegrationName: string, callbacks: RenderPreparationCallbacks): Promise<IntegrationRendererRenderOptions<C>>;
|
|
68
|
+
/**
|
|
69
|
+
* Collects resolved lazy trigger metadata from the component tree.
|
|
70
|
+
*
|
|
71
|
+
* Traversal is depth-first and deduplicated by component identity so shared
|
|
72
|
+
* component dependencies do not emit duplicate trigger sets.
|
|
73
|
+
*
|
|
74
|
+
* @param components Root component set.
|
|
75
|
+
* @param seen Internal visited set for shared graphs.
|
|
76
|
+
* @returns All resolved lazy triggers reachable from the root set.
|
|
77
|
+
*/
|
|
78
|
+
private collectResolvedTriggers;
|
|
79
|
+
/**
|
|
80
|
+
* Collects global integration dependencies used by nested components belonging
|
|
81
|
+
* to integrations other than the current renderer.
|
|
82
|
+
*
|
|
83
|
+
* @param components Root component set.
|
|
84
|
+
* @returns Processed integration dependencies contributed by nested integrations.
|
|
85
|
+
*/
|
|
86
|
+
private collectUsedIntegrationDependencies;
|
|
87
|
+
/**
|
|
88
|
+
* Discovers integration names referenced by the component dependency graph.
|
|
89
|
+
*
|
|
90
|
+
* @param components Root component set.
|
|
91
|
+
* @param seen Internal visited set for shared graphs.
|
|
92
|
+
* @returns Set of integration names found in the graph.
|
|
93
|
+
*/
|
|
94
|
+
private collectIntegrationNames;
|
|
95
|
+
/**
|
|
96
|
+
* Renders the page root through the component-level render contract so any
|
|
97
|
+
* integration-specific assets and root attributes are available before the main
|
|
98
|
+
* document render.
|
|
99
|
+
*
|
|
100
|
+
* @param input Page root render inputs.
|
|
101
|
+
* @returns Structured component render result.
|
|
102
|
+
*/
|
|
103
|
+
private renderPageRoot;
|
|
104
|
+
/**
|
|
105
|
+
* Builds the runtime assets needed to bootstrap global lazy trigger execution.
|
|
106
|
+
*
|
|
107
|
+
* @param triggers Fully resolved lazy trigger definitions.
|
|
108
|
+
* @returns Processed assets that should be merged into the final dependency set.
|
|
109
|
+
*/
|
|
110
|
+
private buildGlobalInjectorAssets;
|
|
111
|
+
}
|
|
112
|
+
export {};
|
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
import { createRequire } from "node:module";
|
|
2
|
+
import {
|
|
3
|
+
AssetFactory
|
|
4
|
+
} from "../services/asset-processing-service/index.js";
|
|
5
|
+
import { buildGlobalInjectorBootstrapContent, buildGlobalInjectorMapScript } from "../eco/global-injector-map.js";
|
|
6
|
+
import { runWithComponentRenderContext } from "../eco/component-render-context.js";
|
|
7
|
+
const coreRequire = createRequire(import.meta.url);
|
|
8
|
+
class RenderPreparationService {
|
|
9
|
+
constructor(appConfig, assetProcessingService) {
|
|
10
|
+
this.appConfig = appConfig;
|
|
11
|
+
this.assetProcessingService = assetProcessingService;
|
|
12
|
+
}
|
|
13
|
+
/**
|
|
14
|
+
* Builds the final render options object used by the integration-specific
|
|
15
|
+
* renderer.
|
|
16
|
+
*
|
|
17
|
+
* The returned object contains normalized page data, processed dependency
|
|
18
|
+
* state, component render artifacts, and the locals contract expected by the
|
|
19
|
+
* rest of the pipeline.
|
|
20
|
+
*
|
|
21
|
+
* @typeParam C Integration render output element type.
|
|
22
|
+
* @param routeOptions Route-level render inputs.
|
|
23
|
+
* @param currentIntegrationName Active integration name for this preparation pass.
|
|
24
|
+
* @param callbacks Renderer-specific hooks used during preparation.
|
|
25
|
+
* @returns Normalized render options.
|
|
26
|
+
*/
|
|
27
|
+
async prepare(routeOptions, currentIntegrationName, callbacks) {
|
|
28
|
+
const pageModule = await callbacks.resolvePageModule(routeOptions.file);
|
|
29
|
+
const { Page, integrationSpecificProps } = pageModule;
|
|
30
|
+
const HtmlTemplate = await callbacks.getHtmlTemplate();
|
|
31
|
+
const { props, metadata } = await callbacks.resolvePageData(pageModule, routeOptions);
|
|
32
|
+
const Layout = Page.config?.layout;
|
|
33
|
+
const componentsToResolve = Layout ? [HtmlTemplate, Layout, Page] : [HtmlTemplate, Page];
|
|
34
|
+
const resolvedDependencies = await callbacks.resolveDependencies(componentsToResolve);
|
|
35
|
+
const usedIntegrationDependencies = this.collectUsedIntegrationDependencies(
|
|
36
|
+
componentsToResolve,
|
|
37
|
+
currentIntegrationName
|
|
38
|
+
);
|
|
39
|
+
const pageDeps = await callbacks.buildRouteRenderAssets(routeOptions.file) || [];
|
|
40
|
+
const allDependencies = [...resolvedDependencies, ...usedIntegrationDependencies, ...pageDeps];
|
|
41
|
+
let componentRender;
|
|
42
|
+
if (callbacks.shouldRenderPageComponent({ Page, Layout, options: routeOptions })) {
|
|
43
|
+
componentRender = await this.renderPageRoot({
|
|
44
|
+
currentIntegrationName,
|
|
45
|
+
Page,
|
|
46
|
+
props,
|
|
47
|
+
routeOptions,
|
|
48
|
+
callbacks
|
|
49
|
+
});
|
|
50
|
+
if (componentRender.assets?.length) {
|
|
51
|
+
allDependencies.push(...componentRender.assets);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
const triggers = this.collectResolvedTriggers(componentsToResolve);
|
|
55
|
+
if (triggers.length > 0) {
|
|
56
|
+
const globalAssets = await this.buildGlobalInjectorAssets(triggers, currentIntegrationName);
|
|
57
|
+
allDependencies.push(...globalAssets);
|
|
58
|
+
}
|
|
59
|
+
callbacks.setProcessedDependencies(callbacks.dedupeProcessedAssets(allDependencies));
|
|
60
|
+
const pageProps = {
|
|
61
|
+
...props,
|
|
62
|
+
params: routeOptions.params || {},
|
|
63
|
+
query: routeOptions.query || {}
|
|
64
|
+
};
|
|
65
|
+
const cacheStrategy = Page.cache;
|
|
66
|
+
const defaultCacheStrategy = this.appConfig.cache?.defaultStrategy ?? "static";
|
|
67
|
+
const effectiveCacheStrategy = cacheStrategy ?? defaultCacheStrategy;
|
|
68
|
+
const localsAvailable = effectiveCacheStrategy === "dynamic" && routeOptions.locals !== void 0;
|
|
69
|
+
const pageLocals = localsAvailable ? routeOptions.locals : callbacks.createPageLocalsProxy(routeOptions.file);
|
|
70
|
+
const locals = localsAvailable ? routeOptions.locals : void 0;
|
|
71
|
+
const preparedOptions = {
|
|
72
|
+
...routeOptions,
|
|
73
|
+
resolvedDependencies,
|
|
74
|
+
componentRender,
|
|
75
|
+
HtmlTemplate,
|
|
76
|
+
Layout,
|
|
77
|
+
props,
|
|
78
|
+
Page,
|
|
79
|
+
metadata,
|
|
80
|
+
params: routeOptions.params || {},
|
|
81
|
+
query: routeOptions.query || {},
|
|
82
|
+
pageProps,
|
|
83
|
+
locals,
|
|
84
|
+
pageLocals,
|
|
85
|
+
cacheStrategy
|
|
86
|
+
};
|
|
87
|
+
return {
|
|
88
|
+
...integrationSpecificProps,
|
|
89
|
+
...preparedOptions
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Collects resolved lazy trigger metadata from the component tree.
|
|
94
|
+
*
|
|
95
|
+
* Traversal is depth-first and deduplicated by component identity so shared
|
|
96
|
+
* component dependencies do not emit duplicate trigger sets.
|
|
97
|
+
*
|
|
98
|
+
* @param components Root component set.
|
|
99
|
+
* @param seen Internal visited set for shared graphs.
|
|
100
|
+
* @returns All resolved lazy triggers reachable from the root set.
|
|
101
|
+
*/
|
|
102
|
+
collectResolvedTriggers(components, seen = /* @__PURE__ */ new Set()) {
|
|
103
|
+
const triggers = [];
|
|
104
|
+
for (const comp of components) {
|
|
105
|
+
const ecoComp = comp;
|
|
106
|
+
if (seen.has(ecoComp)) {
|
|
107
|
+
continue;
|
|
108
|
+
}
|
|
109
|
+
seen.add(ecoComp);
|
|
110
|
+
const ownTriggers = ecoComp.config?._resolvedLazyTriggers;
|
|
111
|
+
if (ownTriggers?.length) {
|
|
112
|
+
triggers.push(...ownTriggers);
|
|
113
|
+
}
|
|
114
|
+
const nested = ecoComp.config?.dependencies?.components;
|
|
115
|
+
if (nested?.length) {
|
|
116
|
+
triggers.push(...this.collectResolvedTriggers(nested, seen));
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
return triggers;
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Collects global integration dependencies used by nested components belonging
|
|
123
|
+
* to integrations other than the current renderer.
|
|
124
|
+
*
|
|
125
|
+
* @param components Root component set.
|
|
126
|
+
* @returns Processed integration dependencies contributed by nested integrations.
|
|
127
|
+
*/
|
|
128
|
+
collectUsedIntegrationDependencies(components, currentIntegrationName) {
|
|
129
|
+
const integrationNames = this.collectIntegrationNames(components);
|
|
130
|
+
const dependencies = [];
|
|
131
|
+
for (const integrationName of integrationNames) {
|
|
132
|
+
if (integrationName === currentIntegrationName) {
|
|
133
|
+
continue;
|
|
134
|
+
}
|
|
135
|
+
const integrationPlugin = this.appConfig.integrations.find(
|
|
136
|
+
(integration) => integration.name === integrationName
|
|
137
|
+
);
|
|
138
|
+
if (!integrationPlugin || typeof integrationPlugin.getResolvedIntegrationDependencies !== "function") {
|
|
139
|
+
continue;
|
|
140
|
+
}
|
|
141
|
+
dependencies.push(...integrationPlugin.getResolvedIntegrationDependencies());
|
|
142
|
+
}
|
|
143
|
+
return dependencies;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Discovers integration names referenced by the component dependency graph.
|
|
147
|
+
*
|
|
148
|
+
* @param components Root component set.
|
|
149
|
+
* @param seen Internal visited set for shared graphs.
|
|
150
|
+
* @returns Set of integration names found in the graph.
|
|
151
|
+
*/
|
|
152
|
+
collectIntegrationNames(components, seen = /* @__PURE__ */ new Set()) {
|
|
153
|
+
const integrationNames = /* @__PURE__ */ new Set();
|
|
154
|
+
for (const comp of components) {
|
|
155
|
+
const ecoComp = comp;
|
|
156
|
+
if (seen.has(ecoComp)) {
|
|
157
|
+
continue;
|
|
158
|
+
}
|
|
159
|
+
seen.add(ecoComp);
|
|
160
|
+
const integrationName = ecoComp.config?.integration ?? ecoComp.config?.__eco?.integration;
|
|
161
|
+
if (integrationName) {
|
|
162
|
+
integrationNames.add(integrationName);
|
|
163
|
+
}
|
|
164
|
+
const nested = ecoComp.config?.dependencies?.components;
|
|
165
|
+
if (nested?.length) {
|
|
166
|
+
const nestedNames = this.collectIntegrationNames(nested, seen);
|
|
167
|
+
for (const nestedName of nestedNames) {
|
|
168
|
+
integrationNames.add(nestedName);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
return integrationNames;
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Renders the page root through the component-level render contract so any
|
|
176
|
+
* integration-specific assets and root attributes are available before the main
|
|
177
|
+
* document render.
|
|
178
|
+
*
|
|
179
|
+
* @param input Page root render inputs.
|
|
180
|
+
* @returns Structured component render result.
|
|
181
|
+
*/
|
|
182
|
+
async renderPageRoot(input) {
|
|
183
|
+
const execution = await runWithComponentRenderContext(
|
|
184
|
+
{
|
|
185
|
+
currentIntegration: input.currentIntegrationName,
|
|
186
|
+
boundaryContext: input.callbacks.getComponentRenderBoundaryContext()
|
|
187
|
+
},
|
|
188
|
+
async () => input.callbacks.renderPageComponent({
|
|
189
|
+
component: input.Page,
|
|
190
|
+
props: {
|
|
191
|
+
...input.props,
|
|
192
|
+
params: input.routeOptions.params || {},
|
|
193
|
+
query: input.routeOptions.query || {}
|
|
194
|
+
}
|
|
195
|
+
})
|
|
196
|
+
);
|
|
197
|
+
return execution.value;
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Builds the runtime assets needed to bootstrap global lazy trigger execution.
|
|
201
|
+
*
|
|
202
|
+
* @param triggers Fully resolved lazy trigger definitions.
|
|
203
|
+
* @returns Processed assets that should be merged into the final dependency set.
|
|
204
|
+
*/
|
|
205
|
+
async buildGlobalInjectorAssets(triggers, currentIntegrationName) {
|
|
206
|
+
const globalInjectorImportPath = coreRequire.resolve("@ecopages/scripts-injector/global");
|
|
207
|
+
const globalInjectorRuntimeAsset = AssetFactory.createNodeModuleScript({
|
|
208
|
+
position: "head",
|
|
209
|
+
name: "ecopages-scripts-injector-global",
|
|
210
|
+
importPath: globalInjectorImportPath,
|
|
211
|
+
excludeFromHtml: true
|
|
212
|
+
});
|
|
213
|
+
const [globalInjectorRuntimeProcessed] = await this.assetProcessingService.processDependencies(
|
|
214
|
+
[globalInjectorRuntimeAsset],
|
|
215
|
+
currentIntegrationName
|
|
216
|
+
);
|
|
217
|
+
const globalInjectorModuleUrl = globalInjectorRuntimeProcessed?.srcUrl;
|
|
218
|
+
if (!globalInjectorModuleUrl) {
|
|
219
|
+
throw new Error("[ecopages] Failed to resolve global injector runtime asset URL.");
|
|
220
|
+
}
|
|
221
|
+
const mapScript = AssetFactory.createInlineContentScript({
|
|
222
|
+
position: "head",
|
|
223
|
+
name: "ecopages-global-injector-map",
|
|
224
|
+
content: buildGlobalInjectorMapScript(triggers),
|
|
225
|
+
attributes: { type: "ecopages/global-injector-map" },
|
|
226
|
+
bundle: false
|
|
227
|
+
});
|
|
228
|
+
const bootstrapScript = AssetFactory.createContentScript({
|
|
229
|
+
position: "head",
|
|
230
|
+
name: "ecopages-global-injector-bootstrap",
|
|
231
|
+
content: buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl),
|
|
232
|
+
attributes: { type: "module" },
|
|
233
|
+
bundle: false
|
|
234
|
+
});
|
|
235
|
+
return this.assetProcessingService.processDependencies(
|
|
236
|
+
[mapScript, bootstrapScript, globalInjectorRuntimeAsset],
|
|
237
|
+
currentIntegrationName
|
|
238
|
+
);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
export {
|
|
242
|
+
RenderPreparationService
|
|
243
|
+
};
|