@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,114 @@
1
+ import type { Server } from 'bun';
2
+ import type {
3
+ ApiHandler,
4
+ ApiHandlerContext,
5
+ Middleware,
6
+ RouteSchema,
7
+ TypedGroupHandlerContext,
8
+ } from '../../public-types';
9
+
10
+ type BunSchemaHandlerContext<
11
+ TSchema extends RouteSchema | undefined,
12
+ WebSocketData,
13
+ TContext extends ApiHandlerContext<Request, Server<WebSocketData>>,
14
+ > = TSchema extends RouteSchema ? TypedGroupHandlerContext<TSchema, TContext> : TContext;
15
+
16
+ /**
17
+ * Helper function specifically for Bun to define an API handler with
18
+ * automatically inferred path type for BunRequest.
19
+ *
20
+ * This returns a self-describing route declaration (`path`, `method`, `handler`,
21
+ * optional `middleware`, optional `schema`) that can be registered on the app.
22
+ *
23
+ * @template TPath The literal string type of the path, inferred from the 'path' property.
24
+ * @param handler The API handler configuration object using BunRequest.
25
+ * @returns The same handler object, strongly typed for Bun.
26
+ */
27
+ export function defineApiHandler<
28
+ TSchema extends RouteSchema | undefined = undefined,
29
+ WebSocketData = undefined,
30
+ TContext extends ApiHandlerContext<Request, Server<WebSocketData>> = ApiHandlerContext<
31
+ Request,
32
+ Server<WebSocketData>
33
+ >,
34
+ >(
35
+ handler: Omit<ApiHandler<string, Request, Server<WebSocketData>>, 'handler' | 'middleware' | 'schema'> & {
36
+ handler: (context: BunSchemaHandlerContext<TSchema, WebSocketData, TContext>) => Promise<Response> | Response;
37
+ middleware?: Middleware<Request, Server<WebSocketData>, TContext>[];
38
+ schema?: TSchema;
39
+ },
40
+ ): ApiHandler<string, Request, Server<WebSocketData>> {
41
+ return handler as ApiHandler<string, Request, Server<WebSocketData>>;
42
+ }
43
+
44
+ /**
45
+ * Configuration for a group of related API handlers with shared prefix and middleware.
46
+ */
47
+ export interface GroupHandler<TPrefix extends string = string, WebSocketData = undefined> {
48
+ prefix: TPrefix;
49
+ middleware?: readonly Middleware<Request, Server<WebSocketData>, any>[];
50
+ routes: readonly ApiHandler<string, Request, Server<WebSocketData>>[];
51
+ }
52
+
53
+ type GroupDefineHandler<WebSocketData, TContext extends ApiHandlerContext<Request, Server<WebSocketData>>> = <
54
+ const TPath extends string,
55
+ TSchema extends RouteSchema | undefined = undefined,
56
+ >(
57
+ handler: Omit<ApiHandler<TPath, Request, Server<WebSocketData>>, 'handler' | 'middleware' | 'schema'> & {
58
+ path: TPath;
59
+ handler: (
60
+ context: TSchema extends RouteSchema ? TypedGroupHandlerContext<TSchema, TContext> : TContext,
61
+ ) => Promise<Response> | Response;
62
+ middleware?: Middleware<Request, Server<WebSocketData>, TContext>[];
63
+ schema?: TSchema;
64
+ },
65
+ ) => ApiHandler<TPath, Request, Server<WebSocketData>>;
66
+
67
+ /**
68
+ * Helper function to define a group of API handlers with shared prefix and middleware.
69
+ * The group middleware context is inferred and passed to all route handlers.
70
+ *
71
+ * @example
72
+ * ```typescript
73
+ * const adminGroup = defineGroupHandler({
74
+ * prefix: '/admin',
75
+ * middleware: [authMiddleware],
76
+ * routes: (define) => [
77
+ * define({ path: '/', method: 'GET', handler: (ctx) => ctx.render(List, {}) }),
78
+ * define({ path: '/posts/:id', method: 'GET', handler: (ctx) => {
79
+ * // ctx.session is typed from authMiddleware!
80
+ * // ctx.params.id is typed from path!
81
+ * return ctx.render(Post, { id: ctx.params.id });
82
+ * }}),
83
+ * ],
84
+ * });
85
+ *
86
+ * app.group(adminGroup);
87
+ * ```
88
+ */
89
+ export function defineGroupHandler<
90
+ TPrefix extends string,
91
+ WebSocketData = undefined,
92
+ TMiddleware extends readonly Middleware<Request, Server<WebSocketData>, any>[] = [],
93
+ TContext extends ApiHandlerContext<Request, Server<WebSocketData>> = TMiddleware extends readonly Middleware<
94
+ Request,
95
+ Server<WebSocketData>,
96
+ infer TGroupContext
97
+ >[]
98
+ ? TGroupContext
99
+ : ApiHandlerContext<Request, Server<WebSocketData>>,
100
+ >(config: {
101
+ prefix: TPrefix;
102
+ middleware?: TMiddleware;
103
+ routes: (
104
+ define: GroupDefineHandler<WebSocketData, TContext>,
105
+ ) => readonly ApiHandler<string, Request, Server<WebSocketData>>[];
106
+ }): GroupHandler<TPrefix, WebSocketData> {
107
+ const define = ((handler) => handler) as GroupDefineHandler<WebSocketData, TContext>;
108
+
109
+ return {
110
+ prefix: config.prefix,
111
+ middleware: config.middleware,
112
+ routes: config.routes(define),
113
+ };
114
+ }
@@ -0,0 +1,84 @@
1
+ import type { WebSocketHandler } from 'bun';
2
+ import type { DefaultHmrContext, EcoPagesAppConfig, IHmrManager } from '../../internal-types';
3
+ import type { EcoBuildPlugin } from '../../build/build-types.js';
4
+ import type { HmrStrategy } from '../../hmr/hmr-strategy';
5
+ import type { ClientBridge } from './client-bridge';
6
+ import type { ClientBridgeEvent } from '../../public-types';
7
+ type BunSocketHandler = WebSocketHandler<unknown>;
8
+ export interface HmrManagerParams {
9
+ appConfig: EcoPagesAppConfig;
10
+ bridge: ClientBridge;
11
+ }
12
+ export declare class HmrManager implements IHmrManager {
13
+ readonly appConfig: EcoPagesAppConfig;
14
+ private readonly bridge;
15
+ /** Keep track of watchers */
16
+ private watchers;
17
+ /** entrypoint -> output path */
18
+ private watchedFiles;
19
+ /** bare specifier -> runtime URL (e.g., 'react' -> '/assets/vendors/react.js') */
20
+ private specifierMap;
21
+ private distDir;
22
+ private plugins;
23
+ private enabled;
24
+ private strategies;
25
+ private wsHandler;
26
+ constructor({ appConfig, bridge }: HmrManagerParams);
27
+ /**
28
+ * Ensures the HMR output directory exists.
29
+ *
30
+ * This must not remove the directory because multiple app processes
31
+ * can share the same dist path during e2e runs.
32
+ */
33
+ private cleanDistDir;
34
+ /**
35
+ * Initializes core HMR strategies.
36
+ * Strategies are evaluated in priority order (highest first).
37
+ */
38
+ private initializeStrategies;
39
+ /**
40
+ * Registers a custom HMR strategy.
41
+ * Used by integrations to provide framework-specific HMR handling.
42
+ * @param strategy - The HMR strategy to register
43
+ */
44
+ registerStrategy(strategy: HmrStrategy): void;
45
+ setPlugins(plugins: EcoBuildPlugin[]): void;
46
+ setEnabled(enabled: boolean): void;
47
+ isEnabled(): boolean;
48
+ /**
49
+ * Registers a mapping from bare specifiers to vendor URLs.
50
+ * Used by integrations to provide their module resolution mappings.
51
+ * @param map - Object mapping bare specifiers to vendor URLs
52
+ */
53
+ registerSpecifierMap(map: Record<string, string>): void;
54
+ getWebSocketHandler(): BunSocketHandler;
55
+ /**
56
+ * Builds the client-side HMR runtime script.
57
+ */
58
+ buildRuntime(): Promise<void>;
59
+ getRuntimePath(): string;
60
+ broadcast(event: ClientBridgeEvent): void;
61
+ /**
62
+ * Handles file changes using registered HMR strategies.
63
+ * Strategies are evaluated in priority order until one matches.
64
+ * @param filePath - Absolute path to the changed file
65
+ */
66
+ handleFileChange(filePath: string): Promise<void>;
67
+ /**
68
+ * Registers a client entrypoint to be built and watched by Bun.
69
+ */
70
+ getOutputUrl(entrypointPath: string): string | undefined;
71
+ getWatchedFiles(): Map<string, string>;
72
+ getSpecifierMap(): Map<string, string>;
73
+ getDistDir(): string;
74
+ getPlugins(): EcoBuildPlugin[];
75
+ getDefaultContext(): DefaultHmrContext;
76
+ registerEntrypoint(entrypointPath: string): Promise<string>;
77
+ /**
78
+ * Encodes dynamic route segments (brackets) in file paths.
79
+ * Converts `[slug]` to `_slug_` to avoid filesystem/URL issues.
80
+ */
81
+ private encodeDynamicSegments;
82
+ stop(): void;
83
+ }
84
+ export {};
@@ -0,0 +1,227 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { RESOLVED_ASSETS_DIR } from "../../constants";
4
+ import { defaultBuildAdapter } from "../../build/build-adapter.js";
5
+ import { fileSystem } from "@ecopages/file-system";
6
+ import { DefaultHmrStrategy } from "../../hmr/strategies/default-hmr-strategy";
7
+ import { JsHmrStrategy } from "../../hmr/strategies/js-hmr-strategy";
8
+ import { appLogger } from "../../global/app-logger";
9
+ class HmrManager {
10
+ appConfig;
11
+ bridge;
12
+ /** Keep track of watchers */
13
+ watchers = /* @__PURE__ */ new Map();
14
+ /** entrypoint -> output path */
15
+ watchedFiles = /* @__PURE__ */ new Map();
16
+ /** bare specifier -> runtime URL (e.g., 'react' -> '/assets/vendors/react.js') */
17
+ specifierMap = /* @__PURE__ */ new Map();
18
+ distDir;
19
+ plugins = [];
20
+ enabled = true;
21
+ strategies = [];
22
+ wsHandler;
23
+ constructor({ appConfig, bridge }) {
24
+ this.appConfig = appConfig;
25
+ this.bridge = bridge;
26
+ this.distDir = path.join(this.appConfig.absolutePaths.distDir, RESOLVED_ASSETS_DIR, "_hmr");
27
+ this.cleanDistDir();
28
+ this.initializeStrategies();
29
+ }
30
+ /**
31
+ * Ensures the HMR output directory exists.
32
+ *
33
+ * This must not remove the directory because multiple app processes
34
+ * can share the same dist path during e2e runs.
35
+ */
36
+ cleanDistDir() {
37
+ fileSystem.ensureDir(this.distDir);
38
+ }
39
+ /**
40
+ * Initializes core HMR strategies.
41
+ * Strategies are evaluated in priority order (highest first).
42
+ */
43
+ initializeStrategies() {
44
+ const jsContext = {
45
+ getWatchedFiles: () => this.watchedFiles,
46
+ getSpecifierMap: () => this.specifierMap,
47
+ getDistDir: () => this.distDir,
48
+ getPlugins: () => this.plugins,
49
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir
50
+ };
51
+ this.strategies = [new JsHmrStrategy(jsContext), new DefaultHmrStrategy()];
52
+ }
53
+ /**
54
+ * Registers a custom HMR strategy.
55
+ * Used by integrations to provide framework-specific HMR handling.
56
+ * @param strategy - The HMR strategy to register
57
+ */
58
+ registerStrategy(strategy) {
59
+ this.strategies.push(strategy);
60
+ }
61
+ setPlugins(plugins) {
62
+ this.plugins = [...plugins];
63
+ }
64
+ setEnabled(enabled) {
65
+ this.enabled = enabled;
66
+ }
67
+ isEnabled() {
68
+ return this.enabled;
69
+ }
70
+ /**
71
+ * Registers a mapping from bare specifiers to vendor URLs.
72
+ * Used by integrations to provide their module resolution mappings.
73
+ * @param map - Object mapping bare specifiers to vendor URLs
74
+ */
75
+ registerSpecifierMap(map) {
76
+ for (const [specifier, url] of Object.entries(map)) {
77
+ this.specifierMap.set(specifier, url);
78
+ }
79
+ }
80
+ getWebSocketHandler() {
81
+ const open = (ws) => {
82
+ this.bridge.subscribe(ws);
83
+ appLogger.debug(`[HmrManager] Connection opened. Subscribers: ${this.bridge.subscriberCount}`);
84
+ };
85
+ const close = (ws) => {
86
+ this.bridge.unsubscribe(ws);
87
+ appLogger.debug(`[HmrManager] Connection closed. Subscribers: ${this.bridge.subscriberCount}`);
88
+ };
89
+ this.wsHandler = { open, close };
90
+ return {
91
+ open: this.wsHandler.open,
92
+ close: this.wsHandler.close,
93
+ message: (_ws, message) => {
94
+ appLogger.debug("[HMR] Received message from client:", message);
95
+ }
96
+ };
97
+ }
98
+ /**
99
+ * Builds the client-side HMR runtime script.
100
+ */
101
+ async buildRuntime() {
102
+ const runtimeSource = path.resolve(import.meta.dirname, "../../hmr/client/hmr-runtime.js");
103
+ const result = await defaultBuildAdapter.build({
104
+ entrypoints: [runtimeSource],
105
+ outdir: this.distDir,
106
+ naming: "_hmr_runtime.js",
107
+ minify: false,
108
+ ...defaultBuildAdapter.getTranspileOptions("hmr-runtime"),
109
+ plugins: this.plugins
110
+ });
111
+ if (!result.success) {
112
+ appLogger.error("[HMR] Failed to build runtime script:", result.logs);
113
+ }
114
+ }
115
+ getRuntimePath() {
116
+ return path.join(this.distDir, "_hmr_runtime.js");
117
+ }
118
+ broadcast(event) {
119
+ appLogger.debug(
120
+ `[HMR] Broadcasting ${event.type} event, path=${event.path || "all"}, subscribers=${this.bridge.subscriberCount}`
121
+ );
122
+ this.bridge.broadcast(event);
123
+ }
124
+ /**
125
+ * Handles file changes using registered HMR strategies.
126
+ * Strategies are evaluated in priority order until one matches.
127
+ * @param filePath - Absolute path to the changed file
128
+ */
129
+ async handleFileChange(filePath) {
130
+ const sorted = [...this.strategies].sort((a, b) => b.priority - a.priority);
131
+ const strategy = sorted.find((s) => {
132
+ try {
133
+ return s.matches(filePath);
134
+ } catch (err) {
135
+ appLogger.error(`[HmrManager] Error checking match for ${s.constructor.name}:`, err);
136
+ return false;
137
+ }
138
+ });
139
+ if (!strategy) {
140
+ appLogger.warn(`[HMR] No strategy found for ${filePath}`);
141
+ return;
142
+ }
143
+ appLogger.debug(`[HmrManager] Selected strategy: ${strategy.constructor.name}`);
144
+ const action = await strategy.process(filePath);
145
+ if (action.type === "broadcast") {
146
+ if (action.events) {
147
+ for (const event of action.events) {
148
+ this.broadcast(event);
149
+ }
150
+ }
151
+ }
152
+ }
153
+ /**
154
+ * Registers a client entrypoint to be built and watched by Bun.
155
+ */
156
+ getOutputUrl(entrypointPath) {
157
+ return this.watchedFiles.get(entrypointPath);
158
+ }
159
+ getWatchedFiles() {
160
+ return this.watchedFiles;
161
+ }
162
+ getSpecifierMap() {
163
+ return this.specifierMap;
164
+ }
165
+ getDistDir() {
166
+ return this.distDir;
167
+ }
168
+ getPlugins() {
169
+ return this.plugins;
170
+ }
171
+ getDefaultContext() {
172
+ return {
173
+ getWatchedFiles: () => this.watchedFiles,
174
+ getSpecifierMap: () => this.specifierMap,
175
+ getDistDir: () => this.distDir,
176
+ getPlugins: () => this.plugins,
177
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
178
+ getLayoutsDir: () => this.appConfig.absolutePaths.layoutsDir,
179
+ getPagesDir: () => this.appConfig.absolutePaths.pagesDir
180
+ };
181
+ }
182
+ async registerEntrypoint(entrypointPath) {
183
+ if (this.watchedFiles.has(entrypointPath)) {
184
+ return this.watchedFiles.get(entrypointPath);
185
+ }
186
+ const srcDir = this.appConfig.absolutePaths.srcDir;
187
+ const relativePath = path.relative(srcDir, entrypointPath);
188
+ const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, ".js");
189
+ const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
190
+ const urlPath = encodedPathJs.split(path.sep).join("/");
191
+ const outputUrl = `/${path.join(RESOLVED_ASSETS_DIR, "_hmr", urlPath)}`;
192
+ const outputPath = path.join(this.distDir, urlPath);
193
+ this.watchedFiles.set(entrypointPath, outputUrl);
194
+ await this.handleFileChange(entrypointPath);
195
+ if (!fileSystem.exists(outputPath)) {
196
+ const fallback = await defaultBuildAdapter.build({
197
+ entrypoints: [entrypointPath],
198
+ outdir: this.distDir,
199
+ naming: encodedPathJs,
200
+ minify: false,
201
+ external: Array.from(this.specifierMap.keys()),
202
+ ...defaultBuildAdapter.getTranspileOptions("hmr-entrypoint"),
203
+ plugins: this.plugins
204
+ });
205
+ if (!fallback.success) {
206
+ appLogger.error(`[HMR] Fallback build failed for ${entrypointPath}:`, fallback.logs);
207
+ }
208
+ }
209
+ return outputUrl;
210
+ }
211
+ /**
212
+ * Encodes dynamic route segments (brackets) in file paths.
213
+ * Converts `[slug]` to `_slug_` to avoid filesystem/URL issues.
214
+ */
215
+ encodeDynamicSegments(filepath) {
216
+ return filepath.replace(/\[([^\]]+)\]/g, "_$1_");
217
+ }
218
+ stop() {
219
+ for (const watcher of this.watchers.values()) {
220
+ watcher.close();
221
+ }
222
+ this.watchers.clear();
223
+ }
224
+ }
225
+ export {
226
+ HmrManager
227
+ };
@@ -0,0 +1,281 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import type { ServerWebSocket, WebSocketHandler } from 'bun';
4
+ import { RESOLVED_ASSETS_DIR } from '../../constants';
5
+ import { defaultBuildAdapter } from '../../build/build-adapter.ts';
6
+ import type { DefaultHmrContext, EcoPagesAppConfig, IHmrManager } from '../../internal-types';
7
+ import type { EcoBuildPlugin } from '../../build/build-types.ts';
8
+ import { fileSystem } from '@ecopages/file-system';
9
+ import type { HmrStrategy } from '../../hmr/hmr-strategy';
10
+ import { DefaultHmrStrategy } from '../../hmr/strategies/default-hmr-strategy';
11
+ import { JsHmrStrategy } from '../../hmr/strategies/js-hmr-strategy';
12
+ import { appLogger } from '../../global/app-logger';
13
+ import type { ClientBridge } from './client-bridge';
14
+ import type { ClientBridgeEvent } from '../../public-types';
15
+
16
+ type BunSocket = ServerWebSocket<unknown>;
17
+ type BunSocketHandler = WebSocketHandler<unknown>;
18
+
19
+ export interface HmrManagerParams {
20
+ appConfig: EcoPagesAppConfig;
21
+ bridge: ClientBridge;
22
+ }
23
+
24
+ export class HmrManager implements IHmrManager {
25
+ public readonly appConfig: EcoPagesAppConfig;
26
+ private readonly bridge: ClientBridge;
27
+ /** Keep track of watchers */
28
+ private watchers = new Map<string, fs.FSWatcher>();
29
+ /** entrypoint -> output path */
30
+ private watchedFiles = new Map<string, string>();
31
+ /** bare specifier -> runtime URL (e.g., 'react' -> '/assets/vendors/react.js') */
32
+ private specifierMap = new Map<string, string>();
33
+ private distDir: string;
34
+ private plugins: EcoBuildPlugin[] = [];
35
+ private enabled = true;
36
+ private strategies: HmrStrategy[] = [];
37
+ private wsHandler!: {
38
+ open: (ws: BunSocket) => void;
39
+ close: (ws: BunSocket) => void;
40
+ };
41
+
42
+ constructor({ appConfig, bridge }: HmrManagerParams) {
43
+ this.appConfig = appConfig;
44
+ this.bridge = bridge;
45
+ this.distDir = path.join(this.appConfig.absolutePaths.distDir, RESOLVED_ASSETS_DIR, '_hmr');
46
+ this.cleanDistDir();
47
+ this.initializeStrategies();
48
+ }
49
+
50
+ /**
51
+ * Ensures the HMR output directory exists.
52
+ *
53
+ * This must not remove the directory because multiple app processes
54
+ * can share the same dist path during e2e runs.
55
+ */
56
+ private cleanDistDir(): void {
57
+ fileSystem.ensureDir(this.distDir);
58
+ }
59
+
60
+ /**
61
+ * Initializes core HMR strategies.
62
+ * Strategies are evaluated in priority order (highest first).
63
+ */
64
+ private initializeStrategies(): void {
65
+ const jsContext = {
66
+ getWatchedFiles: () => this.watchedFiles,
67
+ getSpecifierMap: () => this.specifierMap,
68
+ getDistDir: () => this.distDir,
69
+ getPlugins: () => this.plugins,
70
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
71
+ };
72
+
73
+ this.strategies = [new JsHmrStrategy(jsContext), new DefaultHmrStrategy()];
74
+ }
75
+
76
+ /**
77
+ * Registers a custom HMR strategy.
78
+ * Used by integrations to provide framework-specific HMR handling.
79
+ * @param strategy - The HMR strategy to register
80
+ */
81
+ public registerStrategy(strategy: HmrStrategy): void {
82
+ this.strategies.push(strategy);
83
+ }
84
+
85
+ public setPlugins(plugins: EcoBuildPlugin[]): void {
86
+ this.plugins = [...plugins];
87
+ }
88
+
89
+ public setEnabled(enabled: boolean): void {
90
+ this.enabled = enabled;
91
+ }
92
+
93
+ public isEnabled(): boolean {
94
+ return this.enabled;
95
+ }
96
+
97
+ /**
98
+ * Registers a mapping from bare specifiers to vendor URLs.
99
+ * Used by integrations to provide their module resolution mappings.
100
+ * @param map - Object mapping bare specifiers to vendor URLs
101
+ */
102
+ public registerSpecifierMap(map: Record<string, string>): void {
103
+ for (const [specifier, url] of Object.entries(map)) {
104
+ this.specifierMap.set(specifier, url);
105
+ }
106
+ }
107
+
108
+ public getWebSocketHandler(): BunSocketHandler {
109
+ const open = (ws: BunSocket) => {
110
+ this.bridge.subscribe(ws);
111
+ appLogger.debug(`[HmrManager] Connection opened. Subscribers: ${this.bridge.subscriberCount}`);
112
+ };
113
+
114
+ const close = (ws: BunSocket) => {
115
+ this.bridge.unsubscribe(ws);
116
+ appLogger.debug(`[HmrManager] Connection closed. Subscribers: ${this.bridge.subscriberCount}`);
117
+ };
118
+
119
+ this.wsHandler = { open, close };
120
+
121
+ return {
122
+ open: this.wsHandler.open,
123
+ close: this.wsHandler.close,
124
+ message: (_ws, message) => {
125
+ appLogger.debug('[HMR] Received message from client:', message);
126
+ },
127
+ };
128
+ }
129
+
130
+ /**
131
+ * Builds the client-side HMR runtime script.
132
+ */
133
+ public async buildRuntime(): Promise<void> {
134
+ const runtimeSource = path.resolve(import.meta.dirname, '../../hmr/client/hmr-runtime.ts');
135
+
136
+ const result = await defaultBuildAdapter.build({
137
+ entrypoints: [runtimeSource],
138
+ outdir: this.distDir,
139
+ naming: '_hmr_runtime.js',
140
+ minify: false,
141
+ ...defaultBuildAdapter.getTranspileOptions('hmr-runtime'),
142
+ plugins: this.plugins,
143
+ });
144
+
145
+ if (!result.success) {
146
+ appLogger.error('[HMR] Failed to build runtime script:', result.logs);
147
+ }
148
+ }
149
+
150
+ public getRuntimePath(): string {
151
+ return path.join(this.distDir, '_hmr_runtime.js');
152
+ }
153
+
154
+ public broadcast(event: ClientBridgeEvent) {
155
+ appLogger.debug(
156
+ `[HMR] Broadcasting ${event.type} event, path=${event.path || 'all'}, subscribers=${this.bridge.subscriberCount}`,
157
+ );
158
+ this.bridge.broadcast(event);
159
+ }
160
+
161
+ /**
162
+ * Handles file changes using registered HMR strategies.
163
+ * Strategies are evaluated in priority order until one matches.
164
+ * @param filePath - Absolute path to the changed file
165
+ */
166
+ public async handleFileChange(filePath: string): Promise<void> {
167
+ const sorted = [...this.strategies].sort((a, b) => b.priority - a.priority);
168
+ const strategy = sorted.find((s) => {
169
+ try {
170
+ return s.matches(filePath);
171
+ } catch (err) {
172
+ appLogger.error(`[HmrManager] Error checking match for ${s.constructor.name}:`, err as Error);
173
+ return false;
174
+ }
175
+ });
176
+
177
+ if (!strategy) {
178
+ appLogger.warn(`[HMR] No strategy found for ${filePath}`);
179
+ return;
180
+ }
181
+
182
+ appLogger.debug(`[HmrManager] Selected strategy: ${strategy.constructor.name}`);
183
+
184
+ const action = await strategy.process(filePath);
185
+
186
+ if (action.type === 'broadcast') {
187
+ if (action.events) {
188
+ for (const event of action.events) {
189
+ this.broadcast(event);
190
+ }
191
+ }
192
+ }
193
+ }
194
+
195
+ /**
196
+ * Registers a client entrypoint to be built and watched by Bun.
197
+ */
198
+ public getOutputUrl(entrypointPath: string): string | undefined {
199
+ return this.watchedFiles.get(entrypointPath);
200
+ }
201
+
202
+ public getWatchedFiles(): Map<string, string> {
203
+ return this.watchedFiles;
204
+ }
205
+
206
+ public getSpecifierMap(): Map<string, string> {
207
+ return this.specifierMap;
208
+ }
209
+
210
+ public getDistDir(): string {
211
+ return this.distDir;
212
+ }
213
+
214
+ public getPlugins(): EcoBuildPlugin[] {
215
+ return this.plugins;
216
+ }
217
+
218
+ public getDefaultContext(): DefaultHmrContext {
219
+ return {
220
+ getWatchedFiles: () => this.watchedFiles,
221
+ getSpecifierMap: () => this.specifierMap,
222
+ getDistDir: () => this.distDir,
223
+ getPlugins: () => this.plugins,
224
+ getSrcDir: () => this.appConfig.absolutePaths.srcDir,
225
+ getLayoutsDir: () => this.appConfig.absolutePaths.layoutsDir,
226
+ getPagesDir: () => this.appConfig.absolutePaths.pagesDir,
227
+ };
228
+ }
229
+
230
+ public async registerEntrypoint(entrypointPath: string): Promise<string> {
231
+ if (this.watchedFiles.has(entrypointPath)) {
232
+ return this.watchedFiles.get(entrypointPath)!;
233
+ }
234
+
235
+ const srcDir = this.appConfig.absolutePaths.srcDir;
236
+ const relativePath = path.relative(srcDir, entrypointPath);
237
+ const relativePathJs = relativePath.replace(/\.(tsx?|jsx?|mdx?)$/, '.js');
238
+ const encodedPathJs = this.encodeDynamicSegments(relativePathJs);
239
+
240
+ const urlPath = encodedPathJs.split(path.sep).join('/');
241
+ const outputUrl = `/${path.join(RESOLVED_ASSETS_DIR, '_hmr', urlPath)}`;
242
+ const outputPath = path.join(this.distDir, urlPath);
243
+
244
+ this.watchedFiles.set(entrypointPath, outputUrl);
245
+
246
+ await this.handleFileChange(entrypointPath);
247
+
248
+ if (!fileSystem.exists(outputPath)) {
249
+ const fallback = await defaultBuildAdapter.build({
250
+ entrypoints: [entrypointPath],
251
+ outdir: this.distDir,
252
+ naming: encodedPathJs,
253
+ minify: false,
254
+ external: Array.from(this.specifierMap.keys()),
255
+ ...defaultBuildAdapter.getTranspileOptions('hmr-entrypoint'),
256
+ plugins: this.plugins,
257
+ });
258
+
259
+ if (!fallback.success) {
260
+ appLogger.error(`[HMR] Fallback build failed for ${entrypointPath}:`, fallback.logs);
261
+ }
262
+ }
263
+
264
+ return outputUrl;
265
+ }
266
+
267
+ /**
268
+ * Encodes dynamic route segments (brackets) in file paths.
269
+ * Converts `[slug]` to `_slug_` to avoid filesystem/URL issues.
270
+ */
271
+ private encodeDynamicSegments(filepath: string): string {
272
+ return filepath.replace(/\[([^\]]+)\]/g, '_$1_');
273
+ }
274
+
275
+ public stop() {
276
+ for (const watcher of this.watchers.values()) {
277
+ watcher.close();
278
+ }
279
+ this.watchers.clear();
280
+ }
281
+ }