@ecopages/core 0.2.0-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (342) hide show
  1. package/CHANGELOG.md +89 -0
  2. package/LICENSE +21 -0
  3. package/README.md +32 -0
  4. package/package.json +279 -0
  5. package/src/adapters/abstract/application-adapter.d.ts +168 -0
  6. package/src/adapters/abstract/application-adapter.js +109 -0
  7. package/src/adapters/abstract/application-adapter.ts +337 -0
  8. package/src/adapters/abstract/router-adapter.d.ts +26 -0
  9. package/src/adapters/abstract/router-adapter.js +5 -0
  10. package/src/adapters/abstract/router-adapter.ts +30 -0
  11. package/src/adapters/abstract/server-adapter.d.ts +69 -0
  12. package/src/adapters/abstract/server-adapter.js +15 -0
  13. package/src/adapters/abstract/server-adapter.ts +79 -0
  14. package/src/adapters/bun/client-bridge.d.ts +34 -0
  15. package/src/adapters/bun/client-bridge.js +48 -0
  16. package/src/adapters/bun/client-bridge.ts +62 -0
  17. package/src/adapters/bun/create-app.d.ts +60 -0
  18. package/src/adapters/bun/create-app.js +117 -0
  19. package/src/adapters/bun/create-app.ts +189 -0
  20. package/src/adapters/bun/define-api-handler.d.ts +61 -0
  21. package/src/adapters/bun/define-api-handler.js +15 -0
  22. package/src/adapters/bun/define-api-handler.ts +114 -0
  23. package/src/adapters/bun/hmr-manager.d.ts +84 -0
  24. package/src/adapters/bun/hmr-manager.js +227 -0
  25. package/src/adapters/bun/hmr-manager.ts +281 -0
  26. package/src/adapters/bun/index.d.ts +3 -0
  27. package/src/adapters/bun/index.js +8 -0
  28. package/src/adapters/bun/index.ts +3 -0
  29. package/src/adapters/bun/server-adapter.d.ts +155 -0
  30. package/src/adapters/bun/server-adapter.js +368 -0
  31. package/src/adapters/bun/server-adapter.ts +492 -0
  32. package/src/adapters/bun/server-lifecycle.d.ts +52 -0
  33. package/src/adapters/bun/server-lifecycle.js +120 -0
  34. package/src/adapters/bun/server-lifecycle.ts +154 -0
  35. package/src/adapters/index.d.ts +6 -0
  36. package/src/adapters/index.js +14 -0
  37. package/src/adapters/index.ts +6 -0
  38. package/src/adapters/node/create-app.d.ts +21 -0
  39. package/src/adapters/node/create-app.js +143 -0
  40. package/src/adapters/node/create-app.ts +179 -0
  41. package/src/adapters/node/index.d.ts +4 -0
  42. package/src/adapters/node/index.js +8 -0
  43. package/src/adapters/node/index.ts +9 -0
  44. package/src/adapters/node/node-client-bridge.d.ts +26 -0
  45. package/src/adapters/node/node-client-bridge.js +66 -0
  46. package/src/adapters/node/node-client-bridge.ts +79 -0
  47. package/src/adapters/node/node-hmr-manager.d.ts +62 -0
  48. package/src/adapters/node/node-hmr-manager.js +221 -0
  49. package/src/adapters/node/node-hmr-manager.ts +271 -0
  50. package/src/adapters/node/server-adapter.d.ts +190 -0
  51. package/src/adapters/node/server-adapter.js +420 -0
  52. package/src/adapters/node/server-adapter.ts +561 -0
  53. package/src/adapters/node/static-content-server.d.ts +24 -0
  54. package/src/adapters/node/static-content-server.js +166 -0
  55. package/src/adapters/node/static-content-server.ts +203 -0
  56. package/src/adapters/shared/api-response.d.ts +52 -0
  57. package/src/adapters/shared/api-response.js +96 -0
  58. package/src/adapters/shared/api-response.ts +104 -0
  59. package/src/adapters/shared/application-adapter.d.ts +18 -0
  60. package/src/adapters/shared/application-adapter.js +90 -0
  61. package/src/adapters/shared/application-adapter.ts +199 -0
  62. package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
  63. package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
  64. package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
  65. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
  66. package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
  67. package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
  68. package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
  69. package/src/adapters/shared/fs-server-response-factory.js +97 -0
  70. package/src/adapters/shared/fs-server-response-factory.ts +118 -0
  71. package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
  72. package/src/adapters/shared/fs-server-response-matcher.js +155 -0
  73. package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
  74. package/src/adapters/shared/render-context.d.ts +14 -0
  75. package/src/adapters/shared/render-context.js +69 -0
  76. package/src/adapters/shared/render-context.ts +105 -0
  77. package/src/adapters/shared/server-adapter.d.ts +87 -0
  78. package/src/adapters/shared/server-adapter.js +353 -0
  79. package/src/adapters/shared/server-adapter.ts +442 -0
  80. package/src/adapters/shared/server-route-handler.d.ts +89 -0
  81. package/src/adapters/shared/server-route-handler.js +120 -0
  82. package/src/adapters/shared/server-route-handler.ts +166 -0
  83. package/src/adapters/shared/server-static-builder.d.ts +38 -0
  84. package/src/adapters/shared/server-static-builder.js +46 -0
  85. package/src/adapters/shared/server-static-builder.ts +82 -0
  86. package/src/build/build-adapter.d.ts +74 -0
  87. package/src/build/build-adapter.js +54 -0
  88. package/src/build/build-adapter.ts +132 -0
  89. package/src/build/build-types.d.ts +57 -0
  90. package/src/build/build-types.js +0 -0
  91. package/src/build/build-types.ts +83 -0
  92. package/src/build/esbuild-build-adapter.d.ts +69 -0
  93. package/src/build/esbuild-build-adapter.js +390 -0
  94. package/src/build/esbuild-build-adapter.ts +510 -0
  95. package/src/config/config-builder.d.ts +227 -0
  96. package/src/config/config-builder.js +392 -0
  97. package/src/config/config-builder.ts +474 -0
  98. package/src/constants.d.ts +32 -0
  99. package/src/constants.js +21 -0
  100. package/src/constants.ts +39 -0
  101. package/src/create-app.d.ts +17 -0
  102. package/src/create-app.js +66 -0
  103. package/src/create-app.ts +87 -0
  104. package/src/declarations.d.ts +26 -0
  105. package/src/define-api-handler.d.ts +25 -0
  106. package/src/define-api-handler.js +15 -0
  107. package/src/define-api-handler.ts +66 -0
  108. package/src/dev/sc-server.d.ts +30 -0
  109. package/src/dev/sc-server.js +111 -0
  110. package/src/dev/sc-server.ts +143 -0
  111. package/src/eco/README.md +636 -0
  112. package/src/eco/component-render-context.d.ts +105 -0
  113. package/src/eco/component-render-context.js +77 -0
  114. package/src/eco/component-render-context.ts +202 -0
  115. package/src/eco/eco.d.ts +9 -0
  116. package/src/eco/eco.js +110 -0
  117. package/src/eco/eco.ts +221 -0
  118. package/src/eco/eco.types.d.ts +170 -0
  119. package/src/eco/eco.types.js +0 -0
  120. package/src/eco/eco.types.ts +202 -0
  121. package/src/eco/eco.utils.d.ts +40 -0
  122. package/src/eco/eco.utils.js +40 -0
  123. package/src/eco/eco.utils.ts +89 -0
  124. package/src/eco/global-injector-map.d.ts +16 -0
  125. package/src/eco/global-injector-map.js +80 -0
  126. package/src/eco/global-injector-map.ts +112 -0
  127. package/src/eco/lazy-injector-map.d.ts +8 -0
  128. package/src/eco/lazy-injector-map.js +70 -0
  129. package/src/eco/lazy-injector-map.ts +120 -0
  130. package/src/eco/module-dependencies.d.ts +18 -0
  131. package/src/eco/module-dependencies.js +49 -0
  132. package/src/eco/module-dependencies.ts +75 -0
  133. package/src/env.d.ts +20 -0
  134. package/src/errors/http-error.d.ts +31 -0
  135. package/src/errors/http-error.js +50 -0
  136. package/src/errors/http-error.ts +72 -0
  137. package/src/errors/index.d.ts +2 -0
  138. package/src/errors/index.js +4 -0
  139. package/src/errors/index.ts +2 -0
  140. package/src/errors/locals-access-error.d.ts +4 -0
  141. package/src/errors/locals-access-error.js +9 -0
  142. package/src/errors/locals-access-error.ts +7 -0
  143. package/src/global/app-logger.d.ts +2 -0
  144. package/src/global/app-logger.js +6 -0
  145. package/src/global/app-logger.ts +4 -0
  146. 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
  147. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  148. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  149. package/src/hmr/client/hmr-runtime.d.ts +10 -0
  150. package/src/hmr/client/hmr-runtime.js +86 -0
  151. package/src/hmr/client/hmr-runtime.ts +121 -0
  152. package/src/hmr/hmr-strategy.d.ts +159 -0
  153. package/src/hmr/hmr-strategy.js +29 -0
  154. package/src/hmr/hmr-strategy.ts +172 -0
  155. package/src/hmr/hmr.test.e2e.d.ts +1 -0
  156. package/src/hmr/hmr.test.e2e.js +50 -0
  157. package/src/hmr/hmr.test.e2e.ts +75 -0
  158. package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
  159. package/src/hmr/strategies/default-hmr-strategy.js +34 -0
  160. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  161. package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
  162. package/src/hmr/strategies/js-hmr-strategy.js +179 -0
  163. package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
  164. package/src/index.browser.d.ts +3 -0
  165. package/src/index.browser.js +4 -0
  166. package/src/index.browser.ts +3 -0
  167. package/src/index.d.ts +5 -0
  168. package/src/index.js +10 -0
  169. package/src/index.ts +5 -0
  170. package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
  171. package/src/integrations/ghtml/ghtml-renderer.js +60 -0
  172. package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
  173. package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
  174. package/src/integrations/ghtml/ghtml.plugin.js +21 -0
  175. package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
  176. package/src/internal-types.d.ts +200 -0
  177. package/src/internal-types.js +0 -0
  178. package/src/internal-types.ts +212 -0
  179. package/src/plugins/alias-resolver-plugin.d.ts +2 -0
  180. package/src/plugins/alias-resolver-plugin.js +39 -0
  181. package/src/plugins/alias-resolver-plugin.ts +45 -0
  182. package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
  183. package/src/plugins/eco-component-meta-plugin.js +157 -0
  184. package/src/plugins/eco-component-meta-plugin.ts +474 -0
  185. package/src/plugins/integration-plugin.d.ts +102 -0
  186. package/src/plugins/integration-plugin.js +100 -0
  187. package/src/plugins/integration-plugin.ts +184 -0
  188. package/src/plugins/processor.d.ts +82 -0
  189. package/src/plugins/processor.js +122 -0
  190. package/src/plugins/processor.ts +220 -0
  191. package/src/public-types.d.ts +1094 -0
  192. package/src/public-types.js +0 -0
  193. package/src/public-types.ts +1255 -0
  194. package/src/route-renderer/GRAPH.md +387 -0
  195. package/src/route-renderer/README.md +135 -0
  196. package/src/route-renderer/component-graph-executor.d.ts +32 -0
  197. package/src/route-renderer/component-graph-executor.js +31 -0
  198. package/src/route-renderer/component-graph-executor.ts +84 -0
  199. package/src/route-renderer/component-graph.d.ts +42 -0
  200. package/src/route-renderer/component-graph.js +72 -0
  201. package/src/route-renderer/component-graph.ts +159 -0
  202. package/src/route-renderer/component-marker.d.ts +52 -0
  203. package/src/route-renderer/component-marker.js +46 -0
  204. package/src/route-renderer/component-marker.ts +117 -0
  205. package/src/route-renderer/dependency-resolver.d.ts +24 -0
  206. package/src/route-renderer/dependency-resolver.js +428 -0
  207. package/src/route-renderer/dependency-resolver.ts +596 -0
  208. package/src/route-renderer/html-post-processing.service.d.ts +40 -0
  209. package/src/route-renderer/html-post-processing.service.js +86 -0
  210. package/src/route-renderer/html-post-processing.service.ts +103 -0
  211. package/src/route-renderer/integration-renderer.d.ts +339 -0
  212. package/src/route-renderer/integration-renderer.js +526 -0
  213. package/src/route-renderer/integration-renderer.ts +696 -0
  214. package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
  215. package/src/route-renderer/marker-graph-resolver.js +93 -0
  216. package/src/route-renderer/marker-graph-resolver.ts +153 -0
  217. package/src/route-renderer/page-module-loader.d.ts +61 -0
  218. package/src/route-renderer/page-module-loader.js +102 -0
  219. package/src/route-renderer/page-module-loader.ts +153 -0
  220. package/src/route-renderer/render-execution.service.d.ts +69 -0
  221. package/src/route-renderer/render-execution.service.js +91 -0
  222. package/src/route-renderer/render-execution.service.ts +158 -0
  223. package/src/route-renderer/render-preparation.service.d.ts +112 -0
  224. package/src/route-renderer/render-preparation.service.js +243 -0
  225. package/src/route-renderer/render-preparation.service.ts +358 -0
  226. package/src/route-renderer/route-renderer.d.ts +26 -0
  227. package/src/route-renderer/route-renderer.js +68 -0
  228. package/src/route-renderer/route-renderer.ts +80 -0
  229. package/src/router/fs-router-scanner.d.ts +41 -0
  230. package/src/router/fs-router-scanner.js +155 -0
  231. package/src/router/fs-router-scanner.ts +217 -0
  232. package/src/router/fs-router.d.ts +26 -0
  233. package/src/router/fs-router.js +100 -0
  234. package/src/router/fs-router.ts +122 -0
  235. package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
  236. package/src/services/asset-processing-service/asset-processing.service.js +250 -0
  237. package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
  238. package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
  239. package/src/services/asset-processing-service/asset.factory.js +82 -0
  240. package/src/services/asset-processing-service/asset.factory.ts +105 -0
  241. package/src/services/asset-processing-service/assets.types.d.ts +88 -0
  242. package/src/services/asset-processing-service/assets.types.js +0 -0
  243. package/src/services/asset-processing-service/assets.types.ts +112 -0
  244. package/src/services/asset-processing-service/index.d.ts +3 -0
  245. package/src/services/asset-processing-service/index.js +3 -0
  246. package/src/services/asset-processing-service/index.ts +3 -0
  247. package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
  248. package/src/services/asset-processing-service/processor.interface.js +6 -0
  249. package/src/services/asset-processing-service/processor.interface.ts +27 -0
  250. package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
  251. package/src/services/asset-processing-service/processor.registry.js +15 -0
  252. package/src/services/asset-processing-service/processor.registry.ts +18 -0
  253. package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
  254. package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
  255. package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
  256. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
  257. package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
  258. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
  259. package/src/services/asset-processing-service/processors/index.d.ts +5 -0
  260. package/src/services/asset-processing-service/processors/index.js +5 -0
  261. package/src/services/asset-processing-service/processors/index.ts +5 -0
  262. package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
  263. package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
  264. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
  265. package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
  266. package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
  267. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
  268. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
  269. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
  270. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
  271. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
  272. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
  273. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
  274. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
  275. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
  276. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
  277. package/src/services/cache/cache.types.d.ts +107 -0
  278. package/src/services/cache/cache.types.js +0 -0
  279. package/src/services/cache/cache.types.ts +126 -0
  280. package/src/services/cache/index.d.ts +7 -0
  281. package/src/services/cache/index.js +7 -0
  282. package/src/services/cache/index.ts +18 -0
  283. package/src/services/cache/memory-cache-store.d.ts +42 -0
  284. package/src/services/cache/memory-cache-store.js +98 -0
  285. package/src/services/cache/memory-cache-store.ts +130 -0
  286. package/src/services/cache/page-cache-service.d.ts +70 -0
  287. package/src/services/cache/page-cache-service.js +152 -0
  288. package/src/services/cache/page-cache-service.ts +202 -0
  289. package/src/services/html-transformer.service.d.ts +50 -0
  290. package/src/services/html-transformer.service.js +163 -0
  291. package/src/services/html-transformer.service.ts +217 -0
  292. package/src/services/page-module-import.service.d.ts +37 -0
  293. package/src/services/page-module-import.service.js +88 -0
  294. package/src/services/page-module-import.service.ts +129 -0
  295. package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
  296. package/src/services/page-request-cache-coordinator.service.js +107 -0
  297. package/src/services/page-request-cache-coordinator.service.ts +128 -0
  298. package/src/services/schema-validation-service.d.ts +122 -0
  299. package/src/services/schema-validation-service.js +101 -0
  300. package/src/services/schema-validation-service.ts +204 -0
  301. package/src/services/validation/standard-schema.types.d.ts +65 -0
  302. package/src/services/validation/standard-schema.types.js +0 -0
  303. package/src/services/validation/standard-schema.types.ts +68 -0
  304. package/src/static-site-generator/static-site-generator.d.ts +57 -0
  305. package/src/static-site-generator/static-site-generator.js +272 -0
  306. package/src/static-site-generator/static-site-generator.ts +359 -0
  307. package/src/utils/css.d.ts +1 -0
  308. package/src/utils/css.js +7 -0
  309. package/src/utils/css.ts +5 -0
  310. package/src/utils/deep-merge.d.ts +14 -0
  311. package/src/utils/deep-merge.js +32 -0
  312. package/src/utils/deep-merge.ts +47 -0
  313. package/src/utils/hash.d.ts +1 -0
  314. package/src/utils/hash.js +7 -0
  315. package/src/utils/hash.ts +5 -0
  316. package/src/utils/html.d.ts +1 -0
  317. package/src/utils/html.js +4 -0
  318. package/src/utils/html.ts +1 -0
  319. package/src/utils/invariant.d.ts +5 -0
  320. package/src/utils/invariant.js +11 -0
  321. package/src/utils/invariant.ts +15 -0
  322. package/src/utils/locals-utils.d.ts +15 -0
  323. package/src/utils/locals-utils.js +24 -0
  324. package/src/utils/locals-utils.ts +37 -0
  325. package/src/utils/parse-cli-args.d.ts +24 -0
  326. package/src/utils/parse-cli-args.js +47 -0
  327. package/src/utils/parse-cli-args.ts +83 -0
  328. package/src/utils/path-utils.module.d.ts +5 -0
  329. package/src/utils/path-utils.module.js +14 -0
  330. package/src/utils/path-utils.module.ts +14 -0
  331. package/src/utils/runtime.d.ts +11 -0
  332. package/src/utils/runtime.js +40 -0
  333. package/src/utils/runtime.ts +44 -0
  334. package/src/utils/server-utils.module.d.ts +19 -0
  335. package/src/utils/server-utils.module.js +56 -0
  336. package/src/utils/server-utils.module.ts +67 -0
  337. package/src/watchers/project-watcher.d.ts +120 -0
  338. package/src/watchers/project-watcher.js +238 -0
  339. package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
  340. package/src/watchers/project-watcher.test-helpers.js +51 -0
  341. package/src/watchers/project-watcher.test-helpers.ts +40 -0
  342. package/src/watchers/project-watcher.ts +306 -0
@@ -0,0 +1,172 @@
1
+ import type { ClientBridgeEvent } from '../public-types';
2
+
3
+ /**
4
+ * HMR Strategy Pattern
5
+ *
6
+ * This module defines the base classes and types for Hot Module Replacement strategies.
7
+ * Each strategy handles a specific type of file change and determines how to process it.
8
+ *
9
+ * @module
10
+ * @example
11
+ * ```typescript
12
+ * class CustomHmrStrategy extends HmrStrategy {
13
+ * readonly type = HmrStrategyType.INTEGRATION;
14
+ *
15
+ * matches(filePath: string): boolean {
16
+ * return filePath.endsWith('.custom');
17
+ * }
18
+ *
19
+ * async process(filePath: string): Promise<HmrAction> {
20
+ * return {
21
+ * type: 'broadcast',
22
+ * events: [{ type: 'update', path: filePath, timestamp: Date.now() }]
23
+ * };
24
+ * }
25
+ * }
26
+
27
+ * Defines the category of an HMR strategy, which determines its execution priority.
28
+ * Strategies are evaluated in descending order: INTEGRATION → ASSET → SCRIPT → FALLBACK.
29
+ *
30
+ * @remarks
31
+ * The numeric values represent base priorities. Strategies can fine-tune their priority
32
+ * using the `priorityOffset` property.
33
+ */
34
+ export enum HmrStrategyType {
35
+ /**
36
+ * Integration-specific strategies (React, Lit, etc.)
37
+ * Highest priority to allow framework-specific HMR handling.
38
+ */
39
+ INTEGRATION = 100,
40
+
41
+ /**
42
+ * Asset processing strategies (CSS, images, etc.)
43
+ * High priority for specialized asset handling.
44
+ */
45
+ ASSET = 50,
46
+
47
+ /**
48
+ * Generic script bundling strategies (JS/TS)
49
+ * Medium priority for standard script processing.
50
+ */
51
+ SCRIPT = 25,
52
+
53
+ /**
54
+ * Fallback strategy for unhandled file types.
55
+ * Lowest priority, triggers full page reload.
56
+ */
57
+ FALLBACK = 0,
58
+ }
59
+
60
+ /**
61
+ * Represents an action to be taken after processing a file change.
62
+ */
63
+ export interface HmrAction {
64
+ /**
65
+ * Whether to broadcast an HMR event to connected clients.
66
+ */
67
+ type: 'broadcast' | 'none';
68
+
69
+ /**
70
+ * The HMR events to broadcast, if type is 'broadcast'.
71
+ * capable of broadcasting multiple events at once.
72
+ */
73
+ events?: ClientBridgeEvent[];
74
+ }
75
+
76
+ /**
77
+ * Base class for HMR strategies.
78
+ *
79
+ * Each strategy handles a specific type of file change and determines how to process it.
80
+ * Strategies are selected based on their priority (higher values are evaluated first) and
81
+ * whether they match the changed file path.
82
+ *
83
+ * @remarks
84
+ * Strategies should be stateless and idempotent. The same file change should always
85
+ * produce the same result when processed by the same strategy.
86
+ *
87
+ * @example
88
+ * ```typescript
89
+ * class MyAssetStrategy extends HmrStrategy {
90
+ * readonly type = HmrStrategyType.ASSET;
91
+ * readonly priorityOffset = 5;
92
+ *
93
+ * matches(filePath: string): boolean {
94
+ * return /\.(png|jpg|svg)$/.test(filePath);
95
+ * }
96
+ *
97
+ * async process(filePath: string): Promise<HmrAction> {
98
+ * await this.optimizeImage(filePath);
99
+ * return {
100
+ * type: 'broadcast',
101
+ * events: [{ type: 'update', path: filePath, timestamp: Date.now() }]
102
+ * };
103
+ * }
104
+ * }
105
+ * ```
106
+ */
107
+ export abstract class HmrStrategy {
108
+ /**
109
+ * The category of this strategy, determining its base priority.
110
+ */
111
+ abstract readonly type: HmrStrategyType;
112
+
113
+ /**
114
+ * Optional offset to fine-tune priority within the same category.
115
+ * Useful when multiple strategies share the same type.
116
+ *
117
+ * @defaultValue 0
118
+ */
119
+ readonly priorityOffset: number = 0;
120
+
121
+ /**
122
+ * Computed priority for strategy selection.
123
+ * Higher values are evaluated first.
124
+ *
125
+ * @returns The sum of the strategy type and priority offset
126
+ */
127
+ get priority(): number {
128
+ return this.type + this.priorityOffset;
129
+ }
130
+
131
+ /**
132
+ * Determines if this strategy can handle the given file path.
133
+ *
134
+ * @param filePath - Absolute path to the changed file
135
+ * @returns True if this strategy should process the file
136
+ *
137
+ * @example
138
+ * ```typescript
139
+ * matches(filePath: string): boolean {
140
+ * return filePath.endsWith('.css');
141
+ * }
142
+ * ```
143
+ */
144
+ abstract matches(filePath: string): boolean;
145
+
146
+ /**
147
+ * Processes a file change and returns the action to take.
148
+ *
149
+ * This method may perform side effects such as:
150
+ * - Rebuilding files
151
+ * - Writing to disk
152
+ * - Transforming code
153
+ * - Updating caches
154
+ *
155
+ * @param filePath - Absolute path to the changed file
156
+ * @returns Action to take (broadcast event or none)
157
+ *
158
+ * @example
159
+ * ```typescript
160
+ * async process(filePath: string): Promise<HmrAction> {
161
+ * const processed = await this.transform(filePath);
162
+ * await Bun.write(outputPath, processed);
163
+ *
164
+ * return {
165
+ * type: 'broadcast',
166
+ * events: [{ type: 'update', path: outputPath, timestamp: Date.now() }]
167
+ * };
168
+ * }
169
+ * ```
170
+ */
171
+ abstract process(filePath: string): Promise<HmrAction>;
172
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,50 @@
1
+ import { test, expect } from "@playwright/test";
2
+ import { readFileSync } from "node:fs";
3
+ import { writeFile } from "node:fs/promises";
4
+ import { resolve } from "node:path";
5
+ const FIXTURE_DIR = resolve(process.cwd(), "packages/core/__fixtures__/app");
6
+ const TEST_CSS_FILE = resolve(FIXTURE_DIR, "src/pages/index.css");
7
+ const BUILT_CSS_FILE = resolve(FIXTURE_DIR, ".eco/assets/pages/index.css");
8
+ const TEST_URL = "http://localhost:3002";
9
+ test.describe("HMR E2E", () => {
10
+ test("should load page with .main-title element", async ({ page }) => {
11
+ await page.goto(TEST_URL, { waitUntil: "networkidle" });
12
+ const title = page.locator(".main-title").first();
13
+ await expect(title).toBeVisible();
14
+ });
15
+ test("should connect to HMR WebSocket", async ({ page }) => {
16
+ await page.goto(TEST_URL);
17
+ const connected = await page.evaluate(async () => {
18
+ return new Promise((resolve2) => {
19
+ const ws = new WebSocket("ws://localhost:3002/_hmr");
20
+ ws.onopen = () => {
21
+ ws.close();
22
+ resolve2(true);
23
+ };
24
+ ws.onerror = () => resolve2(false);
25
+ setTimeout(() => resolve2(false), 5e3);
26
+ });
27
+ });
28
+ expect(connected).toBe(true);
29
+ });
30
+ test("should reload page when CSS file changes", async ({ page }) => {
31
+ const originalCss = readFileSync(TEST_CSS_FILE, "utf-8");
32
+ await page.goto(TEST_URL, { waitUntil: "networkidle" });
33
+ const initialColor = await page.evaluate(() => {
34
+ const el = document.querySelector(".main-title");
35
+ return el ? getComputedStyle(el).color : null;
36
+ });
37
+ expect(initialColor).toBeTruthy();
38
+ const modifiedCss = originalCss.replace(".main-title {", ".main-title {\n color: rgb(255, 0, 0);");
39
+ const loadPromise = page.waitForEvent("load", { timeout: 1e4 });
40
+ await writeFile(TEST_CSS_FILE, modifiedCss, { flush: true });
41
+ await writeFile(BUILT_CSS_FILE, modifiedCss, { flush: true });
42
+ await loadPromise;
43
+ const updatedColor = await page.evaluate(() => {
44
+ const el = document.querySelector(".main-title");
45
+ return el ? getComputedStyle(el).color : null;
46
+ });
47
+ expect(updatedColor).toBe("rgb(255, 0, 0)");
48
+ await writeFile(TEST_CSS_FILE, originalCss, { flush: true });
49
+ });
50
+ });
@@ -0,0 +1,75 @@
1
+ import { test, expect } from '@playwright/test';
2
+ import { readFileSync } from 'node:fs';
3
+ import { writeFile } from 'node:fs/promises';
4
+ import { resolve } from 'node:path';
5
+
6
+ /**
7
+ * HMR E2E Tests
8
+ *
9
+ * True end-to-end tests for Hot Module Replacement using Playwright.
10
+ * These tests modify actual source files and verify the browser updates.
11
+ *
12
+ * Run with: bunx playwright test packages/core/src/hmr/hmr.e2e.test.ts
13
+ */
14
+
15
+ const FIXTURE_DIR = resolve(process.cwd(), 'packages/core/__fixtures__/app');
16
+ const TEST_CSS_FILE = resolve(FIXTURE_DIR, 'src/pages/index.css');
17
+ const BUILT_CSS_FILE = resolve(FIXTURE_DIR, '.eco/assets/pages/index.css');
18
+ const TEST_URL = 'http://localhost:3002';
19
+
20
+ test.describe('HMR E2E', () => {
21
+ test('should load page with .main-title element', async ({ page }) => {
22
+ await page.goto(TEST_URL, { waitUntil: 'networkidle' });
23
+ const title = page.locator('.main-title').first();
24
+ await expect(title).toBeVisible();
25
+ });
26
+
27
+ test('should connect to HMR WebSocket', async ({ page }) => {
28
+ await page.goto(TEST_URL);
29
+
30
+ const connected = await page.evaluate(async () => {
31
+ return new Promise<boolean>((resolve) => {
32
+ const ws = new WebSocket('ws://localhost:3002/_hmr');
33
+ ws.onopen = () => {
34
+ ws.close();
35
+ resolve(true);
36
+ };
37
+ ws.onerror = () => resolve(false);
38
+ setTimeout(() => resolve(false), 5000);
39
+ });
40
+ });
41
+
42
+ expect(connected).toBe(true);
43
+ });
44
+
45
+ test('should reload page when CSS file changes', async ({ page }) => {
46
+ const originalCss = readFileSync(TEST_CSS_FILE, 'utf-8');
47
+
48
+ await page.goto(TEST_URL, { waitUntil: 'networkidle' });
49
+
50
+ const initialColor = await page.evaluate(() => {
51
+ const el = document.querySelector('.main-title');
52
+ return el ? getComputedStyle(el).color : null;
53
+ });
54
+
55
+ expect(initialColor).toBeTruthy();
56
+
57
+ const modifiedCss = originalCss.replace('.main-title {', '.main-title {\n\tcolor: rgb(255, 0, 0);');
58
+
59
+ const loadPromise = page.waitForEvent('load', { timeout: 10000 });
60
+
61
+ await writeFile(TEST_CSS_FILE, modifiedCss, { flush: true });
62
+ await writeFile(BUILT_CSS_FILE, modifiedCss, { flush: true });
63
+
64
+ await loadPromise;
65
+
66
+ const updatedColor = await page.evaluate(() => {
67
+ const el = document.querySelector('.main-title');
68
+ return el ? getComputedStyle(el).color : null;
69
+ });
70
+
71
+ expect(updatedColor).toBe('rgb(255, 0, 0)');
72
+
73
+ await writeFile(TEST_CSS_FILE, originalCss, { flush: true });
74
+ });
75
+ });
@@ -0,0 +1,43 @@
1
+ /**
2
+ * Default HMR Strategy
3
+ *
4
+ * Fallback strategy for file types that don't have specialized handling.
5
+ * Triggers a full page reload to ensure changes are reflected.
6
+ *
7
+ * @module
8
+ */
9
+ import { HmrStrategy, HmrStrategyType, type HmrAction } from '../hmr-strategy';
10
+ /**
11
+ * Default fallback strategy for unhandled file types.
12
+ *
13
+ * This strategy matches all files and triggers a full page reload.
14
+ * It has the lowest priority (FALLBACK) and acts as a catch-all when
15
+ * no other strategy matches the changed file.
16
+ *
17
+ * @remarks
18
+ * This strategy ensures that all file changes result in some action,
19
+ * even if we don't have specialized handling for that file type.
20
+ *
21
+ * @example
22
+ * ```typescript
23
+ * const strategy = new DefaultHmrStrategy();
24
+ * const action = await strategy.process('/path/to/unknown.file');
25
+ * ```
26
+ */
27
+ export declare class DefaultHmrStrategy extends HmrStrategy {
28
+ readonly type = HmrStrategyType.FALLBACK;
29
+ /**
30
+ * Matches all file paths.
31
+ *
32
+ * @param _filePath - Absolute path to the changed file (unused)
33
+ * @returns Always returns true as this is a catch-all strategy
34
+ */
35
+ matches(_filePath: string): boolean;
36
+ /**
37
+ * Processes a file change by triggering a full page reload.
38
+ *
39
+ * @param filePath - Absolute path to the changed file
40
+ * @returns Action to broadcast a reload event
41
+ */
42
+ process(filePath: string): Promise<HmrAction>;
43
+ }
@@ -0,0 +1,34 @@
1
+ import { HmrStrategy, HmrStrategyType } from "../hmr-strategy";
2
+ class DefaultHmrStrategy extends HmrStrategy {
3
+ type = HmrStrategyType.FALLBACK;
4
+ /**
5
+ * Matches all file paths.
6
+ *
7
+ * @param _filePath - Absolute path to the changed file (unused)
8
+ * @returns Always returns true as this is a catch-all strategy
9
+ */
10
+ matches(_filePath) {
11
+ return true;
12
+ }
13
+ /**
14
+ * Processes a file change by triggering a full page reload.
15
+ *
16
+ * @param filePath - Absolute path to the changed file
17
+ * @returns Action to broadcast a reload event
18
+ */
19
+ async process(filePath) {
20
+ return {
21
+ type: "broadcast",
22
+ events: [
23
+ {
24
+ type: "reload",
25
+ path: filePath,
26
+ message: "fallback-strategy"
27
+ }
28
+ ]
29
+ };
30
+ }
31
+ }
32
+ export {
33
+ DefaultHmrStrategy
34
+ };
@@ -0,0 +1,60 @@
1
+ /**
2
+ * Default HMR Strategy
3
+ *
4
+ * Fallback strategy for file types that don't have specialized handling.
5
+ * Triggers a full page reload to ensure changes are reflected.
6
+ *
7
+ * @module
8
+ */
9
+
10
+ import { HmrStrategy, HmrStrategyType, type HmrAction } from '../hmr-strategy';
11
+
12
+ /**
13
+ * Default fallback strategy for unhandled file types.
14
+ *
15
+ * This strategy matches all files and triggers a full page reload.
16
+ * It has the lowest priority (FALLBACK) and acts as a catch-all when
17
+ * no other strategy matches the changed file.
18
+ *
19
+ * @remarks
20
+ * This strategy ensures that all file changes result in some action,
21
+ * even if we don't have specialized handling for that file type.
22
+ *
23
+ * @example
24
+ * ```typescript
25
+ * const strategy = new DefaultHmrStrategy();
26
+ * const action = await strategy.process('/path/to/unknown.file');
27
+ * ```
28
+ */
29
+ export class DefaultHmrStrategy extends HmrStrategy {
30
+ readonly type = HmrStrategyType.FALLBACK;
31
+
32
+ /**
33
+ * Matches all file paths.
34
+ *
35
+ * @param _filePath - Absolute path to the changed file (unused)
36
+ * @returns Always returns true as this is a catch-all strategy
37
+ */
38
+ matches(_filePath: string): boolean {
39
+ return true;
40
+ }
41
+
42
+ /**
43
+ * Processes a file change by triggering a full page reload.
44
+ *
45
+ * @param filePath - Absolute path to the changed file
46
+ * @returns Action to broadcast a reload event
47
+ */
48
+ async process(filePath: string): Promise<HmrAction> {
49
+ return {
50
+ type: 'broadcast',
51
+ events: [
52
+ {
53
+ type: 'reload',
54
+ path: filePath,
55
+ message: 'fallback-strategy',
56
+ },
57
+ ],
58
+ };
59
+ }
60
+ }
@@ -0,0 +1,136 @@
1
+ /**
2
+ * JavaScript HMR Strategy
3
+ *
4
+ * Handles hot module replacement for JavaScript and TypeScript entrypoints.
5
+ * Bundles files, replaces bare specifiers, and injects HMR boilerplate.
6
+ *
7
+ * @module
8
+ */
9
+ import { HmrStrategy, HmrStrategyType, type HmrAction } from '../hmr-strategy';
10
+ import type { EcoBuildPlugin } from '../../build/build-types.js';
11
+ /**
12
+ * Context interface providing access to HmrManager state.
13
+ * Required for JsHmrStrategy to access registered entrypoints and configuration.
14
+ */
15
+ export interface JsHmrContext {
16
+ /**
17
+ * Map of registered entrypoints to their output URLs.
18
+ */
19
+ getWatchedFiles(): Map<string, string>;
20
+ /**
21
+ * Map of bare specifiers to vendor URLs for import resolution.
22
+ */
23
+ getSpecifierMap(): Map<string, string>;
24
+ /**
25
+ * Returns entrypoints impacted by a changed dependency path.
26
+ *
27
+ * @remarks
28
+ * This hook is currently provided by the Node HMR manager where dependency
29
+ * graph metadata is extracted from the Node/esbuild build adapter.
30
+ */
31
+ getDependencyEntrypoints?(filePath: string): Set<string>;
32
+ /**
33
+ * Stores latest dependency set for an entrypoint.
34
+ *
35
+ * @remarks
36
+ * This hook is currently used only by the Node HMR manager to maintain a
37
+ * reverse invalidation index. Runtimes that do not provide it keep rebuild-all
38
+ * fallback semantics.
39
+ */
40
+ setEntrypointDependencies?(entrypointPath: string, dependencies: string[]): void;
41
+ /**
42
+ * Directory where HMR bundles are written.
43
+ */
44
+ getDistDir(): string;
45
+ /**
46
+ * Build plugins to use during bundling.
47
+ */
48
+ getPlugins(): EcoBuildPlugin[];
49
+ /**
50
+ * Absolute path to the source directory.
51
+ */
52
+ getSrcDir(): string;
53
+ }
54
+ /**
55
+ * Strategy for handling JavaScript/TypeScript file changes with hot reloading.
56
+ *
57
+ * This strategy rebuilds all registered entrypoints when any file changes,
58
+ * as we don't currently track dependencies. This is safe but inefficient.
59
+ *
60
+ * The processing steps are:
61
+ * 1. Check if any entrypoints are registered
62
+ * 2. Rebuild all entrypoints (the changed file could be a dependency)
63
+ * 3. Replace bare specifiers with vendor URLs
64
+ * 4. Inject generic HMR boilerplate
65
+ * 5. Broadcast update events for each rebuilt entrypoint
66
+ *
67
+ * @remarks
68
+ * Future enhancement: Track dependencies using Bun's transpiler API to only
69
+ * rebuild affected entrypoints instead of all of them.
70
+ *
71
+ * @see https://bun.sh/docs/runtime/transpiler
72
+ *
73
+ * @example
74
+ * ```typescript
75
+ * const context = {
76
+ * getWatchedFiles: () => watchedFilesMap,
77
+ * getSpecifierMap: () => specifierMap,
78
+ * getDistDir: () => '/path/to/dist/_hmr',
79
+ * getPlugins: () => [],
80
+ * getSrcDir: () => '/path/to/src'
81
+ * };
82
+ * const strategy = new JsHmrStrategy(context);
83
+ * ```
84
+ */
85
+ export declare class JsHmrStrategy extends HmrStrategy {
86
+ private context;
87
+ readonly type = HmrStrategyType.SCRIPT;
88
+ constructor(context: JsHmrContext);
89
+ /**
90
+ * Determines if the file is a JS/TS file that could affect registered entrypoints.
91
+ *
92
+ * Matches if:
93
+ * 1. There are registered entrypoints to rebuild
94
+ * 2. The changed file is a JS/TS file in the src directory
95
+ *
96
+ * @param filePath - Absolute path to the changed file
97
+ * @returns True if this file should trigger entrypoint rebuilds
98
+ */
99
+ matches(filePath: string): boolean;
100
+ /**
101
+ * Processes a file change by rebuilding affected entrypoints.
102
+ *
103
+ * @param filePath - Absolute path to the changed file
104
+ *
105
+ * @remarks
106
+ * If runtime-specific dependency graph hooks are unavailable, this strategy
107
+ * falls back to rebuilding all watched entrypoints.
108
+ * @returns Action to broadcast update events
109
+ */
110
+ process(filePath: string): Promise<HmrAction>;
111
+ /**
112
+ * Bundles a single entrypoint and processes the output.
113
+ *
114
+ * @param entrypointPath - Absolute path to the source file
115
+ * @param outputUrl - URL path for the bundled file
116
+ * @returns True if bundling was successful
117
+ */
118
+ private bundleEntrypoint;
119
+ /**
120
+ * Processes bundled output by replacing specifiers and injecting HMR code.
121
+ *
122
+ * @param filepath - Path to the bundled output file
123
+ * @param url - URL path for the bundled file
124
+ * @returns True if processing was successful and update should be broadcast
125
+ */
126
+ private processOutput;
127
+ /**
128
+ * Replaces bare specifiers with vendor URLs.
129
+ *
130
+ * Handles both static imports and dynamic imports.
131
+ *
132
+ * @param code - The bundled code to transform
133
+ * @returns The transformed code with vendor URLs
134
+ */
135
+ private replaceBareSpecifiers;
136
+ }