@ecopages/core 0.2.0-alpha.9 → 0.2.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 (357) hide show
  1. package/CHANGELOG.md +18 -11
  2. package/README.md +23 -26
  3. package/package.json +76 -58
  4. package/src/adapters/README.md +2 -2
  5. package/src/adapters/abstract/application-adapter.d.ts +28 -2
  6. package/src/adapters/abstract/application-adapter.js +14 -2
  7. package/src/adapters/abstract/router-adapter.d.ts +1 -1
  8. package/src/adapters/abstract/server-adapter.d.ts +2 -2
  9. package/src/adapters/bun/client-bridge.d.ts +1 -1
  10. package/src/adapters/bun/create-app.d.ts +4 -12
  11. package/src/adapters/bun/create-app.js +4 -5
  12. package/src/adapters/bun/hmr-manager.d.ts +4 -4
  13. package/src/adapters/bun/hmr-manager.js +8 -8
  14. package/src/adapters/bun/index.d.ts +1 -1
  15. package/src/adapters/bun/index.js +2 -2
  16. package/src/adapters/bun/server-adapter.d.ts +5 -5
  17. package/src/adapters/bun/server-adapter.js +6 -5
  18. package/src/adapters/bun/server-lifecycle.d.ts +4 -4
  19. package/src/adapters/bun/server-lifecycle.js +2 -2
  20. package/src/{create-app.d.ts → adapters/create-app.d.ts} +9 -6
  21. package/src/{create-app.js → adapters/create-app.js} +4 -4
  22. package/src/adapters/index.d.ts +1 -5
  23. package/src/adapters/index.js +1 -7
  24. package/src/adapters/node/create-app.d.ts +6 -9
  25. package/src/adapters/node/create-app.js +12 -6
  26. package/src/adapters/node/node-client-bridge.d.ts +1 -1
  27. package/src/adapters/node/node-hmr-manager.d.ts +2 -2
  28. package/src/adapters/node/node-hmr-manager.js +3 -3
  29. package/src/adapters/node/server-adapter.d.ts +4 -4
  30. package/src/adapters/node/server-adapter.js +13 -12
  31. package/src/adapters/node/static-content-server.d.ts +1 -1
  32. package/src/adapters/node/static-content-server.js +1 -1
  33. package/src/adapters/shared/application-adapter.d.ts +1 -1
  34. package/src/adapters/shared/define-api-handler.d.ts +1 -1
  35. package/src/adapters/shared/explicit-static-route-matcher.d.ts +2 -2
  36. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +1 -1
  37. package/src/adapters/shared/fs-server-response-factory.d.ts +2 -2
  38. package/src/adapters/shared/fs-server-response-factory.js +1 -1
  39. package/src/adapters/shared/fs-server-response-matcher.d.ts +1 -1
  40. package/src/adapters/shared/hmr-entrypoint-registrar.js +1 -1
  41. package/src/adapters/shared/hmr-html-response.d.ts +1 -1
  42. package/src/adapters/shared/render-context.d.ts +2 -1
  43. package/src/adapters/shared/render-context.js +3 -1
  44. package/src/adapters/shared/runtime-bootstrap.d.ts +1 -1
  45. package/src/adapters/shared/runtime-bootstrap.js +1 -1
  46. package/src/adapters/shared/server-adapter.d.ts +1 -1
  47. package/src/adapters/shared/server-adapter.js +5 -1
  48. package/src/adapters/shared/server-route-handler.d.ts +4 -4
  49. package/src/adapters/shared/server-route-handler.js +3 -3
  50. package/src/adapters/shared/server-static-builder.d.ts +5 -5
  51. package/src/adapters/shared/server-static-builder.js +5 -4
  52. package/src/build/README.md +19 -13
  53. package/src/build/build-adapter.d.ts +67 -20
  54. package/src/build/build-adapter.js +420 -19
  55. package/src/build/dev-build-coordinator.d.ts +10 -12
  56. package/src/build/dev-build-coordinator.js +8 -15
  57. package/src/build/esbuild-build-adapter.d.ts +6 -0
  58. package/src/build/esbuild-build-adapter.js +94 -11
  59. package/src/build/runtime-build-executor.d.ts +5 -4
  60. package/src/build/runtime-build-executor.js +5 -3
  61. package/src/build/runtime-specifier-alias-plugin.js +19 -15
  62. package/src/build/runtime-specifier-aliases.d.ts +5 -0
  63. package/src/build/runtime-specifier-aliases.js +95 -0
  64. package/src/config/README.md +5 -2
  65. package/src/config/config-builder.d.ts +25 -2
  66. package/src/config/config-builder.js +44 -7
  67. package/src/declarations.d.ts +1 -1
  68. package/src/dev/sc-server.d.ts +1 -1
  69. package/src/dev/sc-server.js +1 -1
  70. package/src/eco/eco.browser.d.ts +2 -0
  71. package/src/eco/eco.browser.js +83 -0
  72. package/src/eco/eco.js +19 -48
  73. package/src/eco/eco.types.d.ts +1 -1
  74. package/src/eco/eco.utils.d.ts +1 -40
  75. package/src/eco/eco.utils.js +5 -35
  76. package/src/eco/global-injector-map.d.ts +1 -1
  77. package/src/eco/lazy-injector-map.d.ts +1 -1
  78. package/src/hmr/hmr-strategy.d.ts +16 -13
  79. package/src/hmr/hmr-strategy.js +22 -7
  80. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  81. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  82. package/src/hmr/strategies/js-hmr-strategy.d.ts +2 -2
  83. package/src/hmr/strategies/js-hmr-strategy.js +2 -2
  84. package/src/index.browser.d.ts +2 -2
  85. package/src/index.browser.js +1 -1
  86. package/src/index.d.ts +3 -2
  87. package/src/index.js +15 -4
  88. package/src/integrations/ghtml/ghtml-renderer.d.ts +6 -1
  89. package/src/integrations/ghtml/ghtml-renderer.js +29 -28
  90. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -2
  91. package/src/integrations/ghtml/ghtml.plugin.js +2 -2
  92. package/src/plugins/README.md +1 -0
  93. package/src/plugins/eco-component-meta-plugin.d.ts +12 -1
  94. package/src/plugins/eco-component-meta-plugin.js +26 -20
  95. package/src/plugins/foreign-jsx-override-plugin.d.ts +31 -0
  96. package/src/plugins/foreign-jsx-override-plugin.js +35 -0
  97. package/src/plugins/integration-plugin.d.ts +99 -33
  98. package/src/plugins/integration-plugin.js +68 -21
  99. package/src/plugins/processor.d.ts +2 -2
  100. package/src/plugins/processor.js +2 -2
  101. package/src/plugins/source-transform.d.ts +46 -0
  102. package/src/plugins/source-transform.js +71 -0
  103. package/src/route-renderer/GRAPH.md +54 -84
  104. package/src/route-renderer/README.md +14 -20
  105. package/src/route-renderer/orchestration/component-render-context.d.ts +83 -0
  106. package/src/route-renderer/orchestration/component-render-context.js +147 -0
  107. package/src/route-renderer/orchestration/integration-renderer.d.ts +233 -76
  108. package/src/route-renderer/orchestration/integration-renderer.js +500 -143
  109. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +93 -0
  110. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +155 -0
  111. package/src/route-renderer/orchestration/render-execution.service.d.ts +11 -71
  112. package/src/route-renderer/orchestration/render-execution.service.js +65 -80
  113. package/src/{eco/eco.utils.ts → route-renderer/orchestration/render-output.utils.d.ts} +10 -53
  114. package/src/route-renderer/orchestration/render-output.utils.js +65 -0
  115. package/src/route-renderer/orchestration/render-preparation.service.d.ts +2 -8
  116. package/src/route-renderer/orchestration/render-preparation.service.js +10 -17
  117. package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
  118. package/src/route-renderer/orchestration/template-serialization.js +45 -0
  119. package/src/route-renderer/page-loading/dependency-resolver.d.ts +2 -2
  120. package/src/route-renderer/page-loading/dependency-resolver.js +10 -8
  121. package/src/route-renderer/page-loading/page-module-loader.d.ts +6 -4
  122. package/src/route-renderer/page-loading/page-module-loader.js +7 -5
  123. package/src/route-renderer/route-renderer.d.ts +5 -3
  124. package/src/route-renderer/route-renderer.js +13 -3
  125. package/src/router/README.md +79 -8
  126. package/src/router/client/navigation-coordinator.js +2 -2
  127. package/src/router/server/fs-router-scanner.d.ts +1 -1
  128. package/src/router/server/fs-router-scanner.js +6 -1
  129. package/src/router/server/fs-router.d.ts +1 -1
  130. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +3 -3
  131. package/src/services/assets/asset-processing-service/asset-processing.service.js +6 -6
  132. package/src/services/assets/asset-processing-service/asset.factory.d.ts +1 -1
  133. package/src/services/assets/asset-processing-service/asset.factory.js +2 -2
  134. package/src/services/assets/asset-processing-service/index.d.ts +5 -5
  135. package/src/services/assets/asset-processing-service/index.js +5 -5
  136. package/src/services/assets/asset-processing-service/processor.interface.d.ts +2 -2
  137. package/src/services/assets/asset-processing-service/processor.registry.d.ts +2 -2
  138. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +1 -1
  139. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +1 -1
  140. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +3 -3
  141. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +2 -2
  142. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -5
  143. package/src/services/assets/asset-processing-service/processors/index.js +5 -5
  144. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +2 -2
  145. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +1 -1
  146. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +4 -3
  147. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +15 -3
  148. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +3 -3
  149. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +1 -1
  150. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +2 -2
  151. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +1 -1
  152. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  153. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +1 -1
  154. package/src/services/assets/browser-bundle.service.d.ts +1 -1
  155. package/src/services/assets/browser-bundle.service.js +2 -2
  156. package/src/services/html/html-rewriter-provider.service.js +4 -1
  157. package/src/services/html/html-transformer.service.d.ts +2 -2
  158. package/src/services/html/html-transformer.service.js +4 -10
  159. package/src/services/invalidation/development-invalidation.service.d.ts +1 -1
  160. package/src/services/invalidation/development-invalidation.service.js +1 -0
  161. package/src/services/module-loading/app-module-loader.service.d.ts +25 -0
  162. package/src/services/module-loading/app-module-loader.service.js +31 -0
  163. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +9 -1
  164. package/src/services/module-loading/app-server-module-transpiler.service.js +77 -2
  165. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  166. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  167. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  168. package/src/{adapters/node/bootstrap-dependency-resolver.d.ts → services/module-loading/node-bootstrap-plugin.d.ts} +2 -24
  169. package/src/{adapters/node/bootstrap-dependency-resolver.js → services/module-loading/node-bootstrap-plugin.js} +42 -22
  170. package/src/services/module-loading/page-module-import.service.d.ts +4 -0
  171. package/src/services/module-loading/page-module-import.service.js +38 -9
  172. package/src/services/module-loading/server-module-transpiler.service.d.ts +3 -0
  173. package/src/services/module-loading/server-module-transpiler.service.js +4 -1
  174. package/src/services/runtime-state/dev-graph.service.d.ts +6 -6
  175. package/src/services/runtime-state/dev-graph.service.js +10 -10
  176. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +1 -1
  177. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +1 -1
  178. package/src/services/runtime-state/server-invalidation-state.service.d.ts +1 -1
  179. package/src/static-site-generator/static-site-generator.d.ts +2 -2
  180. package/src/static-site-generator/static-site-generator.js +1 -1
  181. package/src/{internal-types.d.ts → types/internal-types.d.ts} +24 -14
  182. package/src/{public-types.d.ts → types/public-types.d.ts} +30 -14
  183. package/src/types/public-types.js +0 -0
  184. package/src/utils/html-escaping.d.ts +7 -0
  185. package/src/utils/html-escaping.js +6 -0
  186. package/src/utils/locals-utils.d.ts +1 -1
  187. package/src/utils/parse-cli-args.d.ts +4 -1
  188. package/src/utils/parse-cli-args.js +16 -1
  189. package/src/utils/resolve-work-dir.js +1 -1
  190. package/src/watchers/project-watcher.d.ts +4 -4
  191. package/src/watchers/project-watcher.js +4 -10
  192. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  193. package/src/adapters/abstract/application-adapter.ts +0 -337
  194. package/src/adapters/abstract/router-adapter.ts +0 -30
  195. package/src/adapters/abstract/server-adapter.ts +0 -79
  196. package/src/adapters/bun/client-bridge.ts +0 -62
  197. package/src/adapters/bun/create-app.ts +0 -189
  198. package/src/adapters/bun/hmr-manager.ts +0 -409
  199. package/src/adapters/bun/index.ts +0 -2
  200. package/src/adapters/bun/server-adapter.ts +0 -499
  201. package/src/adapters/bun/server-lifecycle.ts +0 -124
  202. package/src/adapters/index.ts +0 -6
  203. package/src/adapters/node/bootstrap-dependency-resolver.ts +0 -301
  204. package/src/adapters/node/create-app.ts +0 -179
  205. package/src/adapters/node/index.d.ts +0 -6
  206. package/src/adapters/node/index.js +0 -11
  207. package/src/adapters/node/index.ts +0 -16
  208. package/src/adapters/node/node-client-bridge.ts +0 -79
  209. package/src/adapters/node/node-hmr-manager.ts +0 -381
  210. package/src/adapters/node/runtime-adapter.d.ts +0 -46
  211. package/src/adapters/node/runtime-adapter.js +0 -306
  212. package/src/adapters/node/runtime-adapter.ts +0 -439
  213. package/src/adapters/node/server-adapter.ts +0 -488
  214. package/src/adapters/node/static-content-server.ts +0 -239
  215. package/src/adapters/node/write-runtime-manifest.d.ts +0 -26
  216. package/src/adapters/node/write-runtime-manifest.js +0 -12
  217. package/src/adapters/node/write-runtime-manifest.ts +0 -38
  218. package/src/adapters/shared/api-response.ts +0 -104
  219. package/src/adapters/shared/application-adapter.ts +0 -199
  220. package/src/adapters/shared/define-api-handler.ts +0 -66
  221. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -140
  222. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -127
  223. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  224. package/src/adapters/shared/fs-server-response-matcher.ts +0 -205
  225. package/src/adapters/shared/hmr-entrypoint-registrar.ts +0 -149
  226. package/src/adapters/shared/hmr-html-response.ts +0 -52
  227. package/src/adapters/shared/render-context.ts +0 -120
  228. package/src/adapters/shared/runtime-bootstrap.ts +0 -79
  229. package/src/adapters/shared/server-adapter.ts +0 -489
  230. package/src/adapters/shared/server-route-handler.ts +0 -153
  231. package/src/adapters/shared/server-static-builder.ts +0 -166
  232. package/src/build/build-adapter.ts +0 -361
  233. package/src/build/build-manifest.ts +0 -54
  234. package/src/build/build-types.ts +0 -83
  235. package/src/build/dev-build-coordinator.ts +0 -221
  236. package/src/build/esbuild-build-adapter.ts +0 -559
  237. package/src/build/runtime-build-executor.ts +0 -34
  238. package/src/build/runtime-specifier-alias-plugin.ts +0 -58
  239. package/src/config/config-builder.ts +0 -706
  240. package/src/constants.ts +0 -54
  241. package/src/create-app.ts +0 -87
  242. package/src/dev/sc-server.ts +0 -143
  243. package/src/eco/component-render-context.d.ts +0 -105
  244. package/src/eco/component-render-context.js +0 -87
  245. package/src/eco/component-render-context.ts +0 -224
  246. package/src/eco/eco.ts +0 -242
  247. package/src/eco/eco.types.ts +0 -221
  248. package/src/eco/global-injector-map.ts +0 -112
  249. package/src/eco/lazy-injector-map.ts +0 -120
  250. package/src/eco/module-dependencies.ts +0 -75
  251. package/src/errors/http-error.ts +0 -72
  252. package/src/errors/index.ts +0 -2
  253. package/src/errors/locals-access-error.ts +0 -7
  254. package/src/global/app-logger.ts +0 -4
  255. package/src/hmr/client/hmr-runtime.ts +0 -152
  256. package/src/hmr/hmr-strategy.ts +0 -172
  257. package/src/hmr/hmr.postcss.test.e2e.ts +0 -41
  258. package/src/hmr/hmr.test.e2e.ts +0 -66
  259. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  260. package/src/hmr/strategies/js-hmr-strategy.ts +0 -320
  261. package/src/index.browser.ts +0 -3
  262. package/src/index.ts +0 -5
  263. package/src/integrations/ghtml/ghtml-renderer.ts +0 -96
  264. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  265. package/src/internal-types.ts +0 -232
  266. package/src/plugins/alias-resolver-plugin.ts +0 -63
  267. package/src/plugins/eco-component-meta-plugin.ts +0 -481
  268. package/src/plugins/integration-plugin.ts +0 -226
  269. package/src/plugins/processor.ts +0 -240
  270. package/src/plugins/runtime-capability.ts +0 -14
  271. package/src/public-types.ts +0 -1317
  272. package/src/route-renderer/component-graph/component-graph-executor.d.ts +0 -32
  273. package/src/route-renderer/component-graph/component-graph-executor.js +0 -31
  274. package/src/route-renderer/component-graph/component-graph-executor.ts +0 -84
  275. package/src/route-renderer/component-graph/component-graph.d.ts +0 -42
  276. package/src/route-renderer/component-graph/component-graph.js +0 -72
  277. package/src/route-renderer/component-graph/component-graph.ts +0 -159
  278. package/src/route-renderer/component-graph/component-marker.d.ts +0 -52
  279. package/src/route-renderer/component-graph/component-marker.js +0 -46
  280. package/src/route-renderer/component-graph/component-marker.ts +0 -117
  281. package/src/route-renderer/component-graph/component-reference.d.ts +0 -10
  282. package/src/route-renderer/component-graph/component-reference.js +0 -19
  283. package/src/route-renderer/component-graph/component-reference.ts +0 -29
  284. package/src/route-renderer/component-graph/marker-graph-resolver.d.ts +0 -77
  285. package/src/route-renderer/component-graph/marker-graph-resolver.js +0 -95
  286. package/src/route-renderer/component-graph/marker-graph-resolver.ts +0 -155
  287. package/src/route-renderer/orchestration/integration-renderer.ts +0 -790
  288. package/src/route-renderer/orchestration/render-execution.service.ts +0 -230
  289. package/src/route-renderer/orchestration/render-preparation.service.ts +0 -476
  290. package/src/route-renderer/page-loading/dependency-resolver.ts +0 -612
  291. package/src/route-renderer/page-loading/page-module-loader.ts +0 -181
  292. package/src/route-renderer/route-renderer.ts +0 -115
  293. package/src/router/client/link-intent.test.browser.ts +0 -51
  294. package/src/router/client/link-intent.ts +0 -92
  295. package/src/router/client/navigation-coordinator.ts +0 -433
  296. package/src/router/server/fs-router-scanner.ts +0 -219
  297. package/src/router/server/fs-router.ts +0 -122
  298. package/src/services/assets/asset-processing-service/asset-processing.service.ts +0 -401
  299. package/src/services/assets/asset-processing-service/asset.factory.ts +0 -105
  300. package/src/services/assets/asset-processing-service/assets.types.ts +0 -113
  301. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +0 -95
  302. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +0 -78
  303. package/src/services/assets/asset-processing-service/index.ts +0 -5
  304. package/src/services/assets/asset-processing-service/processor.interface.ts +0 -27
  305. package/src/services/assets/asset-processing-service/processor.registry.ts +0 -18
  306. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +0 -82
  307. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +0 -95
  308. package/src/services/assets/asset-processing-service/processors/index.ts +0 -5
  309. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  310. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  311. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +0 -85
  312. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  313. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -80
  314. package/src/services/assets/browser-bundle.service.ts +0 -53
  315. package/src/services/cache/cache.types.ts +0 -126
  316. package/src/services/cache/index.ts +0 -18
  317. package/src/services/cache/memory-cache-store.ts +0 -130
  318. package/src/services/cache/page-cache-service.ts +0 -202
  319. package/src/services/cache/page-request-cache-coordinator.service.ts +0 -131
  320. package/src/services/html/html-rewriter-provider.service.ts +0 -103
  321. package/src/services/html/html-transformer.service.ts +0 -279
  322. package/src/services/invalidation/development-invalidation.service.ts +0 -261
  323. package/src/services/module-loading/app-server-module-transpiler.service.ts +0 -52
  324. package/src/services/module-loading/page-module-import.service.ts +0 -200
  325. package/src/services/module-loading/server-loader.service.d.ts +0 -96
  326. package/src/services/module-loading/server-loader.service.js +0 -32
  327. package/src/services/module-loading/server-loader.service.ts +0 -130
  328. package/src/services/module-loading/server-module-transpiler.service.ts +0 -105
  329. package/src/services/runtime-manifest/node-runtime-manifest.service.d.ts +0 -35
  330. package/src/services/runtime-manifest/node-runtime-manifest.service.js +0 -60
  331. package/src/services/runtime-manifest/node-runtime-manifest.service.ts +0 -101
  332. package/src/services/runtime-state/dev-graph.service.ts +0 -217
  333. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +0 -136
  334. package/src/services/runtime-state/runtime-specifier-registry.service.ts +0 -96
  335. package/src/services/runtime-state/server-invalidation-state.service.ts +0 -68
  336. package/src/services/validation/schema-validation-service.ts +0 -204
  337. package/src/services/validation/standard-schema.types.ts +0 -68
  338. package/src/static-site-generator/static-site-generator.ts +0 -462
  339. package/src/utils/css.d.ts +0 -1
  340. package/src/utils/css.js +0 -7
  341. package/src/utils/css.ts +0 -5
  342. package/src/utils/deep-merge.ts +0 -47
  343. package/src/utils/hash.ts +0 -5
  344. package/src/utils/html.ts +0 -1
  345. package/src/utils/invariant.ts +0 -15
  346. package/src/utils/locals-utils.ts +0 -37
  347. package/src/utils/parse-cli-args.ts +0 -83
  348. package/src/utils/path-utils.module.ts +0 -14
  349. package/src/utils/resolve-work-dir.ts +0 -45
  350. package/src/utils/runtime.ts +0 -44
  351. package/src/utils/server-utils.module.ts +0 -67
  352. package/src/watchers/project-watcher.test-helpers.ts +0 -41
  353. package/src/watchers/project-watcher.ts +0 -363
  354. /package/src/{constants.d.ts → config/constants.d.ts} +0 -0
  355. /package/src/{constants.js → config/constants.js} +0 -0
  356. /package/src/{internal-types.js → services/module-loading/module-loading-types.js} +0 -0
  357. /package/src/{public-types.js → types/internal-types.js} +0 -0
@@ -1,790 +0,0 @@
1
- /**
2
- * This module contains the abstract class for the Integration Renderer
3
- * Every integration renderer should extend this class
4
- * @module
5
- */
6
-
7
- import type { EcoPagesAppConfig, IHmrManager } from '../../internal-types.ts';
8
- import type {
9
- ComponentRenderInput,
10
- ComponentRenderResult,
11
- EcoComponent,
12
- EcoComponentDependencies,
13
- EcoFunctionComponent,
14
- EcoPageComponent,
15
- EcoPageFile,
16
- EcoPagesElement,
17
- GetMetadata,
18
- GetMetadataContext,
19
- GetStaticProps,
20
- HtmlTemplateProps,
21
- IntegrationRendererRenderOptions,
22
- PageMetadataProps,
23
- RouteRendererBody,
24
- RouteRendererOptions,
25
- RouteRenderResult,
26
- } from '../../public-types.ts';
27
- import {
28
- type AssetProcessingService,
29
- type ProcessedAsset,
30
- } from '../../services/assets/asset-processing-service/index.ts';
31
- import { HtmlTransformerService } from '../../services/html/html-transformer.service.ts';
32
- import { invariant } from '../../utils/invariant.ts';
33
- import { HttpError } from '../../errors/http-error.ts';
34
- import { LocalsAccessError } from '../../errors/locals-access-error.ts';
35
- import { DependencyResolverService } from '../page-loading/dependency-resolver.ts';
36
- import { PageModuleLoaderService } from '../page-loading/page-module-loader.ts';
37
- import { MarkerGraphResolver } from '../component-graph/marker-graph-resolver.ts';
38
- import { RenderExecutionService, type RenderExecutionGraphContext } from './render-execution.service.ts';
39
- import { RenderPreparationService } from './render-preparation.service.ts';
40
- import type {
41
- BoundaryRenderDecisionInput,
42
- ComponentRenderBoundaryContext,
43
- } from '../../eco/component-render-context.ts';
44
-
45
- export interface FinalizeCapturedHtmlRenderOptions {
46
- html: string;
47
- componentsToResolve: EcoComponent[];
48
- graphContext: RenderExecutionGraphContext;
49
- partial?: boolean;
50
- componentRootAttributes?: Record<string, string>;
51
- documentAttributes?: Record<string, string>;
52
- mergeAssets?: boolean;
53
- transformHtml?: boolean;
54
- }
55
-
56
- /**
57
- * Creates a Proxy that throws LocalsAccessError on any property access.
58
- * Used to protect static pages from accidentally accessing request locals.
59
- *
60
- * @param filePath - The file path of the page attempting to access locals (for error reporting)
61
- * @returns A Proxy object that blocks all property access operations
62
- * @throws {LocalsAccessError} When any property access operation is attempted
63
- */
64
- function createLocalsProxy(filePath: string): Record<string, never> {
65
- const errorMessage = `[ecopages] Request locals are only available during request-time rendering with cache: 'dynamic'. Page: ${filePath}. If you meant to use locals here, set cache: 'dynamic' and provide locals from route middleware/handlers.`;
66
-
67
- return new Proxy(
68
- {},
69
- {
70
- get: () => {
71
- throw new LocalsAccessError(errorMessage);
72
- },
73
- set: () => {
74
- throw new LocalsAccessError(errorMessage);
75
- },
76
- has: () => {
77
- throw new LocalsAccessError(errorMessage);
78
- },
79
- ownKeys: () => {
80
- throw new LocalsAccessError(errorMessage);
81
- },
82
- deleteProperty: () => {
83
- throw new LocalsAccessError(errorMessage);
84
- },
85
- defineProperty: () => {
86
- throw new LocalsAccessError(errorMessage);
87
- },
88
- getOwnPropertyDescriptor: () => {
89
- throw new LocalsAccessError(errorMessage);
90
- },
91
- },
92
- );
93
- }
94
-
95
- /**
96
- * Context for renderToResponse method.
97
- */
98
- export interface RenderToResponseContext {
99
- partial?: boolean;
100
- status?: number;
101
- headers?: HeadersInit;
102
- }
103
-
104
- /**
105
- * The IntegrationRenderer class is an abstract class that provides a base for rendering integration-specific components in the EcoPages framework.
106
- * It handles the import of page files, collection of dependencies, and preparation of render options.
107
- * The class is designed to be extended by specific integration renderers.
108
- */
109
- export abstract class IntegrationRenderer<C = EcoPagesElement> {
110
- abstract name: string;
111
- protected appConfig: EcoPagesAppConfig;
112
- protected assetProcessingService: AssetProcessingService;
113
- protected htmlTransformer: HtmlTransformerService;
114
- protected hmrManager?: IHmrManager;
115
- protected resolvedIntegrationDependencies: ProcessedAsset[] = [];
116
- declare protected options: Required<IntegrationRendererRenderOptions>;
117
- protected runtimeOrigin: string;
118
- protected dependencyResolverService: DependencyResolverService;
119
- protected pageModuleLoaderService: PageModuleLoaderService;
120
- protected markerGraphResolver: MarkerGraphResolver;
121
- protected renderPreparationService: RenderPreparationService;
122
- protected renderExecutionService: RenderExecutionService;
123
-
124
- protected DOC_TYPE = '<!DOCTYPE html>';
125
-
126
- public setHmrManager(hmrManager: IHmrManager) {
127
- this.hmrManager = hmrManager;
128
- if (this.assetProcessingService) {
129
- this.assetProcessingService.setHmrManager(hmrManager);
130
- }
131
- }
132
-
133
- /**
134
- * Build response headers with optional custom headers.
135
- * @param contentType - The Content-Type header value
136
- * @param customHeaders - Optional custom headers to merge
137
- * @returns Headers object
138
- */
139
- protected buildHeaders(contentType: string, customHeaders?: HeadersInit): Headers {
140
- const headers = new Headers({ 'Content-Type': contentType });
141
- if (customHeaders) {
142
- const incoming = new Headers(customHeaders);
143
- incoming.forEach((value, key) => headers.set(key, value));
144
- }
145
- return headers;
146
- }
147
-
148
- /**
149
- * Create an HTML Response.
150
- * @param body - Response body (string or ReadableStream)
151
- * @param ctx - Render context with status and headers
152
- * @returns Response object
153
- */
154
- protected createHtmlResponse(body: BodyInit, ctx: RenderToResponseContext): Response {
155
- return new Response(body, {
156
- status: ctx.status ?? 200,
157
- headers: this.buildHeaders('text/html; charset=utf-8', ctx.headers),
158
- });
159
- }
160
-
161
- /**
162
- * Create an HttpError for render failures.
163
- * @param message - Error message
164
- * @param cause - Original error if available
165
- * @returns HttpError with 500 status
166
- */
167
- protected createRenderError(message: string, cause?: unknown): HttpError {
168
- const errorMessage = cause instanceof Error ? `${message}: ${cause.message}` : message;
169
- return HttpError.InternalServerError(errorMessage);
170
- }
171
-
172
- /**
173
- * Prepares dependencies for renderToResponse by resolving component dependencies
174
- * and configuring the HTML transformer.
175
- * @param view - The view component being rendered
176
- * @param layout - Optional layout component
177
- * @returns Resolved processed assets
178
- */
179
- protected async prepareViewDependencies(view: EcoComponent, layout?: EcoComponent): Promise<ProcessedAsset[]> {
180
- const HtmlTemplate = await this.getHtmlTemplate();
181
- const componentsToResolve = layout ? [HtmlTemplate, layout, view] : [HtmlTemplate, view];
182
- const resolvedDependencies = this.htmlTransformer.dedupeProcessedAssets(
183
- await this.resolveDependencies(componentsToResolve),
184
- );
185
- this.htmlTransformer.setProcessedDependencies(resolvedDependencies);
186
- return resolvedDependencies;
187
- }
188
-
189
- constructor({
190
- appConfig,
191
- assetProcessingService,
192
- resolvedIntegrationDependencies,
193
- runtimeOrigin,
194
- }: {
195
- appConfig: EcoPagesAppConfig;
196
- assetProcessingService: AssetProcessingService;
197
- resolvedIntegrationDependencies?: ProcessedAsset[];
198
- runtimeOrigin: string;
199
- }) {
200
- this.appConfig = appConfig;
201
- this.assetProcessingService = assetProcessingService;
202
- this.htmlTransformer = new HtmlTransformerService();
203
- this.resolvedIntegrationDependencies = resolvedIntegrationDependencies || [];
204
- this.runtimeOrigin = runtimeOrigin;
205
- this.dependencyResolverService = new DependencyResolverService(appConfig, assetProcessingService);
206
- this.pageModuleLoaderService = new PageModuleLoaderService(appConfig, runtimeOrigin);
207
- this.markerGraphResolver = new MarkerGraphResolver();
208
- this.renderPreparationService = new RenderPreparationService(appConfig, assetProcessingService);
209
- this.renderExecutionService = new RenderExecutionService();
210
- }
211
-
212
- /**
213
- * Returns the HTML path from the provided file path.
214
- * It extracts the path relative to the pages directory and removes the 'index' part if present.
215
- *
216
- * @param file - The file path to extract the HTML path from.
217
- * @returns The extracted HTML path.
218
- */
219
- protected getHtmlPath({ file }: { file: string }): string {
220
- const pagesDir = this.appConfig.absolutePaths.pagesDir;
221
- const pagesIndex = file.indexOf(pagesDir);
222
- if (pagesIndex === -1) return file;
223
- const startIndex = file.indexOf(pagesDir) + pagesDir.length;
224
- const endIndex = file.lastIndexOf('/');
225
- const path = file.substring(startIndex, endIndex);
226
- if (path === '/index') return '';
227
- return path;
228
- }
229
-
230
- /**
231
- * Returns the HTML template component.
232
- * It imports the HTML template from the specified path in the app configuration.
233
- *
234
- * @returns The HTML template component.
235
- */
236
- protected async getHtmlTemplate(): Promise<EcoComponent<HtmlTemplateProps>> {
237
- const { absolutePaths } = this.appConfig;
238
- try {
239
- const { default: HtmlTemplate } = await this.importPageFile(absolutePaths.htmlTemplatePath);
240
- return HtmlTemplate as EcoComponent<HtmlTemplateProps>;
241
- } catch (error) {
242
- invariant(false, `Error importing HtmlTemplate: ${error}`);
243
- }
244
- }
245
-
246
- /**
247
- * Returns the static props for the page.
248
- * It calls the provided getStaticProps function with the given options.
249
- *
250
- * @param getStaticProps - The function to get static props.
251
- * @param options - The options to pass to the getStaticProps function.
252
- * @returns The static props and metadata.
253
- */
254
- protected async getStaticProps(
255
- getStaticProps?: GetStaticProps<Record<string, unknown>>,
256
- options?: Pick<RouteRendererOptions, 'params'>,
257
- ): Promise<{
258
- props: Record<string, unknown>;
259
- metadata?: PageMetadataProps;
260
- }> {
261
- return this.pageModuleLoaderService.getStaticPropsForPage({
262
- getStaticProps,
263
- params: options?.params,
264
- });
265
- }
266
-
267
- /**
268
- * Returns the metadata properties for the page.
269
- * It calls the provided getMetadata function with the given context.
270
- *
271
- * @param getMetadata - The function to get metadata.
272
- * @param context - The context to pass to the getMetadata function.
273
- * @returns The metadata properties.
274
- */
275
- protected async getMetadataProps(
276
- getMetadata: GetMetadata | undefined,
277
- { props, params, query }: GetMetadataContext,
278
- ): Promise<PageMetadataProps> {
279
- return this.pageModuleLoaderService.getMetadataPropsForPage({
280
- getMetadata,
281
- context: { props, params, query } as GetMetadataContext,
282
- });
283
- }
284
-
285
- /**
286
- * Imports the page file from the specified path.
287
- * It uses dynamic import to load the file and returns the imported module.
288
- *
289
- * @param file - The file path to import.
290
- * @returns The imported module.
291
- */
292
- protected async importPageFile(file: string): Promise<EcoPageFile> {
293
- return this.pageModuleLoaderService.importPageFile(file);
294
- }
295
-
296
- /**
297
- * Resolves the dependency path based on the component directory.
298
- * It combines the component directory with the provided path URL.
299
- *
300
- * @param componentDir - The component directory path.
301
- * @param pathUrl - The path URL to resolve.
302
- * @returns The resolved dependency path.
303
- */
304
- protected resolveDependencyPath(componentDir: string, pathUrl: string): string {
305
- return this.dependencyResolverService.resolveDependencyPath(componentDir, pathUrl);
306
- }
307
-
308
- /**
309
- * Extracts the dependencies from the provided component configuration.
310
- * It resolves the paths for scripts and stylesheets based on the component directory.
311
- *
312
- * @param componentDir - The component directory path.
313
- * @param scripts - The scripts to extract.
314
- * @param stylesheets - The stylesheets to extract.
315
- * @returns The extracted dependencies.
316
- */
317
- protected extractDependencies({
318
- componentDir,
319
- scripts,
320
- stylesheets,
321
- }: {
322
- componentDir: string;
323
- } & EcoComponentDependencies): EcoComponentDependencies {
324
- const scriptsPaths = [
325
- ...new Set(
326
- (scripts ?? [])
327
- .filter((script) => (typeof script === 'string' ? true : !script.lazy))
328
- .map((script) => (typeof script === 'string' ? script : script.src))
329
- .filter((script): script is string => Boolean(script))
330
- .map((script) => this.resolveDependencyPath(componentDir, script)),
331
- ),
332
- ];
333
-
334
- const stylesheetsPaths = [
335
- ...new Set(
336
- (stylesheets ?? [])
337
- .map((style) => (typeof style === 'string' ? style : style.src))
338
- .filter((style): style is string => Boolean(style))
339
- .map((style) => this.resolveDependencyPath(componentDir, style)),
340
- ),
341
- ];
342
-
343
- return {
344
- scripts: scriptsPaths,
345
- stylesheets: stylesheetsPaths,
346
- };
347
- }
348
-
349
- /**
350
- * Resolves lazy script paths to public asset URLs.
351
- * Converts source paths to their final bundled output paths.
352
- *
353
- * @param componentDir - The component directory path.
354
- * @param scripts - The lazy script paths to resolve.
355
- * @returns Comma-separated string of resolved public script paths.
356
- */
357
- protected resolveLazyScripts(componentDir: string, scripts: string[]): string {
358
- return this.dependencyResolverService.resolveLazyScripts(componentDir, scripts);
359
- }
360
-
361
- /**
362
- * Collects the dependencies for the provided components.
363
- * Combines component-specific dependencies with global integration dependencies.
364
- *
365
- * @param components - The components to collect dependencies from.
366
- */
367
- protected async resolveDependencies(
368
- components: (EcoComponent | Partial<EcoComponent>)[],
369
- ): Promise<ProcessedAsset[]> {
370
- const componentDeps = await this.processComponentDependencies(components);
371
- return this.resolvedIntegrationDependencies.concat(componentDeps);
372
- }
373
-
374
- /**
375
- * Processes component-specific dependencies WITHOUT prepending global integration dependencies.
376
- * Use this method when you need only the component's own assets.
377
- *
378
- * @param components - The components to collect dependencies from.
379
- */
380
- protected async processComponentDependencies(
381
- components: (EcoComponent | Partial<EcoComponent>)[],
382
- ): Promise<ProcessedAsset[]> {
383
- return this.dependencyResolverService.processComponentDependencies(components, this.name);
384
- }
385
-
386
- /**
387
- * Prepares the render options for the integration renderer.
388
- * It imports the page file, collects dependencies, and prepares the render options.
389
- *
390
- * @param options - The route renderer options.
391
- * @returns The prepared render options.
392
- */
393
- protected async prepareRenderOptions(options: RouteRendererOptions): Promise<IntegrationRendererRenderOptions> {
394
- return this.renderPreparationService.prepare(options, this.name, {
395
- resolvePageModule: (file) => this.resolvePageModule(file),
396
- getHtmlTemplate: () => this.getHtmlTemplate(),
397
- resolvePageData: (pageModule, routeOptions) => this.resolvePageData(pageModule, routeOptions),
398
- resolveDependencies: (components) => this.resolveDependencies(components),
399
- buildRouteRenderAssets: (file) => this.buildRouteRenderAssets(file),
400
- shouldRenderPageComponent: (input) => this.shouldRenderPageComponent(input),
401
- renderPageComponent: ({ component, props }) =>
402
- this.renderComponent({
403
- component,
404
- props,
405
- integrationContext: {
406
- componentInstanceId: 'eco-page-root',
407
- },
408
- }),
409
- getComponentRenderBoundaryContext: () => this.getComponentRenderBoundaryContext(),
410
- setProcessedDependencies: (dependencies) => this.htmlTransformer.setProcessedDependencies(dependencies),
411
- dedupeProcessedAssets: (assets) => this.htmlTransformer.dedupeProcessedAssets(assets),
412
- createPageLocalsProxy: (filePath) =>
413
- createLocalsProxy(filePath) as unknown as RouteRendererOptions['locals'],
414
- });
415
- }
416
-
417
- /**
418
- * Controls whether the page root should be rendered through `renderComponent()`
419
- * during route option preparation in component-capable modes.
420
- *
421
- * Integrations that already own page-level hydration (for example router-driven
422
- * React rendering) can override this and return `false` to avoid duplicate root
423
- * mount assets and competing hydration entrypoints.
424
- */
425
- protected shouldRenderPageComponent(_input: {
426
- Page: EcoComponent;
427
- Layout?: EcoComponent;
428
- options: RouteRendererOptions;
429
- }): boolean {
430
- return true;
431
- }
432
-
433
- /**
434
- * Resolves the page module and normalizes exports.
435
- */
436
- protected async resolvePageModule(file: string): Promise<{
437
- Page: EcoPageFile['default'] | EcoPageComponent<any>;
438
- getStaticProps?: GetStaticProps<Record<string, unknown>>;
439
- getMetadata?: GetMetadata;
440
- integrationSpecificProps: Record<string, unknown>;
441
- }> {
442
- return this.pageModuleLoaderService.resolvePageModule({
443
- file,
444
- importPageFileFn: (targetFile) => this.importPageFile(targetFile),
445
- });
446
- }
447
-
448
- /**
449
- * Resolves static props and metadata for the page.
450
- */
451
- protected async resolvePageData(
452
- pageModule: {
453
- getStaticProps?: GetStaticProps<Record<string, unknown>>;
454
- getMetadata?: GetMetadata;
455
- },
456
- options: RouteRendererOptions,
457
- ): Promise<{
458
- props: Record<string, unknown>;
459
- metadata: PageMetadataProps;
460
- }> {
461
- return this.pageModuleLoaderService.resolvePageData({
462
- pageModule,
463
- routeOptions: options,
464
- });
465
- }
466
-
467
- /**
468
- * Executes the integration renderer with the provided options.
469
- *
470
- * Execution flow:
471
- * 1. Build normalized render options (`prepareRenderOptions`).
472
- * 2. Render once inside component render context to capture marker graph refs.
473
- * 3. Merge captured refs with optional explicit page-module graph context.
474
- * 4. Resolve any `eco-marker` graph bottom-up and merge produced assets.
475
- * 5. Optionally apply root attributes for page/component root boundaries.
476
- * 6. Run HTML transformer with final dependency set.
477
- *
478
- * Stream-safety note: the first render result is normalized to a string once,
479
- * then the pipeline continues with that immutable HTML value to avoid disturbed
480
- * response-body errors.
481
- *
482
- * @param options Route renderer options.
483
- * @returns Rendered route body plus effective cache strategy.
484
- */
485
- public async execute(options: RouteRendererOptions): Promise<RouteRenderResult> {
486
- return this.renderExecutionService.execute(options, this.name, {
487
- prepareRenderOptions: (routeOptions) =>
488
- this.prepareRenderOptions(routeOptions) as Promise<IntegrationRendererRenderOptions<C>>,
489
- render: (renderOptions) => this.render(renderOptions),
490
- getComponentRenderBoundaryContext: () => this.getComponentRenderBoundaryContext(),
491
- resolveMarkerGraphHtml: (input) =>
492
- this.resolveMarkerGraphHtml({
493
- html: input.html,
494
- componentsToResolve: input.componentsToResolve,
495
- graphContext: input.graphContext,
496
- }),
497
- getDocumentAttributes: (renderOptions) => this.getDocumentAttributes(renderOptions),
498
- dedupeProcessedAssets: (assets) => this.htmlTransformer.dedupeProcessedAssets(assets),
499
- getProcessedDependencies: () => this.htmlTransformer.getProcessedDependencies(),
500
- setProcessedDependencies: (dependencies) => this.htmlTransformer.setProcessedDependencies(dependencies),
501
- applyAttributesToHtmlElement: (html, attributes) =>
502
- this.htmlTransformer.applyAttributesToHtmlElement(html, attributes),
503
- applyAttributesToFirstBodyElement: (html, attributes) =>
504
- this.htmlTransformer.applyAttributesToFirstBodyElement(html, attributes),
505
- transformHtml: async (html) => {
506
- const response = await this.htmlTransformer.transform(
507
- new Response(html, {
508
- headers: {
509
- 'Content-Type': 'text/html',
510
- },
511
- }),
512
- );
513
- return response.body as RouteRendererBody;
514
- },
515
- });
516
- }
517
-
518
- /**
519
- * Captures a render pass as immutable HTML along with the graph context needed
520
- * for deferred marker resolution.
521
- *
522
- * This is the shared entry point for direct `renderToResponse()` flows that
523
- * need the same component graph capture semantics as route execution without
524
- * going through `prepareRenderOptions()`.
525
- */
526
- protected async captureHtmlRender(render: () => Promise<RouteRendererBody>): Promise<{
527
- html: string;
528
- graphContext: RenderExecutionGraphContext;
529
- }> {
530
- return this.renderExecutionService.captureHtmlRender(
531
- this.name,
532
- this.getComponentRenderBoundaryContext(),
533
- render,
534
- );
535
- }
536
-
537
- /**
538
- * Finalizes previously captured HTML by resolving deferred markers, merging
539
- * any emitted assets, stamping optional attributes, and optionally running the
540
- * HTML transformer for full-document flows.
541
- */
542
- protected async finalizeCapturedHtmlRender(options: FinalizeCapturedHtmlRenderOptions): Promise<string> {
543
- const finalization = await this.renderExecutionService.finalizeHtmlRender(
544
- {
545
- html: options.html,
546
- graphContext: options.graphContext,
547
- componentsToResolve: options.componentsToResolve,
548
- componentRootAttributes: options.componentRootAttributes,
549
- documentAttributes: options.documentAttributes,
550
- mergeAssets: options.mergeAssets ?? !options.partial,
551
- },
552
- {
553
- resolveMarkerGraphHtml: (input) =>
554
- this.resolveMarkerGraphHtml({
555
- html: input.html,
556
- componentsToResolve: input.componentsToResolve,
557
- graphContext: input.graphContext,
558
- }),
559
- dedupeProcessedAssets: (assets) => this.htmlTransformer.dedupeProcessedAssets(assets),
560
- getProcessedDependencies: () => this.htmlTransformer.getProcessedDependencies(),
561
- setProcessedDependencies: (dependencies) => this.htmlTransformer.setProcessedDependencies(dependencies),
562
- applyAttributesToHtmlElement: (html, attributes) =>
563
- this.htmlTransformer.applyAttributesToHtmlElement(html, attributes),
564
- applyAttributesToFirstBodyElement: (html, attributes) =>
565
- this.htmlTransformer.applyAttributesToFirstBodyElement(html, attributes),
566
- },
567
- );
568
-
569
- const shouldTransform = options.transformHtml ?? !options.partial;
570
- if (!shouldTransform) {
571
- return finalization.html;
572
- }
573
-
574
- const transformedResponse = await this.htmlTransformer.transform(
575
- new Response(finalization.html, {
576
- headers: { 'Content-Type': 'text/html' },
577
- }),
578
- );
579
-
580
- return await transformedResponse.text();
581
- }
582
-
583
- /**
584
- * Returns document-level attributes to stamp onto the rendered `<html>` tag.
585
- *
586
- * Integrations can override this to expose explicit document ownership or
587
- * other runtime coordination markers without relying on script sniffing.
588
- */
589
- protected getDocumentAttributes(
590
- _renderOptions: IntegrationRendererRenderOptions<C>,
591
- ): Record<string, string> | undefined {
592
- return undefined;
593
- }
594
-
595
- /**
596
- * Resolves all `eco-marker` placeholders in rendered HTML using integration
597
- * dispatch and bottom-up graph execution.
598
- *
599
- * Responsibility split:
600
- * - core decodes markers into component refs, props, slot children, and target
601
- * integration dispatch
602
- * - the selected integration renderer performs the actual component render via
603
- * `renderComponent()`
604
- *
605
- * Resolver callback behavior per marker:
606
- * - resolve component definition by `componentRef`
607
- * - resolve serialized props by `propsRef`
608
- * - stitch resolved child HTML when `slotRef` is present
609
- * - dispatch to target integration `renderComponent`
610
- * - collect produced assets and apply root attributes when attachable
611
- *
612
- * @param options.html HTML that may still contain marker tokens.
613
- * @param options.componentsToResolve Component set used to build component ref registry.
614
- * @param options.graphContext Props/slot linkage captured during render.
615
- * @returns Resolved HTML plus any component-scoped assets produced while resolving nodes.
616
- * @throws Error when marker component refs or props refs cannot be resolved.
617
- */
618
- private async resolveMarkerGraphHtml(options: {
619
- html: string;
620
- componentsToResolve: EcoComponent[];
621
- graphContext: RenderExecutionGraphContext;
622
- }): Promise<{ html: string; assets: ProcessedAsset[] }> {
623
- const integrationRendererCache = new Map<string, IntegrationRenderer>();
624
- return this.markerGraphResolver.resolve({
625
- html: options.html,
626
- componentsToResolve: options.componentsToResolve,
627
- graphContext: options.graphContext,
628
- resolveRenderer: (integrationName) =>
629
- this.getIntegrationRendererForName(integrationName, integrationRendererCache),
630
- applyAttributesToFirstElement: (html, attributes) =>
631
- this.htmlTransformer.applyAttributesToFirstElement(html, attributes),
632
- });
633
- }
634
-
635
- /**
636
- * Returns a renderer instance for a given integration name.
637
- *
638
- * Uses a per-execution cache to avoid repeated renderer initialization.
639
- *
640
- * @param integrationName Target integration name.
641
- * @param cache Render-pass renderer cache.
642
- * @returns Renderer for the requested integration.
643
- * @throws Error when no integration plugin matches `integrationName`.
644
- */
645
- private getIntegrationRendererForName(
646
- integrationName: string,
647
- cache: Map<string, IntegrationRenderer<any>>,
648
- ): IntegrationRenderer<any> {
649
- if (cache.has(integrationName)) {
650
- return cache.get(integrationName) as IntegrationRenderer<any>;
651
- }
652
-
653
- if (integrationName === this.name) {
654
- cache.set(integrationName, this);
655
- return this;
656
- }
657
-
658
- const integrationPlugin = this.appConfig.integrations.find(
659
- (integration) => integration.name === integrationName,
660
- );
661
- invariant(!!integrationPlugin, `[ecopages] Integration not found for marker: ${integrationName}`);
662
- const renderer = integrationPlugin.initializeRenderer();
663
- cache.set(integrationName, renderer);
664
- return renderer;
665
- }
666
-
667
- /**
668
- * Abstract method to render the integration-specific component.
669
- * This method should be implemented by the specific integration renderer.
670
- *
671
- * @param options - The integration renderer render options.
672
- * @returns The rendered body.
673
- */
674
- abstract render(options: IntegrationRendererRenderOptions<C>): Promise<RouteRendererBody>;
675
-
676
- /**
677
- * Render a view directly to a Response object.
678
- * Used for explicit routing where views are rendered from route handlers.
679
- *
680
- * @param view - The eco.page component to render
681
- * @param props - Props to pass to the view
682
- * @param ctx - Render context with partial flag and response options
683
- * @returns A Response object with the rendered content
684
- */
685
- abstract renderToResponse<P = Record<string, unknown>>(
686
- view: EcoComponent<P>,
687
- props: P,
688
- ctx: RenderToResponseContext,
689
- ): Promise<Response>;
690
-
691
- /**
692
- * Render a single component and return structured output for orchestration paths.
693
- *
694
- * Default behavior delegates to `renderToResponse` in partial mode and wraps
695
- * the resulting HTML into the `ComponentRenderResult` contract.
696
- *
697
- * In marker resolution, this method is the integration-owned step that turns an
698
- * already-resolved deferred boundary into concrete HTML, assets, and optional
699
- * root attributes.
700
- *
701
- * Integrations can override this for richer behavior (asset emission,
702
- * root attributes, integration-specific hydration metadata).
703
- *
704
- * @param input Component render request.
705
- * @returns Structured render result used by marker/page orchestration.
706
- */
707
- async renderComponent(input: ComponentRenderInput): Promise<ComponentRenderResult> {
708
- const response = await this.renderToResponse(
709
- input.component as EcoFunctionComponent<Record<string, unknown>, EcoPagesElement>,
710
- input.props,
711
- { partial: true },
712
- );
713
- const html = await response.text();
714
-
715
- return {
716
- html,
717
- canAttachAttributes: true,
718
- rootTag: this.getRootTagName(html),
719
- integrationName: this.name,
720
- };
721
- }
722
-
723
- /**
724
- * Extracts the first root element tag name from HTML output.
725
- *
726
- * @param html HTML fragment.
727
- * @returns Root tag name when present; otherwise `undefined`.
728
- */
729
- protected getRootTagName(html: string): string | undefined {
730
- const rootTag = html.match(/^\s*<([a-zA-Z][a-zA-Z0-9:-]*)\b/);
731
- return rootTag?.[1];
732
- }
733
-
734
- /**
735
- * Method to build route render assets.
736
- * This method can be optionally overridden by the specific integration renderer.
737
- *
738
- * @param file - The file path to build assets for.
739
- * @returns The processed assets or undefined.
740
- */
741
- protected buildRouteRenderAssets(_file: string): Promise<ProcessedAsset[]> | undefined {
742
- return undefined;
743
- }
744
-
745
- /**
746
- * Builds the narrow boundary policy facade injected into component render
747
- * context for this render pass.
748
- *
749
- * `eco.component()` consumes this facade without knowing about integration
750
- * registries or plugin instances.
751
- *
752
- * @returns Boundary policy context for the active integration renderer.
753
- */
754
- protected getComponentRenderBoundaryContext(): ComponentRenderBoundaryContext {
755
- return {
756
- decideBoundaryRender: (input) => (this.shouldDeferComponentBoundary(input) ? 'defer' : 'inline'),
757
- };
758
- }
759
-
760
- /**
761
- * Resolves whether a component boundary should be deferred by consulting the
762
- * target integration plugin.
763
- *
764
- * Boundaries targeting the current integration always render inline. Cross-
765
- * integration boundaries delegate the decision to the target integration's
766
- * `shouldDeferComponentBoundary()` policy.
767
- *
768
- * @param input Boundary metadata for the active render pass.
769
- * @returns `true` when the boundary should emit a marker; otherwise `false`.
770
- */
771
- protected shouldDeferComponentBoundary(input: BoundaryRenderDecisionInput): boolean {
772
- if (!input.targetIntegration || input.targetIntegration === input.currentIntegration) {
773
- return false;
774
- }
775
-
776
- const targetIntegration = this.appConfig.integrations.find(
777
- (integration) => integration.name === input.targetIntegration,
778
- );
779
- invariant(
780
- !!targetIntegration,
781
- `[ecopages] Integration not found for component boundary: ${input.targetIntegration}`,
782
- );
783
-
784
- return targetIntegration.shouldDeferComponentBoundary({
785
- currentIntegration: input.currentIntegration,
786
- targetIntegration: input.targetIntegration,
787
- component: input.component,
788
- });
789
- }
790
- }