@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,306 @@
1
+ import path from 'node:path';
2
+ import chokidar, { type FSWatcher } from 'chokidar';
3
+ import { fileSystem } from '@ecopages/file-system';
4
+ import { appLogger } from '../global/app-logger.ts';
5
+ import type { EcoPagesAppConfig, IHmrManager, IClientBridge } from '../internal-types.ts';
6
+ import type { ProcessorWatchContext } from '../plugins/processor.ts';
7
+
8
+ /**
9
+ * Configuration options for the ProjectWatcher
10
+ * @interface ProjectWatcherConfig
11
+ * @property {EcoPagesAppConfig} config - The application configuration
12
+ * @property {() => void} refreshRouterRoutesCallback - Callback to refresh router routes
13
+ * @property {IHmrManager} hmrManager - The HMR manager instance
14
+ * @property {ClientBridge} bridge - The client bridge instance
15
+ */
16
+ export interface ProjectWatcherConfig {
17
+ config: EcoPagesAppConfig;
18
+ refreshRouterRoutesCallback: () => void;
19
+ hmrManager: IHmrManager;
20
+ bridge: IClientBridge;
21
+ }
22
+
23
+ /**
24
+ * ProjectWatcher handles file system changes for hot module replacement (HMR).
25
+ * It uses chokidar to watch for file changes and triggers appropriate actions:
26
+ * - Uncaches modules when files change
27
+ * - Refreshes router routes for page files
28
+ * - Triggers HMR server reload
29
+ * - Handles processor-specific file changes
30
+ *
31
+ * The watcher uses chokidar's built-in debouncing through `awaitWriteFinish`
32
+ * to handle rapid file changes efficiently:
33
+ * - stabilityThreshold: 50ms - Time to wait for writes to stabilize
34
+ * - pollInterval: 50ms - Interval to poll for file changes
35
+ *
36
+ * @class ProjectWatcher
37
+ */
38
+ export class ProjectWatcher {
39
+ /**
40
+ * Duplicate identical watcher events within this window are ignored.
41
+ *
42
+ * Some editors or save pipelines emit two near-identical filesystem change
43
+ * notifications for the same file. Ecopages should treat those as one logical
44
+ * update so HMR and route refresh work are not repeated unnecessarily.
45
+ */
46
+ private static readonly duplicateChangeWindowMs = 150;
47
+ private appConfig: EcoPagesAppConfig;
48
+ private refreshRouterRoutesCallback: () => void;
49
+ private hmrManager: IHmrManager;
50
+ private bridge: IClientBridge;
51
+ private watcher: FSWatcher | null = null;
52
+ private lastHandledChange = new Map<string, number>();
53
+
54
+ constructor({ config, refreshRouterRoutesCallback, hmrManager, bridge }: ProjectWatcherConfig) {
55
+ this.appConfig = config;
56
+ this.refreshRouterRoutesCallback = refreshRouterRoutesCallback;
57
+ this.hmrManager = hmrManager;
58
+ this.bridge = bridge;
59
+ this.triggerRouterRefresh = this.triggerRouterRefresh.bind(this);
60
+ this.handleError = this.handleError.bind(this);
61
+ this.handleFileChange = this.handleFileChange.bind(this);
62
+ }
63
+
64
+ /**
65
+ * Uncaches modules in the source directory to ensure fresh imports.
66
+ * This is necessary for hot module replacement to work correctly.
67
+ * @private
68
+ */
69
+ private uncacheModules(): void {
70
+ if (typeof require === 'undefined') return;
71
+
72
+ const { srcDir, rootDir } = this.appConfig;
73
+ const regex = new RegExp(`${rootDir}/${srcDir}/.*`);
74
+
75
+ for (const key in require.cache) {
76
+ if (regex.test(key)) {
77
+ delete require.cache[key];
78
+ }
79
+ }
80
+ }
81
+
82
+ /**
83
+ * Handles public directory file changes by copying only the changed file.
84
+ * @param filePath - Absolute path of the changed file
85
+ */
86
+ private async handlePublicDirFileChange(filePath: string): Promise<void> {
87
+ try {
88
+ const relativePath = path.relative(this.appConfig.absolutePaths.publicDir, filePath);
89
+ const destPath = path.join(this.appConfig.absolutePaths.distDir, relativePath);
90
+
91
+ if (fileSystem.exists(filePath)) {
92
+ const destDir = path.dirname(destPath);
93
+ fileSystem.ensureDir(destDir);
94
+ await fileSystem.copyFileAsync(filePath, destPath);
95
+ }
96
+
97
+ this.bridge.reload();
98
+ } catch (error) {
99
+ appLogger.error(`Failed to copy public file: ${error instanceof Error ? error.message : String(error)}`);
100
+ this.bridge.reload();
101
+ }
102
+ }
103
+
104
+ /**
105
+ * Handles file changes by uncaching modules, refreshing routes, and delegating appropriately.
106
+ * Follows 4-rule priority:
107
+ * 0. Public directory match? → copy file and reload
108
+ * 1. additionalWatchPaths match? → reload
109
+ * 2. Processor extension match? → processor handles (skip HMR)
110
+ * 3. Otherwise → HMR strategies
111
+ *
112
+ * Duplicate identical watcher events for the same file are coalesced within a
113
+ * short window before any of the priority rules run.
114
+ * @param rawPath - Path of the changed file
115
+ */
116
+ private async handleFileChange(rawPath: string): Promise<void> {
117
+ const filePath = path.resolve(rawPath);
118
+ const now = Date.now();
119
+ const lastHandledAt = this.lastHandledChange.get(filePath);
120
+ if (lastHandledAt !== undefined && now - lastHandledAt < ProjectWatcher.duplicateChangeWindowMs) {
121
+ return;
122
+ }
123
+ this.lastHandledChange.set(filePath, now);
124
+
125
+ try {
126
+ if (this.isPublicDirFile(filePath)) {
127
+ await this.handlePublicDirFileChange(filePath);
128
+ return;
129
+ }
130
+
131
+ this.uncacheModules();
132
+ const isPageFile = filePath.startsWith(this.appConfig.absolutePaths.pagesDir);
133
+
134
+ if (isPageFile) {
135
+ this.refreshRouterRoutesCallback();
136
+ }
137
+
138
+ if (this.matchesAdditionalWatchPaths(filePath)) {
139
+ this.bridge.reload();
140
+ return;
141
+ }
142
+
143
+ if (this.isHandledByProcessor(filePath)) {
144
+ return;
145
+ }
146
+
147
+ await this.hmrManager.handleFileChange(filePath);
148
+ } catch (error) {
149
+ if (error instanceof Error) {
150
+ this.bridge.error(error.message);
151
+ this.handleError(error);
152
+ }
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Checks if a file is in the public directory.
158
+ */
159
+ private isPublicDirFile(filePath: string): boolean {
160
+ return filePath.startsWith(this.appConfig.absolutePaths.publicDir);
161
+ }
162
+
163
+ /**
164
+ * Checks if file path matches any additionalWatchPaths patterns.
165
+ */
166
+ private matchesAdditionalWatchPaths(filePath: string): boolean {
167
+ const patterns = this.appConfig.additionalWatchPaths;
168
+ if (!patterns.length) return false;
169
+
170
+ for (const pattern of patterns) {
171
+ if (pattern.includes('*')) {
172
+ const ext = pattern.replace(/\*\*?\/\*/, '');
173
+ if (filePath.endsWith(ext)) return true;
174
+ } else {
175
+ if (filePath.endsWith(pattern) || filePath === path.resolve(pattern)) return true;
176
+ }
177
+ }
178
+ return false;
179
+ }
180
+
181
+ /**
182
+ * Checks if a file is handled by a processor.
183
+ * Processors that declare extensions own those file types.
184
+ */
185
+ private isHandledByProcessor(filePath: string): boolean {
186
+ for (const processor of this.appConfig.processors.values()) {
187
+ const watchConfig = processor.getWatchConfig();
188
+ if (!watchConfig) continue;
189
+
190
+ const { extensions = [] } = watchConfig;
191
+ if (extensions.length && extensions.some((ext) => filePath.endsWith(ext))) {
192
+ return true;
193
+ }
194
+ }
195
+ return false;
196
+ }
197
+
198
+ /**
199
+ * Triggers router refresh for page directory changes.
200
+ * This ensures the router is updated when pages are added or removed.
201
+ *
202
+ * @param {string} path - Path of the changed directory
203
+ */
204
+ triggerRouterRefresh(path: string) {
205
+ const isPageDir = path.startsWith(this.appConfig.absolutePaths.pagesDir);
206
+ if (isPageDir) {
207
+ this.refreshRouterRoutesCallback();
208
+ }
209
+ }
210
+
211
+ /**
212
+ * Handles and logs errors that occur during file watching.
213
+ *
214
+ * @param {unknown} error - The error to handle
215
+ */
216
+ handleError(error: unknown) {
217
+ if (error instanceof Error) {
218
+ this.hmrManager.broadcast({ type: 'error', message: error.message });
219
+ }
220
+ appLogger.error(`Watcher error: ${error}`);
221
+ }
222
+
223
+ /**
224
+ * Processes file changes for specific file extensions.
225
+ * Used by processors to handle their specific file types.
226
+ *
227
+ * @private
228
+ * @param {string} path - Path of the changed file
229
+ * @param {string[]} extensions - File extensions to process
230
+ * @param {(ctx: ProcessorWatchContext) => void} handler - Handler function for the file change
231
+ */
232
+ private shouldProcess(path: string, extensions: string[], handler: (ctx: ProcessorWatchContext) => void) {
233
+ if (!extensions.length || extensions.some((ext) => path.endsWith(ext))) {
234
+ handler({ path, bridge: this.bridge });
235
+ }
236
+ }
237
+
238
+ /**
239
+ * Creates and configures the file system watcher.
240
+ * This sets up:
241
+ * 1. Processor-specific file watching
242
+ * 2. Page file watching
243
+ * 3. Directory watching
244
+ * 4. Error handling
245
+ *
246
+ * Uses chokidar's built-in debouncing through `awaitWriteFinish` to handle
247
+ * rapid file changes efficiently.
248
+ */
249
+ public async createWatcherSubscription() {
250
+ if (!this.watcher) {
251
+ const processorPaths: string[] = [];
252
+ for (const processor of this.appConfig.processors.values()) {
253
+ const watchConfig = processor.getWatchConfig();
254
+ if (!watchConfig) continue;
255
+ processorPaths.push(...watchConfig.paths);
256
+ }
257
+
258
+ if (fileSystem.exists(this.appConfig.absolutePaths.pagesDir)) {
259
+ processorPaths.push(this.appConfig.absolutePaths.pagesDir);
260
+ }
261
+
262
+ if (fileSystem.exists(this.appConfig.absolutePaths.publicDir)) {
263
+ processorPaths.push(this.appConfig.absolutePaths.publicDir);
264
+ }
265
+
266
+ if (this.appConfig.additionalWatchPaths.length) {
267
+ processorPaths.push(...this.appConfig.additionalWatchPaths);
268
+ }
269
+
270
+ this.watcher = chokidar.watch(processorPaths, {
271
+ ignoreInitial: true,
272
+ ignorePermissionErrors: true,
273
+ awaitWriteFinish: {
274
+ stabilityThreshold: 50,
275
+ pollInterval: 50,
276
+ },
277
+ });
278
+ }
279
+
280
+ for (const processor of this.appConfig.processors.values()) {
281
+ const watchConfig = processor.getWatchConfig();
282
+ if (!watchConfig) continue;
283
+ const { extensions = [], onCreate, onChange, onDelete, onError } = watchConfig;
284
+
285
+ if (onCreate) this.watcher.on('add', (path) => this.shouldProcess(path, extensions, onCreate));
286
+ if (onChange) this.watcher.on('change', (path) => this.shouldProcess(path, extensions, onChange));
287
+ if (onDelete) this.watcher.on('unlink', (path) => this.shouldProcess(path, extensions, onDelete));
288
+ if (onError) this.watcher.on('error', onError as (error: unknown) => void);
289
+ }
290
+
291
+ this.watcher.add(this.appConfig.absolutePaths.srcDir);
292
+
293
+ this.watcher
294
+ .on('change', (path) => this.handleFileChange(path))
295
+ .on('add', (path) => {
296
+ this.handleFileChange(path);
297
+ this.triggerRouterRefresh(path);
298
+ })
299
+ .on('addDir', (path) => this.triggerRouterRefresh(path))
300
+ .on('unlink', (path) => this.triggerRouterRefresh(path))
301
+ .on('unlinkDir', (path) => this.triggerRouterRefresh(path))
302
+ .on('error', (error) => this.handleError(error));
303
+
304
+ return this.watcher;
305
+ }
306
+ }