@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,79 @@
1
+ import type { WebSocket } from 'ws';
2
+ import type { ClientBridgeEvent, IClientBridge } from '../../public-types.ts';
3
+
4
+ const HEARTBEAT_INTERVAL_MS = 30_000;
5
+
6
+ export class NodeClientBridge implements IClientBridge {
7
+ private subscribers = new Set<WebSocket>();
8
+ private heartbeatTimer: ReturnType<typeof setInterval> | null = null;
9
+
10
+ constructor() {
11
+ this.heartbeatTimer = setInterval(() => this.sweep(), HEARTBEAT_INTERVAL_MS);
12
+ /* Allow the process to exit even if the timer is still active */
13
+ this.heartbeatTimer.unref?.();
14
+ }
15
+
16
+ /**
17
+ * Pings all open subscribers and removes any that are no longer in OPEN state.
18
+ * This prevents a slow memory leak caused by zombie connections that never
19
+ * send a close event (e.g. abruptly killed browsers, orphaned tabs, network drops).
20
+ */
21
+ private sweep(): void {
22
+ for (const ws of this.subscribers) {
23
+ if (ws.readyState !== 1 /* OPEN */) {
24
+ this.subscribers.delete(ws);
25
+ continue;
26
+ }
27
+ ws.ping();
28
+ }
29
+ }
30
+
31
+ subscribe(ws: WebSocket): void {
32
+ this.subscribers.add(ws);
33
+ }
34
+
35
+ unsubscribe(ws: WebSocket): void {
36
+ this.subscribers.delete(ws);
37
+ }
38
+
39
+ broadcast(event: ClientBridgeEvent): void {
40
+ const payload = JSON.stringify(event);
41
+ for (const ws of this.subscribers) {
42
+ if (ws.readyState === 1) {
43
+ ws.send(payload);
44
+ }
45
+ }
46
+ }
47
+
48
+ reload(): void {
49
+ this.broadcast({ type: 'reload' });
50
+ }
51
+
52
+ cssUpdate(path: string): void {
53
+ this.broadcast({ type: 'css-update', path, timestamp: Date.now() });
54
+ }
55
+
56
+ update(path: string): void {
57
+ this.broadcast({ type: 'update', path, timestamp: Date.now() });
58
+ }
59
+
60
+ error(message: string): void {
61
+ this.broadcast({ type: 'error', message });
62
+ }
63
+
64
+ get subscriberCount(): number {
65
+ return this.subscribers.size;
66
+ }
67
+
68
+ /**
69
+ * Stops the heartbeat timer and clears all subscribers.
70
+ * Call this when the dev server is shutting down.
71
+ */
72
+ destroy(): void {
73
+ if (this.heartbeatTimer !== null) {
74
+ clearInterval(this.heartbeatTimer);
75
+ this.heartbeatTimer = null;
76
+ }
77
+ this.subscribers.clear();
78
+ }
79
+ }
@@ -0,0 +1,62 @@
1
+ import type { DefaultHmrContext, EcoPagesAppConfig, IHmrManager, IClientBridge } from '../../internal-types.js';
2
+ import type { EcoBuildPlugin } from '../../build/build-types.js';
3
+ import type { HmrStrategy } from '../../hmr/hmr-strategy.js';
4
+ import type { ClientBridgeEvent } from '../../public-types.js';
5
+ export interface NodeHmrManagerParams {
6
+ appConfig: EcoPagesAppConfig;
7
+ bridge: IClientBridge;
8
+ }
9
+ export declare class NodeHmrManager implements IHmrManager {
10
+ readonly appConfig: EcoPagesAppConfig;
11
+ private readonly bridge;
12
+ private watchers;
13
+ private watchedFiles;
14
+ private specifierMap;
15
+ /**
16
+ * Node-only reverse invalidation index: dependency file -> affected entrypoints.
17
+ */
18
+ private dependencyEntrypoints;
19
+ /**
20
+ * Node-only forward index: entrypoint -> latest dependency set.
21
+ */
22
+ private entrypointDependencies;
23
+ private distDir;
24
+ private plugins;
25
+ private enabled;
26
+ private strategies;
27
+ constructor({ appConfig, bridge }: NodeHmrManagerParams);
28
+ /**
29
+ * Ensures the HMR output directory exists.
30
+ *
31
+ * This must not remove the directory because multiple app processes
32
+ * can share the same dist path during e2e runs.
33
+ */
34
+ private cleanDistDir;
35
+ private initializeStrategies;
36
+ registerStrategy(strategy: HmrStrategy): void;
37
+ setPlugins(plugins: EcoBuildPlugin[]): void;
38
+ setEnabled(enabled: boolean): void;
39
+ isEnabled(): boolean;
40
+ registerSpecifierMap(map: Record<string, string>): void;
41
+ buildRuntime(): Promise<void>;
42
+ getRuntimePath(): string;
43
+ broadcast(event: ClientBridgeEvent): void;
44
+ handleFileChange(filePath: string): Promise<void>;
45
+ getOutputUrl(entrypointPath: string): string | undefined;
46
+ getWatchedFiles(): Map<string, string>;
47
+ getSpecifierMap(): Map<string, string>;
48
+ getDistDir(): string;
49
+ getPlugins(): EcoBuildPlugin[];
50
+ getDefaultContext(): DefaultHmrContext;
51
+ /**
52
+ * Updates Node HMR dependency indexes for selective invalidation.
53
+ *
54
+ * @remarks
55
+ * Graph data comes from Node/esbuild build metadata and does not affect Bun
56
+ * HMR behavior.
57
+ */
58
+ private setEntrypointDependencies;
59
+ registerEntrypoint(entrypointPath: string): Promise<string>;
60
+ private encodeDynamicSegments;
61
+ stop(): void;
62
+ }
@@ -0,0 +1,221 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ import { RESOLVED_ASSETS_DIR } from "../../constants.js";
5
+ import { defaultBuildAdapter } from "../../build/build-adapter.js";
6
+ import { fileSystem } from "@ecopages/file-system";
7
+ import { DefaultHmrStrategy } from "../../hmr/strategies/default-hmr-strategy.js";
8
+ import { JsHmrStrategy } from "../../hmr/strategies/js-hmr-strategy.js";
9
+ import { appLogger } from "../../global/app-logger.js";
10
+ class NodeHmrManager {
11
+ appConfig;
12
+ bridge;
13
+ watchers = /* @__PURE__ */ new Map();
14
+ watchedFiles = /* @__PURE__ */ new Map();
15
+ specifierMap = /* @__PURE__ */ new Map();
16
+ /**
17
+ * Node-only reverse invalidation index: dependency file -> affected entrypoints.
18
+ */
19
+ dependencyEntrypoints = /* @__PURE__ */ new Map();
20
+ /**
21
+ * Node-only forward index: entrypoint -> latest dependency set.
22
+ */
23
+ entrypointDependencies = /* @__PURE__ */ new Map();
24
+ distDir;
25
+ plugins = [];
26
+ enabled = true;
27
+ strategies = [];
28
+ constructor({ appConfig, bridge }) {
29
+ this.appConfig = appConfig;
30
+ this.bridge = bridge;
31
+ this.distDir = path.join(this.appConfig.absolutePaths.distDir, RESOLVED_ASSETS_DIR, "_hmr");
32
+ this.cleanDistDir();
33
+ this.initializeStrategies();
34
+ }
35
+ /**
36
+ * Ensures the HMR output directory exists.
37
+ *
38
+ * This must not remove the directory because multiple app processes
39
+ * can share the same dist path during e2e runs.
40
+ */
41
+ cleanDistDir() {
42
+ fileSystem.ensureDir(this.distDir);
43
+ }
44
+ initializeStrategies() {
45
+ const jsContext = {
46
+ getWatchedFiles: () => this.watchedFiles,
47
+ getSpecifierMap: () => this.specifierMap,
48
+ getDependencyEntrypoints: (filePath) => new Set(this.dependencyEntrypoints.get(path.resolve(filePath)) ?? []),
49
+ setEntrypointDependencies: (entrypointPath, dependencies) => this.setEntrypointDependencies(entrypointPath, dependencies),
50
+ getDistDir: () => this.distDir,
51
+ getPlugins: () => this.plugins,
52
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir
53
+ };
54
+ this.strategies = [new JsHmrStrategy(jsContext), new DefaultHmrStrategy()];
55
+ }
56
+ registerStrategy(strategy) {
57
+ this.strategies.push(strategy);
58
+ }
59
+ setPlugins(plugins) {
60
+ this.plugins = [...plugins];
61
+ }
62
+ setEnabled(enabled) {
63
+ this.enabled = enabled;
64
+ }
65
+ isEnabled() {
66
+ return this.enabled;
67
+ }
68
+ registerSpecifierMap(map) {
69
+ for (const [specifier, url] of Object.entries(map)) {
70
+ this.specifierMap.set(specifier, url);
71
+ }
72
+ }
73
+ async buildRuntime() {
74
+ const currentDir = path.dirname(fileURLToPath(import.meta.url));
75
+ const runtimeSource = path.resolve(currentDir, "../../hmr/client/hmr-runtime.js");
76
+ const result = await defaultBuildAdapter.build({
77
+ entrypoints: [runtimeSource],
78
+ outdir: this.distDir,
79
+ naming: "_hmr_runtime.js",
80
+ minify: false,
81
+ ...defaultBuildAdapter.getTranspileOptions("hmr-runtime"),
82
+ plugins: this.plugins
83
+ });
84
+ if (!result.success) {
85
+ appLogger.error("[HMR] Failed to build runtime script:", result.logs);
86
+ }
87
+ }
88
+ getRuntimePath() {
89
+ return path.join(this.distDir, "_hmr_runtime.js");
90
+ }
91
+ broadcast(event) {
92
+ appLogger.debug(
93
+ `[HMR] Broadcasting ${event.type} event, path=${event.path || "all"}, subscribers=${this.bridge.subscriberCount}`
94
+ );
95
+ this.bridge.broadcast(event);
96
+ }
97
+ async handleFileChange(filePath) {
98
+ const sorted = [...this.strategies].sort((a, b) => b.priority - a.priority);
99
+ const strategy = sorted.find((s) => {
100
+ try {
101
+ return s.matches(filePath);
102
+ } catch (err) {
103
+ appLogger.error(`[NodeHmrManager] Error checking match for ${s.constructor.name}:`, err);
104
+ return false;
105
+ }
106
+ });
107
+ if (!strategy) {
108
+ appLogger.warn(`[HMR] No strategy found for ${filePath}`);
109
+ return;
110
+ }
111
+ appLogger.debug(`[NodeHmrManager] Selected strategy: ${strategy.constructor.name}`);
112
+ const action = await strategy.process(filePath);
113
+ if (action.type === "broadcast") {
114
+ if (action.events) {
115
+ for (const event of action.events) {
116
+ this.broadcast(event);
117
+ }
118
+ }
119
+ }
120
+ }
121
+ getOutputUrl(entrypointPath) {
122
+ return this.watchedFiles.get(entrypointPath);
123
+ }
124
+ getWatchedFiles() {
125
+ return this.watchedFiles;
126
+ }
127
+ getSpecifierMap() {
128
+ return this.specifierMap;
129
+ }
130
+ getDistDir() {
131
+ return this.distDir;
132
+ }
133
+ getPlugins() {
134
+ return this.plugins;
135
+ }
136
+ getDefaultContext() {
137
+ return {
138
+ getWatchedFiles: () => this.watchedFiles,
139
+ getSpecifierMap: () => this.specifierMap,
140
+ getDistDir: () => this.distDir,
141
+ getPlugins: () => this.plugins,
142
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
143
+ getLayoutsDir: () => this.appConfig.absolutePaths.layoutsDir,
144
+ getPagesDir: () => this.appConfig.absolutePaths.pagesDir
145
+ };
146
+ }
147
+ /**
148
+ * Updates Node HMR dependency indexes for selective invalidation.
149
+ *
150
+ * @remarks
151
+ * Graph data comes from Node/esbuild build metadata and does not affect Bun
152
+ * HMR behavior.
153
+ */
154
+ setEntrypointDependencies(entrypointPath, dependencies) {
155
+ const normalizedEntrypoint = path.resolve(entrypointPath);
156
+ const previousDependencies = this.entrypointDependencies.get(normalizedEntrypoint);
157
+ if (previousDependencies) {
158
+ for (const dependencyPath of previousDependencies) {
159
+ const entrypoints = this.dependencyEntrypoints.get(dependencyPath);
160
+ if (!entrypoints) {
161
+ continue;
162
+ }
163
+ entrypoints.delete(normalizedEntrypoint);
164
+ if (entrypoints.size === 0) {
165
+ this.dependencyEntrypoints.delete(dependencyPath);
166
+ }
167
+ }
168
+ }
169
+ const normalizedDependencies = /* @__PURE__ */ new Set([
170
+ normalizedEntrypoint,
171
+ ...dependencies.map((dependencyPath) => path.resolve(dependencyPath))
172
+ ]);
173
+ this.entrypointDependencies.set(normalizedEntrypoint, normalizedDependencies);
174
+ for (const dependencyPath of normalizedDependencies) {
175
+ const entrypoints = this.dependencyEntrypoints.get(dependencyPath) ?? /* @__PURE__ */ new Set();
176
+ entrypoints.add(normalizedEntrypoint);
177
+ this.dependencyEntrypoints.set(dependencyPath, entrypoints);
178
+ }
179
+ }
180
+ async registerEntrypoint(entrypointPath) {
181
+ if (this.watchedFiles.has(entrypointPath)) {
182
+ return this.watchedFiles.get(entrypointPath);
183
+ }
184
+ const srcDir = this.appConfig.absolutePaths.srcDir;
185
+ const relativePath = path.relative(srcDir, entrypointPath);
186
+ const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, ".js");
187
+ const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
188
+ const urlPath = encodedPathJs.split(path.sep).join("/");
189
+ const outputUrl = `/${path.join(RESOLVED_ASSETS_DIR, "_hmr", urlPath)}`;
190
+ const outputPath = path.join(this.distDir, urlPath);
191
+ this.watchedFiles.set(entrypointPath, outputUrl);
192
+ await this.handleFileChange(entrypointPath);
193
+ if (!fileSystem.exists(outputPath)) {
194
+ const fallback = await defaultBuildAdapter.build({
195
+ entrypoints: [entrypointPath],
196
+ outdir: this.distDir,
197
+ naming: encodedPathJs,
198
+ minify: false,
199
+ external: Array.from(this.specifierMap.keys()),
200
+ ...defaultBuildAdapter.getTranspileOptions("hmr-entrypoint"),
201
+ plugins: this.plugins
202
+ });
203
+ if (!fallback.success) {
204
+ appLogger.error(`[HMR] Fallback build failed for ${entrypointPath}:`, fallback.logs);
205
+ }
206
+ }
207
+ return outputUrl;
208
+ }
209
+ encodeDynamicSegments(filepath) {
210
+ return filepath.replace(/\[([^\]]+)\]/g, "_$1_");
211
+ }
212
+ stop() {
213
+ for (const watcher of this.watchers.values()) {
214
+ watcher.close();
215
+ }
216
+ this.watchers.clear();
217
+ }
218
+ }
219
+ export {
220
+ NodeHmrManager
221
+ };
@@ -0,0 +1,271 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ import { RESOLVED_ASSETS_DIR } from '../../constants.ts';
5
+ import { defaultBuildAdapter } from '../../build/build-adapter.ts';
6
+ import type { DefaultHmrContext, EcoPagesAppConfig, IHmrManager, IClientBridge } from '../../internal-types.ts';
7
+ import type { EcoBuildPlugin } from '../../build/build-types.ts';
8
+ import { fileSystem } from '@ecopages/file-system';
9
+ import type { HmrStrategy } from '../../hmr/hmr-strategy.ts';
10
+ import { DefaultHmrStrategy } from '../../hmr/strategies/default-hmr-strategy.ts';
11
+ import { JsHmrStrategy } from '../../hmr/strategies/js-hmr-strategy.ts';
12
+ import { appLogger } from '../../global/app-logger.ts';
13
+ import type { ClientBridgeEvent } from '../../public-types.ts';
14
+
15
+ export interface NodeHmrManagerParams {
16
+ appConfig: EcoPagesAppConfig;
17
+ bridge: IClientBridge;
18
+ }
19
+
20
+ export class NodeHmrManager implements IHmrManager {
21
+ public readonly appConfig: EcoPagesAppConfig;
22
+ private readonly bridge: IClientBridge;
23
+ private watchers = new Map<string, fs.FSWatcher>();
24
+ private watchedFiles = new Map<string, string>();
25
+ private specifierMap = new Map<string, string>();
26
+ /**
27
+ * Node-only reverse invalidation index: dependency file -> affected entrypoints.
28
+ */
29
+ private dependencyEntrypoints = new Map<string, Set<string>>();
30
+ /**
31
+ * Node-only forward index: entrypoint -> latest dependency set.
32
+ */
33
+ private entrypointDependencies = new Map<string, Set<string>>();
34
+ private distDir: string;
35
+ private plugins: EcoBuildPlugin[] = [];
36
+ private enabled = true;
37
+ private strategies: HmrStrategy[] = [];
38
+
39
+ constructor({ appConfig, bridge }: NodeHmrManagerParams) {
40
+ this.appConfig = appConfig;
41
+ this.bridge = bridge;
42
+ this.distDir = path.join(this.appConfig.absolutePaths.distDir, RESOLVED_ASSETS_DIR, '_hmr');
43
+ this.cleanDistDir();
44
+ this.initializeStrategies();
45
+ }
46
+
47
+ /**
48
+ * Ensures the HMR output directory exists.
49
+ *
50
+ * This must not remove the directory because multiple app processes
51
+ * can share the same dist path during e2e runs.
52
+ */
53
+ private cleanDistDir(): void {
54
+ fileSystem.ensureDir(this.distDir);
55
+ }
56
+
57
+ private initializeStrategies(): void {
58
+ const jsContext = {
59
+ getWatchedFiles: () => this.watchedFiles,
60
+ getSpecifierMap: () => this.specifierMap,
61
+ getDependencyEntrypoints: (filePath: string) =>
62
+ new Set(this.dependencyEntrypoints.get(path.resolve(filePath)) ?? []),
63
+ setEntrypointDependencies: (entrypointPath: string, dependencies: string[]) =>
64
+ this.setEntrypointDependencies(entrypointPath, dependencies),
65
+ getDistDir: () => this.distDir,
66
+ getPlugins: () => this.plugins,
67
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
68
+ };
69
+
70
+ this.strategies = [new JsHmrStrategy(jsContext), new DefaultHmrStrategy()];
71
+ }
72
+
73
+ public registerStrategy(strategy: HmrStrategy): void {
74
+ this.strategies.push(strategy);
75
+ }
76
+
77
+ public setPlugins(plugins: EcoBuildPlugin[]): void {
78
+ this.plugins = [...plugins];
79
+ }
80
+
81
+ public setEnabled(enabled: boolean): void {
82
+ this.enabled = enabled;
83
+ }
84
+
85
+ public isEnabled(): boolean {
86
+ return this.enabled;
87
+ }
88
+
89
+ public registerSpecifierMap(map: Record<string, string>): void {
90
+ for (const [specifier, url] of Object.entries(map)) {
91
+ this.specifierMap.set(specifier, url);
92
+ }
93
+ }
94
+
95
+ public async buildRuntime(): Promise<void> {
96
+ const currentDir = path.dirname(fileURLToPath(import.meta.url));
97
+ const runtimeSource = path.resolve(currentDir, '../../hmr/client/hmr-runtime.ts');
98
+
99
+ const result = await defaultBuildAdapter.build({
100
+ entrypoints: [runtimeSource],
101
+ outdir: this.distDir,
102
+ naming: '_hmr_runtime.js',
103
+ minify: false,
104
+ ...defaultBuildAdapter.getTranspileOptions('hmr-runtime'),
105
+ plugins: this.plugins,
106
+ });
107
+
108
+ if (!result.success) {
109
+ appLogger.error('[HMR] Failed to build runtime script:', result.logs);
110
+ }
111
+ }
112
+
113
+ public getRuntimePath(): string {
114
+ return path.join(this.distDir, '_hmr_runtime.js');
115
+ }
116
+
117
+ public broadcast(event: ClientBridgeEvent) {
118
+ appLogger.debug(
119
+ `[HMR] Broadcasting ${event.type} event, path=${event.path || 'all'}, subscribers=${this.bridge.subscriberCount}`,
120
+ );
121
+ this.bridge.broadcast(event);
122
+ }
123
+
124
+ public async handleFileChange(filePath: string): Promise<void> {
125
+ const sorted = [...this.strategies].sort((a, b) => b.priority - a.priority);
126
+ const strategy = sorted.find((s) => {
127
+ try {
128
+ return s.matches(filePath);
129
+ } catch (err) {
130
+ appLogger.error(`[NodeHmrManager] Error checking match for ${s.constructor.name}:`, err as Error);
131
+ return false;
132
+ }
133
+ });
134
+
135
+ if (!strategy) {
136
+ appLogger.warn(`[HMR] No strategy found for ${filePath}`);
137
+ return;
138
+ }
139
+
140
+ appLogger.debug(`[NodeHmrManager] Selected strategy: ${strategy.constructor.name}`);
141
+
142
+ const action = await strategy.process(filePath);
143
+
144
+ if (action.type === 'broadcast') {
145
+ if (action.events) {
146
+ for (const event of action.events) {
147
+ this.broadcast(event);
148
+ }
149
+ }
150
+ }
151
+ }
152
+
153
+ public getOutputUrl(entrypointPath: string): string | undefined {
154
+ return this.watchedFiles.get(entrypointPath);
155
+ }
156
+
157
+ public getWatchedFiles(): Map<string, string> {
158
+ return this.watchedFiles;
159
+ }
160
+
161
+ public getSpecifierMap(): Map<string, string> {
162
+ return this.specifierMap;
163
+ }
164
+
165
+ public getDistDir(): string {
166
+ return this.distDir;
167
+ }
168
+
169
+ public getPlugins(): EcoBuildPlugin[] {
170
+ return this.plugins;
171
+ }
172
+
173
+ public getDefaultContext(): DefaultHmrContext {
174
+ return {
175
+ getWatchedFiles: () => this.watchedFiles,
176
+ getSpecifierMap: () => this.specifierMap,
177
+ getDistDir: () => this.distDir,
178
+ getPlugins: () => this.plugins,
179
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
180
+ getLayoutsDir: () => this.appConfig.absolutePaths.layoutsDir,
181
+ getPagesDir: () => this.appConfig.absolutePaths.pagesDir,
182
+ };
183
+ }
184
+
185
+ /**
186
+ * Updates Node HMR dependency indexes for selective invalidation.
187
+ *
188
+ * @remarks
189
+ * Graph data comes from Node/esbuild build metadata and does not affect Bun
190
+ * HMR behavior.
191
+ */
192
+ private setEntrypointDependencies(entrypointPath: string, dependencies: string[]): void {
193
+ const normalizedEntrypoint = path.resolve(entrypointPath);
194
+ const previousDependencies = this.entrypointDependencies.get(normalizedEntrypoint);
195
+
196
+ if (previousDependencies) {
197
+ for (const dependencyPath of previousDependencies) {
198
+ const entrypoints = this.dependencyEntrypoints.get(dependencyPath);
199
+ if (!entrypoints) {
200
+ continue;
201
+ }
202
+
203
+ entrypoints.delete(normalizedEntrypoint);
204
+ if (entrypoints.size === 0) {
205
+ this.dependencyEntrypoints.delete(dependencyPath);
206
+ }
207
+ }
208
+ }
209
+
210
+ const normalizedDependencies = new Set<string>([
211
+ normalizedEntrypoint,
212
+ ...dependencies.map((dependencyPath) => path.resolve(dependencyPath)),
213
+ ]);
214
+
215
+ this.entrypointDependencies.set(normalizedEntrypoint, normalizedDependencies);
216
+
217
+ for (const dependencyPath of normalizedDependencies) {
218
+ const entrypoints = this.dependencyEntrypoints.get(dependencyPath) ?? new Set<string>();
219
+ entrypoints.add(normalizedEntrypoint);
220
+ this.dependencyEntrypoints.set(dependencyPath, entrypoints);
221
+ }
222
+ }
223
+
224
+ public async registerEntrypoint(entrypointPath: string): Promise<string> {
225
+ if (this.watchedFiles.has(entrypointPath)) {
226
+ return this.watchedFiles.get(entrypointPath)!;
227
+ }
228
+
229
+ const srcDir = this.appConfig.absolutePaths.srcDir;
230
+ const relativePath = path.relative(srcDir, entrypointPath);
231
+ const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
232
+ const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
233
+
234
+ const urlPath = encodedPathJs.split(path.sep).join('/');
235
+ const outputUrl = `/${path.join(RESOLVED_ASSETS_DIR, '_hmr', urlPath)}`;
236
+ const outputPath = path.join(this.distDir, urlPath);
237
+
238
+ this.watchedFiles.set(entrypointPath, outputUrl);
239
+
240
+ await this.handleFileChange(entrypointPath);
241
+
242
+ if (!fileSystem.exists(outputPath)) {
243
+ const fallback = await defaultBuildAdapter.build({
244
+ entrypoints: [entrypointPath],
245
+ outdir: this.distDir,
246
+ naming: encodedPathJs,
247
+ minify: false,
248
+ external: Array.from(this.specifierMap.keys()),
249
+ ...defaultBuildAdapter.getTranspileOptions('hmr-entrypoint'),
250
+ plugins: this.plugins,
251
+ });
252
+
253
+ if (!fallback.success) {
254
+ appLogger.error(`[HMR] Fallback build failed for ${entrypointPath}:`, fallback.logs);
255
+ }
256
+ }
257
+
258
+ return outputUrl;
259
+ }
260
+
261
+ private encodeDynamicSegments(filepath: string): string {
262
+ return filepath.replace(/\[([^\]]+)\]/g, '_$1_');
263
+ }
264
+
265
+ public stop() {
266
+ for (const watcher of this.watchers.values()) {
267
+ watcher.close();
268
+ }
269
+ this.watchers.clear();
270
+ }
271
+ }