@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
package/src/eco/README.md CHANGED
@@ -7,10 +7,12 @@ A unified API for defining components, pages, and page data in EcoPages.
7
7
  The `eco` namespace provides a consistent, type-safe interface for:
8
8
 
9
9
  1. **`eco.component()`** - Factory for defining reusable components with dependencies and optional lazy-loading
10
- 2. **`eco.page()`** - Factory for defining page components with optional inline `staticPaths`, `staticProps`, and `metadata`
11
- 3. **`eco.metadata()`** - Type-safe wrapper for page metadata (legacy pattern)
12
- 4. **`eco.staticPaths()`** - Type-safe wrapper for dynamic route generation (legacy pattern)
13
- 5. **`eco.staticProps()`** - Type-safe wrapper for static data fetching (legacy pattern)
10
+ 2. **`eco.html()`** - Semantic alias for the document shell component (the outermost HTML wrapper)
11
+ 3. **`eco.layout()`** - Semantic alias for route layout components (page-level wrappers)
12
+ 4. **`eco.page()`** - Factory for defining page components with optional inline `staticPaths`, `staticProps`, and `metadata`
13
+ 5. **`eco.metadata()`** - Type-safe wrapper for page metadata (legacy pattern)
14
+ 6. **`eco.staticPaths()`** - Type-safe wrapper for dynamic route generation (legacy pattern)
15
+ 7. **`eco.staticProps()`** - Type-safe wrapper for static data fetching (legacy pattern)
14
16
 
15
17
  ## Component Patterns
16
18
 
@@ -42,9 +44,9 @@ export function Card({ children, class: className }: CardProps) {
42
44
 
43
45
  > **Note:** If your component requires a dedicated CSS file, use `eco.component()` instead to manage the stylesheet dependency.
44
46
 
45
- ### Plain React Components (With Bun)
47
+ ### Plain React Components
46
48
 
47
- When using Bun, React components with hooks and Tailwind CSS work out of the box without `eco.component()`. Bun auto-imports dependencies, so you can write standard React components:
49
+ React components with hooks and Tailwind CSS work out of the box without `eco.component()`. Standard React components can be used directly:
48
50
 
49
51
  ```tsx
50
52
  import { useEffect, useState } from 'react';
@@ -99,7 +101,6 @@ export function ThemeToggle() {
99
101
  - React components with hooks (`useState`, `useEffect`, etc.)
100
102
  - Components styled with Tailwind CSS classes
101
103
  - Interactive UI that doesn't require external scripts or dedicated stylesheets
102
- - Bun handles all imports automatically
103
104
 
104
105
  > **Note:** Use `eco.component()` only when you need to manage external stylesheets, scripts, or lazy loading. For React components relying solely on hooks and Tailwind, plain functions are simpler and sufficient.
105
106
 
@@ -128,14 +129,14 @@ export const Counter = eco.component({
128
129
 
129
130
  ### Comparison
130
131
 
131
- | Aspect | Simple JSX | Plain React (Bun) | `eco.component()` |
132
- | -------------------- | ---------- | ----------------- | ----------------- |
133
- | React hooks | No | Yes | Yes |
134
- | Scripts/Stylesheets | No | No | Yes |
135
- | Lazy loading | No | No | Yes |
136
- | Hydration strategies | No | No | Yes |
137
- | Runtime cost | Zero | Minimal | Minimal |
138
- | Use case | Static UI | Interactive UI | Advanced UI |
132
+ | Aspect | Simple JSX | Plain React | `eco.component()` |
133
+ | -------------------- | ---------- | -------------- | ----------------- |
134
+ | React hooks | No | Yes | Yes |
135
+ | Scripts/Stylesheets | No | No | Yes |
136
+ | Lazy loading | No | No | Yes |
137
+ | Hydration strategies | No | No | Yes |
138
+ | Runtime cost | Zero | Minimal | Minimal |
139
+ | Use case | Static UI | Interactive UI | Advanced UI |
139
140
 
140
141
  All patterns can coexist in the same project. Use the right tool for the job.
141
142
 
@@ -231,6 +232,53 @@ Both patterns work and can be mixed - the renderer checks for attached propertie
231
232
 
232
233
  ## API Reference
233
234
 
235
+ ### `eco.html()`
236
+
237
+ Creates the document shell component — the outermost HTML wrapper rendered once per page. Semantically equivalent to `eco.component()` but signals intent to tooling and readers that this component owns the full document structure (`<html>`, `<head>`, `<body>`).
238
+
239
+ ```tsx
240
+ import { eco } from '@ecopages/core';
241
+
242
+ export const Document = eco.html({
243
+ dependencies: {
244
+ stylesheets: ['./document.css'],
245
+ },
246
+ render: ({ children, metadata }) => (
247
+ <html lang="en">
248
+ <head>
249
+ <title>{metadata?.title ?? 'EcoPages'}</title>
250
+ </head>
251
+ <body>{children}</body>
252
+ </html>
253
+ ),
254
+ });
255
+ ```
256
+
257
+ ### `eco.layout()`
258
+
259
+ Creates a route layout component — a wrapper rendered around page content. Semantically equivalent to `eco.component()` but clearly communicates that the component is intended to be used as a `layout` in `eco.page()`.
260
+
261
+ ```tsx
262
+ import { eco } from '@ecopages/core';
263
+
264
+ export const BaseLayout = eco.layout({
265
+ dependencies: {
266
+ stylesheets: ['./base-layout.css'],
267
+ scripts: ['./base-layout.script.ts'],
268
+ },
269
+ render: ({ children }) => <main>{children}</main>,
270
+ });
271
+ ```
272
+
273
+ Use `eco.layout()` components as the `layout` option in `eco.page()`:
274
+
275
+ ```tsx
276
+ export default eco.page({
277
+ layout: BaseLayout,
278
+ render: () => <h1>Hello</h1>,
279
+ });
280
+ ```
281
+
234
282
  ### `eco.component()`
235
283
 
236
284
  Define a reusable component with dependencies.
@@ -516,12 +564,18 @@ interface EcoComponentDependencies {
516
564
  components?: EcoComponent[];
517
565
  }
518
566
 
567
+ // Shared base option shape used by component(), html(), and layout()
519
568
  interface ComponentOptions<P, E = EcoPagesElement> {
520
569
  componentDir?: string;
521
570
  dependencies?: EcoComponentDependencies;
522
571
  render: (props: P) => E;
523
572
  }
524
573
 
574
+ // html() and layout() accept the same options as component() but return
575
+ // narrower types to signal intent (EcoHtmlComponent / EcoLayoutComponent).
576
+ type HtmlOptions<E = EcoPagesElement> = ComponentOptions<Record<string, unknown>, E>;
577
+ type LayoutOptions<E = EcoPagesElement> = ComponentOptions<{ children: E }, E>;
578
+
525
579
  interface PageOptions<T, E = EcoPagesElement> {
526
580
  componentDir?: string;
527
581
  dependencies?: EcoComponentDependencies;
@@ -551,7 +605,7 @@ type PagePropsFor<T> =
551
605
  ```tsx
552
606
  import { eco } from '@ecopages/core';
553
607
 
554
- eco. // IDE shows: component, page, metadata, staticPaths, staticProps
608
+ eco. // IDE shows: component, html, layout, page, metadata, staticPaths, staticProps
555
609
  ```
556
610
 
557
611
  ### 2. Type Safety
@@ -0,0 +1,2 @@
1
+ import type { Eco } from './eco.types.js';
2
+ export declare const eco: Eco;
@@ -0,0 +1,88 @@
1
+ function createComponentFactory(options) {
2
+ const component2 = ((props) => options.render(props));
3
+ component2.config = {
4
+ __eco: options.__eco,
5
+ integration: options.integration,
6
+ dependencies: options.dependencies
7
+ };
8
+ return component2;
9
+ }
10
+ function component(options) {
11
+ return createComponentFactory(options);
12
+ }
13
+ function embed(component2, props, children) {
14
+ const nextProps = children === void 0 ? props : { ...props, children };
15
+ return component2(nextProps);
16
+ }
17
+ function html(options) {
18
+ return createComponentFactory(options);
19
+ }
20
+ function layout(options) {
21
+ return createComponentFactory(options);
22
+ }
23
+ function page(options) {
24
+ const {
25
+ layout: pageLayout,
26
+ dependencies,
27
+ render,
28
+ staticPaths: staticPaths2,
29
+ staticProps: staticProps2,
30
+ metadata: metadata2,
31
+ cache,
32
+ requires,
33
+ middleware
34
+ } = options;
35
+ const pageComponent = createComponentFactory({
36
+ __eco: options.__eco,
37
+ integration: options.integration,
38
+ dependencies: pageLayout ? {
39
+ ...dependencies,
40
+ components: [...dependencies?.components ?? [], pageLayout]
41
+ } : dependencies,
42
+ render
43
+ });
44
+ if (pageLayout && pageComponent.config) {
45
+ pageComponent.config.layout = pageLayout;
46
+ }
47
+ if (staticPaths2) {
48
+ pageComponent.staticPaths = staticPaths2;
49
+ }
50
+ if (staticProps2) {
51
+ pageComponent.staticProps = staticProps2;
52
+ }
53
+ if (metadata2) {
54
+ pageComponent.metadata = metadata2;
55
+ }
56
+ if (cache) {
57
+ pageComponent.cache = cache;
58
+ }
59
+ if (requires) {
60
+ pageComponent.requires = requires;
61
+ }
62
+ if (middleware) {
63
+ pageComponent.middleware = middleware;
64
+ }
65
+ return pageComponent;
66
+ }
67
+ function metadata(fn) {
68
+ return fn;
69
+ }
70
+ function staticPaths(fn) {
71
+ return fn;
72
+ }
73
+ function staticProps(fn) {
74
+ return fn;
75
+ }
76
+ const eco = {
77
+ component,
78
+ embed,
79
+ html,
80
+ layout,
81
+ page,
82
+ metadata,
83
+ staticPaths,
84
+ staticProps
85
+ };
86
+ export {
87
+ eco
88
+ };
package/src/eco/eco.js CHANGED
@@ -1,60 +1,40 @@
1
- import { createNodeId, createPropsRef, createSlotRef, getComponentRenderContext } from "./component-render-context.js";
2
- import { createComponentMarker, parseComponentMarkers } from "../route-renderer/component-marker.js";
3
- import { addTriggerAttribute, isThenable, wrapWithScriptsInjector } from "./eco.utils.js";
1
+ import {
2
+ finalizeComponentRender,
3
+ getComponentRenderContext,
4
+ interceptForeignChild
5
+ } from "../route-renderer/orchestration/component-render-context.js";
6
+ import { isThenable } from "../route-renderer/orchestration/render-output.utils.js";
4
7
  function createComponentFactory(options) {
5
8
  const integrationName = options.integration ?? options.__eco?.integration;
6
9
  const comp = ((props) => {
7
- const renderContext = getComponentRenderContext();
8
- const shouldEmitMarker = renderContext !== void 0 && renderContext.boundaryContext.decideBoundaryRender({
9
- currentIntegration: renderContext.currentIntegration,
10
- targetIntegration: integrationName,
11
- component: comp
12
- }) === "defer";
13
- if (shouldEmitMarker && renderContext) {
14
- const nodeId = createNodeId(renderContext);
15
- const propsRef = createPropsRef(renderContext);
16
- const componentRef = comp.config?.__eco?.id ?? comp.config?.__eco?.file;
17
- if (!componentRef) {
18
- throw new Error(
19
- "[ecopages] Missing component reference metadata for cross-integration marker emission."
20
- );
21
- }
22
- const componentProps = props ?? {};
23
- renderContext.propsByRef[propsRef] = componentProps;
24
- let slotRef;
25
- const children = componentProps.children;
26
- if (typeof children === "string" && children.includes("<eco-marker")) {
27
- const childMarkers = parseComponentMarkers(children);
28
- if (childMarkers.length > 0) {
29
- slotRef = createSlotRef(renderContext);
30
- renderContext.slotChildrenByRef[slotRef] = childMarkers.map((marker) => marker.nodeId);
10
+ const componentProps = props ?? {};
11
+ const renderInline = (nextProps = props) => finalizeComponentRender(comp, options.render(nextProps));
12
+ const activeRenderContext = getComponentRenderContext();
13
+ const foreignChildRender = interceptForeignChild({
14
+ component: comp,
15
+ props: componentProps,
16
+ targetIntegration: integrationName
17
+ });
18
+ if (isThenable(foreignChildRender)) {
19
+ return foreignChildRender.then((resolvedForeignChildRender) => {
20
+ if (resolvedForeignChildRender?.kind === "resolved") {
21
+ return resolvedForeignChildRender.value;
31
22
  }
32
- }
33
- return createComponentMarker({
34
- nodeId,
35
- integration: integrationName,
36
- componentRef,
37
- propsRef,
38
- slotRef
23
+ return renderInline(resolvedForeignChildRender?.props ?? props);
39
24
  });
40
25
  }
41
- const content = options.render(props);
42
- const lazyTriggers = comp.config?._resolvedLazyTriggers;
43
- if (lazyTriggers && lazyTriggers.length > 0) {
44
- const triggerId = lazyTriggers[0].triggerId;
45
- if (isThenable(content)) {
46
- return content.then((resolvedContent) => addTriggerAttribute(resolvedContent, triggerId));
47
- }
48
- return addTriggerAttribute(content, triggerId);
26
+ if (foreignChildRender?.kind === "resolved") {
27
+ return foreignChildRender.value;
49
28
  }
50
- const lazyGroups = comp.config?._resolvedLazyScripts;
51
- if (lazyGroups && lazyGroups.length > 0) {
52
- if (isThenable(content)) {
53
- return content.then((resolvedContent) => wrapWithScriptsInjector(resolvedContent, lazyGroups));
54
- }
55
- return wrapWithScriptsInjector(content, lazyGroups);
29
+ if (foreignChildRender?.kind === "inline") {
30
+ return renderInline(foreignChildRender.props ?? props);
56
31
  }
57
- return content;
32
+ if (activeRenderContext && activeRenderContext.foreignChildRuntime && integrationName && integrationName !== activeRenderContext.currentIntegration) {
33
+ throw new Error(
34
+ `[ecopages] Missing foreign-child interception from ${activeRenderContext.currentIntegration} to ${integrationName} for ${options.__eco?.file ?? "unknown component"}.`
35
+ );
36
+ }
37
+ return renderInline();
58
38
  });
59
39
  comp.config = {
60
40
  __eco: options.__eco,
@@ -66,20 +46,46 @@ function createComponentFactory(options) {
66
46
  function component(options) {
67
47
  return createComponentFactory(options);
68
48
  }
49
+ function isMissingForeignChildInterceptionError(error) {
50
+ return error instanceof Error && error.message.startsWith("[ecopages] Missing foreign-child interception from ");
51
+ }
52
+ function embed(component2, props, children) {
53
+ const activeRenderContext = getComponentRenderContext();
54
+ const ecoComponent = component2;
55
+ const targetIntegration = ecoComponent.config?.integration ?? ecoComponent.config?.__eco?.integration;
56
+ const componentFile = ecoComponent.config?.__eco?.file ?? "unknown component";
57
+ const nextProps = children === void 0 ? props : { ...props, children };
58
+ try {
59
+ return component2(nextProps);
60
+ } catch (error) {
61
+ if (!isMissingForeignChildInterceptionError(error)) {
62
+ throw error;
63
+ }
64
+ throw new Error(
65
+ `[ecopages] eco.embed() could not hand off the Foreign Child from ${activeRenderContext?.currentIntegration ?? "unknown integration"} to ${targetIntegration ?? "unknown integration"} for ${componentFile}. The active Integration renderer exposed a foreign-child runtime, but it did not intercept this cross-integration render. Ensure mixed-integration Page, Layout, Html, or Component renders install foreign-child handoff before calling eco.embed().`
66
+ );
67
+ }
68
+ }
69
+ function html(options) {
70
+ return createComponentFactory(options);
71
+ }
72
+ function layout(options) {
73
+ return createComponentFactory(options);
74
+ }
69
75
  function page(options) {
70
- const { layout, dependencies, render, staticPaths: staticPaths2, staticProps: staticProps2, metadata: metadata2, cache, requires, middleware } = options;
76
+ const { layout: layout2, dependencies, render, staticPaths: staticPaths2, staticProps: staticProps2, metadata: metadata2, cache, requires, middleware } = options;
71
77
  const componentOptions = {
72
78
  __eco: options.__eco,
73
79
  integration: options.integration,
74
- dependencies: layout ? {
80
+ dependencies: layout2 ? {
75
81
  ...dependencies,
76
- components: [...dependencies?.components || [], layout]
82
+ components: [...dependencies?.components || [], layout2]
77
83
  } : dependencies,
78
84
  render
79
85
  };
80
86
  const pageComponent = createComponentFactory(componentOptions);
81
- if (layout && pageComponent.config) {
82
- pageComponent.config.layout = layout;
87
+ if (layout2 && pageComponent.config) {
88
+ pageComponent.config.layout = layout2;
83
89
  }
84
90
  if (staticPaths2) pageComponent.staticPaths = staticPaths2;
85
91
  if (staticProps2) pageComponent.staticProps = staticProps2;
@@ -100,6 +106,9 @@ function staticProps(fn) {
100
106
  }
101
107
  const eco = {
102
108
  component,
109
+ embed,
110
+ html,
111
+ layout,
103
112
  page,
104
113
  metadata,
105
114
  staticPaths,
@@ -2,8 +2,53 @@
2
2
  * Type definitions for the eco namespace API
3
3
  * @module
4
4
  */
5
- import type { DependencyLazyTrigger, EcoComponent, EcoComponentDependencies, EcoInjectedMeta, EcoPagesElement, GetMetadata, GetStaticPaths, GetStaticProps, Middleware, RequestLocals, RequestPageContext } from '../public-types.js';
5
+ import type { DependencyLazyTrigger, EcoComponent, EcoComponentDependencies, EcoHtmlComponent, EcoInjectedMeta, EcoLayoutComponent, EcoPageLayoutComponent, EcoPagesElement, FileRouteMiddleware, GetMetadata, GetStaticPaths, GetStaticProps, HtmlTemplateProps, LayoutProps, RequestLocals, RequestPageContext } from '../types/public-types.js';
6
6
  import type { CacheStrategy } from '../services/cache/cache.types.js';
7
+ /**
8
+ * Extracts the props type from one eco component.
9
+ */
10
+ export type PropsOf<TComponent extends EcoComponent> = TComponent extends EcoComponent<infer P, any> ? P : never;
11
+ /**
12
+ * Extracts the render result type from one eco component.
13
+ */
14
+ export type ResultOf<TComponent extends EcoComponent> = TComponent extends EcoComponent<any, infer R> ? R : never;
15
+ /**
16
+ * Narrows an eco component to its callable function signature.
17
+ *
18
+ * This is useful for helper modules such as `EcoEmbed` that invoke a component
19
+ * directly and need to exclude the non-callable metadata shape from the public
20
+ * type surface.
21
+ */
22
+ export type CallableComponentOf<TComponent extends EcoComponent> = Extract<TComponent, (props: any, ...args: any[]) => any>;
23
+ /**
24
+ * Extracts the declared `children` type from one eco component when present.
25
+ */
26
+ export type ChildrenOf<TComponent extends EcoComponent> = PropsOf<TComponent> extends {
27
+ children?: infer T;
28
+ } ? T : PropsOf<TComponent> extends {
29
+ children: infer T;
30
+ } ? T : never;
31
+ /**
32
+ * Extracts the props accepted by `eco.embed()` before optional `children`
33
+ * injection.
34
+ *
35
+ * When the target component already declares `children`, callers pass the rest
36
+ * of the props bag here and provide `children` as the third `eco.embed()`
37
+ * argument or the `EcoEmbed` wrapper children slot.
38
+ */
39
+ export type EmbedPropsOf<TComponent extends EcoComponent> = 'children' extends keyof PropsOf<TComponent> ? Omit<PropsOf<TComponent>, 'children'> : PropsOf<TComponent>;
40
+ /**
41
+ * Props accepted by integration-owned `EcoEmbed` adapters.
42
+ *
43
+ * The `component` field is narrowed to the callable portion of the target eco
44
+ * component so JSX wrappers can invoke `eco.embed()` without repeating local
45
+ * callable-component extraction logic.
46
+ */
47
+ export type EcoEmbedProps<TComponent extends EcoComponent> = {
48
+ component: CallableComponentOf<TComponent>;
49
+ props: EmbedPropsOf<TComponent>;
50
+ children?: unknown;
51
+ };
7
52
  type WithRequiredLocals<K extends keyof RequestLocals> = Omit<RequestLocals, K> & {
8
53
  [P in K]-?: Exclude<RequestLocals[P], null | undefined>;
9
54
  };
@@ -26,6 +71,8 @@ export interface ComponentOptions<P, E = EcoPagesElement> {
26
71
  dependencies?: EcoComponentDependencies;
27
72
  render: (props: P) => E | Promise<E>;
28
73
  }
74
+ export type HtmlOptions<E = EcoPagesElement> = ComponentOptions<HtmlTemplateProps, E>;
75
+ export type LayoutOptions<E = EcoPagesElement> = ComponentOptions<LayoutProps<E>, E>;
29
76
  /**
30
77
  * Base options shared by all page variants
31
78
  */
@@ -34,9 +81,7 @@ export interface PageOptionsBase<T, E = EcoPagesElement> {
34
81
  __eco?: EcoInjectedMeta;
35
82
  integration?: string;
36
83
  dependencies?: EcoComponentDependencies;
37
- layout?: EcoComponent<{
38
- children: E;
39
- } & Partial<RequestPageContext>>;
84
+ layout?: EcoPageLayoutComponent<E>;
40
85
  /**
41
86
  * Define static paths for dynamic routes (e.g., [slug].tsx).
42
87
  * Returns all possible paths that should be pre-rendered at build time.
@@ -86,7 +131,7 @@ interface PageOptionsWithMiddleware<T, E = EcoPagesElement> extends PageOptionsB
86
131
  * Request-time middleware for file-based routes.
87
132
  * Runs before rendering and can short-circuit by returning a Response.
88
133
  */
89
- middleware: Middleware[];
134
+ middleware: FileRouteMiddleware[];
90
135
  }
91
136
  /**
92
137
  * Options for creating a page with eco.page()
@@ -127,7 +172,7 @@ export type EcoPageComponent<T> = EcoComponent<PagePropsFor<T> & Partial<Request
127
172
  metadata?: GetMetadata<T>;
128
173
  cache?: CacheStrategy;
129
174
  requires?: PageRequires;
130
- middleware?: Middleware[];
175
+ middleware?: FileRouteMiddleware[];
131
176
  };
132
177
  /**
133
178
  * The eco namespace interface
@@ -139,6 +184,24 @@ export interface Eco {
139
184
  * @template E - Element/return type (EcoPagesElement for Kita, ReactNode for React)
140
185
  */
141
186
  component: <P = {}, E = EcoPagesElement>(options: ComponentOptions<P, E>) => EcoComponent<P, E>;
187
+ /**
188
+ * Render a component explicitly, with an optional `children` argument.
189
+ *
190
+ * This is useful when authoring crosses integration boundaries and inline JSX
191
+ * would otherwise force one file to model multiple JSX namespaces.
192
+ */
193
+ embed: {
194
+ <TComponent extends EcoComponent>(component: CallableComponentOf<TComponent>, props: PropsOf<TComponent>): ResultOf<TComponent>;
195
+ <TComponent extends EcoComponent>(component: CallableComponentOf<TComponent>, props: EmbedPropsOf<TComponent>, children: ChildrenOf<TComponent> | unknown): ResultOf<TComponent>;
196
+ };
197
+ /**
198
+ * Create a document shell component for the HTML wrapper.
199
+ */
200
+ html: <E = EcoPagesElement>(options: HtmlOptions<E>) => EcoHtmlComponent<E>;
201
+ /**
202
+ * Create a route layout component.
203
+ */
204
+ layout: <E = EcoPagesElement>(options: LayoutOptions<E>) => EcoLayoutComponent<E>;
142
205
  /**
143
206
  * Create a page component with type-safe props from getStaticProps.
144
207
  * Returns an EcoPageComponent with attached staticPaths, staticProps, and metadata.
@@ -1,40 +1 @@
1
- import type { EcoComponent } from '../public-types.js';
2
- /**
3
- * Returns `true` when `value` is a thenable (Promise-like) object.
4
- *
5
- * Used to transparently handle both synchronous and asynchronous component
6
- * render results without requiring every caller to branch on `instanceof Promise`.
7
- *
8
- * @typeParam T Expected resolved type of the thenable.
9
- */
10
- export declare function isThenable<T>(value: unknown): value is PromiseLike<T>;
11
- /**
12
- * Injects `data-eco-trigger` into the first real HTML element opening tag of
13
- * a component's rendered output string.
14
- *
15
- * The scan skips over leading whitespace, HTML comments (`<!-- -->`), CDATA
16
- * sections, and doctype declarations so that the attribute is always placed on
17
- * the first actual element — not spurious markup that can precede it.
18
- *
19
- * The insertion point is the end of the element's tag name, before any existing
20
- * attributes or the closing `>`, which produces output like:
21
- *
22
- * ```html
23
- * <my-element data-eco-trigger="eco-trigger-abc123" class="foo">…</my-element>
24
- * ```
25
- *
26
- * When no eligible opening tag is found the original string is returned
27
- * unchanged so callers never receive a broken fragment.
28
- *
29
- * @param content Rendered HTML string (or any value coercible to string).
30
- * @param triggerId Stable trigger identifier produced by `buildResolvedLazyTriggers`.
31
- */
32
- export declare function addTriggerAttribute(content: unknown, triggerId: string): string;
33
- /**
34
- * Wraps rendered component output in a `<scripts-injector>` element that
35
- * carries an inline injector map for the legacy (non-global-injector) path.
36
- *
37
- * @param content Rendered component HTML.
38
- * @param lazyGroups Resolved lazy script groups attached to the component config.
39
- */
40
- export declare function wrapWithScriptsInjector(content: unknown, lazyGroups: NonNullable<EcoComponent['config']>['_resolvedLazyScripts']): string;
1
+ export { addTriggerAttribute, isThenable, wrapWithScriptsInjector, } from '../route-renderer/orchestration/render-output.utils.js';
@@ -1,38 +1,8 @@
1
- import { buildInjectorMapScript } from "./lazy-injector-map.js";
2
- function isThenable(value) {
3
- return typeof value === "object" && value !== null && "then" in value && typeof value.then === "function";
4
- }
5
- function addTriggerAttribute(content, triggerId) {
6
- const str = String(content);
7
- let i = 0;
8
- while (i < str.length) {
9
- if (str[i] !== "<") {
10
- i++;
11
- continue;
12
- }
13
- const next = str[i + 1];
14
- if (next === "!" || next === "?") {
15
- const end = str.indexOf(">", i);
16
- if (end === -1) break;
17
- i = end + 1;
18
- continue;
19
- }
20
- if (next && /[a-zA-Z]/.test(next)) {
21
- const tagSlice = str.slice(i + 1);
22
- const nameEnd = tagSlice.search(/[\s/>]/);
23
- if (nameEnd === -1) break;
24
- const insertAt = i + 1 + nameEnd;
25
- return `${str.slice(0, insertAt)} data-eco-trigger="${triggerId}"${str.slice(insertAt)}`;
26
- }
27
- break;
28
- }
29
- return str;
30
- }
31
- function wrapWithScriptsInjector(content, lazyGroups) {
32
- const wrappedContent = String(content);
33
- const injectorMapScript = buildInjectorMapScript(lazyGroups ?? []);
34
- return `<scripts-injector><script type="ecopages/injector-map">${injectorMapScript}<\/script>${wrappedContent}<\/scripts-injector>`;
35
- }
1
+ import {
2
+ addTriggerAttribute,
3
+ isThenable,
4
+ wrapWithScriptsInjector
5
+ } from "../route-renderer/orchestration/render-output.utils.js";
36
6
  export {
37
7
  addTriggerAttribute,
38
8
  isThenable,
@@ -1,4 +1,4 @@
1
- import type { ResolvedLazyTrigger } from '../public-types.js';
1
+ import type { ResolvedLazyTrigger } from '../types/public-types.js';
2
2
  /**
3
3
  * Serializes resolved lazy triggers into a JSON string safe for embedding
4
4
  * inside an inline `<script type="ecopages/global-injector-map">` tag.
@@ -10,7 +10,7 @@ import type { ResolvedLazyTrigger } from '../public-types.js';
10
10
  */
11
11
  export declare function buildGlobalInjectorMapScript(triggers: ResolvedLazyTrigger[]): string;
12
12
  /**
13
- * Builds the inline module script that boots the global lazy injector on the client.
13
+ * Builds the module source that boots the global lazy injector on the client.
14
14
  * Emitted once per page alongside the `ecopages/global-injector-map` script block.
15
15
  */
16
- export declare function buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl: string): string;
16
+ export declare function buildGlobalInjectorBootstrapContent(globalInjectorModuleSpecifier: string): string;
@@ -37,8 +37,8 @@ function buildGlobalInjectorMapScript(triggers) {
37
37
  const map = buildGlobalInjectorMap(triggers);
38
38
  return JSON.stringify(map).replace(/<\/script/gi, "<\\/script");
39
39
  }
40
- function buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl) {
41
- return `import { initGlobalInjector } from ${JSON.stringify(globalInjectorModuleUrl)};
40
+ function buildGlobalInjectorBootstrapContent(globalInjectorModuleSpecifier) {
41
+ return `import { initGlobalInjector } from ${JSON.stringify(globalInjectorModuleSpecifier)};
42
42
 
43
43
  function pruneStaleTriggerMaps() {
44
44
  const mapScripts = Array.from(document.querySelectorAll('script[type="ecopages/global-injector-map"]'));
@@ -1,8 +1,8 @@
1
- import type { ResolvedLazyScriptGroup } from '../public-types.js';
1
+ import type { ResolvedLazyScriptGroup } from '../types/public-types.js';
2
2
  /**
3
3
  * Creates a safe JSON payload string for `<script type="ecopages/injector-map">`.
4
4
  *
5
5
  * @param lazyGroups Lazy script groups resolved during dependency processing.
6
6
  * @returns Escaped JSON string safe for inline script embedding.
7
7
  */
8
- export declare function buildInjectorMapScript(lazyGroups: ResolvedLazyScriptGroup[]): string;
8
+ export declare function buildInjectorMapScript(lazyGroups: readonly ResolvedLazyScriptGroup[]): string;
@@ -1,2 +1,3 @@
1
1
  export { HttpError } from './http-error.js';
2
2
  export type { HttpErrorDetails, HttpErrorJson } from './http-error.js';
3
+ export { LocalsAccessError } from './locals-access-error.js';
@@ -1,4 +1,6 @@
1
1
  import { HttpError } from "./http-error.js";
2
+ import { LocalsAccessError } from "./locals-access-error.js";
2
3
  export {
3
- HttpError
4
+ HttpError,
5
+ LocalsAccessError
4
6
  };
@@ -0,0 +1,26 @@
1
+ # HMR Layer
2
+
3
+ This directory contains the framework-owned hot-update strategy contracts used by runtime adapters and integrations.
4
+
5
+ ## Purpose
6
+
7
+ The HMR layer separates change classification from update execution.
8
+
9
+ It is responsible for:
10
+
11
+ - defining the shared HMR manager and strategy contracts
12
+ - letting integrations contribute framework-specific update strategies
13
+ - keeping adapter transports independent from update policy
14
+
15
+ ## How It Fits
16
+
17
+ 1. `ProjectWatcher` observes file changes.
18
+ 2. `DevelopmentInvalidationService` classifies the change.
19
+ 3. The active HMR manager selects a strategy.
20
+ 4. The strategy coordinates browser rebuilds, metadata reloads, and client broadcasts.
21
+
22
+ ## Design Rule
23
+
24
+ Generic invalidation policy belongs in core services.
25
+ Framework-specific update behavior belongs in HMR strategies.
26
+ Runtime-specific WebSocket or event-stream transport belongs in adapters.