@ecopages/core 0.2.0-alpha.26 → 0.2.0-alpha.27

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 (550) hide show
  1. package/README.md +63 -7
  2. package/package.json +73 -249
  3. package/src/adapters/abstract/application-adapter.test.ts +172 -0
  4. package/src/adapters/abstract/application-adapter.ts +379 -0
  5. package/src/adapters/abstract/router-adapter.ts +30 -0
  6. package/src/adapters/abstract/server-adapter.ts +79 -0
  7. package/src/adapters/bun/client-bridge.ts +62 -0
  8. package/src/adapters/bun/create-app.ts +232 -0
  9. package/src/adapters/bun/hmr-manager.test.ts +265 -0
  10. package/src/adapters/bun/hmr-manager.ts +383 -0
  11. package/src/adapters/bun/index.ts +2 -0
  12. package/src/adapters/bun/server-adapter.ts +526 -0
  13. package/src/adapters/bun/server-lifecycle.ts +124 -0
  14. package/src/adapters/create-app.test.ts +10 -0
  15. package/src/adapters/create-app.ts +91 -0
  16. package/src/adapters/index.ts +2 -0
  17. package/src/adapters/node/create-app.test.ts +53 -0
  18. package/src/adapters/node/create-app.ts +183 -0
  19. package/src/adapters/node/node-client-bridge.test.ts +198 -0
  20. package/src/adapters/node/node-client-bridge.ts +79 -0
  21. package/src/adapters/node/node-hmr-manager.test.ts +320 -0
  22. package/src/adapters/node/node-hmr-manager.ts +355 -0
  23. package/src/adapters/node/server-adapter.ts +502 -0
  24. package/src/adapters/node/static-content-server.test.ts +60 -0
  25. package/src/adapters/node/static-content-server.ts +239 -0
  26. package/src/adapters/shared/api-response.test.ts +97 -0
  27. package/src/adapters/shared/api-response.ts +104 -0
  28. package/src/adapters/shared/application-adapter.ts +199 -0
  29. package/src/adapters/shared/define-api-handler.ts +66 -0
  30. package/src/adapters/shared/explicit-static-render-preparation.ts +58 -0
  31. package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
  32. package/src/adapters/shared/explicit-static-route-matcher.ts +131 -0
  33. package/src/adapters/shared/file-route-middleware-pipeline.test.ts +85 -0
  34. package/src/adapters/shared/file-route-middleware-pipeline.ts +118 -0
  35. package/src/adapters/shared/fs-server-response-factory.test.ts +176 -0
  36. package/src/adapters/shared/fs-server-response-factory.ts +96 -0
  37. package/src/adapters/shared/fs-server-response-matcher.test.ts +311 -0
  38. package/src/adapters/shared/fs-server-response-matcher.ts +240 -0
  39. package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
  40. package/src/adapters/shared/hmr-html-response.ts +52 -0
  41. package/src/adapters/shared/hmr-manager.contract.test.ts +228 -0
  42. package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
  43. package/src/adapters/shared/render-context.test.ts +150 -0
  44. package/src/adapters/shared/render-context.ts +123 -0
  45. package/src/adapters/shared/runtime-bootstrap.ts +79 -0
  46. package/src/adapters/shared/server-adapter.test.ts +130 -0
  47. package/src/adapters/shared/server-adapter.ts +562 -0
  48. package/src/adapters/shared/server-route-handler.test.ts +111 -0
  49. package/src/adapters/shared/server-route-handler.ts +153 -0
  50. package/src/adapters/shared/server-static-builder.test.ts +338 -0
  51. package/src/adapters/shared/server-static-builder.ts +170 -0
  52. package/src/build/build-adapter-serialization.test.ts +281 -0
  53. package/src/build/build-adapter.test.ts +1240 -0
  54. package/src/build/build-adapter.ts +1012 -0
  55. package/src/build/build-manifest.ts +54 -0
  56. package/src/build/build-types.ts +83 -0
  57. package/src/build/dev-build-coordinator.ts +220 -0
  58. package/src/build/esbuild-build-adapter.ts +660 -0
  59. package/src/build/runtime-build-executor.test.ts +81 -0
  60. package/src/build/runtime-build-executor.ts +40 -0
  61. package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
  62. package/src/build/runtime-specifier-alias-plugin.ts +62 -0
  63. package/src/build/runtime-specifier-aliases.ts +135 -0
  64. package/src/config/README.md +1 -1
  65. package/src/config/config-builder.test.ts +442 -0
  66. package/src/config/config-builder.ts +737 -0
  67. package/src/config/config-builder.typecheck.test.ts +96 -0
  68. package/src/config/{constants.d.ts → constants.ts} +22 -13
  69. package/src/dev/host-runtime.ts +34 -0
  70. package/src/dev/sc-server.ts +143 -0
  71. package/src/eco/eco.browser.test.ts +43 -0
  72. package/src/eco/eco.browser.ts +118 -0
  73. package/src/eco/eco.test.ts +654 -0
  74. package/src/eco/eco.ts +205 -0
  75. package/src/eco/eco.types.ts +221 -0
  76. package/src/eco/eco.utils.test.ts +219 -0
  77. package/src/eco/eco.utils.ts +5 -0
  78. package/src/eco/global-injector-map.test.ts +42 -0
  79. package/src/eco/global-injector-map.ts +112 -0
  80. package/src/eco/lazy-injector-map.test.ts +66 -0
  81. package/src/eco/lazy-injector-map.ts +120 -0
  82. package/src/eco/module-dependencies.test.ts +30 -0
  83. package/src/eco/module-dependencies.ts +75 -0
  84. package/src/errors/http-error.test.ts +134 -0
  85. package/src/errors/http-error.ts +72 -0
  86. package/src/errors/index.ts +3 -0
  87. package/src/errors/locals-access-error.ts +7 -0
  88. package/src/global/app-logger.ts +4 -0
  89. package/src/global/utils.test.ts +12 -0
  90. 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
  91. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  92. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  93. package/src/hmr/client/hmr-runtime.ts +162 -0
  94. package/src/hmr/hmr-strategy.test.ts +124 -0
  95. package/src/hmr/hmr-strategy.ts +177 -0
  96. package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
  97. package/src/hmr/hmr.test.e2e.ts +66 -0
  98. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  99. package/src/hmr/strategies/js-hmr-strategy.test.ts +334 -0
  100. package/src/hmr/strategies/js-hmr-strategy.ts +314 -0
  101. package/src/index.browser.ts +3 -0
  102. package/src/index.ts +15 -0
  103. package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
  104. package/src/integrations/ghtml/ghtml-renderer.ts +87 -0
  105. package/src/integrations/ghtml/ghtml.constants.ts +1 -0
  106. package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
  107. package/src/plugins/alias-resolver-plugin.test.ts +41 -0
  108. package/src/plugins/alias-resolver-plugin.ts +63 -0
  109. package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
  110. package/src/plugins/eco-component-meta-plugin.ts +494 -0
  111. package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
  112. package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
  113. package/src/plugins/integration-plugin.test.ts +151 -0
  114. package/src/plugins/integration-plugin.ts +323 -0
  115. package/src/plugins/processor.test.ts +148 -0
  116. package/src/plugins/processor.ts +257 -0
  117. package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
  118. package/src/plugins/source-transform.test.ts +82 -0
  119. package/src/plugins/source-transform.ts +123 -0
  120. package/src/route-renderer/GRAPH.md +81 -289
  121. package/src/route-renderer/README.md +67 -105
  122. package/src/route-renderer/orchestration/component-render-context.ts +325 -0
  123. package/src/route-renderer/orchestration/declared-ownership-graph.ts +62 -0
  124. package/src/route-renderer/orchestration/foreign-subtree-execution.service.ts +383 -0
  125. package/src/route-renderer/orchestration/integration-renderer.test.ts +2085 -0
  126. package/src/route-renderer/orchestration/integration-renderer.ts +1244 -0
  127. package/src/route-renderer/orchestration/ownership-planning.service.ts +97 -0
  128. package/src/route-renderer/orchestration/ownership-validation.service.ts +76 -0
  129. package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
  130. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.test.ts +324 -0
  131. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.ts +294 -0
  132. package/src/route-renderer/orchestration/render-output.utils.ts +310 -0
  133. package/src/route-renderer/orchestration/route-render-orchestrator.prepare-render-options.test.ts +644 -0
  134. package/src/route-renderer/orchestration/route-render-orchestrator.test.ts +265 -0
  135. package/src/route-renderer/orchestration/route-render-orchestrator.ts +592 -0
  136. package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
  137. package/src/route-renderer/orchestration/template-serialization.ts +117 -0
  138. package/src/route-renderer/page-loading/component-dependency-collection.ts +202 -0
  139. package/src/route-renderer/page-loading/declared-asset-collection.ts +153 -0
  140. package/src/route-renderer/page-loading/dependency-resolver.test.ts +761 -0
  141. package/src/route-renderer/page-loading/dependency-resolver.ts +144 -0
  142. package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
  143. package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
  144. package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
  145. package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
  146. package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
  147. package/src/route-renderer/page-loading/page-dependency-bundling.ts +244 -0
  148. package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
  149. package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
  150. package/src/route-renderer/route-renderer.ts +133 -0
  151. package/src/router/README.md +16 -19
  152. package/src/router/client/link-intent.test.browser.ts +51 -0
  153. package/src/router/client/link-intent.ts +92 -0
  154. package/src/router/client/navigation-coordinator.test.ts +237 -0
  155. package/src/router/client/navigation-coordinator.ts +453 -0
  156. package/src/router/server/route-registry.test.ts +176 -0
  157. package/src/router/server/route-registry.ts +382 -0
  158. package/src/services/README.md +1 -2
  159. package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
  160. package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +473 -0
  161. package/src/services/assets/asset-processing-service/asset-processing.service.ts +344 -0
  162. package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
  163. package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
  164. package/src/services/assets/asset-processing-service/assets.types.ts +128 -0
  165. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
  166. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
  167. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
  168. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
  169. package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
  170. package/src/services/assets/asset-processing-service/index.ts +6 -0
  171. package/src/services/assets/asset-processing-service/page-package.test.ts +100 -0
  172. package/src/services/assets/asset-processing-service/page-package.ts +93 -0
  173. package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
  174. package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
  175. package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
  176. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
  177. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +173 -0
  178. package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
  179. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +195 -0
  180. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +137 -0
  181. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
  182. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +116 -0
  183. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
  184. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +89 -0
  185. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
  186. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +72 -0
  187. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +83 -0
  188. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
  189. package/src/services/assets/browser-bundle.service.test.ts +66 -0
  190. package/src/services/assets/browser-bundle.service.ts +109 -0
  191. package/src/services/cache/cache.types.ts +126 -0
  192. package/src/services/cache/index.ts +18 -0
  193. package/src/services/cache/memory-cache-store.test.ts +225 -0
  194. package/src/services/cache/memory-cache-store.ts +130 -0
  195. package/src/services/cache/page-cache-service.test.ts +175 -0
  196. package/src/services/cache/page-cache-service.ts +202 -0
  197. package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
  198. package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
  199. package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
  200. package/src/services/html/html-rewriter-provider.service.ts +104 -0
  201. package/src/services/html/html-transformer.service.test.ts +476 -0
  202. package/src/services/html/html-transformer.service.ts +275 -0
  203. package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
  204. package/src/services/invalidation/development-invalidation.service.ts +262 -0
  205. package/src/services/module-loading/app-module-loader.service.ts +9 -0
  206. package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
  207. package/src/services/module-loading/app-server-module-transpiler.service.ts +141 -0
  208. package/src/services/module-loading/host-module-loader-registry.ts +15 -0
  209. package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
  210. package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
  211. package/src/services/module-loading/node-bootstrap-plugin.ts +311 -0
  212. package/src/services/module-loading/page-module-import.service.test.ts +504 -0
  213. package/src/services/module-loading/page-module-import.service.ts +251 -0
  214. package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
  215. package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
  216. package/src/services/module-loading/source-module-support.ts +19 -0
  217. package/src/services/runtime-state/dev-graph.service.ts +217 -0
  218. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
  219. package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
  220. package/src/services/validation/schema-validation-service.test.ts +223 -0
  221. package/src/services/validation/schema-validation-service.ts +204 -0
  222. package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
  223. package/src/static-site-generator/static-site-generator.test.ts +408 -0
  224. package/src/static-site-generator/static-site-generator.ts +445 -0
  225. package/src/types/internal-types.ts +243 -0
  226. package/src/types/public-types.ts +1459 -0
  227. package/src/utils/deep-merge.test.ts +114 -0
  228. package/src/utils/deep-merge.ts +47 -0
  229. package/src/utils/hash.ts +5 -0
  230. package/src/utils/html-escaping.ts +9 -0
  231. package/src/utils/invariant.test.ts +22 -0
  232. package/src/utils/invariant.ts +15 -0
  233. package/src/utils/locals-utils.ts +37 -0
  234. package/src/utils/parse-cli-args.test.ts +69 -0
  235. package/src/utils/parse-cli-args.ts +105 -0
  236. package/src/utils/path-utils.module.ts +14 -0
  237. package/src/utils/path-utils.test.ts +15 -0
  238. package/src/utils/resolve-work-dir.ts +45 -0
  239. package/src/utils/runtime.ts +44 -0
  240. package/src/utils/server-utils.module.ts +67 -0
  241. package/src/utils/server-utils.test.ts +38 -0
  242. package/src/watchers/project-watcher.integration.test.ts +337 -0
  243. package/src/watchers/project-watcher.test-helpers.ts +42 -0
  244. package/src/watchers/project-watcher.test.ts +768 -0
  245. package/src/watchers/project-watcher.ts +357 -0
  246. package/CHANGELOG.md +0 -66
  247. package/src/adapters/abstract/application-adapter.d.ts +0 -194
  248. package/src/adapters/abstract/application-adapter.js +0 -121
  249. package/src/adapters/abstract/router-adapter.d.ts +0 -26
  250. package/src/adapters/abstract/router-adapter.js +0 -5
  251. package/src/adapters/abstract/server-adapter.d.ts +0 -69
  252. package/src/adapters/abstract/server-adapter.js +0 -15
  253. package/src/adapters/bun/client-bridge.d.ts +0 -34
  254. package/src/adapters/bun/client-bridge.js +0 -48
  255. package/src/adapters/bun/create-app.d.ts +0 -52
  256. package/src/adapters/bun/create-app.js +0 -116
  257. package/src/adapters/bun/hmr-manager.d.ts +0 -143
  258. package/src/adapters/bun/hmr-manager.js +0 -333
  259. package/src/adapters/bun/index.d.ts +0 -2
  260. package/src/adapters/bun/index.js +0 -8
  261. package/src/adapters/bun/server-adapter.d.ts +0 -155
  262. package/src/adapters/bun/server-adapter.js +0 -374
  263. package/src/adapters/bun/server-lifecycle.d.ts +0 -63
  264. package/src/adapters/bun/server-lifecycle.js +0 -92
  265. package/src/adapters/create-app.d.ts +0 -20
  266. package/src/adapters/create-app.js +0 -66
  267. package/src/adapters/index.d.ts +0 -2
  268. package/src/adapters/index.js +0 -8
  269. package/src/adapters/node/create-app.d.ts +0 -18
  270. package/src/adapters/node/create-app.js +0 -149
  271. package/src/adapters/node/node-client-bridge.d.ts +0 -26
  272. package/src/adapters/node/node-client-bridge.js +0 -66
  273. package/src/adapters/node/node-hmr-manager.d.ts +0 -133
  274. package/src/adapters/node/node-hmr-manager.js +0 -311
  275. package/src/adapters/node/server-adapter.d.ts +0 -162
  276. package/src/adapters/node/server-adapter.js +0 -368
  277. package/src/adapters/node/static-content-server.d.ts +0 -60
  278. package/src/adapters/node/static-content-server.js +0 -194
  279. package/src/adapters/shared/api-response.d.ts +0 -52
  280. package/src/adapters/shared/api-response.js +0 -96
  281. package/src/adapters/shared/application-adapter.d.ts +0 -18
  282. package/src/adapters/shared/application-adapter.js +0 -90
  283. package/src/adapters/shared/define-api-handler.d.ts +0 -25
  284. package/src/adapters/shared/define-api-handler.js +0 -15
  285. package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
  286. package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
  287. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
  288. package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
  289. package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
  290. package/src/adapters/shared/fs-server-response-factory.js +0 -97
  291. package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
  292. package/src/adapters/shared/fs-server-response-matcher.js +0 -147
  293. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
  294. package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
  295. package/src/adapters/shared/hmr-html-response.d.ts +0 -22
  296. package/src/adapters/shared/hmr-html-response.js +0 -32
  297. package/src/adapters/shared/render-context.d.ts +0 -15
  298. package/src/adapters/shared/render-context.js +0 -72
  299. package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
  300. package/src/adapters/shared/runtime-bootstrap.js +0 -43
  301. package/src/adapters/shared/server-adapter.d.ts +0 -97
  302. package/src/adapters/shared/server-adapter.js +0 -390
  303. package/src/adapters/shared/server-route-handler.d.ts +0 -89
  304. package/src/adapters/shared/server-route-handler.js +0 -111
  305. package/src/adapters/shared/server-static-builder.d.ts +0 -71
  306. package/src/adapters/shared/server-static-builder.js +0 -100
  307. package/src/build/build-adapter.d.ts +0 -239
  308. package/src/build/build-adapter.js +0 -642
  309. package/src/build/build-manifest.d.ts +0 -27
  310. package/src/build/build-manifest.js +0 -30
  311. package/src/build/build-types.d.ts +0 -57
  312. package/src/build/build-types.js +0 -0
  313. package/src/build/dev-build-coordinator.d.ts +0 -72
  314. package/src/build/dev-build-coordinator.js +0 -154
  315. package/src/build/esbuild-build-adapter.d.ts +0 -78
  316. package/src/build/esbuild-build-adapter.js +0 -505
  317. package/src/build/runtime-build-executor.d.ts +0 -14
  318. package/src/build/runtime-build-executor.js +0 -22
  319. package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
  320. package/src/build/runtime-specifier-alias-plugin.js +0 -35
  321. package/src/build/runtime-specifier-aliases.d.ts +0 -5
  322. package/src/build/runtime-specifier-aliases.js +0 -95
  323. package/src/config/config-builder.d.ts +0 -252
  324. package/src/config/config-builder.js +0 -603
  325. package/src/config/constants.js +0 -25
  326. package/src/dev/sc-server.d.ts +0 -30
  327. package/src/dev/sc-server.js +0 -111
  328. package/src/eco/eco.browser.d.ts +0 -2
  329. package/src/eco/eco.browser.js +0 -83
  330. package/src/eco/eco.d.ts +0 -9
  331. package/src/eco/eco.js +0 -85
  332. package/src/eco/eco.types.d.ts +0 -178
  333. package/src/eco/eco.types.js +0 -0
  334. package/src/eco/eco.utils.d.ts +0 -1
  335. package/src/eco/eco.utils.js +0 -10
  336. package/src/eco/global-injector-map.d.ts +0 -16
  337. package/src/eco/global-injector-map.js +0 -80
  338. package/src/eco/lazy-injector-map.d.ts +0 -8
  339. package/src/eco/lazy-injector-map.js +0 -70
  340. package/src/eco/module-dependencies.d.ts +0 -18
  341. package/src/eco/module-dependencies.js +0 -49
  342. package/src/errors/http-error.d.ts +0 -31
  343. package/src/errors/http-error.js +0 -50
  344. package/src/errors/index.d.ts +0 -2
  345. package/src/errors/index.js +0 -4
  346. package/src/errors/locals-access-error.d.ts +0 -4
  347. package/src/errors/locals-access-error.js +0 -9
  348. package/src/global/app-logger.d.ts +0 -2
  349. package/src/global/app-logger.js +0 -6
  350. package/src/hmr/client/hmr-runtime.d.ts +0 -5
  351. package/src/hmr/client/hmr-runtime.js +0 -117
  352. package/src/hmr/hmr-strategy.d.ts +0 -162
  353. package/src/hmr/hmr-strategy.js +0 -44
  354. package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
  355. package/src/hmr/hmr.postcss.test.e2e.js +0 -31
  356. package/src/hmr/hmr.test.e2e.d.ts +0 -1
  357. package/src/hmr/hmr.test.e2e.js +0 -43
  358. package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
  359. package/src/hmr/strategies/default-hmr-strategy.js +0 -34
  360. package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
  361. package/src/hmr/strategies/js-hmr-strategy.js +0 -178
  362. package/src/index.browser.d.ts +0 -3
  363. package/src/index.browser.js +0 -4
  364. package/src/index.d.ts +0 -6
  365. package/src/index.js +0 -21
  366. package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
  367. package/src/integrations/ghtml/ghtml-renderer.js +0 -63
  368. package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
  369. package/src/integrations/ghtml/ghtml.constants.js +0 -4
  370. package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
  371. package/src/integrations/ghtml/ghtml.plugin.js +0 -20
  372. package/src/plugins/alias-resolver-plugin.d.ts +0 -2
  373. package/src/plugins/alias-resolver-plugin.js +0 -53
  374. package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
  375. package/src/plugins/eco-component-meta-plugin.js +0 -163
  376. package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
  377. package/src/plugins/foreign-jsx-override-plugin.js +0 -35
  378. package/src/plugins/integration-plugin.d.ts +0 -219
  379. package/src/plugins/integration-plugin.js +0 -196
  380. package/src/plugins/processor.d.ts +0 -95
  381. package/src/plugins/processor.js +0 -136
  382. package/src/plugins/runtime-capability.js +0 -0
  383. package/src/plugins/source-transform.d.ts +0 -46
  384. package/src/plugins/source-transform.js +0 -71
  385. package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
  386. package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
  387. package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
  388. package/src/route-renderer/orchestration/component-render-context.js +0 -147
  389. package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -556
  390. package/src/route-renderer/orchestration/integration-renderer.js +0 -932
  391. package/src/route-renderer/orchestration/page-packaging.service.d.ts +0 -16
  392. package/src/route-renderer/orchestration/page-packaging.service.js +0 -66
  393. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +0 -2
  394. package/src/route-renderer/orchestration/processed-asset-dedupe.js +0 -23
  395. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
  396. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
  397. package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
  398. package/src/route-renderer/orchestration/render-execution.service.js +0 -106
  399. package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -66
  400. package/src/route-renderer/orchestration/render-output.utils.js +0 -171
  401. package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
  402. package/src/route-renderer/orchestration/render-preparation.service.js +0 -364
  403. package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
  404. package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
  405. package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
  406. package/src/route-renderer/orchestration/template-serialization.js +0 -45
  407. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +0 -37
  408. package/src/route-renderer/page-loading/component-dependency-collection.js +0 -125
  409. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +0 -24
  410. package/src/route-renderer/page-loading/declared-asset-collection.js +0 -106
  411. package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
  412. package/src/route-renderer/page-loading/dependency-resolver.js +0 -117
  413. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +0 -11
  414. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +0 -57
  415. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +0 -45
  416. package/src/route-renderer/page-loading/lazy-entry-collection.js +0 -105
  417. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +0 -19
  418. package/src/route-renderer/page-loading/lazy-trigger-planning.js +0 -40
  419. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +0 -5
  420. package/src/route-renderer/page-loading/module-declaration-aggregation.js +0 -33
  421. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +0 -3
  422. package/src/route-renderer/page-loading/module-declaration-scripts.js +0 -18
  423. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +0 -13
  424. package/src/route-renderer/page-loading/page-dependency-bundling.js +0 -115
  425. package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
  426. package/src/route-renderer/page-loading/page-module-loader.js +0 -127
  427. package/src/route-renderer/route-renderer.d.ts +0 -67
  428. package/src/route-renderer/route-renderer.js +0 -103
  429. package/src/router/client/link-intent.js +0 -34
  430. package/src/router/client/link-intent.test.browser.d.ts +0 -1
  431. package/src/router/client/link-intent.test.browser.js +0 -43
  432. package/src/router/client/navigation-coordinator.d.ts +0 -169
  433. package/src/router/client/navigation-coordinator.js +0 -215
  434. package/src/router/server/fs-router-scanner.d.ts +0 -41
  435. package/src/router/server/fs-router-scanner.js +0 -161
  436. package/src/router/server/fs-router.d.ts +0 -26
  437. package/src/router/server/fs-router.js +0 -100
  438. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +0 -3
  439. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +0 -56
  440. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -103
  441. package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -285
  442. package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
  443. package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
  444. package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -100
  445. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  446. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +0 -55
  447. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -49
  448. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
  449. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
  450. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +0 -30
  451. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +0 -65
  452. package/src/services/assets/asset-processing-service/index.d.ts +0 -5
  453. package/src/services/assets/asset-processing-service/index.js +0 -5
  454. package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
  455. package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
  456. package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
  457. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
  458. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -65
  459. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -22
  460. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -136
  461. package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
  462. package/src/services/assets/asset-processing-service/processors/index.js +0 -5
  463. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -6
  464. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -116
  465. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
  466. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -91
  467. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  468. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -77
  469. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -8
  470. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -58
  471. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
  472. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -67
  473. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +0 -18
  474. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +0 -45
  475. package/src/services/assets/browser-bundle.service.d.ts +0 -73
  476. package/src/services/assets/browser-bundle.service.js +0 -41
  477. package/src/services/cache/cache.types.d.ts +0 -107
  478. package/src/services/cache/cache.types.js +0 -0
  479. package/src/services/cache/index.d.ts +0 -7
  480. package/src/services/cache/index.js +0 -7
  481. package/src/services/cache/memory-cache-store.d.ts +0 -42
  482. package/src/services/cache/memory-cache-store.js +0 -98
  483. package/src/services/cache/page-cache-service.d.ts +0 -70
  484. package/src/services/cache/page-cache-service.js +0 -152
  485. package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
  486. package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
  487. package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
  488. package/src/services/html/html-rewriter-provider.service.js +0 -68
  489. package/src/services/html/html-transformer.service.d.ts +0 -87
  490. package/src/services/html/html-transformer.service.js +0 -216
  491. package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
  492. package/src/services/invalidation/development-invalidation.service.js +0 -190
  493. package/src/services/module-loading/app-module-loader.service.d.ts +0 -7
  494. package/src/services/module-loading/app-module-loader.service.js +0 -0
  495. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
  496. package/src/services/module-loading/app-server-module-transpiler.service.js +0 -115
  497. package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
  498. package/src/services/module-loading/host-module-loader-registry.js +0 -15
  499. package/src/services/module-loading/module-loading-types.js +0 -0
  500. package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
  501. package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
  502. package/src/services/module-loading/page-module-import.service.d.ts +0 -76
  503. package/src/services/module-loading/page-module-import.service.js +0 -170
  504. package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -63
  505. package/src/services/module-loading/server-module-transpiler.service.js +0 -64
  506. package/src/services/module-loading/source-module-support.d.ts +0 -5
  507. package/src/services/module-loading/source-module-support.js +0 -8
  508. package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
  509. package/src/services/runtime-state/dev-graph.service.js +0 -162
  510. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
  511. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
  512. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
  513. package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
  514. package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
  515. package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
  516. package/src/services/validation/schema-validation-service.d.ts +0 -122
  517. package/src/services/validation/schema-validation-service.js +0 -101
  518. package/src/services/validation/standard-schema.types.js +0 -0
  519. package/src/static-site-generator/static-site-generator.d.ts +0 -105
  520. package/src/static-site-generator/static-site-generator.js +0 -349
  521. package/src/types/internal-types.d.ts +0 -231
  522. package/src/types/internal-types.js +0 -0
  523. package/src/types/public-types.d.ts +0 -1257
  524. package/src/types/public-types.js +0 -0
  525. package/src/utils/deep-merge.d.ts +0 -14
  526. package/src/utils/deep-merge.js +0 -32
  527. package/src/utils/hash.d.ts +0 -1
  528. package/src/utils/hash.js +0 -7
  529. package/src/utils/html-escaping.d.ts +0 -7
  530. package/src/utils/html-escaping.js +0 -6
  531. package/src/utils/html.js +0 -4
  532. package/src/utils/invariant.d.ts +0 -5
  533. package/src/utils/invariant.js +0 -11
  534. package/src/utils/locals-utils.d.ts +0 -15
  535. package/src/utils/locals-utils.js +0 -24
  536. package/src/utils/parse-cli-args.d.ts +0 -27
  537. package/src/utils/parse-cli-args.js +0 -62
  538. package/src/utils/path-utils.module.d.ts +0 -5
  539. package/src/utils/path-utils.module.js +0 -14
  540. package/src/utils/resolve-work-dir.d.ts +0 -11
  541. package/src/utils/resolve-work-dir.js +0 -31
  542. package/src/utils/runtime.d.ts +0 -11
  543. package/src/utils/runtime.js +0 -40
  544. package/src/utils/server-utils.module.d.ts +0 -19
  545. package/src/utils/server-utils.module.js +0 -56
  546. package/src/watchers/project-watcher.d.ts +0 -136
  547. package/src/watchers/project-watcher.js +0 -275
  548. package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
  549. package/src/watchers/project-watcher.test-helpers.js +0 -52
  550. /package/src/utils/{html.d.ts → html.ts} +0 -0
@@ -0,0 +1,1459 @@
1
+ import type { Readable } from 'node:stream';
2
+ import type { ApiResponseBuilder } from '../adapters/shared/api-response.ts';
3
+ import type { BuildExecutor } from '../build/build-adapter.ts';
4
+ import type { EcoBuildPlugin } from '../build/build-types.ts';
5
+ import type { ForeignChildRuntime } from '../route-renderer/orchestration/component-render-context.ts';
6
+ import type { EcoPageComponent } from '../eco/eco.types.ts';
7
+ import type { EcoPagesAppConfig } from './internal-types.ts';
8
+ import type { HmrStrategy } from '../hmr/hmr-strategy.ts';
9
+ import type { BrowserBundleExecutor } from '../services/assets/browser-bundle.service.ts';
10
+ import type { ProcessedAsset } from '../services/assets/asset-processing-service/assets.types.ts';
11
+ import type { CacheStats, CacheStrategy } from '../services/cache/cache.types.ts';
12
+ import type { InteractionEventsString as ScriptsInjectorInteractionEventsString } from '@ecopages/scripts-injector/types';
13
+
14
+ export type { EcoPagesAppConfig } from './internal-types.ts';
15
+ export type { EcoPageComponent } from '../eco/eco.types.ts';
16
+ export type { ProcessedAsset } from '../services/assets/asset-processing-service/assets.types.ts';
17
+
18
+ import type {
19
+ StandardSchema,
20
+ StandardSchemaResult,
21
+ StandardSchemaSuccessResult,
22
+ StandardSchemaFailureResult,
23
+ StandardSchemaIssue,
24
+ InferOutput,
25
+ } from '../services/validation/standard-schema.types.ts';
26
+
27
+ export type {
28
+ StandardSchema,
29
+ StandardSchemaResult,
30
+ StandardSchemaSuccessResult,
31
+ StandardSchemaFailureResult,
32
+ StandardSchemaIssue,
33
+ InferOutput,
34
+ ForeignChildRuntime,
35
+ };
36
+
37
+ export type InteractionEventsString = ScriptsInjectorInteractionEventsString;
38
+
39
+ export type DependencyLazyTrigger =
40
+ | { 'on:idle': true }
41
+ | { 'on:interaction': InteractionEventsString }
42
+ | { 'on:visible': true | string };
43
+
44
+ export type DependencyAttributes = Record<string, string>;
45
+
46
+ export type EcoComponentStylesheetEntry = {
47
+ src?: string;
48
+ content?: string;
49
+ attributes?: DependencyAttributes;
50
+ };
51
+
52
+ export type EcoComponentScriptEntry = {
53
+ src?: string;
54
+ content?: string;
55
+ attributes?: DependencyAttributes;
56
+ lazy?: DependencyLazyTrigger;
57
+ ssr?: boolean;
58
+ };
59
+
60
+ export type ResolvedLazyScriptGroup = {
61
+ lazy: DependencyLazyTrigger;
62
+ scripts: string;
63
+ };
64
+
65
+ export type LazyTriggerRule =
66
+ | { 'on:idle': { scripts: string[] } }
67
+ | { 'on:interaction': { value: string; scripts: string[] } }
68
+ | { 'on:visible': { value?: string; scripts: string[] } };
69
+
70
+ export type ResolvedLazyTrigger = {
71
+ triggerId: string;
72
+ rules: LazyTriggerRule[];
73
+ };
74
+
75
+ /**
76
+ * Narrow interface for cache invalidation in API handlers.
77
+ * Exposes only the methods needed for programmatic cache control.
78
+ */
79
+ export interface CacheInvalidator {
80
+ /**
81
+ * Invalidate all cached entries matching any of the provided tags.
82
+ * @param tags - Array of tags to invalidate
83
+ * @returns Number of entries invalidated
84
+ */
85
+ invalidateByTags(tags: string[]): Promise<number>;
86
+
87
+ /**
88
+ * Invalidate cached entries by exact path.
89
+ * @param paths - Array of URL paths to invalidate
90
+ * @returns Number of entries invalidated
91
+ */
92
+ invalidateByPaths(paths: string[]): Promise<number>;
93
+
94
+ /**
95
+ * Clear all cached entries.
96
+ */
97
+ clear(): Promise<void>;
98
+
99
+ /**
100
+ * Get cache statistics for debugging.
101
+ */
102
+ stats(): Promise<CacheStats>;
103
+ }
104
+
105
+ /**
106
+ * Context interface for HMR strategies.
107
+ * Provides access to watched files, registered bare-specifier mappings, and build configuration.
108
+ */
109
+ export interface DefaultHmrContext {
110
+ /**
111
+ * Map of registered entrypoints to their output URLs.
112
+ */
113
+ getWatchedFiles(): Map<string, string>;
114
+
115
+ /**
116
+ * Directory where HMR bundles are written.
117
+ */
118
+ getDistDir(): string;
119
+
120
+ /**
121
+ * Build plugins to use during bundling.
122
+ */
123
+ getPlugins(): EcoBuildPlugin[];
124
+
125
+ /**
126
+ * Absolute path to the source directory.
127
+ */
128
+ getSrcDir(): string;
129
+
130
+ /**
131
+ * Absolute path to the layouts directory.
132
+ * Used to detect layout file changes that require full page reloads.
133
+ */
134
+ getLayoutsDir(): string;
135
+
136
+ /**
137
+ * Absolute path to the pages directory.
138
+ * Used by plugins to identify page files for transformation.
139
+ */
140
+ getPagesDir(): string;
141
+
142
+ /**
143
+ * Build executor owned by the active app/runtime.
144
+ */
145
+ getBuildExecutor(): BuildExecutor;
146
+
147
+ /**
148
+ * Browser bundler owned by the active app/runtime.
149
+ */
150
+ getBrowserBundleService(): BrowserBundleExecutor;
151
+ /**
152
+ * Server-side module loader owned by the active app/runtime.
153
+ */
154
+ importServerModule<T = unknown>(filePath: string): Promise<T>;
155
+ }
156
+
157
+ /**
158
+ * Represents an event broadcast to connected clients via the ClientBridge.
159
+ */
160
+ export type ClientBridgeEvent = {
161
+ /**
162
+ * Event type: 'reload' triggers full refresh, 'update' for JS modules, 'css-update' for stylesheets, 'layout-update' for layout changes
163
+ */
164
+ type: 'reload' | 'error' | 'update' | 'css-update' | 'layout-update';
165
+ /**
166
+ * Path to the changed file
167
+ */
168
+ path?: string;
169
+ /**
170
+ * Optional message for error or debug info
171
+ */
172
+ message?: string;
173
+ /**
174
+ * Timestamp for cache busting
175
+ */
176
+ timestamp?: number;
177
+ };
178
+
179
+ /**
180
+ * Adapter-agnostic interface for broadcasting development events to connected clients.
181
+ * Implemented by both the Bun and Node client bridges.
182
+ */
183
+ export interface IClientBridge {
184
+ broadcast(event: ClientBridgeEvent): void;
185
+ reload(): void;
186
+ cssUpdate(path: string): void;
187
+ update(path: string): void;
188
+ error(message: string): void;
189
+ subscriberCount: number;
190
+ }
191
+
192
+ /**
193
+ * Interface for the HMR Manager.
194
+ * Used by integration plugins to register entrypoints and strategies.
195
+ */
196
+ export interface IHmrManager {
197
+ /**
198
+ * Registers an integration-owned client entrypoint to be built and watched.
199
+ *
200
+ * @remarks
201
+ * This path is strict: the owning integration must emit the expected `_hmr`
202
+ * bundle. Missing output is treated as a development pipeline failure.
203
+ */
204
+ registerEntrypoint(entrypointPath: string): Promise<string>;
205
+
206
+ /**
207
+ * Registers a generic script asset entrypoint to be built and watched.
208
+ *
209
+ * @remarks
210
+ * This path exists for non-page script assets that are not owned by a
211
+ * framework integration. Unlike `registerEntrypoint()`, it may use the generic
212
+ * script bundling path.
213
+ */
214
+ registerScriptEntrypoint(entrypointPath: string): Promise<string>;
215
+
216
+ /**
217
+ * Registers a custom HMR strategy.
218
+ */
219
+ registerStrategy(strategy: HmrStrategy): void;
220
+
221
+ /**
222
+ * Sets the build plugins to use during bundling.
223
+ */
224
+ setPlugins(plugins: EcoBuildPlugin[]): void;
225
+
226
+ /**
227
+ * Enables or disables HMR.
228
+ */
229
+ setEnabled(enabled: boolean): void;
230
+
231
+ /**
232
+ * Returns whether HMR is enabled.
233
+ */
234
+ isEnabled(): boolean;
235
+
236
+ /**
237
+ * Broadcasts an HMR event to connected clients.
238
+ */
239
+ broadcast(event: ClientBridgeEvent): void;
240
+
241
+ /**
242
+ * Gets the output URL for a registered entrypoint.
243
+ */
244
+ getOutputUrl(entrypointPath: string): string | undefined;
245
+
246
+ /**
247
+ * Gets the map of watched files.
248
+ */
249
+ getWatchedFiles(): Map<string, string>;
250
+
251
+ /**
252
+ * Gets the HMR dist directory.
253
+ */
254
+ getDistDir(): string;
255
+
256
+ /**
257
+ * Gets the build plugins.
258
+ */
259
+ getPlugins(): EcoBuildPlugin[];
260
+
261
+ /**
262
+ * Gets the default HMR context.
263
+ */
264
+ getDefaultContext(): DefaultHmrContext;
265
+
266
+ /**
267
+ * Handles a file change event.
268
+ */
269
+ handleFileChange(path: string): Promise<void>;
270
+ }
271
+
272
+ /**
273
+ * Represents the dependencies for an EcoComponent.
274
+ */
275
+ export type EcoComponentDependencies = {
276
+ stylesheets?: Array<string | EcoComponentStylesheetEntry>;
277
+ scripts?: Array<string | EcoComponentScriptEntry>;
278
+ /**
279
+ * Browser module declarations resolved from node_modules.
280
+ *
281
+ * Supports grammar entries such as `react-aria-components{Table,Select}`
282
+ * to express explicit module imports for client bundles.
283
+ */
284
+ modules?: string[];
285
+ components?: EcoComponent[];
286
+ };
287
+
288
+ export type EcoPagesElement = string | Promise<string>;
289
+
290
+ /**
291
+ * Serializable child payloads accepted by cross-integration deferred rendering.
292
+ *
293
+ * This models the broad value shapes that EcoPages already flattens when a
294
+ * foreign component boundary serializes its children for another integration.
295
+ * It is intentionally transport-oriented rather than framework-native, so it
296
+ * can be shared across Kita, Lit, React, and Ecopages JSX authoring surfaces
297
+ * without coupling core types to any one renderer.
298
+ */
299
+ export type EcoChildren =
300
+ | string
301
+ | Promise<string>
302
+ | number
303
+ | bigint
304
+ | boolean
305
+ | null
306
+ | undefined
307
+ | readonly EcoChildren[]
308
+ | {
309
+ strings: readonly string[];
310
+ values?: readonly EcoChildren[];
311
+ }
312
+ | {
313
+ [key: string]: EcoChildren;
314
+ };
315
+
316
+ /**
317
+ * Represents the input configuration for EcoPages.
318
+ */
319
+ export type EcoPagesConfig = Omit<
320
+ Partial<EcoPagesAppConfig>,
321
+ 'baseUrl' | 'derivedPaths' | 'templatesExt' | 'integrationsDependencies'
322
+ > &
323
+ Pick<EcoPagesAppConfig, 'baseUrl' | 'rootDir'>;
324
+
325
+ /**
326
+ * Internal metadata injected by eco-component-meta-plugin.
327
+ * Contains component file path and integration info for dependency resolution.
328
+ * @internal
329
+ */
330
+ export interface EcoInjectedMeta {
331
+ /** Hashed identifier for client-side use (doesn't expose file paths) */
332
+ id: string;
333
+ /** Full file path of the component (use path.dirname() to get directory) */
334
+ file: string;
335
+ /** The integration identifier (e.g., 'react', 'kitajs', 'lit', 'ghtml', 'mdx') */
336
+ integration: string;
337
+ }
338
+
339
+ export type EcoComponentConfig = {
340
+ /** @internal Injected by eco-component-meta-plugin */
341
+ __eco?: EcoInjectedMeta;
342
+ /**
343
+ * Explicit integration override for this component.
344
+ * When provided, this takes precedence over auto-detected integration metadata.
345
+ */
346
+ integration?: string;
347
+ /**
348
+ * The layout component to wrap this page during rendering.
349
+ *
350
+ * The layout receives the page content as `children` and is responsible for
351
+ * providing the page structure (header, footer, navigation, etc.).
352
+ *
353
+ * For React pages with client-side routing, the layout also handles the router context.
354
+ * Use `EcoRouter` and `PageContent` from `@ecopages/react-router` in layouts that
355
+ * need SPA navigation support.
356
+ *
357
+ * @example
358
+ * ```tsx
359
+ * // Simple layout (no routing)
360
+ * const Layout = ({ children }) => <main>{children}</main>;
361
+ *
362
+ * // Layout with SPA routing
363
+ * const Layout = ({ children }) => (
364
+ * <EcoRouter page={...} pageProps={...}>
365
+ * <Header />
366
+ * <PageContent />
367
+ * </EcoRouter>
368
+ * );
369
+ *
370
+ * // Page using the layout
371
+ * const MyPage = () => <h1>Hello</h1>;
372
+ * MyPage.config = { layout: Layout };
373
+ * ```
374
+ */
375
+ layout?: EcoPageLayoutComponent<any>;
376
+ dependencies?: EcoComponentDependencies;
377
+ /**
378
+ * Internal: Resolved lazy scripts grouped by trigger.
379
+ * Set by the renderer, used by eco.component() for multi-trigger auto-wrapping.
380
+ * @internal
381
+ */
382
+ _resolvedLazyScripts?: ResolvedLazyScriptGroup[];
383
+ /**
384
+ * Internal: Resolved lazy triggers for the global injector map.
385
+ * Set by the renderer in the default full orchestration flow.
386
+ * @internal
387
+ */
388
+ _resolvedLazyTriggers?: ResolvedLazyTrigger[];
389
+ };
390
+
391
+ /**
392
+ * The base structure for any EcoPages component.
393
+ */
394
+ export type EcoComponentBase = {
395
+ /**
396
+ * The configuration options for the EcoComponent.
397
+ */
398
+ config?: EcoComponentConfig;
399
+
400
+ /**
401
+ * Static paths for dynamic routes (consolidated eco.page API).
402
+ * @internal Used by the renderer to retrieve static paths from the page component.
403
+ */
404
+ staticPaths?: GetStaticPaths;
405
+
406
+ /**
407
+ * Static props fetcher (consolidated eco.page API).
408
+ * @internal Used by the renderer to retrieve static props from the page component.
409
+ */
410
+ staticProps?: GetStaticProps<any>;
411
+
412
+ /**
413
+ * Metadata generator (consolidated eco.page API).
414
+ * @internal Used by the renderer to retrieve metadata from the page component.
415
+ */
416
+ metadata?: GetMetadata<any>;
417
+ };
418
+
419
+ /**
420
+ * Checks if a type is `any`.
421
+ */
422
+ export type IsAny<T> = 0 extends 1 & T ? true : false;
423
+
424
+ /**
425
+ * A function component type that is framework-agnostic.
426
+ * Uses a broader signature to support both direct calls and HOC wrappers.
427
+ */
428
+ export type EcoFunctionComponent<P, R> = {
429
+ (props: P, ...args: any[]): R;
430
+ } & EcoComponentBase;
431
+
432
+ /**
433
+ * Represents an EcoComponent.
434
+ *
435
+ * It can be defined by passing the props type as the first generic,
436
+ * or by passing the component type itself to infer the signature.
437
+ *
438
+ * @template T - The type of the props object or the component function itself.
439
+ * @template C - The type of the rendered element.
440
+ *
441
+ * @example
442
+ * //1. Simplest usage
443
+ * export const MyComponent: EcoComponent<{prop1: string}> = ({prop1}) => {
444
+ * return <div>...</div>;
445
+ * };
446
+ *
447
+ * @example
448
+ * // 2. Using with HOCs like MobX observer (passing the props type)
449
+ * export const MyObservedComponent: EcoComponent<object> = observer(function MyObservedComponent() {
450
+ * return <div>...</div>;
451
+ * });
452
+ *
453
+ * @example
454
+ * // 3. Passing the full function signature (Perfect for Generic Components)
455
+ * export const Select: EcoComponent<<T extends object>(props: SelectProps<T>) => JSX.Element> = <T extends object>({
456
+ * label,
457
+ * items,
458
+ * }: SelectProps<T>) => {
459
+ * return <select>...</select>;
460
+ * };
461
+ */
462
+ /**
463
+ * A function component type that is framework-agnostic.
464
+ * Uses a broader signature to support both direct calls and HOC wrappers.
465
+ */
466
+ export type EcoComponent<P = any, R = any> =
467
+ IsAny<P> extends true
468
+ ? EcoFunctionComponent<any, any> | EcoComponentBase
469
+ : P extends (props: infer Props, ...args: any[]) => infer Return
470
+ ? EcoFunctionComponent<Props, Return>
471
+ : EcoFunctionComponent<P, R>;
472
+
473
+ /**
474
+ * Represents a page in EcoPages.
475
+ */
476
+ export type PageProps<T = unknown> = T & StaticPageContext & { locals?: RequestLocals };
477
+
478
+ /**
479
+ * Represents the metadata for a page.
480
+ */
481
+ export interface PageMetadataProps {
482
+ title: string;
483
+ description: string;
484
+ image?: string;
485
+ url?: string;
486
+ keywords?: string[];
487
+ }
488
+
489
+ /**
490
+ * Represents the props for the head of a page.
491
+ */
492
+ export interface PageHeadProps<T = EcoPagesElement> {
493
+ metadata: PageMetadataProps;
494
+ dependencies?: EcoComponentDependencies;
495
+ children?: T;
496
+ }
497
+
498
+ /**
499
+ * Represents the props for a route layout.
500
+ */
501
+ export interface LayoutProps<T = EcoPagesElement> extends Partial<RequestPageContext> {
502
+ children: T;
503
+ }
504
+
505
+ /**
506
+ * Represents the props for the HTML template of a page.
507
+ */
508
+ export interface HtmlTemplateProps extends PageHeadProps {
509
+ children: EcoPagesElement;
510
+ language?: string;
511
+ headContent?: EcoPagesElement;
512
+ pageProps: Record<string, unknown>;
513
+ }
514
+
515
+ /**
516
+ * Layout components accepted by pages.
517
+ *
518
+ * This preserves compatibility with existing `eco.component()` layouts while
519
+ * also supporting semantic `eco.layout()` declarations.
520
+ */
521
+ export type EcoPageLayoutComponent<T = EcoPagesElement> = EcoLayoutComponent<T> | EcoComponent<any, T>;
522
+
523
+ /**
524
+ * Represents a layout component created with eco.layout().
525
+ */
526
+ export type EcoLayoutComponent<T = EcoPagesElement> = EcoComponent<LayoutProps<T>, T>;
527
+
528
+ /**
529
+ * Represents an HTML shell component created with eco.html().
530
+ */
531
+ export type EcoHtmlComponent<T = EcoPagesElement> = EcoComponent<HtmlTemplateProps, T>;
532
+
533
+ /**
534
+ * Represents the props for the error 404 template.
535
+ */
536
+ export interface Error404TemplateProps extends Omit<HtmlTemplateProps, 'children'> {
537
+ message: string;
538
+ stack?: string;
539
+ }
540
+
541
+ /**
542
+ * Represents the parameters for a page.
543
+ * The keys are strings, and the values can be either a string or an array of strings.
544
+ */
545
+ export type PageParams = Record<string, string | string[]>;
546
+
547
+ /**
548
+ * Represents a query object for a page.
549
+ * The keys are strings and the values can be either a string or an array of strings.
550
+ */
551
+ export type PageQuery = Record<string, string | string[]>;
552
+
553
+ /**
554
+ * Request-scoped data that is only available during request-time rendering.
555
+ *
556
+ * Apps should augment this interface via module augmentation:
557
+ *
558
+ * declare module '@ecopages/core' {
559
+ * interface RequestLocals { session?: Session | null }
560
+ * }
561
+ */
562
+ export interface RequestLocals {}
563
+
564
+ /**
565
+ * Represents the context object for a static page.
566
+ */
567
+ export type StaticPageContext = {
568
+ params?: PageParams;
569
+ query?: PageQuery;
570
+ };
571
+
572
+ /**
573
+ * Request-time page context.
574
+ *
575
+ * This is only populated during SSR. Static generation must not access locals.
576
+ */
577
+ export type RequestPageContext = {
578
+ locals: RequestLocals;
579
+ };
580
+
581
+ /**
582
+ * Adds optional `locals` to a props type.
583
+ * Primarily used for layouts that may receive request-scoped data from middleware.
584
+ *
585
+ * @template P - The base props type
586
+ *
587
+ * @example
588
+ * ```ts
589
+ * type MyLayoutProps = WithLocals<{ children: ReactNode }>;
590
+ * // { children: ReactNode; locals?: RequestLocals }
591
+ * ```
592
+ */
593
+ export type WithLocals<P> = P & { locals?: RequestLocals };
594
+
595
+ /**
596
+ * Represents the params for a static path.
597
+ */
598
+ export type StaticPath = { params: PageParams };
599
+
600
+ /**
601
+ * The function that returns the static paths for a page.
602
+ */
603
+ export type GetStaticPaths = (context: { appConfig: EcoPagesAppConfig; runtimeOrigin: string }) => Promise<{
604
+ paths: StaticPath[];
605
+ }>;
606
+
607
+ /**
608
+ * The context object for the getMetadata function.
609
+ */
610
+ export type GetMetadataContext<T = Record<string, unknown>> = Required<StaticPageContext> & {
611
+ props: T;
612
+ appConfig: EcoPagesAppConfig;
613
+ };
614
+
615
+ /**
616
+ * The function that returns the metadata for a page.
617
+ */
618
+ export type GetMetadata<T = Record<string, unknown>> = (
619
+ context: GetMetadataContext<T>,
620
+ ) => PageMetadataProps | Promise<PageMetadataProps>;
621
+
622
+ /**
623
+ * The function that returns the static props for a page.
624
+ */
625
+ export type GetStaticProps<T> = (context: {
626
+ pathname: StaticPath;
627
+ appConfig: EcoPagesAppConfig;
628
+ runtimeOrigin: string;
629
+ }) => Promise<{
630
+ props: T;
631
+ }>;
632
+
633
+ /**
634
+ * Represents a page file in EcoPages.
635
+ * @template T - The type of the page props.
636
+ */
637
+ export type EcoPageFile<T = unknown> = T & {
638
+ default: EcoComponent<any, any>;
639
+ getStaticPaths?: GetStaticPaths;
640
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
641
+ getMetadata?: GetMetadata;
642
+ cache?: CacheStrategy;
643
+ };
644
+
645
+ /**
646
+ * Represents a CSS processor.
647
+ */
648
+ export interface CssProcessor {
649
+ /**
650
+ * Processes a CSS file at the specified path.
651
+ * @param path - The path to the CSS file.
652
+ * @returns A promise that resolves to the processed CSS as a string.
653
+ */
654
+ processPath: (path: string, options?: any) => Promise<string>;
655
+
656
+ /**
657
+ * Processes a CSS string or buffer.
658
+ * @param contents - The CSS contents as a string or buffer.
659
+ * @returns A promise that resolves to the processed CSS as a string.
660
+ */
661
+ processStringOrBuffer: (contents: string | Buffer, options?: any) => Promise<string>;
662
+ }
663
+
664
+ /**
665
+ * The options for the route renderer.
666
+ */
667
+ export type RouteRendererOptions = {
668
+ file: string;
669
+ params?: PageParams;
670
+ query?: PageQuery;
671
+ locals?: RequestLocals;
672
+ };
673
+
674
+ /**
675
+ * The body of the route renderer.
676
+ */
677
+ export type RouteRendererBody = BodyInit | Readable;
678
+
679
+ /**
680
+ * Result of rendering a route, including body and optional cache configuration.
681
+ */
682
+ export type RouteRenderResult = {
683
+ body: RouteRendererBody;
684
+ /** Cache strategy from page component's eco.page({ cache }) option */
685
+ cacheStrategy?: CacheStrategy;
686
+ };
687
+
688
+ /**
689
+ * Represents the dependencies required for an integration plugin.
690
+ * It combines the base integration plugin dependencies with specific integration plugin dependencies.
691
+ */
692
+ export type IntegrationPluginDependencies = BaseIntegrationPluginDependencies & SpecificIntegrationPluginDependencies;
693
+
694
+ type BaseIntegrationPluginDependencies = {
695
+ inline?: boolean;
696
+ };
697
+
698
+ /**
699
+ * Represents the dependencies required for a specific integration plugin.
700
+ * It can be one of the following types:
701
+ * {@link ScriptImportIntegrationPluginDependencies}
702
+ * {@link ScriptContentIntegrationPluginDependencies}
703
+ * {@link StylesheetImportIntegrationPluginDependencies}
704
+ * {@link StylesheetContentIntegrationPluginDependencies}
705
+ */
706
+ type SpecificIntegrationPluginDependencies =
707
+ | ScriptImportIntegrationPluginDependencies
708
+ | ScriptContentIntegrationPluginDependencies
709
+ | StylesheetImportIntegrationPluginDependencies
710
+ | StylesheetContentIntegrationPluginDependencies;
711
+
712
+ /**
713
+ * Script dependencies for an integration plugin with an import path.
714
+ */
715
+ type ScriptImportIntegrationPluginDependencies = {
716
+ kind: 'script';
717
+ importPath: string;
718
+ position?: 'head' | 'body';
719
+ /** @default true */
720
+ minify?: boolean;
721
+ };
722
+
723
+ /**
724
+ * Script dependencies for an integration plugin with content.
725
+ */
726
+ type ScriptContentIntegrationPluginDependencies = {
727
+ kind: 'script';
728
+ content: string;
729
+ position?: 'head' | 'body';
730
+ /** @default true */
731
+ minify?: boolean;
732
+ };
733
+
734
+ /**
735
+ * Stylesheet dependencies for an integration plugin with an import path.
736
+ */
737
+ type StylesheetImportIntegrationPluginDependencies = {
738
+ kind: 'stylesheet';
739
+ importPath: string;
740
+ };
741
+
742
+ /**
743
+ * Stylesheet dependencies for an integration plugin with content.
744
+ */
745
+ type StylesheetContentIntegrationPluginDependencies = {
746
+ kind: 'stylesheet';
747
+ content: string;
748
+ };
749
+
750
+ /**
751
+ * The options for the integration renderer.
752
+ */
753
+ export type IntegrationRendererRenderOptions<C = EcoPagesElement> = RouteRendererOptions & {
754
+ props?: Record<string, unknown>;
755
+ metadata: PageMetadataProps;
756
+ HtmlTemplate: EcoHtmlComponent<C>;
757
+ Page: EcoComponent<PageProps, C>;
758
+ Layout?: EcoPageLayoutComponent<any>;
759
+ dependencies?: EcoComponentDependencies;
760
+ resolvedDependencies: ProcessedAsset[];
761
+ pagePackage?: PagePackageResult;
762
+ componentRender?: ComponentRenderResult;
763
+ pageProps?: Record<string, unknown>;
764
+ cacheStrategy?: CacheStrategy;
765
+ pageLocals?: RequestLocals;
766
+ ownershipPlan?: OwnershipPlan;
767
+ };
768
+
769
+ /**
770
+ * Structured page asset package produced after dependency resolution.
771
+ *
772
+ * `assets` retains the full processed asset list, while the remaining fields
773
+ * split that list into the subsets used during final HTML injection and
774
+ * renderer-owned follow-up work.
775
+ */
776
+ export interface PagePackageResult {
777
+ /**
778
+ * Full processed asset list before any page-level partitioning.
779
+ */
780
+ assets: ProcessedAsset[];
781
+ /**
782
+ * Assets that should still be injected into the final HTML document.
783
+ */
784
+ htmlAssets: ProcessedAsset[];
785
+ /**
786
+ * Primary page-owned browser entry when one was identified during packaging.
787
+ */
788
+ pageScript?: ProcessedAsset;
789
+ /**
790
+ * Primary page-owned stylesheet when one was identified during packaging.
791
+ */
792
+ pageStylesheet?: ProcessedAsset;
793
+ /**
794
+ * Inline assets that remain embedded directly in the document.
795
+ */
796
+ inlineAssets: ProcessedAsset[];
797
+ /**
798
+ * Assets kept outside the main page package so callers can manage them explicitly.
799
+ */
800
+ separateAssets: ProcessedAsset[];
801
+ /**
802
+ * Browser chunks needed after initial page bootstrap, including eager lazy-entry bundles.
803
+ */
804
+ dynamicChunks: ProcessedAsset[];
805
+ }
806
+
807
+ /**
808
+ * Page-scoped browser output planned before final HTML packaging.
809
+ *
810
+ * The initial seam keeps the graph payload intentionally small: integrations
811
+ * return the processed assets that belong to the Page browser graph, while the
812
+ * surrounding route pipeline remains free to evolve toward richer entry/lazy/
813
+ * shared chunk structure later.
814
+ */
815
+ export interface PageBrowserGraphResult {
816
+ /**
817
+ * Processed assets owned by the current Page browser graph.
818
+ */
819
+ assets: ProcessedAsset[];
820
+ }
821
+
822
+ export type OwnershipValidationErrorCode = 'UNKNOWN_INTEGRATION_OWNER' | 'MISSING_COMPONENT_METADATA';
823
+
824
+ export interface OwnershipValidationError {
825
+ code: OwnershipValidationErrorCode;
826
+ message: string;
827
+ componentId?: string;
828
+ componentFile?: string;
829
+ integrationName?: string;
830
+ }
831
+
832
+ export type OwnershipPlanNodeSource = 'route' | 'page' | 'layout' | 'html-template' | 'dependency';
833
+
834
+ export interface IntegrationOwnership {
835
+ integrationName: string;
836
+ componentId: string;
837
+ componentFile?: string;
838
+ isPageEntry: boolean;
839
+ isForeignToParent: boolean;
840
+ }
841
+
842
+ export interface OwnershipPlanNode {
843
+ id: string;
844
+ source: OwnershipPlanNodeSource;
845
+ ownership: IntegrationOwnership;
846
+ children: OwnershipPlanNode[];
847
+ declaredDependenciesValid: boolean;
848
+ }
849
+
850
+ export interface OwnershipPlan {
851
+ root: OwnershipPlanNode;
852
+ rendererNames: string[];
853
+ foreignEdgeCount: number;
854
+ hasValidationErrors: boolean;
855
+ validationErrors: OwnershipValidationError[];
856
+ }
857
+
858
+ export type ForeignSubtreeAttachmentPolicy = { kind: 'none' } | { kind: 'first-element' };
859
+
860
+ export interface ForeignSubtreeRenderPayload {
861
+ html: string;
862
+ assets: ProcessedAsset[];
863
+ rootTag?: string;
864
+ rootAttributes?: Record<string, string>;
865
+ attachmentPolicy: ForeignSubtreeAttachmentPolicy;
866
+ integrationName: string;
867
+ }
868
+
869
+ /**
870
+ * Shared execution-scoped context threaded through foreign-child renders.
871
+ *
872
+ * Integrations can extend this with renderer-local runtime keys, but the cache
873
+ * and optional component instance identity are shared across all renderers.
874
+ */
875
+ export interface BaseIntegrationContext {
876
+ rendererCache?: Map<string, unknown>;
877
+ componentInstanceId?: string;
878
+ }
879
+
880
+ export interface ComponentRenderInput<TIntegrationContext extends BaseIntegrationContext = BaseIntegrationContext> {
881
+ component: EcoComponent;
882
+ props: Record<string, unknown>;
883
+ children?: unknown;
884
+ integrationContext?: TIntegrationContext;
885
+ }
886
+
887
+ export interface ComponentRenderResult {
888
+ html: string;
889
+ canAttachAttributes: boolean;
890
+ rootTag?: string;
891
+ integrationName: string;
892
+ rootAttributes?: Record<string, string>;
893
+ assets?: ProcessedAsset[];
894
+ }
895
+
896
+ /**
897
+ * Represents a deep required type for a given object
898
+ */
899
+ export type DeepRequired<T> = Required<{
900
+ [K in keyof T]: T[K] extends Required<T[K]> ? T[K] : DeepRequired<T[K]>;
901
+ }>;
902
+
903
+ /**
904
+ * The Prettify helper is a utility type that takes an object type and makes the hover overlay more readable.
905
+ */
906
+ export type Prettify<T> = {
907
+ [K in keyof T]: T[K];
908
+ } & {};
909
+
910
+ /**
911
+ * Services available to API handlers.
912
+ */
913
+ export interface ApiHandlerServices {
914
+ /**
915
+ * Cache invalidation service.
916
+ * Null when caching is disabled.
917
+ */
918
+ cache: CacheInvalidator | null;
919
+ }
920
+
921
+ /**
922
+ * Options for rendering a view.
923
+ */
924
+ export interface RenderOptions {
925
+ status?: number;
926
+ headers?: HeadersInit;
927
+ }
928
+
929
+ /**
930
+ * Options for JSON/HTML response helpers.
931
+ */
932
+ export interface ResponseOptions {
933
+ status?: number;
934
+ headers?: HeadersInit;
935
+ }
936
+
937
+ /**
938
+ * Context for rendering views in route handlers.
939
+ * Provides methods to render eco.page views and return formatted responses.
940
+ */
941
+ export interface RenderContext {
942
+ /**
943
+ * Render an eco.page view with full layout and includes.
944
+ * @param view - The eco.page component to render
945
+ * @param props - Props to pass to the view
946
+ * @param options - Optional status code and headers
947
+ */
948
+ render<P = Record<string, unknown>>(view: EcoComponent<P>, props?: P, options?: RenderOptions): Promise<Response>;
949
+
950
+ /**
951
+ * Render an eco.page view without layout (for partials/fragments).
952
+ * @param view - The eco.page component to render
953
+ * @param props - Props to pass to the view
954
+ * @param options - Optional status code and headers
955
+ */
956
+ renderPartial<P = Record<string, unknown>>(
957
+ view: EcoComponent<P>,
958
+ props: P,
959
+ options?: RenderOptions,
960
+ ): Promise<Response>;
961
+
962
+ /**
963
+ * Return a JSON response.
964
+ * @param data - Data to serialize as JSON
965
+ * @param options - Optional status code and headers
966
+ */
967
+ json(data: unknown, options?: ResponseOptions): Response;
968
+
969
+ /**
970
+ * Return an HTML response.
971
+ * @param content - HTML string content
972
+ * @param options - Optional status code and headers
973
+ */
974
+ html(content: string, options?: ResponseOptions): Response;
975
+ }
976
+
977
+ /**
978
+ * Context provided to the API handler.
979
+ */
980
+ export interface ApiHandlerContext<TRequest extends Request = Request, TServer = any> extends RenderContext {
981
+ request: TRequest;
982
+ params: Record<string, string>;
983
+ response: ApiResponseBuilder;
984
+ server: TServer;
985
+ /**
986
+ * Request-scoped data store.
987
+ * Only valid during request-time handling (SSR/API). Must not be used for static generation.
988
+ */
989
+ locals: RequestLocals;
990
+ /**
991
+ * Require one or more locals keys. If missing, executes `onMissing` to produce a terminating Response.
992
+ */
993
+ require: {
994
+ <K extends keyof RequestLocals>(key: K, onMissing: () => Response): Exclude<RequestLocals[K], null | undefined>;
995
+ <K extends keyof RequestLocals>(
996
+ keys: readonly K[],
997
+ onMissing: () => Response,
998
+ ): { [P in K]-?: Exclude<RequestLocals[P], null | undefined> };
999
+ };
1000
+ /**
1001
+ * Services available to the API handler.
1002
+ */
1003
+ services: ApiHandlerServices;
1004
+ /**
1005
+ * Parsed and optionally validated request body.
1006
+ *
1007
+ * - Without schema: Contains the parsed JSON body as `unknown`
1008
+ * - With schema: Contains the validated and type-safe body data
1009
+ *
1010
+ * For raw access to the request body stream, use `ctx.request.body`.
1011
+ *
1012
+ * @example Without validation
1013
+ * ```typescript
1014
+ * app.post('/posts', async (ctx) => {
1015
+ * const data = ctx.body; // [unknown] - parsed JSON
1016
+ * return ctx.json({ received: data });
1017
+ * });
1018
+ * ```
1019
+ *
1020
+ * @example With validation
1021
+ * ```typescript
1022
+ * import { z } from 'zod';
1023
+ *
1024
+ * app.post('/posts', async (ctx) => {
1025
+ * const { title, author } = ctx.body; // [type-safe]
1026
+ * return ctx.json({ title, author });
1027
+ * }, {
1028
+ * schema: {
1029
+ * body: z.object({ title: z.string(), author: z.string() })
1030
+ * }
1031
+ * });
1032
+ * ```
1033
+ *
1034
+ * Validation runs before the handler executes. Invalid requests receive a 400 response.
1035
+ */
1036
+ body?: unknown;
1037
+ /**
1038
+ * Parsed and optionally validated query parameters.
1039
+ *
1040
+ * - Without schema: Contains the parsed query params as a string record
1041
+ * - With schema: Contains the validated and type-safe query data
1042
+ *
1043
+ * For raw access to query parameters, use `ctx.request.url` and parse manually.
1044
+ *
1045
+ * @example Pagination with type coercion
1046
+ * ```typescript
1047
+ * import { z } from 'zod';
1048
+ *
1049
+ * app.get('/posts', async (ctx) => {
1050
+ * const { page, limit, sortBy } = ctx.query;
1051
+ * // page: number, limit: number, sortBy: 'date' | 'title' | 'views'
1052
+ * return ctx.json({ page, limit, sortBy });
1053
+ * }, {
1054
+ * schema: {
1055
+ * query: z.object({
1056
+ * page: z.coerce.number().min(1).default(1),
1057
+ * limit: z.coerce.number().min(1).max(100).default(20),
1058
+ * sortBy: z.enum(['date', 'title', 'views']).default('date')
1059
+ * })
1060
+ * }
1061
+ * });
1062
+ * ```
1063
+ *
1064
+ * @example Search with filters
1065
+ * ```typescript
1066
+ * app.get('/search', async (ctx) => {
1067
+ * const { q, category } = ctx.query;
1068
+ * return ctx.json({ results: await search(q, category) });
1069
+ * }, {
1070
+ * schema: {
1071
+ * query: z.object({
1072
+ * q: z.string().min(2).max(100),
1073
+ * category: z.enum(['posts', 'users', 'comments']).optional()
1074
+ * })
1075
+ * }
1076
+ * });
1077
+ * ```
1078
+ */
1079
+ query?: unknown;
1080
+ /**
1081
+ * Parsed and optionally validated request headers.
1082
+ *
1083
+ * - Without schema: Contains the parsed headers as a string record
1084
+ * - With schema: Contains the validated and type-safe header data
1085
+ *
1086
+ * For raw access to headers, use `ctx.request.headers`.
1087
+ *
1088
+ * @example API key authentication
1089
+ * ```typescript
1090
+ * import { z } from 'zod';
1091
+ *
1092
+ * app.post('/api/webhooks', async (ctx) => {
1093
+ * const apiKey = ctx.headers['x-api-key'];
1094
+ * // apiKey is guaranteed to be a valid UUID
1095
+ * return ctx.json({ received: true });
1096
+ * }, {
1097
+ * schema: {
1098
+ * headers: z.object({
1099
+ * 'x-api-key': z.string().uuid()
1100
+ * })
1101
+ * }
1102
+ * });
1103
+ * ```
1104
+ *
1105
+ * @example Webhook signature validation
1106
+ * ```typescript
1107
+ * app.post('/webhooks/stripe', async (ctx) => {
1108
+ * const signature = ctx.headers['stripe-signature'];
1109
+ * // signature is guaranteed to exist
1110
+ * const isValid = verifyStripeSignature(ctx.body, signature);
1111
+ * return ctx.json({ verified: isValid });
1112
+ * }, {
1113
+ * schema: {
1114
+ * headers: z.object({
1115
+ * 'stripe-signature': z.string().min(1)
1116
+ * })
1117
+ * }
1118
+ * });
1119
+ * ```
1120
+ *
1121
+ * @example Content negotiation
1122
+ * ```typescript
1123
+ * app.post('/api/data', async (ctx) => {
1124
+ * const { accept } = ctx.headers;
1125
+ * if (accept === 'application/xml') {
1126
+ * return ctx.html(toXml(data), { headers: { 'Content-Type': 'application/xml' } });
1127
+ * }
1128
+ * return ctx.json(data);
1129
+ * }, {
1130
+ * schema: {
1131
+ * headers: z.object({
1132
+ * 'content-type': z.literal('application/json'),
1133
+ * accept: z.enum(['application/json', 'application/xml']).optional()
1134
+ * })
1135
+ * }
1136
+ * });
1137
+ * ```
1138
+ */
1139
+ headers?: unknown;
1140
+ }
1141
+
1142
+ /**
1143
+ * Context available to file-route page middleware.
1144
+ *
1145
+ * Page middleware can mutate locals, short-circuit the request, and use the
1146
+ * response helpers, but final document rendering stays owned by the page route
1147
+ * execution path.
1148
+ */
1149
+ export interface FileRouteMiddlewareContext<TRequest extends Request = Request, TServer = any> extends Omit<
1150
+ ApiHandlerContext<TRequest, TServer>,
1151
+ 'render' | 'renderPartial'
1152
+ > {}
1153
+
1154
+ /**
1155
+ * Next function for middleware chain.
1156
+ * Call to continue to the next middleware or final handler.
1157
+ */
1158
+ export type MiddlewareNext = () => Promise<Response>;
1159
+
1160
+ /**
1161
+ * Middleware function signature.
1162
+ * Receives the request context and a next function to continue the chain.
1163
+ * Can short-circuit by returning a Response directly without calling next().
1164
+ *
1165
+ * @typeParam TRequest - Request type
1166
+ * @typeParam TServer - Server type
1167
+ * @typeParam TContext - Extended context type, defaults to base ApiHandlerContext
1168
+ *
1169
+ * @example Basic middleware (no context extension)
1170
+ * ```typescript
1171
+ * const loggingMiddleware: Middleware = async (ctx, next) => {
1172
+ * console.log(`${ctx.request.method} ${ctx.request.url}`);
1173
+ * return next();
1174
+ * };
1175
+ * ```
1176
+ *
1177
+ * @example Middleware with extended context (use EcoMiddleware helper)
1178
+ * ```typescript
1179
+ * type AuthContext = ApiHandlerContext<BunRequest<string>, Server> & { user: User };
1180
+ *
1181
+ * const authMiddleware: EcoMiddleware<AuthContext> = async (ctx, next) => {
1182
+ * const user = await authenticate(ctx.request);
1183
+ * if (!user) return ctx.response.status(401).json({ error: 'Unauthorized' });
1184
+ * ctx.user = user;
1185
+ * return next();
1186
+ * };
1187
+ * ```
1188
+ */
1189
+ export type Middleware<
1190
+ TRequest extends Request = Request,
1191
+ TServer = any,
1192
+ TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
1193
+ > = (context: TContext, next: MiddlewareNext) => Promise<Response> | Response;
1194
+
1195
+ /**
1196
+ * Middleware contract for file-based page routes.
1197
+ */
1198
+ export type FileRouteMiddleware<
1199
+ TRequest extends Request = Request,
1200
+ TServer = any,
1201
+ TContext extends FileRouteMiddlewareContext<TRequest, TServer> = FileRouteMiddlewareContext<TRequest, TServer>,
1202
+ > = (context: TContext, next: MiddlewareNext) => Promise<Response> | Response;
1203
+
1204
+ /**
1205
+ * Helper type for defining middleware with extended context.
1206
+ * Automatically infers TRequest and TServer from the provided context type.
1207
+ *
1208
+ * @typeParam TContext - The extended context type that includes TRequest and TServer
1209
+ *
1210
+ * @example
1211
+ * ```typescript
1212
+ * type AuthContext = ApiHandlerContext<BunRequest<string>, Server> & {
1213
+ * session: { user: User };
1214
+ * };
1215
+ *
1216
+ * const authMiddleware: EcoMiddleware<AuthContext> = async (ctx, next) => {
1217
+ * const session = await authenticate(ctx.request);
1218
+ * if (!session) return Response.redirect('/login');
1219
+ * ctx.session = session;
1220
+ * return next();
1221
+ * };
1222
+ * ```
1223
+ */
1224
+ export type EcoMiddleware<TContext extends ApiHandlerContext<any, any>> =
1225
+ TContext extends ApiHandlerContext<infer TRequest, infer TServer> ? Middleware<TRequest, TServer, TContext> : never;
1226
+
1227
+ /**
1228
+ * Represents an API handler in EcoPages.
1229
+ * Defines the path, method, handler function, optional middleware, and validation schemas.
1230
+ *
1231
+ * @example Basic handler
1232
+ * ```typescript
1233
+ * app.get('/users/:id', async (ctx) => {
1234
+ * return ctx.json({ id: ctx.params.id });
1235
+ * });
1236
+ * ```
1237
+ *
1238
+ * @example With validation
1239
+ * ```typescript
1240
+ * import { z } from 'zod';
1241
+ *
1242
+ * app.post('/posts', async (ctx) => {
1243
+ * const { title, content } = ctx.body;
1244
+ * return ctx.json({ success: true, title, content });
1245
+ * }, {
1246
+ * schema: {
1247
+ * body: z.object({
1248
+ * title: z.string().min(3),
1249
+ * content: z.string()
1250
+ * })
1251
+ * }
1252
+ * });
1253
+ * ```
1254
+ *
1255
+ * @example With middleware
1256
+ * ```typescript
1257
+ * const authMiddleware = async (ctx, next) => {
1258
+ * if (!ctx.request.headers.get('authorization')) {
1259
+ * return ctx.response.status(401).json({ error: 'Unauthorized' });
1260
+ * }
1261
+ * return next();
1262
+ * };
1263
+ *
1264
+ * app.get('/protected', async (ctx) => {
1265
+ * return ctx.json({ message: 'Secret data' });
1266
+ * }, {
1267
+ * middleware: [authMiddleware]
1268
+ * });
1269
+ * ```
1270
+ *
1271
+ * @example Multiple validations
1272
+ * ```typescript
1273
+ * import { z } from 'zod';
1274
+ *
1275
+ * app.post('/api/search', async (ctx) => {
1276
+ * const { q, page } = ctx.query;
1277
+ * const filters = ctx.body;
1278
+ * return ctx.json({ query: q, page, filters });
1279
+ * }, {
1280
+ * schema: {
1281
+ * query: z.object({ q: z.string(), page: z.string() }),
1282
+ * body: z.object({ category: z.string().optional() })
1283
+ * }
1284
+ * });
1285
+ * ```
1286
+ */
1287
+ export interface ApiHandler<TPath extends string = string, TRequest extends Request = Request, TServer = any> {
1288
+ path: TPath;
1289
+ method: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH' | 'OPTIONS' | 'HEAD';
1290
+ handler: (context: ApiHandlerContext<TRequest, TServer>) => Promise<Response> | Response;
1291
+ /** Optional middleware chain executed before the handler */
1292
+ middleware?: Middleware<TRequest, TServer, any>[];
1293
+ /** Optional validation schemas for request body, query parameters, and headers */
1294
+ schema?: {
1295
+ body?: StandardSchema;
1296
+ query?: StandardSchema;
1297
+ headers?: StandardSchema;
1298
+ params?: StandardSchema;
1299
+ };
1300
+ }
1301
+
1302
+ /**
1303
+ * Global error handler for centralized error handling across all routes.
1304
+ * Receives the error and full request context, returns a Response.
1305
+ * Falls back to default handling if the handler itself throws.
1306
+ */
1307
+ export type ErrorHandler<TRequest extends Request = Request, TServer = any> = (
1308
+ error: unknown,
1309
+ context: ApiHandlerContext<TRequest, TServer>,
1310
+ ) => Promise<Response> | Response;
1311
+
1312
+ /**
1313
+ * Options for route handlers.
1314
+ */
1315
+ export interface RouteOptions<
1316
+ TRequest extends Request = Request,
1317
+ TServer = any,
1318
+ TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
1319
+ > {
1320
+ middleware?: Middleware<TRequest, TServer, TContext>[];
1321
+ schema?: RouteSchema;
1322
+ }
1323
+
1324
+ export interface RouteSchema {
1325
+ body?: StandardSchema;
1326
+ query?: StandardSchema;
1327
+ headers?: StandardSchema;
1328
+ params?: StandardSchema;
1329
+ }
1330
+
1331
+ /**
1332
+ * Helper type to extract inferred types from a schema, with fallback to unknown.
1333
+ */
1334
+ export type InferSchemaOutput<T> = T extends StandardSchema ? InferOutput<T> : unknown;
1335
+
1336
+ /**
1337
+ * Context with typed body/query/headers based on the provided schema.
1338
+ */
1339
+ export type TypedApiHandlerContext<
1340
+ TSchema extends RouteSchema,
1341
+ TRequest extends Request = Request,
1342
+ TServer = any,
1343
+ > = Omit<ApiHandlerContext<TRequest, TServer>, 'body' | 'query' | 'headers'> & {
1344
+ body: InferSchemaOutput<TSchema['body']>;
1345
+ query: InferSchemaOutput<TSchema['query']>;
1346
+ headers: InferSchemaOutput<TSchema['headers']>;
1347
+ params: InferSchemaOutput<TSchema['params']>;
1348
+ };
1349
+
1350
+ /**
1351
+ * Options for the group method.
1352
+ *
1353
+ * @typeParam TContext - Extended context type that middleware provides to handlers
1354
+ *
1355
+ * @example Group with auth middleware extending context
1356
+ * ```typescript
1357
+ * type AuthContext = ApiHandlerContext<BunRequest<string>, Server> & { user: User };
1358
+ *
1359
+ * app.group<AuthContext>('/api', (r) => {
1360
+ * r.get('/profile', (ctx) => {
1361
+ * // ctx.user is properly typed!
1362
+ * return ctx.json({ name: ctx.user.name });
1363
+ * });
1364
+ * }, { middleware: [authMiddleware] });
1365
+ * ```
1366
+ */
1367
+ export interface GroupOptions<
1368
+ TRequest extends Request = Request,
1369
+ TServer = any,
1370
+ TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
1371
+ > {
1372
+ middleware?: Middleware<TRequest, TServer, TContext>[];
1373
+ }
1374
+
1375
+ /**
1376
+ * Context type that combines schema-typed fields with an extended base context.
1377
+ * Used by RouteGroupBuilder to merge schema validation types with middleware-extended context.
1378
+ * Note: Path parameter typing (e.g., :id -> params.id) comes from the TContext.request type.
1379
+ */
1380
+ export type TypedGroupHandlerContext<TSchema extends RouteSchema, TContext extends ApiHandlerContext<any, any>> = Omit<
1381
+ TContext,
1382
+ 'body' | 'query' | 'headers' | 'params'
1383
+ > & {
1384
+ body: InferSchemaOutput<TSchema['body']>;
1385
+ query: InferSchemaOutput<TSchema['query']>;
1386
+ headers: InferSchemaOutput<TSchema['headers']>;
1387
+ params: InferSchemaOutput<TSchema['params']>;
1388
+ };
1389
+
1390
+ /**
1391
+ * Builder interface for defining routes within a group.
1392
+ * Provides chainable methods for registering routes with shared prefix and middleware.
1393
+ *
1394
+ * @typeParam TRequest - The request type
1395
+ * @typeParam TServer - The server type
1396
+ * @typeParam TContext - Extended context type from group middleware
1397
+ */
1398
+ export interface RouteGroupBuilder<
1399
+ TRequest extends Request = Request,
1400
+ TServer = any,
1401
+ TContext extends ApiHandlerContext<TRequest, TServer> = ApiHandlerContext<TRequest, TServer>,
1402
+ > {
1403
+ get<P extends string, TSchema extends RouteSchema = RouteSchema>(
1404
+ path: P,
1405
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1406
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1407
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1408
+
1409
+ post<P extends string, TSchema extends RouteSchema = RouteSchema>(
1410
+ path: P,
1411
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1412
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1413
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1414
+
1415
+ put<P extends string, TSchema extends RouteSchema = RouteSchema>(
1416
+ path: P,
1417
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1418
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1419
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1420
+
1421
+ delete<P extends string, TSchema extends RouteSchema = RouteSchema>(
1422
+ path: P,
1423
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1424
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1425
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1426
+
1427
+ patch<P extends string, TSchema extends RouteSchema = RouteSchema>(
1428
+ path: P,
1429
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1430
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1431
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1432
+
1433
+ options<P extends string, TSchema extends RouteSchema = RouteSchema>(
1434
+ path: P,
1435
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1436
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1437
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1438
+
1439
+ head<P extends string, TSchema extends RouteSchema = RouteSchema>(
1440
+ path: P,
1441
+ handler: (context: TypedGroupHandlerContext<TSchema, TContext>) => Promise<Response> | Response,
1442
+ options?: RouteOptions<TRequest, TServer, TContext> & { schema?: TSchema },
1443
+ ): RouteGroupBuilder<TRequest, TServer, TContext>;
1444
+ }
1445
+
1446
+ /**
1447
+ * A function that dynamically imports a view module.
1448
+ * Used by app.static() to enable HMR in development.
1449
+ */
1450
+ export type ViewLoader<P = any> = () => Promise<{ default: EcoPageComponent<P> }>;
1451
+
1452
+ /**
1453
+ * Represents a static route registered via app.static().
1454
+ * Uses a loader function to enable HMR in development mode.
1455
+ */
1456
+ export interface StaticRoute<P = any> {
1457
+ path: string;
1458
+ loader: ViewLoader<P>;
1459
+ }