@ecopages/core 0.2.0-alpha.3 → 0.2.0-alpha.31

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 (436) hide show
  1. package/CHANGELOG.md +66 -66
  2. package/README.md +268 -14
  3. package/package.json +72 -108
  4. package/src/adapters/README.md +39 -0
  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 +5 -12
  11. package/src/adapters/bun/create-app.js +43 -7
  12. package/src/adapters/bun/hmr-manager.d.ts +72 -26
  13. package/src/adapters/bun/hmr-manager.js +153 -81
  14. package/src/adapters/bun/index.d.ts +2 -3
  15. package/src/adapters/bun/index.js +3 -3
  16. package/src/adapters/bun/server-adapter.d.ts +5 -5
  17. package/src/adapters/bun/server-adapter.js +63 -38
  18. package/src/adapters/bun/server-lifecycle.d.ts +28 -17
  19. package/src/adapters/bun/server-lifecycle.js +34 -62
  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 +2 -6
  23. package/src/adapters/index.js +2 -8
  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 +79 -21
  28. package/src/adapters/node/node-hmr-manager.js +167 -114
  29. package/src/adapters/node/server-adapter.d.ts +7 -35
  30. package/src/adapters/node/server-adapter.js +55 -107
  31. package/src/adapters/node/static-content-server.d.ts +37 -1
  32. package/src/adapters/node/static-content-server.js +29 -1
  33. package/src/adapters/shared/application-adapter.d.ts +1 -1
  34. package/src/{define-api-handler.d.ts → adapters/shared/define-api-handler.d.ts} +1 -1
  35. package/src/adapters/shared/explicit-static-render-preparation.d.ts +25 -0
  36. package/src/adapters/shared/explicit-static-render-preparation.js +26 -0
  37. package/src/adapters/shared/explicit-static-route-matcher.d.ts +7 -4
  38. package/src/adapters/shared/explicit-static-route-matcher.js +14 -13
  39. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +7 -10
  40. package/src/adapters/shared/file-route-middleware-pipeline.js +3 -11
  41. package/src/adapters/shared/fs-server-response-factory.d.ts +14 -10
  42. package/src/adapters/shared/fs-server-response-factory.js +11 -27
  43. package/src/adapters/shared/fs-server-response-matcher.d.ts +20 -16
  44. package/src/adapters/shared/fs-server-response-matcher.js +76 -45
  45. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
  46. package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
  47. package/src/adapters/shared/hmr-html-response.d.ts +22 -0
  48. package/src/adapters/shared/hmr-html-response.js +32 -0
  49. package/src/adapters/shared/render-context.d.ts +4 -3
  50. package/src/adapters/shared/render-context.js +6 -3
  51. package/src/adapters/shared/runtime-bootstrap.d.ts +38 -0
  52. package/src/adapters/shared/runtime-bootstrap.js +43 -0
  53. package/src/adapters/shared/server-adapter.d.ts +33 -12
  54. package/src/adapters/shared/server-adapter.js +208 -132
  55. package/src/adapters/shared/server-route-handler.d.ts +5 -5
  56. package/src/adapters/shared/server-route-handler.js +7 -16
  57. package/src/adapters/shared/server-static-builder.d.ts +41 -8
  58. package/src/adapters/shared/server-static-builder.js +65 -11
  59. package/src/build/README.md +107 -0
  60. package/src/build/build-adapter.d.ts +168 -3
  61. package/src/build/build-adapter.js +604 -16
  62. package/src/build/build-manifest.d.ts +27 -0
  63. package/src/build/build-manifest.js +30 -0
  64. package/src/build/dev-build-coordinator.d.ts +72 -0
  65. package/src/build/dev-build-coordinator.js +154 -0
  66. package/src/build/esbuild-build-adapter.d.ts +15 -6
  67. package/src/build/esbuild-build-adapter.js +189 -74
  68. package/src/build/runtime-build-executor.d.ts +14 -0
  69. package/src/build/runtime-build-executor.js +22 -0
  70. package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
  71. package/src/build/runtime-specifier-alias-plugin.js +35 -0
  72. package/src/build/runtime-specifier-aliases.d.ts +5 -0
  73. package/src/build/runtime-specifier-aliases.js +95 -0
  74. package/src/config/README.md +36 -0
  75. package/src/config/config-builder.d.ts +54 -29
  76. package/src/config/config-builder.js +255 -49
  77. package/src/{constants.d.ts → config/constants.d.ts} +13 -0
  78. package/src/{constants.js → config/constants.js} +4 -0
  79. package/src/declarations.d.ts +19 -14
  80. package/src/dev/host-runtime.d.ts +10 -0
  81. package/src/dev/host-runtime.js +24 -0
  82. package/src/dev/sc-server.d.ts +1 -1
  83. package/src/dev/sc-server.js +1 -1
  84. package/src/eco/README.md +70 -16
  85. package/src/eco/eco.browser.d.ts +2 -0
  86. package/src/eco/eco.browser.js +83 -0
  87. package/src/eco/eco.js +32 -57
  88. package/src/eco/eco.types.d.ts +14 -6
  89. package/src/eco/eco.utils.d.ts +1 -40
  90. package/src/eco/eco.utils.js +5 -35
  91. package/src/eco/global-injector-map.d.ts +3 -3
  92. package/src/eco/global-injector-map.js +2 -2
  93. package/src/eco/lazy-injector-map.d.ts +2 -2
  94. package/src/errors/index.d.ts +1 -0
  95. package/src/errors/index.js +3 -1
  96. package/src/hmr/README.md +26 -0
  97. package/src/hmr/client/hmr-runtime.d.ts +1 -6
  98. package/src/hmr/client/hmr-runtime.js +38 -7
  99. package/src/hmr/hmr-strategy.d.ts +16 -13
  100. package/src/hmr/hmr-strategy.js +22 -7
  101. package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
  102. package/src/hmr/hmr.postcss.test.e2e.js +31 -0
  103. package/src/hmr/hmr.test.e2e.js +26 -33
  104. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  105. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  106. package/src/hmr/strategies/js-hmr-strategy.d.ts +46 -48
  107. package/src/hmr/strategies/js-hmr-strategy.js +64 -74
  108. package/src/index.browser.d.ts +2 -2
  109. package/src/index.browser.js +1 -1
  110. package/src/index.d.ts +4 -3
  111. package/src/index.js +16 -5
  112. package/src/integrations/ghtml/ghtml-renderer.d.ts +3 -2
  113. package/src/integrations/ghtml/ghtml-renderer.js +27 -30
  114. package/src/integrations/ghtml/ghtml.constants.d.ts +1 -0
  115. package/src/integrations/ghtml/ghtml.constants.js +4 -0
  116. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -6
  117. package/src/integrations/ghtml/ghtml.plugin.js +3 -4
  118. package/src/plugins/README.md +35 -0
  119. package/src/plugins/alias-resolver-plugin.js +17 -3
  120. package/src/plugins/eco-component-meta-plugin.d.ts +14 -1
  121. package/src/plugins/eco-component-meta-plugin.js +27 -22
  122. package/src/plugins/foreign-jsx-override-plugin.d.ts +31 -0
  123. package/src/plugins/foreign-jsx-override-plugin.js +35 -0
  124. package/src/plugins/integration-plugin.d.ts +142 -29
  125. package/src/plugins/integration-plugin.js +103 -14
  126. package/src/plugins/processor.d.ts +17 -2
  127. package/src/plugins/processor.js +22 -3
  128. package/src/plugins/runtime-capability.d.ts +9 -0
  129. package/src/plugins/source-transform.d.ts +46 -0
  130. package/src/plugins/source-transform.js +71 -0
  131. package/src/route-renderer/GRAPH.md +83 -325
  132. package/src/route-renderer/README.md +73 -90
  133. package/src/route-renderer/orchestration/component-render-context.d.ts +89 -0
  134. package/src/route-renderer/orchestration/component-render-context.js +147 -0
  135. package/src/route-renderer/orchestration/declared-ownership-graph.d.ts +18 -0
  136. package/src/route-renderer/orchestration/declared-ownership-graph.js +34 -0
  137. package/src/route-renderer/orchestration/foreign-subtree-execution.service.d.ts +108 -0
  138. package/src/route-renderer/orchestration/foreign-subtree-execution.service.js +206 -0
  139. package/src/route-renderer/orchestration/integration-renderer.d.ts +516 -0
  140. package/src/route-renderer/orchestration/integration-renderer.js +909 -0
  141. package/src/route-renderer/orchestration/ownership-planning.service.d.ts +24 -0
  142. package/src/route-renderer/orchestration/ownership-planning.service.js +63 -0
  143. package/src/route-renderer/orchestration/ownership-validation.service.d.ts +29 -0
  144. package/src/route-renderer/orchestration/ownership-validation.service.js +53 -0
  145. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +2 -0
  146. package/src/route-renderer/orchestration/processed-asset-dedupe.js +23 -0
  147. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.d.ts +90 -0
  148. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.js +158 -0
  149. package/src/route-renderer/orchestration/render-output.utils.d.ts +66 -0
  150. package/src/route-renderer/orchestration/render-output.utils.js +171 -0
  151. package/src/route-renderer/orchestration/route-render-orchestrator.d.ts +120 -0
  152. package/src/route-renderer/orchestration/route-render-orchestrator.js +388 -0
  153. package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
  154. package/src/route-renderer/orchestration/template-serialization.js +45 -0
  155. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +37 -0
  156. package/src/route-renderer/page-loading/component-dependency-collection.js +132 -0
  157. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +24 -0
  158. package/src/route-renderer/page-loading/declared-asset-collection.js +106 -0
  159. package/src/route-renderer/{dependency-resolver.d.ts → page-loading/dependency-resolver.d.ts} +15 -4
  160. package/src/route-renderer/page-loading/dependency-resolver.js +115 -0
  161. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +11 -0
  162. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +57 -0
  163. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +45 -0
  164. package/src/route-renderer/page-loading/lazy-entry-collection.js +105 -0
  165. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +19 -0
  166. package/src/route-renderer/page-loading/lazy-trigger-planning.js +40 -0
  167. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +5 -0
  168. package/src/route-renderer/page-loading/module-declaration-aggregation.js +33 -0
  169. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +3 -0
  170. package/src/route-renderer/page-loading/module-declaration-scripts.js +18 -0
  171. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +13 -0
  172. package/src/route-renderer/page-loading/page-dependency-bundling.js +137 -0
  173. package/src/route-renderer/page-loading/page-module-loader.d.ts +90 -0
  174. package/src/route-renderer/{page-module-loader.js → page-loading/page-module-loader.js} +39 -14
  175. package/src/route-renderer/route-renderer.d.ts +57 -14
  176. package/src/route-renderer/route-renderer.js +30 -18
  177. package/src/router/README.md +94 -0
  178. package/src/router/client/link-intent.d.ts +53 -0
  179. package/src/router/client/link-intent.js +34 -0
  180. package/src/router/client/link-intent.test.browser.d.ts +1 -0
  181. package/src/router/client/link-intent.test.browser.js +43 -0
  182. package/src/router/client/navigation-coordinator.d.ts +169 -0
  183. package/src/router/client/navigation-coordinator.js +215 -0
  184. package/src/router/server/route-registry.d.ts +78 -0
  185. package/src/router/server/route-registry.js +262 -0
  186. package/src/services/README.md +28 -0
  187. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +3 -0
  188. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +56 -0
  189. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +103 -0
  190. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.js +124 -89
  191. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.d.ts +1 -1
  192. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.js +2 -2
  193. package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.d.ts +16 -1
  194. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +55 -0
  195. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +49 -0
  196. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
  197. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
  198. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +30 -0
  199. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +65 -0
  200. package/src/services/assets/asset-processing-service/index.d.ts +6 -0
  201. package/src/services/assets/asset-processing-service/index.js +6 -0
  202. package/src/services/assets/asset-processing-service/page-package.d.ts +3 -0
  203. package/src/services/assets/asset-processing-service/page-package.js +74 -0
  204. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.d.ts +2 -2
  205. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.d.ts +2 -2
  206. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.d.ts +1 -1
  207. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.js +11 -5
  208. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +22 -0
  209. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +136 -0
  210. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
  211. package/src/services/assets/asset-processing-service/processors/index.js +5 -0
  212. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.d.ts +3 -2
  213. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +119 -0
  214. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.d.ts +4 -3
  215. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.js +28 -7
  216. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.d.ts +3 -3
  217. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.js +12 -7
  218. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.d.ts +5 -2
  219. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +59 -0
  220. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  221. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.js +9 -3
  222. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +18 -0
  223. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +45 -0
  224. package/src/services/assets/browser-bundle.service.d.ts +73 -0
  225. package/src/services/assets/browser-bundle.service.js +41 -0
  226. package/src/services/{page-request-cache-coordinator.service.d.ts → cache/page-request-cache-coordinator.service.d.ts} +2 -2
  227. package/src/services/{page-request-cache-coordinator.service.js → cache/page-request-cache-coordinator.service.js} +3 -1
  228. package/src/services/html/html-rewriter-provider.service.d.ts +37 -0
  229. package/src/services/html/html-rewriter-provider.service.js +68 -0
  230. package/src/services/html/html-transformer.service.d.ts +87 -0
  231. package/src/services/html/html-transformer.service.js +216 -0
  232. package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
  233. package/src/services/invalidation/development-invalidation.service.js +190 -0
  234. package/src/services/module-loading/app-module-loader.service.d.ts +7 -0
  235. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +24 -0
  236. package/src/services/module-loading/app-server-module-transpiler.service.js +115 -0
  237. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  238. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  239. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  240. package/src/services/module-loading/module-loading-types.js +0 -0
  241. package/src/services/module-loading/node-bootstrap-plugin.d.ts +42 -0
  242. package/src/services/module-loading/node-bootstrap-plugin.js +216 -0
  243. package/src/services/module-loading/page-module-import.service.d.ts +76 -0
  244. package/src/services/module-loading/page-module-import.service.js +170 -0
  245. package/src/services/module-loading/server-module-transpiler.service.d.ts +63 -0
  246. package/src/services/module-loading/server-module-transpiler.service.js +64 -0
  247. package/src/services/module-loading/source-module-support.d.ts +5 -0
  248. package/src/services/module-loading/source-module-support.js +8 -0
  249. package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
  250. package/src/services/runtime-state/dev-graph.service.js +162 -0
  251. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
  252. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
  253. package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
  254. package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
  255. package/src/services/{schema-validation-service.d.ts → validation/schema-validation-service.d.ts} +1 -1
  256. package/src/static-site-generator/README.md +26 -0
  257. package/src/static-site-generator/static-site-generator.d.ts +67 -20
  258. package/src/static-site-generator/static-site-generator.js +182 -138
  259. package/src/{internal-types.d.ts → types/internal-types.d.ts} +62 -30
  260. package/src/types/internal-types.js +0 -0
  261. package/src/{public-types.d.ts → types/public-types.d.ts} +201 -32
  262. package/src/types/public-types.js +0 -0
  263. package/src/utils/html-escaping.d.ts +7 -0
  264. package/src/utils/html-escaping.js +6 -0
  265. package/src/utils/locals-utils.d.ts +1 -1
  266. package/src/utils/parse-cli-args.d.ts +4 -1
  267. package/src/utils/parse-cli-args.js +16 -1
  268. package/src/utils/resolve-work-dir.d.ts +11 -0
  269. package/src/utils/resolve-work-dir.js +31 -0
  270. package/src/watchers/project-watcher.d.ts +40 -29
  271. package/src/watchers/project-watcher.js +126 -116
  272. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  273. package/src/watchers/project-watcher.test-helpers.js +6 -6
  274. package/src/adapters/abstract/application-adapter.ts +0 -337
  275. package/src/adapters/abstract/router-adapter.ts +0 -30
  276. package/src/adapters/abstract/server-adapter.ts +0 -79
  277. package/src/adapters/bun/client-bridge.ts +0 -62
  278. package/src/adapters/bun/create-app.ts +0 -189
  279. package/src/adapters/bun/define-api-handler.d.ts +0 -61
  280. package/src/adapters/bun/define-api-handler.ts +0 -114
  281. package/src/adapters/bun/hmr-manager.ts +0 -296
  282. package/src/adapters/bun/index.ts +0 -3
  283. package/src/adapters/bun/server-adapter.ts +0 -492
  284. package/src/adapters/bun/server-lifecycle.ts +0 -154
  285. package/src/adapters/index.ts +0 -6
  286. package/src/adapters/node/create-app.ts +0 -179
  287. package/src/adapters/node/index.d.ts +0 -4
  288. package/src/adapters/node/index.js +0 -8
  289. package/src/adapters/node/index.ts +0 -9
  290. package/src/adapters/node/node-client-bridge.ts +0 -79
  291. package/src/adapters/node/node-hmr-manager.ts +0 -289
  292. package/src/adapters/node/server-adapter.ts +0 -561
  293. package/src/adapters/node/static-content-server.ts +0 -203
  294. package/src/adapters/shared/api-response.ts +0 -104
  295. package/src/adapters/shared/application-adapter.ts +0 -199
  296. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -134
  297. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -123
  298. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  299. package/src/adapters/shared/fs-server-response-matcher.ts +0 -198
  300. package/src/adapters/shared/render-context.ts +0 -105
  301. package/src/adapters/shared/server-adapter.ts +0 -442
  302. package/src/adapters/shared/server-route-handler.ts +0 -166
  303. package/src/adapters/shared/server-static-builder.ts +0 -82
  304. package/src/build/build-adapter.ts +0 -132
  305. package/src/build/build-types.ts +0 -83
  306. package/src/build/esbuild-build-adapter.ts +0 -510
  307. package/src/config/config-builder.ts +0 -474
  308. package/src/constants.ts +0 -39
  309. package/src/create-app.ts +0 -87
  310. package/src/define-api-handler.js +0 -15
  311. package/src/define-api-handler.ts +0 -66
  312. package/src/dev/sc-server.ts +0 -143
  313. package/src/eco/component-render-context.d.ts +0 -105
  314. package/src/eco/component-render-context.js +0 -77
  315. package/src/eco/component-render-context.ts +0 -202
  316. package/src/eco/eco.ts +0 -221
  317. package/src/eco/eco.types.ts +0 -202
  318. package/src/eco/eco.utils.ts +0 -89
  319. package/src/eco/global-injector-map.ts +0 -112
  320. package/src/eco/lazy-injector-map.ts +0 -120
  321. package/src/eco/module-dependencies.ts +0 -75
  322. package/src/errors/http-error.ts +0 -72
  323. package/src/errors/index.ts +0 -2
  324. package/src/errors/locals-access-error.ts +0 -7
  325. package/src/global/app-logger.ts +0 -4
  326. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
  327. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  328. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  329. package/src/hmr/client/hmr-runtime.ts +0 -121
  330. package/src/hmr/hmr-strategy.ts +0 -172
  331. package/src/hmr/hmr.test.e2e.ts +0 -75
  332. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  333. package/src/hmr/strategies/js-hmr-strategy.ts +0 -320
  334. package/src/index.browser.ts +0 -3
  335. package/src/index.ts +0 -5
  336. package/src/integrations/ghtml/ghtml-renderer.ts +0 -93
  337. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  338. package/src/internal-types.ts +0 -212
  339. package/src/plugins/alias-resolver-plugin.ts +0 -45
  340. package/src/plugins/eco-component-meta-plugin.ts +0 -474
  341. package/src/plugins/integration-plugin.ts +0 -184
  342. package/src/plugins/processor.ts +0 -220
  343. package/src/public-types.ts +0 -1260
  344. package/src/route-renderer/component-graph-executor.d.ts +0 -32
  345. package/src/route-renderer/component-graph-executor.js +0 -31
  346. package/src/route-renderer/component-graph-executor.ts +0 -84
  347. package/src/route-renderer/component-graph.d.ts +0 -42
  348. package/src/route-renderer/component-graph.js +0 -72
  349. package/src/route-renderer/component-graph.ts +0 -159
  350. package/src/route-renderer/component-marker.d.ts +0 -52
  351. package/src/route-renderer/component-marker.js +0 -46
  352. package/src/route-renderer/component-marker.ts +0 -117
  353. package/src/route-renderer/dependency-resolver.js +0 -428
  354. package/src/route-renderer/dependency-resolver.ts +0 -596
  355. package/src/route-renderer/html-post-processing.service.d.ts +0 -40
  356. package/src/route-renderer/html-post-processing.service.js +0 -86
  357. package/src/route-renderer/html-post-processing.service.ts +0 -103
  358. package/src/route-renderer/integration-renderer.d.ts +0 -339
  359. package/src/route-renderer/integration-renderer.js +0 -526
  360. package/src/route-renderer/integration-renderer.ts +0 -696
  361. package/src/route-renderer/marker-graph-resolver.d.ts +0 -76
  362. package/src/route-renderer/marker-graph-resolver.js +0 -93
  363. package/src/route-renderer/marker-graph-resolver.ts +0 -153
  364. package/src/route-renderer/page-module-loader.d.ts +0 -61
  365. package/src/route-renderer/page-module-loader.ts +0 -153
  366. package/src/route-renderer/render-execution.service.d.ts +0 -69
  367. package/src/route-renderer/render-execution.service.js +0 -91
  368. package/src/route-renderer/render-execution.service.ts +0 -158
  369. package/src/route-renderer/render-preparation.service.d.ts +0 -112
  370. package/src/route-renderer/render-preparation.service.js +0 -243
  371. package/src/route-renderer/render-preparation.service.ts +0 -358
  372. package/src/route-renderer/route-renderer.ts +0 -80
  373. package/src/router/fs-router-scanner.d.ts +0 -41
  374. package/src/router/fs-router-scanner.js +0 -155
  375. package/src/router/fs-router-scanner.ts +0 -217
  376. package/src/router/fs-router.d.ts +0 -26
  377. package/src/router/fs-router.js +0 -100
  378. package/src/router/fs-router.ts +0 -122
  379. package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
  380. package/src/services/asset-processing-service/asset-processing.service.ts +0 -306
  381. package/src/services/asset-processing-service/asset.factory.ts +0 -105
  382. package/src/services/asset-processing-service/assets.types.ts +0 -112
  383. package/src/services/asset-processing-service/index.d.ts +0 -3
  384. package/src/services/asset-processing-service/index.js +0 -3
  385. package/src/services/asset-processing-service/index.ts +0 -3
  386. package/src/services/asset-processing-service/processor.interface.ts +0 -27
  387. package/src/services/asset-processing-service/processor.registry.ts +0 -18
  388. package/src/services/asset-processing-service/processors/base/base-processor.ts +0 -76
  389. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +0 -16
  390. package/src/services/asset-processing-service/processors/base/base-script-processor.js +0 -80
  391. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +0 -105
  392. package/src/services/asset-processing-service/processors/index.d.ts +0 -5
  393. package/src/services/asset-processing-service/processors/index.js +0 -5
  394. package/src/services/asset-processing-service/processors/index.ts +0 -5
  395. package/src/services/asset-processing-service/processors/script/content-script.processor.js +0 -57
  396. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  397. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  398. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +0 -84
  399. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
  400. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  401. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -77
  402. package/src/services/cache/cache.types.ts +0 -126
  403. package/src/services/cache/index.ts +0 -18
  404. package/src/services/cache/memory-cache-store.ts +0 -130
  405. package/src/services/cache/page-cache-service.ts +0 -202
  406. package/src/services/html-transformer.service.d.ts +0 -50
  407. package/src/services/html-transformer.service.js +0 -163
  408. package/src/services/html-transformer.service.ts +0 -217
  409. package/src/services/page-module-import.service.d.ts +0 -37
  410. package/src/services/page-module-import.service.js +0 -88
  411. package/src/services/page-module-import.service.ts +0 -129
  412. package/src/services/page-request-cache-coordinator.service.ts +0 -128
  413. package/src/services/schema-validation-service.ts +0 -204
  414. package/src/services/validation/standard-schema.types.ts +0 -68
  415. package/src/static-site-generator/static-site-generator.ts +0 -359
  416. package/src/utils/css.d.ts +0 -1
  417. package/src/utils/css.js +0 -7
  418. package/src/utils/css.ts +0 -5
  419. package/src/utils/deep-merge.ts +0 -47
  420. package/src/utils/hash.ts +0 -5
  421. package/src/utils/html.ts +0 -1
  422. package/src/utils/invariant.ts +0 -15
  423. package/src/utils/locals-utils.ts +0 -37
  424. package/src/utils/parse-cli-args.ts +0 -83
  425. package/src/utils/path-utils.module.ts +0 -14
  426. package/src/utils/runtime.ts +0 -44
  427. package/src/utils/server-utils.module.ts +0 -67
  428. package/src/watchers/project-watcher.test-helpers.ts +0 -41
  429. package/src/watchers/project-watcher.ts +0 -344
  430. /package/src/adapters/{bun → shared}/define-api-handler.js +0 -0
  431. /package/src/{internal-types.js → plugins/runtime-capability.js} +0 -0
  432. /package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.js +0 -0
  433. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.js +0 -0
  434. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.js +0 -0
  435. /package/src/{public-types.js → services/module-loading/app-module-loader.service.js} +0 -0
  436. /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,83 @@
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 html(options) {
14
+ return createComponentFactory(options);
15
+ }
16
+ function layout(options) {
17
+ return createComponentFactory(options);
18
+ }
19
+ function page(options) {
20
+ const {
21
+ layout: pageLayout,
22
+ dependencies,
23
+ render,
24
+ staticPaths: staticPaths2,
25
+ staticProps: staticProps2,
26
+ metadata: metadata2,
27
+ cache,
28
+ requires,
29
+ middleware
30
+ } = options;
31
+ const pageComponent = createComponentFactory({
32
+ __eco: options.__eco,
33
+ integration: options.integration,
34
+ dependencies: pageLayout ? {
35
+ ...dependencies,
36
+ components: [...dependencies?.components ?? [], pageLayout]
37
+ } : dependencies,
38
+ render
39
+ });
40
+ if (pageLayout && pageComponent.config) {
41
+ pageComponent.config.layout = pageLayout;
42
+ }
43
+ if (staticPaths2) {
44
+ pageComponent.staticPaths = staticPaths2;
45
+ }
46
+ if (staticProps2) {
47
+ pageComponent.staticProps = staticProps2;
48
+ }
49
+ if (metadata2) {
50
+ pageComponent.metadata = metadata2;
51
+ }
52
+ if (cache) {
53
+ pageComponent.cache = cache;
54
+ }
55
+ if (requires) {
56
+ pageComponent.requires = requires;
57
+ }
58
+ if (middleware) {
59
+ pageComponent.middleware = middleware;
60
+ }
61
+ return pageComponent;
62
+ }
63
+ function metadata(fn) {
64
+ return fn;
65
+ }
66
+ function staticPaths(fn) {
67
+ return fn;
68
+ }
69
+ function staticProps(fn) {
70
+ return fn;
71
+ }
72
+ const eco = {
73
+ component,
74
+ html,
75
+ layout,
76
+ page,
77
+ metadata,
78
+ staticPaths,
79
+ staticProps
80
+ };
81
+ export {
82
+ eco
83
+ };
package/src/eco/eco.js CHANGED
@@ -1,60 +1,27 @@
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
+ interceptForeignChild
4
+ } from "../route-renderer/orchestration/component-render-context.js";
5
+ import { isThenable } from "../route-renderer/orchestration/render-output.utils.js";
4
6
  function createComponentFactory(options) {
5
7
  const integrationName = options.integration ?? options.__eco?.integration;
6
8
  const comp = ((props) => {
7
- const renderContext = getComponentRenderContext();
8
- const shouldEmitMarker = renderContext !== void 0 && renderContext.boundaryContext.decideBoundaryRender({
9
- currentIntegration: renderContext.currentIntegration,
10
- targetIntegration: integrationName,
11
- component: comp
12
- }) === "defer";
13
- if (shouldEmitMarker && renderContext) {
14
- const nodeId = createNodeId(renderContext);
15
- const propsRef = createPropsRef(renderContext);
16
- const componentRef = comp.config?.__eco?.id ?? comp.config?.__eco?.file;
17
- if (!componentRef) {
18
- throw new Error(
19
- "[ecopages] Missing component reference metadata for cross-integration marker emission."
20
- );
21
- }
22
- const componentProps = props ?? {};
23
- renderContext.propsByRef[propsRef] = componentProps;
24
- let slotRef;
25
- const children = componentProps.children;
26
- if (typeof children === "string" && children.includes("<eco-marker")) {
27
- const childMarkers = parseComponentMarkers(children);
28
- if (childMarkers.length > 0) {
29
- slotRef = createSlotRef(renderContext);
30
- renderContext.slotChildrenByRef[slotRef] = childMarkers.map((marker) => marker.nodeId);
31
- }
32
- }
33
- return createComponentMarker({
34
- nodeId,
35
- integration: integrationName,
36
- componentRef,
37
- propsRef,
38
- slotRef
39
- });
9
+ const componentProps = props ?? {};
10
+ const renderInline = () => finalizeComponentRender(comp, options.render(props));
11
+ const foreignChildRender = interceptForeignChild({
12
+ component: comp,
13
+ props: componentProps,
14
+ targetIntegration: integrationName
15
+ });
16
+ if (isThenable(foreignChildRender)) {
17
+ return foreignChildRender.then(
18
+ (resolvedForeignChildRender) => resolvedForeignChildRender !== void 0 ? resolvedForeignChildRender : renderInline()
19
+ );
40
20
  }
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);
21
+ if (foreignChildRender !== void 0) {
22
+ return foreignChildRender;
49
23
  }
50
- const lazyGroups = comp.config?._resolvedLazyScripts;
51
- if (lazyGroups && lazyGroups.length > 0) {
52
- if (isThenable(content)) {
53
- return content.then((resolvedContent) => wrapWithScriptsInjector(resolvedContent, lazyGroups));
54
- }
55
- return wrapWithScriptsInjector(content, lazyGroups);
56
- }
57
- return content;
24
+ return renderInline();
58
25
  });
59
26
  comp.config = {
60
27
  __eco: options.__eco,
@@ -66,20 +33,26 @@ function createComponentFactory(options) {
66
33
  function component(options) {
67
34
  return createComponentFactory(options);
68
35
  }
36
+ function html(options) {
37
+ return createComponentFactory(options);
38
+ }
39
+ function layout(options) {
40
+ return createComponentFactory(options);
41
+ }
69
42
  function page(options) {
70
- const { layout, dependencies, render, staticPaths: staticPaths2, staticProps: staticProps2, metadata: metadata2, cache, requires, middleware } = options;
43
+ const { layout: layout2, dependencies, render, staticPaths: staticPaths2, staticProps: staticProps2, metadata: metadata2, cache, requires, middleware } = options;
71
44
  const componentOptions = {
72
45
  __eco: options.__eco,
73
46
  integration: options.integration,
74
- dependencies: layout ? {
47
+ dependencies: layout2 ? {
75
48
  ...dependencies,
76
- components: [...dependencies?.components || [], layout]
49
+ components: [...dependencies?.components || [], layout2]
77
50
  } : dependencies,
78
51
  render
79
52
  };
80
53
  const pageComponent = createComponentFactory(componentOptions);
81
- if (layout && pageComponent.config) {
82
- pageComponent.config.layout = layout;
54
+ if (layout2 && pageComponent.config) {
55
+ pageComponent.config.layout = layout2;
83
56
  }
84
57
  if (staticPaths2) pageComponent.staticPaths = staticPaths2;
85
58
  if (staticProps2) pageComponent.staticProps = staticProps2;
@@ -100,6 +73,8 @@ function staticProps(fn) {
100
73
  }
101
74
  const eco = {
102
75
  component,
76
+ html,
77
+ layout,
103
78
  page,
104
79
  metadata,
105
80
  staticPaths,
@@ -2,7 +2,7 @@
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
7
  type WithRequiredLocals<K extends keyof RequestLocals> = Omit<RequestLocals, K> & {
8
8
  [P in K]-?: Exclude<RequestLocals[P], null | undefined>;
@@ -26,6 +26,8 @@ export interface ComponentOptions<P, E = EcoPagesElement> {
26
26
  dependencies?: EcoComponentDependencies;
27
27
  render: (props: P) => E | Promise<E>;
28
28
  }
29
+ export type HtmlOptions<E = EcoPagesElement> = ComponentOptions<HtmlTemplateProps, E>;
30
+ export type LayoutOptions<E = EcoPagesElement> = ComponentOptions<LayoutProps<E>, E>;
29
31
  /**
30
32
  * Base options shared by all page variants
31
33
  */
@@ -34,9 +36,7 @@ export interface PageOptionsBase<T, E = EcoPagesElement> {
34
36
  __eco?: EcoInjectedMeta;
35
37
  integration?: string;
36
38
  dependencies?: EcoComponentDependencies;
37
- layout?: EcoComponent<{
38
- children: E;
39
- } & Partial<RequestPageContext>>;
39
+ layout?: EcoPageLayoutComponent<E>;
40
40
  /**
41
41
  * Define static paths for dynamic routes (e.g., [slug].tsx).
42
42
  * Returns all possible paths that should be pre-rendered at build time.
@@ -86,7 +86,7 @@ interface PageOptionsWithMiddleware<T, E = EcoPagesElement> extends PageOptionsB
86
86
  * Request-time middleware for file-based routes.
87
87
  * Runs before rendering and can short-circuit by returning a Response.
88
88
  */
89
- middleware: Middleware[];
89
+ middleware: FileRouteMiddleware[];
90
90
  }
91
91
  /**
92
92
  * Options for creating a page with eco.page()
@@ -127,7 +127,7 @@ export type EcoPageComponent<T> = EcoComponent<PagePropsFor<T> & Partial<Request
127
127
  metadata?: GetMetadata<T>;
128
128
  cache?: CacheStrategy;
129
129
  requires?: PageRequires;
130
- middleware?: Middleware[];
130
+ middleware?: FileRouteMiddleware[];
131
131
  };
132
132
  /**
133
133
  * The eco namespace interface
@@ -139,6 +139,14 @@ export interface Eco {
139
139
  * @template E - Element/return type (EcoPagesElement for Kita, ReactNode for React)
140
140
  */
141
141
  component: <P = {}, E = EcoPagesElement>(options: ComponentOptions<P, E>) => EcoComponent<P, E>;
142
+ /**
143
+ * Create a document shell component for the HTML wrapper.
144
+ */
145
+ html: <E = EcoPagesElement>(options: HtmlOptions<E>) => EcoHtmlComponent<E>;
146
+ /**
147
+ * Create a route layout component.
148
+ */
149
+ layout: <E = EcoPagesElement>(options: LayoutOptions<E>) => EcoLayoutComponent<E>;
142
150
  /**
143
151
  * Create a page component with type-safe props from getStaticProps.
144
152
  * 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.
@@ -2,9 +2,4 @@
2
2
  * Ecopages HMR Runtime
3
3
  * Injected into the browser to handle Hot Module Replacement updates.
4
4
  */
5
- interface HMRPayload {
6
- type: 'reload' | 'error' | 'update' | 'css-update' | 'layout-update';
7
- path?: string;
8
- message?: string;
9
- timestamp?: number;
10
- }
5
+ export {};
@@ -1,3 +1,4 @@
1
+ import { getEcoNavigationRuntime } from "../../router/client/navigation-coordinator.js";
1
2
  (function() {
2
3
  const WS_URL = "ws://" + location.host + "/_hmr";
3
4
  let socket;
@@ -24,14 +25,15 @@
24
25
  });
25
26
  }
26
27
  async function handleMessage(payload) {
28
+ const navigationRuntime = getEcoNavigationRuntime(window);
27
29
  switch (payload.type) {
28
30
  case "reload":
31
+ await waitForNavigationToSettle(navigationRuntime);
29
32
  location.reload();
30
33
  break;
31
34
  case "layout-update": {
32
- const reloadFn = window.__ecopages_reload_current_page__;
33
- if (typeof reloadFn === "function") {
34
- await reloadFn({ clearCache: true });
35
+ await waitForNavigationToSettle(navigationRuntime);
36
+ if (await navigationRuntime.reloadCurrentPage({ clearCache: true, moduleUrl: getActiveHmrModuleUrl() })) {
35
37
  } else {
36
38
  location.reload();
37
39
  }
@@ -55,20 +57,49 @@
55
57
  async function applyUpdate(path, timestamp) {
56
58
  try {
57
59
  const url = path + "?t=" + (timestamp || Date.now());
58
- const handlers = window.__ecopages_hmr_handlers__;
60
+ const handlers = window.__ECO_PAGES__?.hmrHandlers;
61
+ const navigationRuntime = getEcoNavigationRuntime(window);
62
+ await waitForNavigationToSettle(navigationRuntime);
59
63
  if (handlers?.[path]) {
60
64
  await handlers[path](url);
61
65
  return;
62
66
  }
63
67
  await import(url);
64
- const reloadFn = window.__ecopages_reload_current_page__;
65
- if (typeof reloadFn === "function") {
66
- await reloadFn({ clearCache: false });
68
+ if (await navigationRuntime.reloadCurrentPage({ clearCache: false })) {
67
69
  }
68
70
  } catch (e) {
69
71
  console.error("[ecopages] Failed to apply HMR update:", e);
70
72
  }
71
73
  }
74
+ function getActiveHmrModuleUrl() {
75
+ const pageModule = window.__ECO_PAGES__?.page?.module;
76
+ if (pageModule && pageModule.includes("/assets/_hmr/")) {
77
+ return pageModule.split("?")[0];
78
+ }
79
+ const handlerPaths = Object.keys(window.__ECO_PAGES__?.hmrHandlers ?? {});
80
+ return handlerPaths[handlerPaths.length - 1];
81
+ }
82
+ async function waitForNavigationToSettle(navigationRuntime) {
83
+ if (!navigationRuntime.hasPendingNavigationTransaction()) {
84
+ return;
85
+ }
86
+ await new Promise((resolve) => {
87
+ const startedAt = performance.now();
88
+ const timeoutMs = 2e3;
89
+ const poll = () => {
90
+ if (!navigationRuntime.hasPendingNavigationTransaction()) {
91
+ resolve();
92
+ return;
93
+ }
94
+ if (performance.now() - startedAt >= timeoutMs) {
95
+ resolve();
96
+ return;
97
+ }
98
+ requestAnimationFrame(poll);
99
+ };
100
+ requestAnimationFrame(poll);
101
+ });
102
+ }
72
103
  function refreshStylesheet(path) {
73
104
  const filename = path.split("/").pop() || "";
74
105
  const links = document.querySelectorAll('link[rel="stylesheet"]');