@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.
Files changed (342) hide show
  1. package/CHANGELOG.md +89 -0
  2. package/LICENSE +21 -0
  3. package/README.md +32 -0
  4. package/package.json +279 -0
  5. package/src/adapters/abstract/application-adapter.d.ts +168 -0
  6. package/src/adapters/abstract/application-adapter.js +109 -0
  7. package/src/adapters/abstract/application-adapter.ts +337 -0
  8. package/src/adapters/abstract/router-adapter.d.ts +26 -0
  9. package/src/adapters/abstract/router-adapter.js +5 -0
  10. package/src/adapters/abstract/router-adapter.ts +30 -0
  11. package/src/adapters/abstract/server-adapter.d.ts +69 -0
  12. package/src/adapters/abstract/server-adapter.js +15 -0
  13. package/src/adapters/abstract/server-adapter.ts +79 -0
  14. package/src/adapters/bun/client-bridge.d.ts +34 -0
  15. package/src/adapters/bun/client-bridge.js +48 -0
  16. package/src/adapters/bun/client-bridge.ts +62 -0
  17. package/src/adapters/bun/create-app.d.ts +60 -0
  18. package/src/adapters/bun/create-app.js +117 -0
  19. package/src/adapters/bun/create-app.ts +189 -0
  20. package/src/adapters/bun/define-api-handler.d.ts +61 -0
  21. package/src/adapters/bun/define-api-handler.js +15 -0
  22. package/src/adapters/bun/define-api-handler.ts +114 -0
  23. package/src/adapters/bun/hmr-manager.d.ts +84 -0
  24. package/src/adapters/bun/hmr-manager.js +227 -0
  25. package/src/adapters/bun/hmr-manager.ts +281 -0
  26. package/src/adapters/bun/index.d.ts +3 -0
  27. package/src/adapters/bun/index.js +8 -0
  28. package/src/adapters/bun/index.ts +3 -0
  29. package/src/adapters/bun/server-adapter.d.ts +155 -0
  30. package/src/adapters/bun/server-adapter.js +368 -0
  31. package/src/adapters/bun/server-adapter.ts +492 -0
  32. package/src/adapters/bun/server-lifecycle.d.ts +52 -0
  33. package/src/adapters/bun/server-lifecycle.js +120 -0
  34. package/src/adapters/bun/server-lifecycle.ts +154 -0
  35. package/src/adapters/index.d.ts +6 -0
  36. package/src/adapters/index.js +14 -0
  37. package/src/adapters/index.ts +6 -0
  38. package/src/adapters/node/create-app.d.ts +21 -0
  39. package/src/adapters/node/create-app.js +143 -0
  40. package/src/adapters/node/create-app.ts +179 -0
  41. package/src/adapters/node/index.d.ts +4 -0
  42. package/src/adapters/node/index.js +8 -0
  43. package/src/adapters/node/index.ts +9 -0
  44. package/src/adapters/node/node-client-bridge.d.ts +26 -0
  45. package/src/adapters/node/node-client-bridge.js +66 -0
  46. package/src/adapters/node/node-client-bridge.ts +79 -0
  47. package/src/adapters/node/node-hmr-manager.d.ts +62 -0
  48. package/src/adapters/node/node-hmr-manager.js +221 -0
  49. package/src/adapters/node/node-hmr-manager.ts +271 -0
  50. package/src/adapters/node/server-adapter.d.ts +190 -0
  51. package/src/adapters/node/server-adapter.js +420 -0
  52. package/src/adapters/node/server-adapter.ts +561 -0
  53. package/src/adapters/node/static-content-server.d.ts +24 -0
  54. package/src/adapters/node/static-content-server.js +166 -0
  55. package/src/adapters/node/static-content-server.ts +203 -0
  56. package/src/adapters/shared/api-response.d.ts +52 -0
  57. package/src/adapters/shared/api-response.js +96 -0
  58. package/src/adapters/shared/api-response.ts +104 -0
  59. package/src/adapters/shared/application-adapter.d.ts +18 -0
  60. package/src/adapters/shared/application-adapter.js +90 -0
  61. package/src/adapters/shared/application-adapter.ts +199 -0
  62. package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
  63. package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
  64. package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
  65. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
  66. package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
  67. package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
  68. package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
  69. package/src/adapters/shared/fs-server-response-factory.js +97 -0
  70. package/src/adapters/shared/fs-server-response-factory.ts +118 -0
  71. package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
  72. package/src/adapters/shared/fs-server-response-matcher.js +155 -0
  73. package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
  74. package/src/adapters/shared/render-context.d.ts +14 -0
  75. package/src/adapters/shared/render-context.js +69 -0
  76. package/src/adapters/shared/render-context.ts +105 -0
  77. package/src/adapters/shared/server-adapter.d.ts +87 -0
  78. package/src/adapters/shared/server-adapter.js +353 -0
  79. package/src/adapters/shared/server-adapter.ts +442 -0
  80. package/src/adapters/shared/server-route-handler.d.ts +89 -0
  81. package/src/adapters/shared/server-route-handler.js +120 -0
  82. package/src/adapters/shared/server-route-handler.ts +166 -0
  83. package/src/adapters/shared/server-static-builder.d.ts +38 -0
  84. package/src/adapters/shared/server-static-builder.js +46 -0
  85. package/src/adapters/shared/server-static-builder.ts +82 -0
  86. package/src/build/build-adapter.d.ts +74 -0
  87. package/src/build/build-adapter.js +54 -0
  88. package/src/build/build-adapter.ts +132 -0
  89. package/src/build/build-types.d.ts +57 -0
  90. package/src/build/build-types.js +0 -0
  91. package/src/build/build-types.ts +83 -0
  92. package/src/build/esbuild-build-adapter.d.ts +69 -0
  93. package/src/build/esbuild-build-adapter.js +390 -0
  94. package/src/build/esbuild-build-adapter.ts +510 -0
  95. package/src/config/config-builder.d.ts +227 -0
  96. package/src/config/config-builder.js +392 -0
  97. package/src/config/config-builder.ts +474 -0
  98. package/src/constants.d.ts +32 -0
  99. package/src/constants.js +21 -0
  100. package/src/constants.ts +39 -0
  101. package/src/create-app.d.ts +17 -0
  102. package/src/create-app.js +66 -0
  103. package/src/create-app.ts +87 -0
  104. package/src/declarations.d.ts +26 -0
  105. package/src/define-api-handler.d.ts +25 -0
  106. package/src/define-api-handler.js +15 -0
  107. package/src/define-api-handler.ts +66 -0
  108. package/src/dev/sc-server.d.ts +30 -0
  109. package/src/dev/sc-server.js +111 -0
  110. package/src/dev/sc-server.ts +143 -0
  111. package/src/eco/README.md +636 -0
  112. package/src/eco/component-render-context.d.ts +105 -0
  113. package/src/eco/component-render-context.js +77 -0
  114. package/src/eco/component-render-context.ts +202 -0
  115. package/src/eco/eco.d.ts +9 -0
  116. package/src/eco/eco.js +110 -0
  117. package/src/eco/eco.ts +221 -0
  118. package/src/eco/eco.types.d.ts +170 -0
  119. package/src/eco/eco.types.js +0 -0
  120. package/src/eco/eco.types.ts +202 -0
  121. package/src/eco/eco.utils.d.ts +40 -0
  122. package/src/eco/eco.utils.js +40 -0
  123. package/src/eco/eco.utils.ts +89 -0
  124. package/src/eco/global-injector-map.d.ts +16 -0
  125. package/src/eco/global-injector-map.js +80 -0
  126. package/src/eco/global-injector-map.ts +112 -0
  127. package/src/eco/lazy-injector-map.d.ts +8 -0
  128. package/src/eco/lazy-injector-map.js +70 -0
  129. package/src/eco/lazy-injector-map.ts +120 -0
  130. package/src/eco/module-dependencies.d.ts +18 -0
  131. package/src/eco/module-dependencies.js +49 -0
  132. package/src/eco/module-dependencies.ts +75 -0
  133. package/src/env.d.ts +20 -0
  134. package/src/errors/http-error.d.ts +31 -0
  135. package/src/errors/http-error.js +50 -0
  136. package/src/errors/http-error.ts +72 -0
  137. package/src/errors/index.d.ts +2 -0
  138. package/src/errors/index.js +4 -0
  139. package/src/errors/index.ts +2 -0
  140. package/src/errors/locals-access-error.d.ts +4 -0
  141. package/src/errors/locals-access-error.js +9 -0
  142. package/src/errors/locals-access-error.ts +7 -0
  143. package/src/global/app-logger.d.ts +2 -0
  144. package/src/global/app-logger.js +6 -0
  145. package/src/global/app-logger.ts +4 -0
  146. 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
  147. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  148. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  149. package/src/hmr/client/hmr-runtime.d.ts +10 -0
  150. package/src/hmr/client/hmr-runtime.js +86 -0
  151. package/src/hmr/client/hmr-runtime.ts +121 -0
  152. package/src/hmr/hmr-strategy.d.ts +159 -0
  153. package/src/hmr/hmr-strategy.js +29 -0
  154. package/src/hmr/hmr-strategy.ts +172 -0
  155. package/src/hmr/hmr.test.e2e.d.ts +1 -0
  156. package/src/hmr/hmr.test.e2e.js +50 -0
  157. package/src/hmr/hmr.test.e2e.ts +75 -0
  158. package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
  159. package/src/hmr/strategies/default-hmr-strategy.js +34 -0
  160. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  161. package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
  162. package/src/hmr/strategies/js-hmr-strategy.js +179 -0
  163. package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
  164. package/src/index.browser.d.ts +3 -0
  165. package/src/index.browser.js +4 -0
  166. package/src/index.browser.ts +3 -0
  167. package/src/index.d.ts +5 -0
  168. package/src/index.js +10 -0
  169. package/src/index.ts +5 -0
  170. package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
  171. package/src/integrations/ghtml/ghtml-renderer.js +60 -0
  172. package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
  173. package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
  174. package/src/integrations/ghtml/ghtml.plugin.js +21 -0
  175. package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
  176. package/src/internal-types.d.ts +200 -0
  177. package/src/internal-types.js +0 -0
  178. package/src/internal-types.ts +212 -0
  179. package/src/plugins/alias-resolver-plugin.d.ts +2 -0
  180. package/src/plugins/alias-resolver-plugin.js +39 -0
  181. package/src/plugins/alias-resolver-plugin.ts +45 -0
  182. package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
  183. package/src/plugins/eco-component-meta-plugin.js +157 -0
  184. package/src/plugins/eco-component-meta-plugin.ts +474 -0
  185. package/src/plugins/integration-plugin.d.ts +102 -0
  186. package/src/plugins/integration-plugin.js +100 -0
  187. package/src/plugins/integration-plugin.ts +184 -0
  188. package/src/plugins/processor.d.ts +82 -0
  189. package/src/plugins/processor.js +122 -0
  190. package/src/plugins/processor.ts +220 -0
  191. package/src/public-types.d.ts +1094 -0
  192. package/src/public-types.js +0 -0
  193. package/src/public-types.ts +1255 -0
  194. package/src/route-renderer/GRAPH.md +387 -0
  195. package/src/route-renderer/README.md +135 -0
  196. package/src/route-renderer/component-graph-executor.d.ts +32 -0
  197. package/src/route-renderer/component-graph-executor.js +31 -0
  198. package/src/route-renderer/component-graph-executor.ts +84 -0
  199. package/src/route-renderer/component-graph.d.ts +42 -0
  200. package/src/route-renderer/component-graph.js +72 -0
  201. package/src/route-renderer/component-graph.ts +159 -0
  202. package/src/route-renderer/component-marker.d.ts +52 -0
  203. package/src/route-renderer/component-marker.js +46 -0
  204. package/src/route-renderer/component-marker.ts +117 -0
  205. package/src/route-renderer/dependency-resolver.d.ts +24 -0
  206. package/src/route-renderer/dependency-resolver.js +428 -0
  207. package/src/route-renderer/dependency-resolver.ts +596 -0
  208. package/src/route-renderer/html-post-processing.service.d.ts +40 -0
  209. package/src/route-renderer/html-post-processing.service.js +86 -0
  210. package/src/route-renderer/html-post-processing.service.ts +103 -0
  211. package/src/route-renderer/integration-renderer.d.ts +339 -0
  212. package/src/route-renderer/integration-renderer.js +526 -0
  213. package/src/route-renderer/integration-renderer.ts +696 -0
  214. package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
  215. package/src/route-renderer/marker-graph-resolver.js +93 -0
  216. package/src/route-renderer/marker-graph-resolver.ts +153 -0
  217. package/src/route-renderer/page-module-loader.d.ts +61 -0
  218. package/src/route-renderer/page-module-loader.js +102 -0
  219. package/src/route-renderer/page-module-loader.ts +153 -0
  220. package/src/route-renderer/render-execution.service.d.ts +69 -0
  221. package/src/route-renderer/render-execution.service.js +91 -0
  222. package/src/route-renderer/render-execution.service.ts +158 -0
  223. package/src/route-renderer/render-preparation.service.d.ts +112 -0
  224. package/src/route-renderer/render-preparation.service.js +243 -0
  225. package/src/route-renderer/render-preparation.service.ts +358 -0
  226. package/src/route-renderer/route-renderer.d.ts +26 -0
  227. package/src/route-renderer/route-renderer.js +68 -0
  228. package/src/route-renderer/route-renderer.ts +80 -0
  229. package/src/router/fs-router-scanner.d.ts +41 -0
  230. package/src/router/fs-router-scanner.js +155 -0
  231. package/src/router/fs-router-scanner.ts +217 -0
  232. package/src/router/fs-router.d.ts +26 -0
  233. package/src/router/fs-router.js +100 -0
  234. package/src/router/fs-router.ts +122 -0
  235. package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
  236. package/src/services/asset-processing-service/asset-processing.service.js +250 -0
  237. package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
  238. package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
  239. package/src/services/asset-processing-service/asset.factory.js +82 -0
  240. package/src/services/asset-processing-service/asset.factory.ts +105 -0
  241. package/src/services/asset-processing-service/assets.types.d.ts +88 -0
  242. package/src/services/asset-processing-service/assets.types.js +0 -0
  243. package/src/services/asset-processing-service/assets.types.ts +112 -0
  244. package/src/services/asset-processing-service/index.d.ts +3 -0
  245. package/src/services/asset-processing-service/index.js +3 -0
  246. package/src/services/asset-processing-service/index.ts +3 -0
  247. package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
  248. package/src/services/asset-processing-service/processor.interface.js +6 -0
  249. package/src/services/asset-processing-service/processor.interface.ts +27 -0
  250. package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
  251. package/src/services/asset-processing-service/processor.registry.js +15 -0
  252. package/src/services/asset-processing-service/processor.registry.ts +18 -0
  253. package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
  254. package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
  255. package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
  256. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
  257. package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
  258. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
  259. package/src/services/asset-processing-service/processors/index.d.ts +5 -0
  260. package/src/services/asset-processing-service/processors/index.js +5 -0
  261. package/src/services/asset-processing-service/processors/index.ts +5 -0
  262. package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
  263. package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
  264. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
  265. package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
  266. package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
  267. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
  268. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
  269. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
  270. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
  271. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
  272. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
  273. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
  274. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
  275. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
  276. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
  277. package/src/services/cache/cache.types.d.ts +107 -0
  278. package/src/services/cache/cache.types.js +0 -0
  279. package/src/services/cache/cache.types.ts +126 -0
  280. package/src/services/cache/index.d.ts +7 -0
  281. package/src/services/cache/index.js +7 -0
  282. package/src/services/cache/index.ts +18 -0
  283. package/src/services/cache/memory-cache-store.d.ts +42 -0
  284. package/src/services/cache/memory-cache-store.js +98 -0
  285. package/src/services/cache/memory-cache-store.ts +130 -0
  286. package/src/services/cache/page-cache-service.d.ts +70 -0
  287. package/src/services/cache/page-cache-service.js +152 -0
  288. package/src/services/cache/page-cache-service.ts +202 -0
  289. package/src/services/html-transformer.service.d.ts +50 -0
  290. package/src/services/html-transformer.service.js +163 -0
  291. package/src/services/html-transformer.service.ts +217 -0
  292. package/src/services/page-module-import.service.d.ts +37 -0
  293. package/src/services/page-module-import.service.js +88 -0
  294. package/src/services/page-module-import.service.ts +129 -0
  295. package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
  296. package/src/services/page-request-cache-coordinator.service.js +107 -0
  297. package/src/services/page-request-cache-coordinator.service.ts +128 -0
  298. package/src/services/schema-validation-service.d.ts +122 -0
  299. package/src/services/schema-validation-service.js +101 -0
  300. package/src/services/schema-validation-service.ts +204 -0
  301. package/src/services/validation/standard-schema.types.d.ts +65 -0
  302. package/src/services/validation/standard-schema.types.js +0 -0
  303. package/src/services/validation/standard-schema.types.ts +68 -0
  304. package/src/static-site-generator/static-site-generator.d.ts +57 -0
  305. package/src/static-site-generator/static-site-generator.js +272 -0
  306. package/src/static-site-generator/static-site-generator.ts +359 -0
  307. package/src/utils/css.d.ts +1 -0
  308. package/src/utils/css.js +7 -0
  309. package/src/utils/css.ts +5 -0
  310. package/src/utils/deep-merge.d.ts +14 -0
  311. package/src/utils/deep-merge.js +32 -0
  312. package/src/utils/deep-merge.ts +47 -0
  313. package/src/utils/hash.d.ts +1 -0
  314. package/src/utils/hash.js +7 -0
  315. package/src/utils/hash.ts +5 -0
  316. package/src/utils/html.d.ts +1 -0
  317. package/src/utils/html.js +4 -0
  318. package/src/utils/html.ts +1 -0
  319. package/src/utils/invariant.d.ts +5 -0
  320. package/src/utils/invariant.js +11 -0
  321. package/src/utils/invariant.ts +15 -0
  322. package/src/utils/locals-utils.d.ts +15 -0
  323. package/src/utils/locals-utils.js +24 -0
  324. package/src/utils/locals-utils.ts +37 -0
  325. package/src/utils/parse-cli-args.d.ts +24 -0
  326. package/src/utils/parse-cli-args.js +47 -0
  327. package/src/utils/parse-cli-args.ts +83 -0
  328. package/src/utils/path-utils.module.d.ts +5 -0
  329. package/src/utils/path-utils.module.js +14 -0
  330. package/src/utils/path-utils.module.ts +14 -0
  331. package/src/utils/runtime.d.ts +11 -0
  332. package/src/utils/runtime.js +40 -0
  333. package/src/utils/runtime.ts +44 -0
  334. package/src/utils/server-utils.module.d.ts +19 -0
  335. package/src/utils/server-utils.module.js +56 -0
  336. package/src/utils/server-utils.module.ts +67 -0
  337. package/src/watchers/project-watcher.d.ts +120 -0
  338. package/src/watchers/project-watcher.js +238 -0
  339. package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
  340. package/src/watchers/project-watcher.test-helpers.js +51 -0
  341. package/src/watchers/project-watcher.test-helpers.ts +40 -0
  342. 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
+ };