@ecopages/core 0.2.0-alpha.5 → 0.2.0-alpha.51

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 (456) hide show
  1. package/README.md +267 -14
  2. package/package.json +98 -108
  3. package/src/adapters/README.md +39 -0
  4. package/src/adapters/abstract/application-adapter.d.ts +28 -2
  5. package/src/adapters/abstract/application-adapter.js +14 -2
  6. package/src/adapters/abstract/router-adapter.d.ts +1 -1
  7. package/src/adapters/abstract/server-adapter.d.ts +2 -2
  8. package/src/adapters/bun/client-bridge.d.ts +1 -1
  9. package/src/adapters/bun/create-app.d.ts +12 -12
  10. package/src/adapters/bun/create-app.js +64 -41
  11. package/src/adapters/bun/hmr-manager.d.ts +30 -62
  12. package/src/adapters/bun/hmr-manager.js +30 -194
  13. package/src/adapters/bun/index.d.ts +2 -3
  14. package/src/adapters/bun/index.js +3 -3
  15. package/src/adapters/bun/runtime-host.d.ts +52 -0
  16. package/src/adapters/bun/runtime-host.js +56 -0
  17. package/src/adapters/bun/server-adapter.d.ts +93 -32
  18. package/src/adapters/bun/server-adapter.js +166 -89
  19. package/src/adapters/bun/static-preview-host.d.ts +28 -0
  20. package/src/adapters/bun/static-preview-host.js +45 -0
  21. package/src/{create-app.d.ts → adapters/create-app.d.ts} +9 -6
  22. package/src/{create-app.js → adapters/create-app.js} +4 -4
  23. package/src/adapters/index.d.ts +2 -6
  24. package/src/adapters/index.js +2 -8
  25. package/src/adapters/node/create-app.d.ts +15 -12
  26. package/src/adapters/node/create-app.js +34 -85
  27. package/src/adapters/node/http-request-bridge.d.ts +57 -0
  28. package/src/adapters/node/http-request-bridge.js +118 -0
  29. package/src/adapters/node/node-client-bridge.d.ts +1 -1
  30. package/src/adapters/node/node-hmr-manager.d.ts +38 -48
  31. package/src/adapters/node/node-hmr-manager.js +31 -203
  32. package/src/adapters/node/runtime-host.d.ts +57 -0
  33. package/src/adapters/node/runtime-host.js +92 -0
  34. package/src/adapters/node/server-adapter-dependencies.d.ts +19 -0
  35. package/src/adapters/node/server-adapter-dependencies.js +18 -0
  36. package/src/adapters/node/server-adapter.d.ts +20 -72
  37. package/src/adapters/node/server-adapter.js +98 -203
  38. package/src/adapters/node/static-content-server.d.ts +37 -1
  39. package/src/adapters/node/static-content-server.js +29 -1
  40. package/src/adapters/node/static-preview-host.d.ts +55 -0
  41. package/src/adapters/node/static-preview-host.js +68 -0
  42. package/src/adapters/shared/application-adapter.d.ts +1 -1
  43. package/src/{define-api-handler.d.ts → adapters/shared/define-api-handler.d.ts} +1 -1
  44. package/src/adapters/shared/explicit-static-render-preparation.d.ts +25 -0
  45. package/src/adapters/shared/explicit-static-render-preparation.js +26 -0
  46. package/src/adapters/shared/explicit-static-route-matcher.d.ts +7 -4
  47. package/src/adapters/shared/explicit-static-route-matcher.js +14 -13
  48. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +7 -10
  49. package/src/adapters/shared/file-route-middleware-pipeline.js +3 -11
  50. package/src/adapters/shared/fs-server-response-factory.d.ts +14 -10
  51. package/src/adapters/shared/fs-server-response-factory.js +11 -27
  52. package/src/adapters/shared/fs-server-response-matcher.d.ts +20 -16
  53. package/src/adapters/shared/fs-server-response-matcher.js +76 -45
  54. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
  55. package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
  56. package/src/adapters/shared/hmr-html-response.d.ts +22 -0
  57. package/src/adapters/shared/hmr-html-response.js +33 -0
  58. package/src/adapters/shared/render-context.d.ts +5 -3
  59. package/src/adapters/shared/render-context.js +27 -3
  60. package/src/adapters/shared/runtime-app-bootstrap.d.ts +26 -0
  61. package/src/adapters/shared/runtime-app-bootstrap.js +46 -0
  62. package/src/adapters/shared/runtime-host.d.ts +12 -0
  63. package/src/adapters/shared/server-adapter.d.ts +33 -12
  64. package/src/adapters/shared/server-adapter.js +215 -132
  65. package/src/adapters/shared/server-route-handler.d.ts +5 -5
  66. package/src/adapters/shared/server-route-handler.js +7 -16
  67. package/src/adapters/shared/server-static-builder.d.ts +41 -8
  68. package/src/adapters/shared/server-static-builder.js +65 -11
  69. package/src/adapters/shared/shared-hmr-manager.d.ts +59 -0
  70. package/src/adapters/shared/shared-hmr-manager.js +240 -0
  71. package/src/adapters/shared/static-preview-host.d.ts +10 -0
  72. package/src/build/README.md +107 -0
  73. package/src/build/browser-runtime-import-rewrite-plugin.d.ts +26 -0
  74. package/src/build/browser-runtime-import-rewrite-plugin.js +162 -0
  75. package/src/build/browser-runtime-manifest.d.ts +31 -0
  76. package/src/build/browser-runtime-manifest.js +61 -0
  77. package/src/build/build-adapter.d.ts +175 -3
  78. package/src/build/build-adapter.js +633 -16
  79. package/src/build/build-manifest.d.ts +33 -0
  80. package/src/build/build-manifest.js +52 -0
  81. package/src/build/dev-build-coordinator.d.ts +72 -0
  82. package/src/build/dev-build-coordinator.js +154 -0
  83. package/src/build/esbuild-build-adapter.d.ts +16 -6
  84. package/src/build/esbuild-build-adapter.js +205 -75
  85. package/src/build/runtime-build-executor.d.ts +14 -0
  86. package/src/build/runtime-build-executor.js +22 -0
  87. package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
  88. package/src/build/runtime-specifier-alias-plugin.js +31 -0
  89. package/src/config/README.md +36 -0
  90. package/src/config/config-builder.d.ts +54 -29
  91. package/src/config/config-builder.js +256 -50
  92. package/src/{constants.d.ts → config/constants.d.ts} +13 -0
  93. package/src/{constants.js → config/constants.js} +4 -0
  94. package/src/declarations.d.ts +19 -14
  95. package/src/dev/host-runtime.d.ts +10 -0
  96. package/src/dev/host-runtime.js +24 -0
  97. package/src/dev/sc-server.d.ts +1 -1
  98. package/src/dev/sc-server.js +1 -1
  99. package/src/eco/README.md +70 -16
  100. package/src/eco/eco.browser.d.ts +2 -0
  101. package/src/eco/eco.browser.js +88 -0
  102. package/src/eco/eco.js +63 -54
  103. package/src/eco/eco.types.d.ts +69 -6
  104. package/src/eco/eco.utils.d.ts +1 -40
  105. package/src/eco/eco.utils.js +5 -35
  106. package/src/eco/global-injector-map.d.ts +3 -3
  107. package/src/eco/global-injector-map.js +2 -2
  108. package/src/eco/lazy-injector-map.d.ts +2 -2
  109. package/src/errors/index.d.ts +1 -0
  110. package/src/errors/index.js +3 -1
  111. package/src/hmr/README.md +26 -0
  112. package/src/hmr/client/hmr-runtime.d.ts +1 -6
  113. package/src/hmr/client/hmr-runtime.js +38 -7
  114. package/src/hmr/hmr-strategy.d.ts +16 -13
  115. package/src/hmr/hmr-strategy.js +22 -7
  116. package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
  117. package/src/hmr/hmr.postcss.test.e2e.js +31 -0
  118. package/src/hmr/hmr.test.e2e.js +26 -33
  119. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  120. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  121. package/src/hmr/strategies/js-hmr-strategy.d.ts +40 -42
  122. package/src/hmr/strategies/js-hmr-strategy.js +24 -43
  123. package/src/index.browser.d.ts +2 -2
  124. package/src/index.browser.js +1 -1
  125. package/src/index.d.ts +4 -3
  126. package/src/index.js +16 -5
  127. package/src/integrations/ghtml/ghtml-renderer.d.ts +3 -2
  128. package/src/integrations/ghtml/ghtml-renderer.js +27 -30
  129. package/src/integrations/ghtml/ghtml.constants.d.ts +1 -0
  130. package/src/integrations/ghtml/ghtml.constants.js +4 -0
  131. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -6
  132. package/src/integrations/ghtml/ghtml.plugin.js +3 -4
  133. package/src/plugins/README.md +35 -0
  134. package/src/plugins/alias-resolver-plugin.d.ts +1 -0
  135. package/src/plugins/alias-resolver-plugin.js +27 -5
  136. package/src/plugins/eco-component-meta-plugin.d.ts +14 -1
  137. package/src/plugins/eco-component-meta-plugin.js +42 -24
  138. package/src/plugins/foreign-jsx-override-plugin.d.ts +33 -0
  139. package/src/plugins/foreign-jsx-override-plugin.js +41 -0
  140. package/src/plugins/integration-plugin.d.ts +157 -29
  141. package/src/plugins/integration-plugin.js +115 -14
  142. package/src/plugins/processor.d.ts +17 -2
  143. package/src/plugins/processor.js +22 -3
  144. package/src/plugins/runtime-capability.d.ts +9 -0
  145. package/src/plugins/source-transform.d.ts +46 -0
  146. package/src/plugins/source-transform.js +71 -0
  147. package/src/route-renderer/GRAPH.md +83 -325
  148. package/src/route-renderer/README.md +73 -90
  149. package/src/route-renderer/orchestration/component-render-context.d.ts +97 -0
  150. package/src/route-renderer/orchestration/component-render-context.js +147 -0
  151. package/src/route-renderer/orchestration/declared-ownership-graph.d.ts +18 -0
  152. package/src/route-renderer/orchestration/declared-ownership-graph.js +34 -0
  153. package/src/route-renderer/orchestration/foreign-subtree-execution.service.d.ts +110 -0
  154. package/src/route-renderer/orchestration/foreign-subtree-execution.service.js +233 -0
  155. package/src/route-renderer/orchestration/integration-renderer.d.ts +534 -0
  156. package/src/route-renderer/orchestration/integration-renderer.js +991 -0
  157. package/src/route-renderer/orchestration/ownership-planning.service.d.ts +24 -0
  158. package/src/route-renderer/orchestration/ownership-planning.service.js +63 -0
  159. package/src/route-renderer/orchestration/ownership-validation.service.d.ts +29 -0
  160. package/src/route-renderer/orchestration/ownership-validation.service.js +53 -0
  161. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +3 -0
  162. package/src/route-renderer/orchestration/processed-asset-dedupe.js +27 -0
  163. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.d.ts +91 -0
  164. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.js +170 -0
  165. package/src/route-renderer/orchestration/render-output.utils.d.ts +66 -0
  166. package/src/route-renderer/orchestration/render-output.utils.js +171 -0
  167. package/src/route-renderer/orchestration/route-render-orchestrator.d.ts +156 -0
  168. package/src/route-renderer/orchestration/route-render-orchestrator.js +577 -0
  169. package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
  170. package/src/route-renderer/orchestration/template-serialization.js +45 -0
  171. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +37 -0
  172. package/src/route-renderer/page-loading/component-dependency-collection.js +132 -0
  173. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +24 -0
  174. package/src/route-renderer/page-loading/declared-asset-collection.js +106 -0
  175. package/src/route-renderer/{dependency-resolver.d.ts → page-loading/dependency-resolver.d.ts} +15 -4
  176. package/src/route-renderer/page-loading/dependency-resolver.js +115 -0
  177. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +11 -0
  178. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +57 -0
  179. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +45 -0
  180. package/src/route-renderer/page-loading/lazy-entry-collection.js +105 -0
  181. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +19 -0
  182. package/src/route-renderer/page-loading/lazy-trigger-planning.js +40 -0
  183. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +5 -0
  184. package/src/route-renderer/page-loading/module-declaration-aggregation.js +33 -0
  185. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +3 -0
  186. package/src/route-renderer/page-loading/module-declaration-scripts.js +18 -0
  187. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +13 -0
  188. package/src/route-renderer/page-loading/page-dependency-bundling.js +137 -0
  189. package/src/route-renderer/page-loading/page-module-loader.d.ts +90 -0
  190. package/src/route-renderer/{page-module-loader.js → page-loading/page-module-loader.js} +39 -14
  191. package/src/route-renderer/route-renderer.d.ts +57 -14
  192. package/src/route-renderer/route-renderer.js +30 -18
  193. package/src/router/README.md +94 -0
  194. package/src/router/client/link-intent.d.ts +53 -0
  195. package/src/router/client/link-intent.js +34 -0
  196. package/src/router/client/link-intent.test.browser.d.ts +1 -0
  197. package/src/router/client/link-intent.test.browser.js +43 -0
  198. package/src/router/client/navigation-coordinator.d.ts +169 -0
  199. package/src/router/client/navigation-coordinator.js +215 -0
  200. package/src/router/server/route-registry.d.ts +78 -0
  201. package/src/router/server/route-registry.js +262 -0
  202. package/src/services/README.md +28 -0
  203. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +3 -0
  204. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +56 -0
  205. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +103 -0
  206. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.js +124 -89
  207. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.d.ts +1 -1
  208. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.js +2 -2
  209. package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.d.ts +16 -1
  210. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  211. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +57 -0
  212. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +49 -0
  213. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
  214. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
  215. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +30 -0
  216. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +65 -0
  217. package/src/services/assets/asset-processing-service/index.d.ts +6 -0
  218. package/src/services/assets/asset-processing-service/index.js +6 -0
  219. package/src/services/assets/asset-processing-service/page-package.d.ts +6 -0
  220. package/src/services/assets/asset-processing-service/page-package.js +80 -0
  221. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.d.ts +2 -2
  222. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.d.ts +2 -2
  223. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.d.ts +1 -1
  224. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.js +11 -5
  225. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +22 -0
  226. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +136 -0
  227. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
  228. package/src/services/assets/asset-processing-service/processors/index.js +5 -0
  229. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.d.ts +3 -2
  230. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +120 -0
  231. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.d.ts +4 -3
  232. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.js +28 -7
  233. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +42 -0
  234. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +126 -0
  235. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.d.ts +5 -2
  236. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +59 -0
  237. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  238. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.js +9 -3
  239. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +18 -0
  240. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +45 -0
  241. package/src/services/assets/browser-bundle.service.d.ts +73 -0
  242. package/src/services/assets/browser-bundle.service.js +41 -0
  243. package/src/services/{page-request-cache-coordinator.service.d.ts → cache/page-request-cache-coordinator.service.d.ts} +2 -2
  244. package/src/services/{page-request-cache-coordinator.service.js → cache/page-request-cache-coordinator.service.js} +3 -1
  245. package/src/services/html/html-rewriter-provider.service.d.ts +40 -0
  246. package/src/services/html/html-rewriter-provider.service.js +68 -0
  247. package/src/services/html/html-transformer.service.d.ts +96 -0
  248. package/src/services/html/html-transformer.service.js +287 -0
  249. package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
  250. package/src/services/invalidation/development-invalidation.service.js +190 -0
  251. package/src/services/module-loading/app-module-loader.service.d.ts +7 -0
  252. package/src/services/module-loading/app-module-loader.service.js +0 -0
  253. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +24 -0
  254. package/src/services/module-loading/app-server-module-transpiler.service.js +151 -0
  255. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  256. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  257. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  258. package/src/services/module-loading/module-loading-types.js +0 -0
  259. package/src/services/module-loading/node-bootstrap-plugin.d.ts +38 -0
  260. package/src/services/module-loading/node-bootstrap-plugin.js +215 -0
  261. package/src/services/module-loading/page-module-import.service.d.ts +95 -0
  262. package/src/services/module-loading/page-module-import.service.js +191 -0
  263. package/src/services/module-loading/server-module-transpiler.service.d.ts +63 -0
  264. package/src/services/module-loading/server-module-transpiler.service.js +64 -0
  265. package/src/services/module-loading/source-module-support.d.ts +5 -0
  266. package/src/services/module-loading/source-module-support.js +8 -0
  267. package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
  268. package/src/services/runtime-state/dev-graph.service.js +162 -0
  269. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
  270. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
  271. package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
  272. package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
  273. package/src/services/{schema-validation-service.d.ts → validation/schema-validation-service.d.ts} +1 -1
  274. package/src/static-site-generator/README.md +26 -0
  275. package/src/static-site-generator/static-site-generator.d.ts +67 -20
  276. package/src/static-site-generator/static-site-generator.js +182 -138
  277. package/src/{internal-types.d.ts → types/internal-types.d.ts} +62 -30
  278. package/src/types/internal-types.js +0 -0
  279. package/src/{public-types.d.ts → types/public-types.d.ts} +239 -31
  280. package/src/types/public-types.js +0 -0
  281. package/src/utils/html-escaping.d.ts +7 -0
  282. package/src/utils/html-escaping.js +6 -0
  283. package/src/utils/locals-utils.d.ts +1 -1
  284. package/src/utils/parse-cli-args.d.ts +4 -1
  285. package/src/utils/parse-cli-args.js +16 -1
  286. package/src/utils/resolve-work-dir.d.ts +11 -0
  287. package/src/utils/resolve-work-dir.js +31 -0
  288. package/src/watchers/project-watcher.d.ts +11 -7
  289. package/src/watchers/project-watcher.js +69 -75
  290. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  291. package/src/watchers/project-watcher.test-helpers.js +6 -5
  292. package/CHANGELOG.md +0 -94
  293. package/src/adapters/abstract/application-adapter.ts +0 -337
  294. package/src/adapters/abstract/router-adapter.ts +0 -30
  295. package/src/adapters/abstract/server-adapter.ts +0 -79
  296. package/src/adapters/bun/client-bridge.ts +0 -62
  297. package/src/adapters/bun/create-app.ts +0 -189
  298. package/src/adapters/bun/define-api-handler.d.ts +0 -61
  299. package/src/adapters/bun/define-api-handler.ts +0 -114
  300. package/src/adapters/bun/hmr-manager.ts +0 -276
  301. package/src/adapters/bun/index.ts +0 -3
  302. package/src/adapters/bun/server-adapter.ts +0 -492
  303. package/src/adapters/bun/server-lifecycle.d.ts +0 -52
  304. package/src/adapters/bun/server-lifecycle.js +0 -120
  305. package/src/adapters/bun/server-lifecycle.ts +0 -154
  306. package/src/adapters/index.ts +0 -6
  307. package/src/adapters/node/create-app.ts +0 -179
  308. package/src/adapters/node/index.d.ts +0 -4
  309. package/src/adapters/node/index.js +0 -8
  310. package/src/adapters/node/index.ts +0 -9
  311. package/src/adapters/node/node-client-bridge.ts +0 -79
  312. package/src/adapters/node/node-hmr-manager.ts +0 -271
  313. package/src/adapters/node/server-adapter.ts +0 -561
  314. package/src/adapters/node/static-content-server.ts +0 -203
  315. package/src/adapters/shared/api-response.ts +0 -104
  316. package/src/adapters/shared/application-adapter.ts +0 -199
  317. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -134
  318. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -123
  319. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  320. package/src/adapters/shared/fs-server-response-matcher.ts +0 -198
  321. package/src/adapters/shared/render-context.ts +0 -105
  322. package/src/adapters/shared/server-adapter.ts +0 -442
  323. package/src/adapters/shared/server-route-handler.ts +0 -166
  324. package/src/adapters/shared/server-static-builder.ts +0 -82
  325. package/src/build/build-adapter.ts +0 -133
  326. package/src/build/build-types.ts +0 -83
  327. package/src/build/esbuild-build-adapter.ts +0 -511
  328. package/src/config/config-builder.ts +0 -474
  329. package/src/constants.ts +0 -39
  330. package/src/create-app.ts +0 -87
  331. package/src/define-api-handler.js +0 -15
  332. package/src/define-api-handler.ts +0 -66
  333. package/src/dev/sc-server.ts +0 -143
  334. package/src/eco/component-render-context.d.ts +0 -105
  335. package/src/eco/component-render-context.js +0 -77
  336. package/src/eco/component-render-context.ts +0 -202
  337. package/src/eco/eco.ts +0 -221
  338. package/src/eco/eco.types.ts +0 -202
  339. package/src/eco/eco.utils.ts +0 -89
  340. package/src/eco/global-injector-map.ts +0 -112
  341. package/src/eco/lazy-injector-map.ts +0 -120
  342. package/src/eco/module-dependencies.ts +0 -75
  343. package/src/errors/http-error.ts +0 -72
  344. package/src/errors/index.ts +0 -2
  345. package/src/errors/locals-access-error.ts +0 -7
  346. package/src/global/app-logger.ts +0 -4
  347. package/src/hmr/client/hmr-runtime.ts +0 -121
  348. package/src/hmr/hmr-strategy.ts +0 -172
  349. package/src/hmr/hmr.test.e2e.ts +0 -75
  350. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  351. package/src/hmr/strategies/js-hmr-strategy.ts +0 -327
  352. package/src/index.browser.ts +0 -3
  353. package/src/index.ts +0 -5
  354. package/src/integrations/ghtml/ghtml-renderer.ts +0 -93
  355. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  356. package/src/internal-types.ts +0 -212
  357. package/src/plugins/alias-resolver-plugin.ts +0 -45
  358. package/src/plugins/eco-component-meta-plugin.ts +0 -474
  359. package/src/plugins/integration-plugin.ts +0 -184
  360. package/src/plugins/processor.ts +0 -220
  361. package/src/public-types.ts +0 -1255
  362. package/src/route-renderer/component-graph-executor.d.ts +0 -32
  363. package/src/route-renderer/component-graph-executor.js +0 -31
  364. package/src/route-renderer/component-graph-executor.ts +0 -84
  365. package/src/route-renderer/component-graph.d.ts +0 -42
  366. package/src/route-renderer/component-graph.js +0 -72
  367. package/src/route-renderer/component-graph.ts +0 -159
  368. package/src/route-renderer/component-marker.d.ts +0 -52
  369. package/src/route-renderer/component-marker.js +0 -46
  370. package/src/route-renderer/component-marker.ts +0 -117
  371. package/src/route-renderer/dependency-resolver.js +0 -428
  372. package/src/route-renderer/dependency-resolver.ts +0 -596
  373. package/src/route-renderer/html-post-processing.service.d.ts +0 -40
  374. package/src/route-renderer/html-post-processing.service.js +0 -86
  375. package/src/route-renderer/html-post-processing.service.ts +0 -103
  376. package/src/route-renderer/integration-renderer.d.ts +0 -339
  377. package/src/route-renderer/integration-renderer.js +0 -526
  378. package/src/route-renderer/integration-renderer.ts +0 -696
  379. package/src/route-renderer/marker-graph-resolver.d.ts +0 -76
  380. package/src/route-renderer/marker-graph-resolver.js +0 -93
  381. package/src/route-renderer/marker-graph-resolver.ts +0 -153
  382. package/src/route-renderer/page-module-loader.d.ts +0 -61
  383. package/src/route-renderer/page-module-loader.ts +0 -153
  384. package/src/route-renderer/render-execution.service.d.ts +0 -69
  385. package/src/route-renderer/render-execution.service.js +0 -91
  386. package/src/route-renderer/render-execution.service.ts +0 -158
  387. package/src/route-renderer/render-preparation.service.d.ts +0 -112
  388. package/src/route-renderer/render-preparation.service.js +0 -243
  389. package/src/route-renderer/render-preparation.service.ts +0 -358
  390. package/src/route-renderer/route-renderer.ts +0 -80
  391. package/src/router/fs-router-scanner.d.ts +0 -41
  392. package/src/router/fs-router-scanner.js +0 -155
  393. package/src/router/fs-router-scanner.ts +0 -217
  394. package/src/router/fs-router.d.ts +0 -26
  395. package/src/router/fs-router.js +0 -100
  396. package/src/router/fs-router.ts +0 -122
  397. package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
  398. package/src/services/asset-processing-service/asset-processing.service.ts +0 -306
  399. package/src/services/asset-processing-service/asset.factory.ts +0 -105
  400. package/src/services/asset-processing-service/assets.types.ts +0 -112
  401. package/src/services/asset-processing-service/index.d.ts +0 -3
  402. package/src/services/asset-processing-service/index.js +0 -3
  403. package/src/services/asset-processing-service/index.ts +0 -3
  404. package/src/services/asset-processing-service/processor.interface.ts +0 -27
  405. package/src/services/asset-processing-service/processor.registry.ts +0 -18
  406. package/src/services/asset-processing-service/processors/base/base-processor.ts +0 -76
  407. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +0 -16
  408. package/src/services/asset-processing-service/processors/base/base-script-processor.js +0 -80
  409. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +0 -105
  410. package/src/services/asset-processing-service/processors/index.d.ts +0 -5
  411. package/src/services/asset-processing-service/processors/index.js +0 -5
  412. package/src/services/asset-processing-service/processors/index.ts +0 -5
  413. package/src/services/asset-processing-service/processors/script/content-script.processor.js +0 -57
  414. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  415. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  416. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  417. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +0 -74
  418. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +0 -84
  419. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
  420. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  421. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -77
  422. package/src/services/cache/cache.types.ts +0 -126
  423. package/src/services/cache/index.ts +0 -18
  424. package/src/services/cache/memory-cache-store.ts +0 -130
  425. package/src/services/cache/page-cache-service.ts +0 -202
  426. package/src/services/html-transformer.service.d.ts +0 -50
  427. package/src/services/html-transformer.service.js +0 -163
  428. package/src/services/html-transformer.service.ts +0 -217
  429. package/src/services/page-module-import.service.d.ts +0 -37
  430. package/src/services/page-module-import.service.js +0 -88
  431. package/src/services/page-module-import.service.ts +0 -129
  432. package/src/services/page-request-cache-coordinator.service.ts +0 -128
  433. package/src/services/schema-validation-service.ts +0 -204
  434. package/src/services/validation/standard-schema.types.ts +0 -68
  435. package/src/static-site-generator/static-site-generator.ts +0 -359
  436. package/src/utils/css.d.ts +0 -1
  437. package/src/utils/css.js +0 -7
  438. package/src/utils/css.ts +0 -5
  439. package/src/utils/deep-merge.ts +0 -47
  440. package/src/utils/hash.ts +0 -5
  441. package/src/utils/html.ts +0 -1
  442. package/src/utils/invariant.ts +0 -15
  443. package/src/utils/locals-utils.ts +0 -37
  444. package/src/utils/parse-cli-args.ts +0 -83
  445. package/src/utils/path-utils.module.ts +0 -14
  446. package/src/utils/runtime.ts +0 -44
  447. package/src/utils/server-utils.module.ts +0 -67
  448. package/src/watchers/project-watcher.test-helpers.ts +0 -40
  449. package/src/watchers/project-watcher.ts +0 -364
  450. /package/src/adapters/{bun → shared}/define-api-handler.js +0 -0
  451. /package/src/{internal-types.js → adapters/shared/runtime-host.js} +0 -0
  452. /package/src/{public-types.js → adapters/shared/static-preview-host.js} +0 -0
  453. /package/src/{services/asset-processing-service/assets.types.js → plugins/runtime-capability.js} +0 -0
  454. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.js +0 -0
  455. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.js +0 -0
  456. /package/src/services/{schema-validation-service.js → validation/schema-validation-service.js} +0 -0
@@ -0,0 +1,991 @@
1
+ import {
2
+ createPagePackage
3
+ } from "../../services/assets/asset-processing-service/index.js";
4
+ import { HtmlTransformerService } from "../../services/html/html-transformer.service.js";
5
+ import { invariant } from "../../utils/invariant.js";
6
+ import { HttpError } from "../../errors/http-error.js";
7
+ import { DependencyResolverService } from "../page-loading/dependency-resolver.js";
8
+ import { PageModuleLoaderService } from "../page-loading/page-module-loader.js";
9
+ import { OwnershipValidationService } from "./ownership-validation.service.js";
10
+ import {
11
+ RouteRenderOrchestrator
12
+ } from "./route-render-orchestrator.js";
13
+ import { normalizeUnresolvedMarkerArtifactHtml } from "./render-output.utils.js";
14
+ import {
15
+ ForeignSubtreeExecutionService
16
+ } from "./foreign-subtree-execution.service.js";
17
+ import {} from "./queued-foreign-subtree-resolution.service.js";
18
+ import { buildProcessedAssetDedupeKey } from "./processed-asset-dedupe.js";
19
+ function isMarkupNodeLike(value) {
20
+ return typeof value === "object" && value !== null && "nodeType" in value && typeof value.nodeType === "number" && "outerHTML" in value && typeof value.outerHTML === "string";
21
+ }
22
+ class IntegrationRenderer {
23
+ appConfig;
24
+ assetProcessingService;
25
+ htmlTransformer;
26
+ hmrManager;
27
+ resolvedIntegrationDependencies = [];
28
+ rendererModules;
29
+ runtimeOrigin;
30
+ dependencyResolverService;
31
+ pageModuleLoaderService;
32
+ routeRenderOrchestrator;
33
+ foreignSubtreeExecutionService = new ForeignSubtreeExecutionService();
34
+ DOC_TYPE = "<!DOCTYPE html>";
35
+ /**
36
+ * Loads one route module through the owning renderer's import path.
37
+ *
38
+ * Request-time infrastructure may need page metadata such as cache strategy or
39
+ * middleware before full rendering starts. Exposing this narrow entrypoint lets
40
+ * those callers reuse integration-specific import setup instead of bypassing it
41
+ * with raw transpiler access.
42
+ */
43
+ async loadPageModule(file, options) {
44
+ return this.importPageFile(file, options);
45
+ }
46
+ /**
47
+ * Reads the execution-scoped owning-renderer cache from one render input.
48
+ *
49
+ * Shared page/layout/document shell helpers pass one cache through
50
+ * `integrationContext` so repeated delegation to the same foreign integration
51
+ * can reuse a single initialized renderer instance during one render flow.
52
+ * The cache is deliberately scoped to the current render execution rather than
53
+ * stored on the renderer, which avoids leaking mutable integration state across
54
+ * requests while still preventing redundant renderer initialization.
55
+ *
56
+ * @param integrationContext - Optional render context carried with one render input.
57
+ * @returns The current execution cache when present.
58
+ */
59
+ getOwningRendererCache(integrationContext) {
60
+ if (integrationContext?.rendererCache instanceof Map) {
61
+ return integrationContext.rendererCache;
62
+ }
63
+ return void 0;
64
+ }
65
+ getForeignOwnerIntegrationName(component) {
66
+ const integrationName = component.config?.integration ?? component.config?.__eco?.integration;
67
+ if (!integrationName || integrationName === this.name) {
68
+ return void 0;
69
+ }
70
+ return this.appConfig.integrations.some((integration) => integration.name === integrationName) ? integrationName : void 0;
71
+ }
72
+ /**
73
+ * Attaches an execution-scoped owning-renderer cache to one render input.
74
+ *
75
+ * Foreign-owned page, layout, or document shells may delegate several times in
76
+ * the same render flow. Threading the cache through `integrationContext`
77
+ * preserves renderer reuse without changing the public render input contract.
78
+ * Existing integration-specific context is preserved and augmented.
79
+ *
80
+ * @param input - Original render input.
81
+ * @param rendererCache - Execution-scoped renderer cache to propagate.
82
+ * @returns Render input augmented with the shared renderer cache.
83
+ */
84
+ withOwningRendererCache(input, rendererCache) {
85
+ const integrationContext = input.integrationContext;
86
+ const sharedRendererCache = rendererCache;
87
+ return {
88
+ ...input,
89
+ integrationContext: integrationContext ? { ...integrationContext, rendererCache: sharedRendererCache } : { rendererCache: sharedRendererCache }
90
+ };
91
+ }
92
+ getRendererModuleValue(key) {
93
+ if (!this.rendererModules || typeof this.rendererModules !== "object") {
94
+ return void 0;
95
+ }
96
+ return this.rendererModules[key];
97
+ }
98
+ getRendererModuleString(key) {
99
+ const value = this.getRendererModuleValue(key);
100
+ return typeof value === "string" && value.length > 0 ? value : void 0;
101
+ }
102
+ getRendererBootstrapDependencies(partial = false) {
103
+ if (partial) {
104
+ return [];
105
+ }
106
+ const islandClientModuleId = this.getRendererModuleString("islandClientModuleId");
107
+ if (!islandClientModuleId) {
108
+ return [];
109
+ }
110
+ return [
111
+ {
112
+ attributes: {
113
+ crossorigin: "anonymous",
114
+ "data-ecopages-runtime": "islands",
115
+ type: "module"
116
+ },
117
+ content: `import ${JSON.stringify(islandClientModuleId)};`,
118
+ inline: true,
119
+ kind: "script",
120
+ packageRole: "keep-separate",
121
+ position: "body"
122
+ }
123
+ ];
124
+ }
125
+ setHmrManager(hmrManager) {
126
+ this.hmrManager = hmrManager;
127
+ if (this.assetProcessingService) {
128
+ this.assetProcessingService.setHmrManager(hmrManager);
129
+ }
130
+ }
131
+ /**
132
+ * Build response headers with optional custom headers.
133
+ * @param contentType - The Content-Type header value
134
+ * @param customHeaders - Optional custom headers to merge
135
+ * @returns Headers object
136
+ */
137
+ buildHeaders(contentType, customHeaders) {
138
+ const headers = new Headers({ "Content-Type": contentType });
139
+ if (customHeaders) {
140
+ const incoming = new Headers(customHeaders);
141
+ incoming.forEach((value, key) => headers.set(key, value));
142
+ }
143
+ return headers;
144
+ }
145
+ /**
146
+ * Create an HTML Response.
147
+ * @param body - Response body (string or ReadableStream)
148
+ * @param ctx - Render context with status and headers
149
+ * @returns Response object
150
+ */
151
+ createHtmlResponse(body, ctx) {
152
+ return new Response(body, {
153
+ status: ctx.status ?? 200,
154
+ headers: this.buildHeaders("text/html; charset=utf-8", ctx.headers)
155
+ });
156
+ }
157
+ /**
158
+ * Create an HttpError for render failures.
159
+ * @param message - Error message
160
+ * @param cause - Original error if available
161
+ * @returns HttpError with 500 status
162
+ */
163
+ createRenderError(message, cause) {
164
+ const errorMessage = cause instanceof Error ? `${message}: ${cause.message}` : message;
165
+ return HttpError.InternalServerError(errorMessage);
166
+ }
167
+ /**
168
+ * Prepares dependencies for renderToResponse by resolving component dependencies
169
+ * and configuring the HTML transformer.
170
+ * @param view - The view component being rendered
171
+ * @param layout - Optional layout component
172
+ * @returns Resolved processed assets
173
+ */
174
+ async prepareViewDependencies(view, layout) {
175
+ const HtmlTemplate = await this.getHtmlTemplate();
176
+ const componentsToResolve = layout ? [HtmlTemplate, layout, view] : [HtmlTemplate, view];
177
+ const resolvedDependencies = this.htmlTransformer.dedupeProcessedAssets(
178
+ await this.resolveDependencies(componentsToResolve)
179
+ );
180
+ this.htmlTransformer.setPagePackage(createPagePackage(resolvedDependencies));
181
+ return resolvedDependencies;
182
+ }
183
+ async resolvePageBrowserGraphForFile(filePath) {
184
+ return await this.routeRenderOrchestrator.resolveDeclaredPageBrowserGraph({
185
+ routeFile: filePath,
186
+ integrationName: this.name,
187
+ collectContribution: async (routeFile) => {
188
+ const pageModule = await this.importPageFile(routeFile);
189
+ return await this.collectPageBrowserGraphContribution({ file: routeFile, pageModule });
190
+ }
191
+ });
192
+ }
193
+ mergePageBrowserGraphIntoPagePackage(pageBrowserGraph) {
194
+ if (!pageBrowserGraph) {
195
+ return void 0;
196
+ }
197
+ const currentPagePackage = this.htmlTransformer.getPagePackage();
198
+ const mergedPageBrowserGraph = currentPagePackage?.pageBrowserGraph ? {
199
+ entryAssets: this.htmlTransformer.dedupeProcessedAssets([
200
+ ...currentPagePackage.pageBrowserGraph.entryAssets,
201
+ ...pageBrowserGraph.entryAssets
202
+ ]),
203
+ chunkAssets: this.htmlTransformer.dedupeProcessedAssets([
204
+ ...currentPagePackage.pageBrowserGraph.chunkAssets,
205
+ ...pageBrowserGraph.chunkAssets
206
+ ])
207
+ } : pageBrowserGraph;
208
+ const pageBrowserGraphAssetKeys = new Set(
209
+ [...mergedPageBrowserGraph.entryAssets, ...mergedPageBrowserGraph.chunkAssets].map(
210
+ (asset) => buildProcessedAssetDedupeKey(asset)
211
+ )
212
+ );
213
+ const baseAssets = currentPagePackage ? currentPagePackage.assets.filter(
214
+ (asset) => !pageBrowserGraphAssetKeys.has(buildProcessedAssetDedupeKey(asset))
215
+ ) : this.htmlTransformer.getProcessedDependencies();
216
+ this.htmlTransformer.setPagePackage(
217
+ createPagePackage(this.htmlTransformer.dedupeProcessedAssets(baseAssets), {
218
+ pageBrowserGraph: mergedPageBrowserGraph
219
+ })
220
+ );
221
+ return mergedPageBrowserGraph;
222
+ }
223
+ /**
224
+ * Merges component-scoped assets into the active HTML transformer state.
225
+ *
226
+ * Explicit page, layout, and document shell composition can produce assets at
227
+ * each foreign subtree. This helper deduplicates those groups and folds them back into
228
+ * the transformer so downstream HTML finalization sees one canonical asset set.
229
+ *
230
+ * @param assetGroups - Optional groups of processed assets to merge.
231
+ * @returns The deduplicated asset subset contributed by this merge operation.
232
+ */
233
+ appendProcessedDependencies(...assetGroups) {
234
+ const nextDependencies = this.htmlTransformer.dedupeProcessedAssets(
235
+ assetGroups.flatMap((assets) => assets ?? [])
236
+ );
237
+ if (nextDependencies.length === 0) {
238
+ return nextDependencies;
239
+ }
240
+ const mergedDependencies = this.htmlTransformer.dedupeProcessedAssets([
241
+ ...this.htmlTransformer.getProcessedDependencies(),
242
+ ...nextDependencies
243
+ ]);
244
+ const currentPageBrowserGraph = this.htmlTransformer.getPagePackage()?.pageBrowserGraph;
245
+ this.htmlTransformer.setPagePackage(
246
+ createPagePackage(mergedDependencies, {
247
+ pageBrowserGraph: currentPageBrowserGraph
248
+ })
249
+ );
250
+ return nextDependencies;
251
+ }
252
+ /**
253
+ * Resolves metadata for explicit view rendering.
254
+ *
255
+ * When a view declares a `metadata()` function, that contract owns the final
256
+ * metadata for the explicit render. Otherwise the app-level default metadata is
257
+ * reused so explicit routes and page-module routes share the same fallback.
258
+ *
259
+ * @param view - View component being rendered.
260
+ * @param props - Props passed to the view.
261
+ * @returns Resolved metadata for the final document shell.
262
+ */
263
+ async resolveViewMetadata(view, props) {
264
+ return view.metadata ? await view.metadata({
265
+ params: {},
266
+ query: {},
267
+ props,
268
+ appConfig: this.appConfig
269
+ }) : this.appConfig.defaultMetadata;
270
+ }
271
+ /**
272
+ * Renders one explicit view response in partial mode.
273
+ *
274
+ * Same-integration views can optionally stream or render inline via the caller's
275
+ * `renderInline()` hook. Once a view may cross integration boundaries, this
276
+ * helper routes the render through `renderComponentWithForeignChildren()` instead so mixed
277
+ * shells can reuse the execution-scoped renderer cache and resolve nested
278
+ * foreign ownership before the partial response is returned.
279
+ *
280
+ * @param input - View render options for the partial response.
281
+ * @returns HTML response for the partial render.
282
+ */
283
+ async renderPartialViewResponse(input) {
284
+ if (input.renderInline && !this.hasForeignChildDescendants(input.view)) {
285
+ return this.createHtmlResponse(await input.renderInline(), input.ctx);
286
+ }
287
+ const rendererCache = /* @__PURE__ */ new Map();
288
+ const viewRender = await this.renderComponentWithForeignChildren({
289
+ component: input.view,
290
+ props: input.props ?? {},
291
+ integrationContext: { rendererCache }
292
+ });
293
+ const html = input.transformHtml ? input.transformHtml(viewRender.html) : viewRender.html;
294
+ return this.createHtmlResponse(html, input.ctx);
295
+ }
296
+ /**
297
+ * Renders an explicit view through optional layout and document shells.
298
+ *
299
+ * This helper is the shared explicit-route path for string-oriented and mixed
300
+ * integrations. It prepares view dependencies, resolves metadata, and composes
301
+ * view, layout, and html template boundaries with one execution-scoped renderer
302
+ * cache so repeated foreign shell delegation can reuse initialized renderers
303
+ * during the same render flow.
304
+ *
305
+ * @param input - View, props, and optional layout metadata for the render.
306
+ * @returns HTML response for the explicit view render.
307
+ */
308
+ async renderViewWithDocumentShell(input) {
309
+ const normalizedProps = input.props ?? {};
310
+ if (input.ctx.partial) {
311
+ return this.renderPartialViewResponse(input);
312
+ }
313
+ await this.prepareViewDependencies(input.view, input.layout);
314
+ const HtmlTemplate = await this.getHtmlTemplate();
315
+ const metadata = await this.resolveViewMetadata(input.view, input.props);
316
+ const { documentHtml } = await this.composeDocumentShell({
317
+ primaryComponent: input.view,
318
+ primaryProps: normalizedProps,
319
+ layout: input.layout ? {
320
+ component: input.layout,
321
+ props: {}
322
+ } : void 0,
323
+ htmlTemplate: HtmlTemplate,
324
+ documentProps: {
325
+ metadata,
326
+ pageProps: normalizedProps
327
+ }
328
+ });
329
+ const html = await this.finalizeResolvedHtml({
330
+ html: `${this.DOC_TYPE}${documentHtml}`,
331
+ partial: false
332
+ });
333
+ return this.createHtmlResponse(html, input.ctx);
334
+ }
335
+ /**
336
+ * Renders a route page through optional layout and document shells.
337
+ *
338
+ * Route rendering and explicit view rendering now share the same renderer-owned
339
+ * shell composition model. This helper composes page, layout, and html template
340
+ * renders while threading one execution-scoped renderer cache through every
341
+ * delegated foreign subtree so foreign shell ownership remains stable and renderer
342
+ * initialization is reused inside the current request.
343
+ *
344
+ * @param input - Page, layout, document, and metadata inputs for the route render.
345
+ * @returns Final serialized document HTML including the doctype prefix.
346
+ */
347
+ async renderPageWithDocumentShell(input) {
348
+ const { documentHtml: composedDocumentHtml } = await this.composeDocumentShell({
349
+ primaryComponent: input.page.component,
350
+ primaryProps: input.page.props,
351
+ layout: input.layout,
352
+ htmlTemplate: input.htmlTemplate,
353
+ documentProps: {
354
+ metadata: input.metadata,
355
+ pageProps: input.pageProps,
356
+ ...input.documentProps ?? {}
357
+ }
358
+ });
359
+ const documentHtml = input.transformDocumentHtml ? input.transformDocumentHtml(composedDocumentHtml) : composedDocumentHtml;
360
+ return `${this.DOC_TYPE}${documentHtml}`;
361
+ }
362
+ async composeDocumentShell(input) {
363
+ const rendererCache = /* @__PURE__ */ new Map();
364
+ const primaryRender = await this.renderComponentWithForeignChildren({
365
+ component: input.primaryComponent,
366
+ props: input.primaryProps,
367
+ integrationContext: { rendererCache }
368
+ });
369
+ const layoutRender = input.layout ? await this.renderComponentWithForeignChildren({
370
+ component: input.layout.component,
371
+ props: input.layout.props ?? {},
372
+ children: primaryRender.html,
373
+ integrationContext: { rendererCache }
374
+ }) : void 0;
375
+ const documentRender = await this.renderComponentWithForeignChildren({
376
+ component: input.htmlTemplate,
377
+ props: input.documentProps,
378
+ children: layoutRender?.html ?? primaryRender.html,
379
+ integrationContext: { rendererCache }
380
+ });
381
+ this.appendProcessedDependencies(primaryRender.assets, layoutRender?.assets, documentRender.assets);
382
+ return {
383
+ documentHtml: documentRender.html
384
+ };
385
+ }
386
+ /**
387
+ * Renders one string-first component with serialized children and collects its assets.
388
+ *
389
+ * String-oriented integrations frequently share the same component contract:
390
+ * pass serialized children through props, coerce the render result to HTML, and
391
+ * attach any component-scoped dependencies. This helper centralizes that flow
392
+ * so integrations can opt into shared orchestration without repeating the same
393
+ * string-render boilerplate.
394
+ *
395
+ * @param input - Component render input.
396
+ * @param component - String-oriented component implementation to execute.
397
+ * @returns Structured component render result for orchestration paths.
398
+ */
399
+ async renderStringComponentWithSerializedChildren(input, component) {
400
+ const serializedChildren = input.children === void 0 ? void 0 : typeof input.children === "string" ? input.children : isMarkupNodeLike(input.children) ? input.children.outerHTML : void 0;
401
+ if (input.children !== void 0 && serializedChildren === void 0) {
402
+ const componentFile = input.component.config?.__eco?.file ?? "unknown component";
403
+ const childTag = Object.prototype.toString.call(input.children);
404
+ throw new TypeError(
405
+ `[ecopages] ${this.name} renderer expected serialized children for ${componentFile}, received ${childTag}.`
406
+ );
407
+ }
408
+ const props = serializedChildren === void 0 ? input.props : { ...input.props, children: serializedChildren };
409
+ const content = await component(props);
410
+ const html = String(content);
411
+ const assets = input.component.config?.dependencies && typeof this.assetProcessingService?.processDependencies === "function" ? await this.processComponentDependencies([input.component]) : void 0;
412
+ return {
413
+ html,
414
+ canAttachAttributes: true,
415
+ rootTag: this.getRootTagName(html),
416
+ integrationName: this.name,
417
+ assets
418
+ };
419
+ }
420
+ getForeignSubtreeTokenPrefix() {
421
+ return `__${this.name}_foreign_subtree__`;
422
+ }
423
+ getForeignSubtreeResolutionContextKey() {
424
+ return `__${this.name}_foreign_subtree_runtime__`;
425
+ }
426
+ createQueuedForeignSubtreeExecutionRuntime(options) {
427
+ return this.foreignSubtreeExecutionService.createQueuedRuntime({
428
+ renderInput: options.renderInput,
429
+ rendererCache: options.rendererCache,
430
+ runtimeContextKey: options.runtimeContextKey ?? this.getForeignSubtreeResolutionContextKey(),
431
+ tokenPrefix: options.tokenPrefix ?? this.getForeignSubtreeTokenPrefix(),
432
+ createRuntimeContext: options.createRuntimeContext
433
+ });
434
+ }
435
+ getQueuedForeignSubtreeResolutionContext(input) {
436
+ return this.foreignSubtreeExecutionService.getQueuedRuntimeContext(
437
+ input,
438
+ this.getForeignSubtreeResolutionContextKey()
439
+ );
440
+ }
441
+ /**
442
+ * Renders a string-first component, then resolves any queued foreign
443
+ * boundaries before returning final component HTML.
444
+ */
445
+ async renderStringComponentWithQueuedForeignSubtrees(input, component) {
446
+ const componentRender = await this.renderStringComponentWithSerializedChildren(input, component);
447
+ const queuedForeignSubtreeResolution = await this.foreignSubtreeExecutionService.resolveStringQueuedHtml({
448
+ currentIntegrationName: this.name,
449
+ renderInput: input,
450
+ html: componentRender.html,
451
+ runtimeContextKey: this.getForeignSubtreeResolutionContextKey(),
452
+ queueLabel: "String",
453
+ getOwningRenderer: (integrationName, rendererCache) => this.getIntegrationRendererForName(integrationName, rendererCache),
454
+ applyAttributesToFirstElement: (html, attributes) => this.htmlTransformer.applyAttributesToFirstElement(html, attributes),
455
+ dedupeProcessedAssets: (assets) => this.htmlTransformer.dedupeProcessedAssets(assets)
456
+ });
457
+ const mergedAssets = this.htmlTransformer.dedupeProcessedAssets([
458
+ ...componentRender.assets ?? [],
459
+ ...queuedForeignSubtreeResolution.assets
460
+ ]);
461
+ return {
462
+ ...componentRender,
463
+ html: queuedForeignSubtreeResolution.html,
464
+ rootTag: this.getRootTagName(queuedForeignSubtreeResolution.html),
465
+ assets: mergedAssets.length > 0 ? mergedAssets : void 0
466
+ };
467
+ }
468
+ constructor({
469
+ appConfig,
470
+ assetProcessingService,
471
+ resolvedIntegrationDependencies,
472
+ rendererModules,
473
+ runtimeOrigin
474
+ }) {
475
+ this.appConfig = appConfig;
476
+ this.assetProcessingService = assetProcessingService;
477
+ this.htmlTransformer = new HtmlTransformerService();
478
+ this.resolvedIntegrationDependencies = resolvedIntegrationDependencies || [];
479
+ this.rendererModules = rendererModules ?? appConfig.runtime?.rendererModuleContext;
480
+ this.runtimeOrigin = runtimeOrigin;
481
+ this.dependencyResolverService = new DependencyResolverService(appConfig, assetProcessingService);
482
+ this.pageModuleLoaderService = new PageModuleLoaderService(appConfig, runtimeOrigin);
483
+ this.routeRenderOrchestrator = new RouteRenderOrchestrator(appConfig, assetProcessingService, {
484
+ ownershipValidationService: new OwnershipValidationService(appConfig)
485
+ });
486
+ }
487
+ /**
488
+ * Returns the HTML path from the provided file path.
489
+ * It extracts the path relative to the pages directory and removes the 'index' part if present.
490
+ *
491
+ * @param file - The file path to extract the HTML path from.
492
+ * @returns The extracted HTML path.
493
+ */
494
+ getHtmlPath({ file }) {
495
+ const pagesDir = this.appConfig.absolutePaths.pagesDir;
496
+ const pagesIndex = file.indexOf(pagesDir);
497
+ if (pagesIndex === -1) return file;
498
+ const startIndex = file.indexOf(pagesDir) + pagesDir.length;
499
+ const endIndex = file.lastIndexOf("/");
500
+ const path = file.substring(startIndex, endIndex);
501
+ if (path === "/index") return "";
502
+ return path;
503
+ }
504
+ /**
505
+ * Returns the HTML template component.
506
+ * It imports the HTML template from the specified path in the app configuration.
507
+ *
508
+ * @returns The HTML template component.
509
+ */
510
+ async getHtmlTemplate() {
511
+ const htmlTemplatePath = this.getRendererModuleString("htmlTemplateModulePath") ?? this.appConfig.absolutePaths.htmlTemplatePath;
512
+ try {
513
+ const { default: HtmlTemplate } = await this.importPageFile(htmlTemplatePath);
514
+ return HtmlTemplate;
515
+ } catch (error) {
516
+ invariant(false, `Error importing HtmlTemplate: ${error}`);
517
+ }
518
+ }
519
+ usesIntegrationPageImporter(_file) {
520
+ return false;
521
+ }
522
+ async importIntegrationPageFile(_file, _options) {
523
+ invariant(false, "Integration page importer must be implemented when enabled");
524
+ }
525
+ normalizeImportedPageFile(_file, pageModule) {
526
+ return pageModule;
527
+ }
528
+ /**
529
+ * Imports the page file from the specified path.
530
+ * It uses dynamic import to load the file and returns the imported module.
531
+ *
532
+ * @param file - The file path to import.
533
+ * @returns The imported module.
534
+ */
535
+ async importPageFile(file, options) {
536
+ const bypassCache = options?.bypassCache ?? (typeof Bun !== "undefined" && process.env.NODE_ENV === "development");
537
+ const pageModule = this.usesIntegrationPageImporter(file) ? await this.importIntegrationPageFile(file, {
538
+ bypassCache,
539
+ cacheScope: options?.cacheScope
540
+ }) : await this.pageModuleLoaderService.importPageFile(file, {
541
+ bypassCache,
542
+ cacheScope: options?.cacheScope
543
+ });
544
+ return this.normalizeImportedPageFile(file, pageModule);
545
+ }
546
+ /**
547
+ * Resolves the dependency path based on the component directory.
548
+ * It combines the component directory with the provided path URL.
549
+ *
550
+ * @param componentDir - The component directory path.
551
+ * @param pathUrl - The path URL to resolve.
552
+ * @returns The resolved dependency path.
553
+ */
554
+ resolveDependencyPath(componentDir, pathUrl) {
555
+ return this.dependencyResolverService.resolveDependencyPath(componentDir, pathUrl);
556
+ }
557
+ /**
558
+ * Extracts the dependencies from the provided component configuration.
559
+ * It resolves the paths for scripts and stylesheets based on the component directory.
560
+ *
561
+ * @param componentDir - The component directory path.
562
+ * @param scripts - The scripts to extract.
563
+ * @param stylesheets - The stylesheets to extract.
564
+ * @returns The extracted dependencies.
565
+ */
566
+ extractDependencies({
567
+ componentDir,
568
+ scripts,
569
+ stylesheets
570
+ }) {
571
+ const scriptsPaths = [
572
+ ...new Set(
573
+ (scripts ?? []).filter((script) => typeof script === "string" ? true : !script.lazy).map((script) => typeof script === "string" ? script : script.src).filter((script) => Boolean(script)).map((script) => this.resolveDependencyPath(componentDir, script))
574
+ )
575
+ ];
576
+ const stylesheetsPaths = [
577
+ ...new Set(
578
+ (stylesheets ?? []).map((style) => typeof style === "string" ? style : style.src).filter((style) => Boolean(style)).map((style) => this.resolveDependencyPath(componentDir, style))
579
+ )
580
+ ];
581
+ return {
582
+ scripts: scriptsPaths,
583
+ stylesheets: stylesheetsPaths
584
+ };
585
+ }
586
+ /**
587
+ * Resolves lazy script paths to public asset URLs.
588
+ * Converts source paths to their final bundled output paths.
589
+ *
590
+ * @param componentDir - The component directory path.
591
+ * @param scripts - The lazy script paths to resolve.
592
+ * @returns Comma-separated string of resolved public script paths.
593
+ */
594
+ resolveLazyScripts(componentDir, scripts) {
595
+ return this.dependencyResolverService.resolveLazyScripts(componentDir, scripts);
596
+ }
597
+ /**
598
+ * Collects the dependencies for the provided components.
599
+ * Combines component-specific dependencies with global integration dependencies.
600
+ *
601
+ * @param components - The components to collect dependencies from.
602
+ */
603
+ async resolveDependencies(components) {
604
+ const componentDeps = await this.processComponentDependencies(components);
605
+ return this.resolvedIntegrationDependencies.concat(componentDeps);
606
+ }
607
+ /**
608
+ * Processes component-specific dependencies WITHOUT prepending global integration dependencies.
609
+ * Use this method when you need only the component's own assets.
610
+ *
611
+ * @param components - The components to collect dependencies from.
612
+ */
613
+ async processComponentDependencies(components) {
614
+ return this.dependencyResolverService.processComponentDependencies(components, this.name);
615
+ }
616
+ /**
617
+ * Builds the internal route-render adapter consumed by `RouteRenderOrchestrator`.
618
+ *
619
+ * The route orchestrator needs a narrow orchestration contract, but those hooks should
620
+ * not become public API on the renderer base class. Keeping the adapter object
621
+ * local to the execution path lets the orchestrator depend on one explicit seam while
622
+ * subclasses continue to override protected renderer behavior directly.
623
+ */
624
+ createRouteRenderOrchestratorAdapter() {
625
+ return {
626
+ name: this.name,
627
+ resolveRouteRenderInputs: (routeOptions) => this.resolveRouteRenderInputs(routeOptions),
628
+ resolveRouteDependencies: (input) => this.resolveRouteDependencies(input),
629
+ collectPageBrowserGraphContribution: async (routeFile) => {
630
+ const pageModule = await this.importPageFile(routeFile);
631
+ return await this.collectPageBrowserGraphContribution({ file: routeFile, pageModule });
632
+ },
633
+ resolveRoutePageComponentRender: (input) => this.resolveRoutePageComponentRender(input),
634
+ renderRouteBody: (renderOptions) => this.renderRouteBody(renderOptions),
635
+ getRouteHtmlFinalization: (renderOptions) => this.getRouteHtmlFinalization(renderOptions),
636
+ transformRouteResponse: (response, htmlContributions) => this.transformRouteResponse(response, htmlContributions)
637
+ };
638
+ }
639
+ async resolveRouteRenderInputs(routeOptions) {
640
+ const pageModule = await this.pageModuleLoaderService.resolvePageModule({
641
+ file: routeOptions.file,
642
+ importPageFileFn: (targetFile) => this.importPageFile(targetFile)
643
+ });
644
+ const { Page, integrationSpecificProps } = pageModule;
645
+ const HtmlTemplate = await this.getHtmlTemplate();
646
+ const Layout = Page.config?.layout;
647
+ const { props, metadata } = await this.pageModuleLoaderService.resolvePageData({
648
+ pageModule,
649
+ routeOptions
650
+ });
651
+ return {
652
+ Page,
653
+ HtmlTemplate,
654
+ Layout,
655
+ props,
656
+ metadata,
657
+ integrationSpecificProps
658
+ };
659
+ }
660
+ async resolveRouteDependencies(input) {
661
+ return {
662
+ resolvedDependencies: await this.resolveDependencies(input.components)
663
+ };
664
+ }
665
+ async resolveRoutePageComponentRender(input) {
666
+ if (!this.shouldRenderPageComponent({ Page: input.Page, Layout: input.Layout, options: input.routeOptions })) {
667
+ return void 0;
668
+ }
669
+ return this.renderComponentWithForeignChildren({
670
+ component: input.Page,
671
+ props: {
672
+ ...input.props,
673
+ params: input.routeOptions.params || {},
674
+ query: input.routeOptions.query || {}
675
+ },
676
+ integrationContext: {
677
+ componentInstanceId: "eco-page-root"
678
+ }
679
+ });
680
+ }
681
+ async renderRouteBody(renderOptions) {
682
+ return this.render(renderOptions);
683
+ }
684
+ getRouteHtmlFinalization(renderOptions) {
685
+ const componentRootAttributes = renderOptions.componentRender?.canAttachAttributes && renderOptions.componentRender.rootAttributes && Object.keys(renderOptions.componentRender.rootAttributes).length > 0 ? renderOptions.componentRender.rootAttributes : void 0;
686
+ const documentAttributes = this.getDocumentAttributes(renderOptions);
687
+ const htmlContributions = this.getHtmlDocumentContributions({ renderOptions, partial: false });
688
+ const hasStructuralFinalization = componentRootAttributes && Object.keys(componentRootAttributes).length > 0 || documentAttributes && Object.keys(documentAttributes).length > 0;
689
+ if (!hasStructuralFinalization && (!htmlContributions || htmlContributions.length === 0)) {
690
+ return {};
691
+ }
692
+ return {
693
+ htmlContributions,
694
+ finalizeHtml: (html) => {
695
+ let renderedHtml = html;
696
+ if (componentRootAttributes) {
697
+ renderedHtml = this.htmlTransformer.applyAttributesToFirstBodyElement(
698
+ renderedHtml,
699
+ componentRootAttributes
700
+ );
701
+ }
702
+ if (documentAttributes) {
703
+ renderedHtml = this.htmlTransformer.applyAttributesToHtmlElement(renderedHtml, documentAttributes);
704
+ }
705
+ return renderedHtml;
706
+ }
707
+ };
708
+ }
709
+ async transformRouteResponse(response, htmlContributions) {
710
+ const transformedResponse = await this.htmlTransformer.transform(response, htmlContributions);
711
+ return transformedResponse.body ?? await transformedResponse.text();
712
+ }
713
+ /**
714
+ * Prepares the render options for the integration renderer.
715
+ * It imports the page file, collects dependencies, and prepares the render options.
716
+ *
717
+ * @param options - The route renderer options.
718
+ * @returns The prepared render options.
719
+ */
720
+ async prepareRenderOptions(options, adapter = this.createRouteRenderOrchestratorAdapter()) {
721
+ const renderOptions = await this.routeRenderOrchestrator.prepareRenderOptions(options, adapter);
722
+ invariant(renderOptions.pagePackage !== void 0, "Expected render preparation to produce a page package");
723
+ this.htmlTransformer.setPagePackage(renderOptions.pagePackage);
724
+ return renderOptions;
725
+ }
726
+ /**
727
+ * Controls whether the page root should be rendered through `renderComponent()`
728
+ * during route option preparation in component-capable modes.
729
+ *
730
+ * Integrations that already own page-level hydration (for example router-driven
731
+ * React rendering) can override this and return `false` to avoid duplicate root
732
+ * mount assets and competing hydration entrypoints.
733
+ */
734
+ shouldRenderPageComponent(_input) {
735
+ return true;
736
+ }
737
+ /**
738
+ * Executes the integration renderer with the provided options.
739
+ *
740
+ * Execution flow:
741
+ * 1. Build normalized render options (`prepareRenderOptions`).
742
+ * 2. Render the route body once.
743
+ * 3. Reject unresolved route-level eco-marker artifacts.
744
+ * 4. Optionally apply root attributes for page/component root boundaries.
745
+ * 5. Run HTML transformer with final dependency set.
746
+ *
747
+ * Stream-safety note: the first render result is normalized to a string once,
748
+ * then the pipeline continues with that immutable HTML value to avoid disturbed
749
+ * response-body errors.
750
+ *
751
+ * @param options Route renderer options.
752
+ * @returns Rendered route body plus effective cache strategy.
753
+ */
754
+ async execute(options) {
755
+ const adapter = this.createRouteRenderOrchestratorAdapter();
756
+ const renderOptions = await this.prepareRenderOptions(options, adapter);
757
+ return this.routeRenderOrchestrator.executePrepared(renderOptions, adapter);
758
+ }
759
+ /**
760
+ * Finalizes already-resolved HTML for explicit renderer-owned paths.
761
+ *
762
+ * This keeps document and root-attribute stamping plus HTML transformation
763
+ * available after a renderer has completed nested foreign-subtree resolution without
764
+ * routing back through shared route execution.
765
+ */
766
+ async finalizeResolvedHtml(options) {
767
+ const rendererBootstrapDependencies = this.getRendererBootstrapDependencies(options.partial);
768
+ this.appendProcessedDependencies(rendererBootstrapDependencies);
769
+ let html = options.html;
770
+ if (options.componentRootAttributes && Object.keys(options.componentRootAttributes).length > 0) {
771
+ html = this.htmlTransformer.applyAttributesToFirstBodyElement(html, options.componentRootAttributes);
772
+ }
773
+ if (options.documentAttributes && Object.keys(options.documentAttributes).length > 0) {
774
+ html = this.htmlTransformer.applyAttributesToHtmlElement(html, options.documentAttributes);
775
+ }
776
+ const shouldTransform = options.transformHtml ?? !options.partial;
777
+ if (!shouldTransform) {
778
+ return html;
779
+ }
780
+ const htmlContributions = options.htmlContributions ?? this.getHtmlDocumentContributions({
781
+ partial: options.partial
782
+ });
783
+ const transformedResponse = await this.htmlTransformer.transform(
784
+ new Response(html, {
785
+ headers: { "Content-Type": "text/html" }
786
+ }),
787
+ htmlContributions
788
+ );
789
+ return await transformedResponse.text();
790
+ }
791
+ /**
792
+ * Returns document-level attributes to stamp onto the rendered `<html>` tag.
793
+ *
794
+ * Integrations can override this to expose explicit document ownership or
795
+ * other runtime coordination markers without relying on script sniffing.
796
+ */
797
+ getDocumentAttributes(_renderOptions) {
798
+ return void 0;
799
+ }
800
+ /**
801
+ * Returns declarative HTML fragments that core should inject into the final document.
802
+ *
803
+ * @remarks
804
+ * Integrations may contribute document markup here, but core retains ownership
805
+ * of the final HTML rewrite pipeline and placement semantics. This is the
806
+ * supported document-markup extension point for integrations instead of custom
807
+ * response finalization logic.
808
+ */
809
+ getHtmlDocumentContributions(_options) {
810
+ return void 0;
811
+ }
812
+ /**
813
+ * Returns a renderer instance for a given integration name.
814
+ *
815
+ * Uses a per-execution cache to avoid repeated renderer initialization.
816
+ *
817
+ * @param integrationName Target integration name.
818
+ * @param cache Render-pass renderer cache.
819
+ * @returns Renderer for the requested integration.
820
+ * @throws Error when no integration plugin matches `integrationName`.
821
+ */
822
+ getIntegrationRendererForName(integrationName, cache) {
823
+ if (cache.has(integrationName)) {
824
+ return cache.get(integrationName);
825
+ }
826
+ if (integrationName === this.name) {
827
+ cache.set(integrationName, this);
828
+ return this;
829
+ }
830
+ const integrationPlugin = this.appConfig.integrations.find(
831
+ (integration) => integration.name === integrationName
832
+ );
833
+ invariant(!!integrationPlugin, `[ecopages] Integration not found for foreign owner: ${integrationName}`);
834
+ const renderer = integrationPlugin.initializeRenderer({
835
+ rendererModules: this.appConfig.runtime?.rendererModuleContext
836
+ });
837
+ cache.set(integrationName, renderer);
838
+ return renderer;
839
+ }
840
+ /**
841
+ * Renders one component under this integration's foreign-child runtime and resolves
842
+ * any nested foreign children captured during that render.
843
+ *
844
+ * Without this wrapper, a component tree with foreign-owned descendants would
845
+ * render them with no active foreign-child runtime, which bypasses the owning
846
+ * renderer's nested foreign-child handoff.
847
+ */
848
+ async renderComponentWithForeignChildren(input) {
849
+ return await this.foreignSubtreeExecutionService.executeComponentRender({
850
+ currentIntegrationName: this.name,
851
+ input,
852
+ renderComponent: (renderInput) => this.renderComponent(renderInput),
853
+ normalizeComponentRenderOutput: (result) => this.normalizeComponentRenderOutput(result),
854
+ hasForeignChildDescendants: (component) => this.hasForeignChildDescendants(component),
855
+ createForeignChildRuntime: ({ renderInput, rendererCache }) => this.createForeignChildRuntime({
856
+ renderInput,
857
+ rendererCache
858
+ }),
859
+ getOwningRenderer: (integrationName, rendererCache) => this.getIntegrationRendererForName(integrationName, rendererCache)
860
+ });
861
+ }
862
+ /**
863
+ * Compatibility foreign-subtree contract that exposes a narrower payload shape for
864
+ * future route-composition work while preserving the current
865
+ * `renderComponentWithForeignChildren()` runtime semantics.
866
+ */
867
+ async renderForeignSubtree(input) {
868
+ const result = await this.renderComponentWithForeignChildren(input);
869
+ return {
870
+ html: result.html,
871
+ assets: result.assets ?? [],
872
+ rootTag: result.rootTag,
873
+ rootAttributes: result.rootAttributes,
874
+ attachmentPolicy: result.canAttachAttributes ? { kind: "first-element" } : { kind: "none" },
875
+ integrationName: result.integrationName
876
+ };
877
+ }
878
+ normalizeComponentRenderOutput(result) {
879
+ const normalizedHtml = this.normalizeUnresolvedMarkerArtifactHtml(result.html);
880
+ return normalizedHtml === result.html ? result : {
881
+ ...result,
882
+ html: normalizedHtml
883
+ };
884
+ }
885
+ normalizeUnresolvedMarkerArtifactHtml(html) {
886
+ return normalizeUnresolvedMarkerArtifactHtml(html);
887
+ }
888
+ /**
889
+ * Returns whether the component dependency tree crosses into another
890
+ * integration.
891
+ *
892
+ * This keeps foreign-child runtime setup narrow: same-integration trees can render
893
+ * directly without paying the queue orchestration cost.
894
+ */
895
+ hasForeignChildDescendants(component) {
896
+ const stack = [component];
897
+ const seen = /* @__PURE__ */ new Set();
898
+ while (stack.length > 0) {
899
+ const current = stack.pop();
900
+ if (!current || seen.has(current)) {
901
+ continue;
902
+ }
903
+ seen.add(current);
904
+ const integrationName = current.config?.integration ?? current.config?.__eco?.integration;
905
+ if (integrationName && integrationName !== this.name) {
906
+ return true;
907
+ }
908
+ stack.push(...current.config?.dependencies?.components ?? []);
909
+ }
910
+ return false;
911
+ }
912
+ /**
913
+ * Render a single component and return structured output for orchestration paths.
914
+ *
915
+ * Default behavior delegates to `renderToResponse` in partial mode and wraps
916
+ * the resulting HTML into the `ComponentRenderResult` contract.
917
+ *
918
+ * In foreign-subtree resolution, this method is the integration-owned step that turns an
919
+ * already-resolved deferred foreign subtree into concrete HTML, assets, and optional
920
+ * root attributes.
921
+ *
922
+ * Integrations can override this for richer behavior (asset emission,
923
+ * root attributes, integration-specific hydration metadata).
924
+ *
925
+ * @param input Component render request.
926
+ * @returns Structured render result used by component/page orchestration.
927
+ */
928
+ async renderComponent(input) {
929
+ const response = await this.renderToResponse(
930
+ input.component,
931
+ input.props,
932
+ { partial: true }
933
+ );
934
+ const html = await response.text();
935
+ return {
936
+ html,
937
+ canAttachAttributes: true,
938
+ rootTag: this.getRootTagName(html),
939
+ integrationName: this.name
940
+ };
941
+ }
942
+ /**
943
+ * Extracts the first root element tag name from HTML output.
944
+ *
945
+ * @param html HTML fragment.
946
+ * @returns Root tag name when present; otherwise `undefined`.
947
+ */
948
+ getRootTagName(html) {
949
+ const rootTag = html.match(/^(?:\s|<!--[\s\S]*?-->)*<([a-zA-Z][a-zA-Z0-9:-]*)\b/);
950
+ return rootTag?.[1];
951
+ }
952
+ /**
953
+ * Collects declarative Page Browser Graph contributions for one Page.
954
+ *
955
+ * @remarks
956
+ * Integrations may describe page-scoped browser requirements here, while core
957
+ * retains ownership of dependency processing and final graph assembly. This is
958
+ * the supported page-browser extension point for integrations.
959
+ *
960
+ * @param context - The route file path and already imported page module.
961
+ * @returns Declarative dependencies or pre-resolved assets for the Page.
962
+ */
963
+ async collectPageBrowserGraphContribution(_context) {
964
+ return void 0;
965
+ }
966
+ /**
967
+ * Creates the per-render foreign-child runtime adopted by the shared component
968
+ * render context.
969
+ *
970
+ * The default runtime queues delegated foreign subtrees inside the owning
971
+ * renderer so string and markup renderers do not need to re-declare the same
972
+ * handoff boilerplate. Override only when a renderer needs custom runtime
973
+ * context or a different foreign-child execution strategy.
974
+ */
975
+ createForeignChildRuntime(options) {
976
+ return this.createQueuedForeignSubtreeExecutionRuntime({
977
+ renderInput: options.renderInput,
978
+ rendererCache: options.rendererCache
979
+ });
980
+ }
981
+ /**
982
+ * Creates an explicit fail-fast runtime for tests or renderers that do not
983
+ * support cross-integration foreign-child execution.
984
+ */
985
+ createFailFastForeignChildRuntime() {
986
+ return this.foreignSubtreeExecutionService.createFailFastRuntime(this.name);
987
+ }
988
+ }
989
+ export {
990
+ IntegrationRenderer
991
+ };