@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,105 @@
|
|
|
1
|
+
import type { EcoComponent } from '../public-types.js';
|
|
2
|
+
import type { MarkerNodeId } from '../route-renderer/component-marker.js';
|
|
3
|
+
/**
|
|
4
|
+
* Outcome returned by boundary policy during one component render pass.
|
|
5
|
+
*
|
|
6
|
+
* - `inline`: render the target component immediately in the current pass
|
|
7
|
+
* - `defer`: emit an `eco-marker` and resolve it during the marker graph phase
|
|
8
|
+
*/
|
|
9
|
+
export type BoundaryRenderMode = 'inline' | 'defer';
|
|
10
|
+
/**
|
|
11
|
+
* Input provided to boundary policy when a component boundary is reached.
|
|
12
|
+
*
|
|
13
|
+
* This keeps `eco.component()` decoupled from concrete integration/plugin
|
|
14
|
+
* objects while still giving policy enough information to decide whether the
|
|
15
|
+
* boundary should render immediately or be deferred.
|
|
16
|
+
*/
|
|
17
|
+
export type BoundaryRenderDecisionInput = {
|
|
18
|
+
currentIntegration: string;
|
|
19
|
+
targetIntegration?: string;
|
|
20
|
+
component: EcoComponent;
|
|
21
|
+
};
|
|
22
|
+
/**
|
|
23
|
+
* Narrow render-pass facade used by `eco.component()` for boundary decisions.
|
|
24
|
+
*
|
|
25
|
+
* The boundary context is intentionally small so component rendering can remain
|
|
26
|
+
* unaware of integration registries, plugin instances, or renderer lifecycles.
|
|
27
|
+
*/
|
|
28
|
+
export type ComponentRenderBoundaryContext = {
|
|
29
|
+
/**
|
|
30
|
+
* Decides whether the next component boundary should render inline or defer to
|
|
31
|
+
* the marker graph stage.
|
|
32
|
+
*
|
|
33
|
+
* @param input Boundary metadata for the current render pass.
|
|
34
|
+
* @returns Boundary rendering mode for the target component.
|
|
35
|
+
*/
|
|
36
|
+
decideBoundaryRender(input: BoundaryRenderDecisionInput): BoundaryRenderMode;
|
|
37
|
+
};
|
|
38
|
+
/**
|
|
39
|
+
* Per-render mutable state used while collecting marker graph references.
|
|
40
|
+
*
|
|
41
|
+
* Counters generate deterministic ids within one render execution.
|
|
42
|
+
*/
|
|
43
|
+
type ComponentRenderContext = {
|
|
44
|
+
currentIntegration: string;
|
|
45
|
+
boundaryContext: ComponentRenderBoundaryContext;
|
|
46
|
+
nextNodeId: number;
|
|
47
|
+
nextPropsRefId: number;
|
|
48
|
+
nextSlotRefId: number;
|
|
49
|
+
propsByRef: Record<string, Record<string, unknown>>;
|
|
50
|
+
slotChildrenByRef: Record<string, MarkerNodeId[]>;
|
|
51
|
+
};
|
|
52
|
+
/**
|
|
53
|
+
* Serializable graph context captured from one render execution.
|
|
54
|
+
*
|
|
55
|
+
* This payload is merged with explicit page-module graph context before marker
|
|
56
|
+
* resolution in the route renderer.
|
|
57
|
+
*/
|
|
58
|
+
export type ComponentGraphContext = {
|
|
59
|
+
propsByRef: Record<string, Record<string, unknown>>;
|
|
60
|
+
slotChildrenByRef: Record<string, MarkerNodeId[]>;
|
|
61
|
+
};
|
|
62
|
+
/**
|
|
63
|
+
* Returns the current component render context, if one is active.
|
|
64
|
+
*
|
|
65
|
+
* @returns Active render context or `undefined` outside render execution.
|
|
66
|
+
*/
|
|
67
|
+
export declare function getComponentRenderContext(): ComponentRenderContext | undefined;
|
|
68
|
+
/**
|
|
69
|
+
* Allocates the next marker node id in the active render context.
|
|
70
|
+
*
|
|
71
|
+
* @param context Active render context.
|
|
72
|
+
* @returns Stable marker node id for this render pass.
|
|
73
|
+
*/
|
|
74
|
+
export declare function createNodeId(context: ComponentRenderContext): MarkerNodeId;
|
|
75
|
+
/**
|
|
76
|
+
* Allocates the next props reference id in the active render context.
|
|
77
|
+
*
|
|
78
|
+
* @param context Active render context.
|
|
79
|
+
* @returns Props reference key.
|
|
80
|
+
*/
|
|
81
|
+
export declare function createPropsRef(context: ComponentRenderContext): string;
|
|
82
|
+
/**
|
|
83
|
+
* Allocates the next slot reference id in the active render context.
|
|
84
|
+
*
|
|
85
|
+
* @param context Active render context.
|
|
86
|
+
* @returns Slot reference key.
|
|
87
|
+
*/
|
|
88
|
+
export declare function createSlotRef(context: ComponentRenderContext): string;
|
|
89
|
+
/**
|
|
90
|
+
* Runs render work under a fresh component render context and returns both:
|
|
91
|
+
* - the render result value
|
|
92
|
+
* - captured graph reference maps for downstream marker resolution
|
|
93
|
+
*
|
|
94
|
+
* @param input Execution metadata for current integration and boundary policy.
|
|
95
|
+
* @param render Async render function to execute inside the context.
|
|
96
|
+
* @returns Render result and captured graph context maps.
|
|
97
|
+
*/
|
|
98
|
+
export declare function runWithComponentRenderContext<T>(input: {
|
|
99
|
+
currentIntegration: string;
|
|
100
|
+
boundaryContext: ComponentRenderBoundaryContext;
|
|
101
|
+
}, render: () => Promise<T>): Promise<{
|
|
102
|
+
value: T;
|
|
103
|
+
graphContext: ComponentGraphContext;
|
|
104
|
+
}>;
|
|
105
|
+
export {};
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
const contextStack = [];
|
|
2
|
+
let nodeContextStorage = null;
|
|
3
|
+
let nodeContextStorageLoader = null;
|
|
4
|
+
async function getContextStorage() {
|
|
5
|
+
if (nodeContextStorage) {
|
|
6
|
+
return nodeContextStorage;
|
|
7
|
+
}
|
|
8
|
+
if (nodeContextStorageLoader) {
|
|
9
|
+
return nodeContextStorageLoader;
|
|
10
|
+
}
|
|
11
|
+
nodeContextStorageLoader = import("node:async_hooks").then((module) => {
|
|
12
|
+
const storage = new module.AsyncLocalStorage();
|
|
13
|
+
nodeContextStorage = {
|
|
14
|
+
getStore: () => storage.getStore(),
|
|
15
|
+
run: (store, callback) => storage.run(store, callback)
|
|
16
|
+
};
|
|
17
|
+
return nodeContextStorage;
|
|
18
|
+
}).catch(() => {
|
|
19
|
+
nodeContextStorage = null;
|
|
20
|
+
return null;
|
|
21
|
+
}).finally(() => {
|
|
22
|
+
nodeContextStorageLoader = null;
|
|
23
|
+
});
|
|
24
|
+
return nodeContextStorageLoader;
|
|
25
|
+
}
|
|
26
|
+
function getComponentRenderContext() {
|
|
27
|
+
return nodeContextStorage?.getStore() ?? contextStack[contextStack.length - 1];
|
|
28
|
+
}
|
|
29
|
+
function createNodeId(context) {
|
|
30
|
+
context.nextNodeId += 1;
|
|
31
|
+
return `n_${context.nextNodeId}`;
|
|
32
|
+
}
|
|
33
|
+
function createPropsRef(context) {
|
|
34
|
+
context.nextPropsRefId += 1;
|
|
35
|
+
return `p_${context.nextPropsRefId}`;
|
|
36
|
+
}
|
|
37
|
+
function createSlotRef(context) {
|
|
38
|
+
context.nextSlotRefId += 1;
|
|
39
|
+
return `s_${context.nextSlotRefId}`;
|
|
40
|
+
}
|
|
41
|
+
async function runWithComponentRenderContext(input, render) {
|
|
42
|
+
const context = {
|
|
43
|
+
currentIntegration: input.currentIntegration,
|
|
44
|
+
boundaryContext: input.boundaryContext,
|
|
45
|
+
nextNodeId: 0,
|
|
46
|
+
nextPropsRefId: 0,
|
|
47
|
+
nextSlotRefId: 0,
|
|
48
|
+
propsByRef: {},
|
|
49
|
+
slotChildrenByRef: {}
|
|
50
|
+
};
|
|
51
|
+
const storage = await getContextStorage();
|
|
52
|
+
let value;
|
|
53
|
+
if (storage) {
|
|
54
|
+
value = await storage.run(context, render);
|
|
55
|
+
} else {
|
|
56
|
+
contextStack.push(context);
|
|
57
|
+
try {
|
|
58
|
+
value = await render();
|
|
59
|
+
} finally {
|
|
60
|
+
contextStack.pop();
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
return {
|
|
64
|
+
value,
|
|
65
|
+
graphContext: {
|
|
66
|
+
propsByRef: context.propsByRef,
|
|
67
|
+
slotChildrenByRef: context.slotChildrenByRef
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
export {
|
|
72
|
+
createNodeId,
|
|
73
|
+
createPropsRef,
|
|
74
|
+
createSlotRef,
|
|
75
|
+
getComponentRenderContext,
|
|
76
|
+
runWithComponentRenderContext
|
|
77
|
+
};
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import type { EcoComponent } from '../public-types.ts';
|
|
2
|
+
import type { MarkerNodeId } from '../route-renderer/component-marker.ts';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Outcome returned by boundary policy during one component render pass.
|
|
6
|
+
*
|
|
7
|
+
* - `inline`: render the target component immediately in the current pass
|
|
8
|
+
* - `defer`: emit an `eco-marker` and resolve it during the marker graph phase
|
|
9
|
+
*/
|
|
10
|
+
export type BoundaryRenderMode = 'inline' | 'defer';
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Input provided to boundary policy when a component boundary is reached.
|
|
14
|
+
*
|
|
15
|
+
* This keeps `eco.component()` decoupled from concrete integration/plugin
|
|
16
|
+
* objects while still giving policy enough information to decide whether the
|
|
17
|
+
* boundary should render immediately or be deferred.
|
|
18
|
+
*/
|
|
19
|
+
export type BoundaryRenderDecisionInput = {
|
|
20
|
+
currentIntegration: string;
|
|
21
|
+
targetIntegration?: string;
|
|
22
|
+
component: EcoComponent;
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Narrow render-pass facade used by `eco.component()` for boundary decisions.
|
|
27
|
+
*
|
|
28
|
+
* The boundary context is intentionally small so component rendering can remain
|
|
29
|
+
* unaware of integration registries, plugin instances, or renderer lifecycles.
|
|
30
|
+
*/
|
|
31
|
+
export type ComponentRenderBoundaryContext = {
|
|
32
|
+
/**
|
|
33
|
+
* Decides whether the next component boundary should render inline or defer to
|
|
34
|
+
* the marker graph stage.
|
|
35
|
+
*
|
|
36
|
+
* @param input Boundary metadata for the current render pass.
|
|
37
|
+
* @returns Boundary rendering mode for the target component.
|
|
38
|
+
*/
|
|
39
|
+
decideBoundaryRender(input: BoundaryRenderDecisionInput): BoundaryRenderMode;
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Per-render mutable state used while collecting marker graph references.
|
|
44
|
+
*
|
|
45
|
+
* Counters generate deterministic ids within one render execution.
|
|
46
|
+
*/
|
|
47
|
+
type ComponentRenderContext = {
|
|
48
|
+
currentIntegration: string;
|
|
49
|
+
boundaryContext: ComponentRenderBoundaryContext;
|
|
50
|
+
nextNodeId: number;
|
|
51
|
+
nextPropsRefId: number;
|
|
52
|
+
nextSlotRefId: number;
|
|
53
|
+
propsByRef: Record<string, Record<string, unknown>>;
|
|
54
|
+
slotChildrenByRef: Record<string, MarkerNodeId[]>;
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Serializable graph context captured from one render execution.
|
|
59
|
+
*
|
|
60
|
+
* This payload is merged with explicit page-module graph context before marker
|
|
61
|
+
* resolution in the route renderer.
|
|
62
|
+
*/
|
|
63
|
+
export type ComponentGraphContext = {
|
|
64
|
+
propsByRef: Record<string, Record<string, unknown>>;
|
|
65
|
+
slotChildrenByRef: Record<string, MarkerNodeId[]>;
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
type ContextStorage = {
|
|
69
|
+
getStore(): ComponentRenderContext | undefined;
|
|
70
|
+
run<T>(store: ComponentRenderContext, callback: () => Promise<T>): Promise<T>;
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
const contextStack: ComponentRenderContext[] = [];
|
|
74
|
+
let nodeContextStorage: ContextStorage | null = null;
|
|
75
|
+
let nodeContextStorageLoader: Promise<ContextStorage | null> | null = null;
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Lazily initializes async context storage for Node runtimes.
|
|
79
|
+
*
|
|
80
|
+
* Falls back to in-memory stack mode when async_hooks is unavailable.
|
|
81
|
+
*
|
|
82
|
+
* @returns Async context storage when available; otherwise `null`.
|
|
83
|
+
*/
|
|
84
|
+
async function getContextStorage(): Promise<ContextStorage | null> {
|
|
85
|
+
if (nodeContextStorage) {
|
|
86
|
+
return nodeContextStorage;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
if (nodeContextStorageLoader) {
|
|
90
|
+
return nodeContextStorageLoader;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
nodeContextStorageLoader = import('node:async_hooks')
|
|
94
|
+
.then((module) => {
|
|
95
|
+
const storage = new module.AsyncLocalStorage<ComponentRenderContext>();
|
|
96
|
+
nodeContextStorage = {
|
|
97
|
+
getStore: () => storage.getStore(),
|
|
98
|
+
run: (store, callback) => storage.run(store, callback),
|
|
99
|
+
};
|
|
100
|
+
return nodeContextStorage;
|
|
101
|
+
})
|
|
102
|
+
.catch(() => {
|
|
103
|
+
nodeContextStorage = null;
|
|
104
|
+
return null;
|
|
105
|
+
})
|
|
106
|
+
.finally(() => {
|
|
107
|
+
nodeContextStorageLoader = null;
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
return nodeContextStorageLoader;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Returns the current component render context, if one is active.
|
|
115
|
+
*
|
|
116
|
+
* @returns Active render context or `undefined` outside render execution.
|
|
117
|
+
*/
|
|
118
|
+
export function getComponentRenderContext(): ComponentRenderContext | undefined {
|
|
119
|
+
return nodeContextStorage?.getStore() ?? contextStack[contextStack.length - 1];
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/**
|
|
123
|
+
* Allocates the next marker node id in the active render context.
|
|
124
|
+
*
|
|
125
|
+
* @param context Active render context.
|
|
126
|
+
* @returns Stable marker node id for this render pass.
|
|
127
|
+
*/
|
|
128
|
+
export function createNodeId(context: ComponentRenderContext): MarkerNodeId {
|
|
129
|
+
context.nextNodeId += 1;
|
|
130
|
+
return `n_${context.nextNodeId}`;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Allocates the next props reference id in the active render context.
|
|
135
|
+
*
|
|
136
|
+
* @param context Active render context.
|
|
137
|
+
* @returns Props reference key.
|
|
138
|
+
*/
|
|
139
|
+
export function createPropsRef(context: ComponentRenderContext): string {
|
|
140
|
+
context.nextPropsRefId += 1;
|
|
141
|
+
return `p_${context.nextPropsRefId}`;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Allocates the next slot reference id in the active render context.
|
|
146
|
+
*
|
|
147
|
+
* @param context Active render context.
|
|
148
|
+
* @returns Slot reference key.
|
|
149
|
+
*/
|
|
150
|
+
export function createSlotRef(context: ComponentRenderContext): string {
|
|
151
|
+
context.nextSlotRefId += 1;
|
|
152
|
+
return `s_${context.nextSlotRefId}`;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Runs render work under a fresh component render context and returns both:
|
|
157
|
+
* - the render result value
|
|
158
|
+
* - captured graph reference maps for downstream marker resolution
|
|
159
|
+
*
|
|
160
|
+
* @param input Execution metadata for current integration and boundary policy.
|
|
161
|
+
* @param render Async render function to execute inside the context.
|
|
162
|
+
* @returns Render result and captured graph context maps.
|
|
163
|
+
*/
|
|
164
|
+
export async function runWithComponentRenderContext<T>(
|
|
165
|
+
input: {
|
|
166
|
+
currentIntegration: string;
|
|
167
|
+
boundaryContext: ComponentRenderBoundaryContext;
|
|
168
|
+
},
|
|
169
|
+
render: () => Promise<T>,
|
|
170
|
+
): Promise<{ value: T; graphContext: ComponentGraphContext }> {
|
|
171
|
+
const context: ComponentRenderContext = {
|
|
172
|
+
currentIntegration: input.currentIntegration,
|
|
173
|
+
boundaryContext: input.boundaryContext,
|
|
174
|
+
nextNodeId: 0,
|
|
175
|
+
nextPropsRefId: 0,
|
|
176
|
+
nextSlotRefId: 0,
|
|
177
|
+
propsByRef: {},
|
|
178
|
+
slotChildrenByRef: {},
|
|
179
|
+
};
|
|
180
|
+
|
|
181
|
+
const storage = await getContextStorage();
|
|
182
|
+
|
|
183
|
+
let value: T;
|
|
184
|
+
if (storage) {
|
|
185
|
+
value = await storage.run(context, render);
|
|
186
|
+
} else {
|
|
187
|
+
contextStack.push(context);
|
|
188
|
+
try {
|
|
189
|
+
value = await render();
|
|
190
|
+
} finally {
|
|
191
|
+
contextStack.pop();
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
return {
|
|
196
|
+
value,
|
|
197
|
+
graphContext: {
|
|
198
|
+
propsByRef: context.propsByRef,
|
|
199
|
+
slotChildrenByRef: context.slotChildrenByRef,
|
|
200
|
+
},
|
|
201
|
+
};
|
|
202
|
+
}
|
package/src/eco/eco.d.ts
ADDED
package/src/eco/eco.js
ADDED
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import { createNodeId, createPropsRef, createSlotRef, getComponentRenderContext } from "./component-render-context.js";
|
|
2
|
+
import { createComponentMarker, parseComponentMarkers } from "../route-renderer/component-marker.js";
|
|
3
|
+
import { addTriggerAttribute, isThenable, wrapWithScriptsInjector } from "./eco.utils.js";
|
|
4
|
+
function createComponentFactory(options) {
|
|
5
|
+
const integrationName = options.integration ?? options.__eco?.integration;
|
|
6
|
+
const comp = ((props) => {
|
|
7
|
+
const renderContext = getComponentRenderContext();
|
|
8
|
+
const shouldEmitMarker = renderContext !== void 0 && renderContext.boundaryContext.decideBoundaryRender({
|
|
9
|
+
currentIntegration: renderContext.currentIntegration,
|
|
10
|
+
targetIntegration: integrationName,
|
|
11
|
+
component: comp
|
|
12
|
+
}) === "defer";
|
|
13
|
+
if (shouldEmitMarker && renderContext) {
|
|
14
|
+
const nodeId = createNodeId(renderContext);
|
|
15
|
+
const propsRef = createPropsRef(renderContext);
|
|
16
|
+
const componentRef = comp.config?.__eco?.id ?? comp.config?.__eco?.file;
|
|
17
|
+
if (!componentRef) {
|
|
18
|
+
throw new Error(
|
|
19
|
+
"[ecopages] Missing component reference metadata for cross-integration marker emission."
|
|
20
|
+
);
|
|
21
|
+
}
|
|
22
|
+
const componentProps = props ?? {};
|
|
23
|
+
renderContext.propsByRef[propsRef] = componentProps;
|
|
24
|
+
let slotRef;
|
|
25
|
+
const children = componentProps.children;
|
|
26
|
+
if (typeof children === "string" && children.includes("<eco-marker")) {
|
|
27
|
+
const childMarkers = parseComponentMarkers(children);
|
|
28
|
+
if (childMarkers.length > 0) {
|
|
29
|
+
slotRef = createSlotRef(renderContext);
|
|
30
|
+
renderContext.slotChildrenByRef[slotRef] = childMarkers.map((marker) => marker.nodeId);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
return createComponentMarker({
|
|
34
|
+
nodeId,
|
|
35
|
+
integration: integrationName,
|
|
36
|
+
componentRef,
|
|
37
|
+
propsRef,
|
|
38
|
+
slotRef
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
const content = options.render(props);
|
|
42
|
+
const lazyTriggers = comp.config?._resolvedLazyTriggers;
|
|
43
|
+
if (lazyTriggers && lazyTriggers.length > 0) {
|
|
44
|
+
const triggerId = lazyTriggers[0].triggerId;
|
|
45
|
+
if (isThenable(content)) {
|
|
46
|
+
return content.then((resolvedContent) => addTriggerAttribute(resolvedContent, triggerId));
|
|
47
|
+
}
|
|
48
|
+
return addTriggerAttribute(content, triggerId);
|
|
49
|
+
}
|
|
50
|
+
const lazyGroups = comp.config?._resolvedLazyScripts;
|
|
51
|
+
if (lazyGroups && lazyGroups.length > 0) {
|
|
52
|
+
if (isThenable(content)) {
|
|
53
|
+
return content.then((resolvedContent) => wrapWithScriptsInjector(resolvedContent, lazyGroups));
|
|
54
|
+
}
|
|
55
|
+
return wrapWithScriptsInjector(content, lazyGroups);
|
|
56
|
+
}
|
|
57
|
+
return content;
|
|
58
|
+
});
|
|
59
|
+
comp.config = {
|
|
60
|
+
__eco: options.__eco,
|
|
61
|
+
integration: options.integration,
|
|
62
|
+
dependencies: options.dependencies
|
|
63
|
+
};
|
|
64
|
+
return comp;
|
|
65
|
+
}
|
|
66
|
+
function component(options) {
|
|
67
|
+
return createComponentFactory(options);
|
|
68
|
+
}
|
|
69
|
+
function page(options) {
|
|
70
|
+
const { layout, dependencies, render, staticPaths: staticPaths2, staticProps: staticProps2, metadata: metadata2, cache, requires, middleware } = options;
|
|
71
|
+
const componentOptions = {
|
|
72
|
+
__eco: options.__eco,
|
|
73
|
+
integration: options.integration,
|
|
74
|
+
dependencies: layout ? {
|
|
75
|
+
...dependencies,
|
|
76
|
+
components: [...dependencies?.components || [], layout]
|
|
77
|
+
} : dependencies,
|
|
78
|
+
render
|
|
79
|
+
};
|
|
80
|
+
const pageComponent = createComponentFactory(componentOptions);
|
|
81
|
+
if (layout && pageComponent.config) {
|
|
82
|
+
pageComponent.config.layout = layout;
|
|
83
|
+
}
|
|
84
|
+
if (staticPaths2) pageComponent.staticPaths = staticPaths2;
|
|
85
|
+
if (staticProps2) pageComponent.staticProps = staticProps2;
|
|
86
|
+
if (metadata2) pageComponent.metadata = metadata2;
|
|
87
|
+
if (cache) pageComponent.cache = cache;
|
|
88
|
+
if (requires) pageComponent.requires = requires;
|
|
89
|
+
if (middleware) pageComponent.middleware = middleware;
|
|
90
|
+
return pageComponent;
|
|
91
|
+
}
|
|
92
|
+
function metadata(fn) {
|
|
93
|
+
return fn;
|
|
94
|
+
}
|
|
95
|
+
function staticPaths(fn) {
|
|
96
|
+
return fn;
|
|
97
|
+
}
|
|
98
|
+
function staticProps(fn) {
|
|
99
|
+
return fn;
|
|
100
|
+
}
|
|
101
|
+
const eco = {
|
|
102
|
+
component,
|
|
103
|
+
page,
|
|
104
|
+
metadata,
|
|
105
|
+
staticPaths,
|
|
106
|
+
staticProps
|
|
107
|
+
};
|
|
108
|
+
export {
|
|
109
|
+
eco
|
|
110
|
+
};
|