@ecopages/core 0.2.0-alpha.23 → 0.2.0-alpha.24

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 (515) hide show
  1. package/package.json +93 -226
  2. package/src/adapters/abstract/application-adapter.test.ts +172 -0
  3. package/src/adapters/abstract/application-adapter.ts +379 -0
  4. package/src/adapters/abstract/router-adapter.ts +30 -0
  5. package/src/adapters/abstract/server-adapter.ts +79 -0
  6. package/src/adapters/bun/client-bridge.ts +62 -0
  7. package/src/adapters/bun/create-app.ts +180 -0
  8. package/src/adapters/bun/hmr-manager.test.ts +267 -0
  9. package/src/adapters/bun/hmr-manager.ts +406 -0
  10. package/src/adapters/bun/index.ts +2 -0
  11. package/src/adapters/bun/server-adapter.ts +500 -0
  12. package/src/adapters/bun/server-lifecycle.ts +124 -0
  13. package/src/adapters/create-app.test.ts +10 -0
  14. package/src/adapters/create-app.ts +91 -0
  15. package/src/adapters/index.ts +2 -0
  16. package/src/adapters/node/create-app.test.ts +53 -0
  17. package/src/adapters/node/create-app.ts +183 -0
  18. package/src/adapters/node/node-client-bridge.test.ts +198 -0
  19. package/src/adapters/node/node-client-bridge.ts +79 -0
  20. package/src/adapters/node/node-hmr-manager.test.ts +322 -0
  21. package/src/adapters/node/node-hmr-manager.ts +378 -0
  22. package/src/adapters/node/server-adapter.ts +502 -0
  23. package/src/adapters/node/static-content-server.test.ts +60 -0
  24. package/src/adapters/node/static-content-server.ts +239 -0
  25. package/src/adapters/shared/api-response.test.ts +97 -0
  26. package/src/adapters/shared/api-response.ts +104 -0
  27. package/src/adapters/shared/application-adapter.ts +199 -0
  28. package/src/adapters/shared/define-api-handler.ts +66 -0
  29. package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
  30. package/src/adapters/shared/explicit-static-route-matcher.ts +140 -0
  31. package/src/adapters/shared/file-route-middleware-pipeline.test.ts +90 -0
  32. package/src/adapters/shared/file-route-middleware-pipeline.ts +127 -0
  33. package/src/adapters/shared/fs-server-response-factory.test.ts +187 -0
  34. package/src/adapters/shared/fs-server-response-factory.ts +118 -0
  35. package/src/adapters/shared/fs-server-response-matcher.test.ts +285 -0
  36. package/src/adapters/shared/fs-server-response-matcher.ts +189 -0
  37. package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
  38. package/src/adapters/shared/hmr-html-response.ts +52 -0
  39. package/src/adapters/shared/hmr-manager.contract.test.ts +232 -0
  40. package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
  41. package/src/adapters/shared/render-context.test.ts +150 -0
  42. package/src/adapters/shared/render-context.ts +123 -0
  43. package/src/adapters/shared/runtime-bootstrap.ts +79 -0
  44. package/src/adapters/shared/server-adapter.test.ts +77 -0
  45. package/src/adapters/shared/server-adapter.ts +493 -0
  46. package/src/adapters/shared/server-route-handler.test.ts +110 -0
  47. package/src/adapters/shared/server-route-handler.ts +153 -0
  48. package/src/adapters/shared/server-static-builder.test.ts +338 -0
  49. package/src/adapters/shared/server-static-builder.ts +170 -0
  50. package/src/build/build-adapter-serialization.test.ts +281 -0
  51. package/src/build/build-adapter.test.ts +1240 -0
  52. package/src/build/build-adapter.ts +1012 -0
  53. package/src/build/build-manifest.ts +54 -0
  54. package/src/build/build-types.ts +83 -0
  55. package/src/build/dev-build-coordinator.ts +220 -0
  56. package/src/build/esbuild-build-adapter.ts +660 -0
  57. package/src/build/runtime-build-executor.test.ts +81 -0
  58. package/src/build/runtime-build-executor.ts +40 -0
  59. package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
  60. package/src/build/runtime-specifier-alias-plugin.ts +62 -0
  61. package/src/build/runtime-specifier-aliases.ts +135 -0
  62. package/src/config/config-builder.test.ts +443 -0
  63. package/src/config/config-builder.ts +742 -0
  64. package/src/config/config-builder.typecheck.test.ts +96 -0
  65. package/src/config/{constants.d.ts → constants.ts} +22 -13
  66. package/src/dev/sc-server.ts +143 -0
  67. package/src/eco/eco.browser.test.ts +43 -0
  68. package/src/eco/eco.browser.ts +118 -0
  69. package/src/eco/eco.test.ts +654 -0
  70. package/src/eco/eco.ts +205 -0
  71. package/src/eco/eco.types.ts +221 -0
  72. package/src/eco/eco.utils.test.ts +219 -0
  73. package/src/eco/eco.utils.ts +5 -0
  74. package/src/eco/global-injector-map.test.ts +42 -0
  75. package/src/eco/global-injector-map.ts +112 -0
  76. package/src/eco/lazy-injector-map.test.ts +66 -0
  77. package/src/eco/lazy-injector-map.ts +120 -0
  78. package/src/eco/module-dependencies.test.ts +30 -0
  79. package/src/eco/module-dependencies.ts +75 -0
  80. package/src/errors/http-error.test.ts +134 -0
  81. package/src/errors/http-error.ts +72 -0
  82. package/src/errors/{index.d.ts → index.ts} +2 -2
  83. package/src/errors/locals-access-error.ts +7 -0
  84. package/src/global/app-logger.ts +4 -0
  85. package/src/global/utils.test.ts +12 -0
  86. 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
  87. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  88. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  89. package/src/hmr/client/hmr-runtime.ts +160 -0
  90. package/src/hmr/hmr-strategy.test.ts +124 -0
  91. package/src/hmr/hmr-strategy.ts +177 -0
  92. package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
  93. package/src/hmr/hmr.test.e2e.ts +66 -0
  94. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  95. package/src/hmr/strategies/js-hmr-strategy.test.ts +335 -0
  96. package/src/hmr/strategies/js-hmr-strategy.ts +320 -0
  97. package/src/index.browser.ts +3 -0
  98. package/src/index.ts +15 -0
  99. package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
  100. package/src/integrations/ghtml/ghtml-renderer.ts +97 -0
  101. package/src/integrations/ghtml/ghtml.constants.ts +1 -0
  102. package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
  103. package/src/plugins/alias-resolver-plugin.test.ts +41 -0
  104. package/src/plugins/alias-resolver-plugin.ts +63 -0
  105. package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
  106. package/src/plugins/eco-component-meta-plugin.ts +495 -0
  107. package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
  108. package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
  109. package/src/plugins/integration-plugin.test.ts +156 -0
  110. package/src/plugins/integration-plugin.ts +311 -0
  111. package/src/plugins/processor.test.ts +148 -0
  112. package/src/plugins/processor.ts +240 -0
  113. package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
  114. package/src/plugins/source-transform.test.ts +82 -0
  115. package/src/plugins/source-transform.ts +123 -0
  116. package/src/route-renderer/orchestration/boundary-planning.service.ts +146 -0
  117. package/src/route-renderer/orchestration/component-render-context.ts +318 -0
  118. package/src/route-renderer/orchestration/integration-renderer.test.ts +2088 -0
  119. package/src/route-renderer/orchestration/integration-renderer.ts +1285 -0
  120. package/src/route-renderer/orchestration/page-packaging.service.test.ts +76 -0
  121. package/src/route-renderer/orchestration/page-packaging.service.ts +85 -0
  122. package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
  123. package/src/route-renderer/orchestration/queued-boundary-runtime.service.test.ts +319 -0
  124. package/src/route-renderer/orchestration/queued-boundary-runtime.service.ts +289 -0
  125. package/src/route-renderer/orchestration/render-execution.service.test.ts +196 -0
  126. package/src/route-renderer/orchestration/render-execution.service.ts +182 -0
  127. package/src/route-renderer/orchestration/render-output.utils.ts +302 -0
  128. package/src/route-renderer/orchestration/render-preparation.service.test.ts +569 -0
  129. package/src/route-renderer/orchestration/render-preparation.service.ts +508 -0
  130. package/src/route-renderer/orchestration/route-shell-composer.service.ts +162 -0
  131. package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
  132. package/src/route-renderer/orchestration/template-serialization.ts +117 -0
  133. package/src/route-renderer/page-loading/component-dependency-collection.ts +196 -0
  134. package/src/route-renderer/page-loading/declared-asset-collection.ts +156 -0
  135. package/src/route-renderer/page-loading/dependency-resolver.test.ts +665 -0
  136. package/src/route-renderer/page-loading/dependency-resolver.ts +150 -0
  137. package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
  138. package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
  139. package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
  140. package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
  141. package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
  142. package/src/route-renderer/page-loading/page-dependency-bundling.ts +205 -0
  143. package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
  144. package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
  145. package/src/route-renderer/route-renderer.ts +136 -0
  146. package/src/router/client/link-intent.test.browser.ts +51 -0
  147. package/src/router/client/link-intent.ts +92 -0
  148. package/src/router/client/navigation-coordinator.test.ts +237 -0
  149. package/src/router/client/navigation-coordinator.ts +453 -0
  150. package/src/router/server/fs-router-scanner.test.ts +83 -0
  151. package/src/router/server/fs-router-scanner.ts +224 -0
  152. package/src/router/server/fs-router.test.ts +214 -0
  153. package/src/router/server/fs-router.ts +122 -0
  154. package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
  155. package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +476 -0
  156. package/src/services/assets/asset-processing-service/asset-processing.service.ts +345 -0
  157. package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
  158. package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
  159. package/src/services/assets/asset-processing-service/assets.types.ts +125 -0
  160. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
  161. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
  162. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
  163. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
  164. package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
  165. package/src/services/assets/asset-processing-service/index.ts +5 -0
  166. package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
  167. package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
  168. package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
  169. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
  170. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +174 -0
  171. package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
  172. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +192 -0
  173. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +134 -0
  174. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
  175. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +110 -0
  176. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
  177. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +87 -0
  178. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
  179. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +71 -0
  180. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +81 -0
  181. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
  182. package/src/services/assets/browser-bundle.service.test.ts +66 -0
  183. package/src/services/assets/browser-bundle.service.ts +109 -0
  184. package/src/services/cache/cache.types.ts +126 -0
  185. package/src/services/cache/index.ts +18 -0
  186. package/src/services/cache/memory-cache-store.test.ts +225 -0
  187. package/src/services/cache/memory-cache-store.ts +130 -0
  188. package/src/services/cache/page-cache-service.test.ts +175 -0
  189. package/src/services/cache/page-cache-service.ts +202 -0
  190. package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
  191. package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
  192. package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
  193. package/src/services/html/html-rewriter-provider.service.ts +104 -0
  194. package/src/services/html/html-transformer.service.test.ts +479 -0
  195. package/src/services/html/html-transformer.service.ts +275 -0
  196. package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
  197. package/src/services/invalidation/development-invalidation.service.ts +262 -0
  198. package/src/services/module-loading/app-module-loader.service.ts +9 -0
  199. package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
  200. package/src/services/module-loading/app-server-module-transpiler.service.ts +143 -0
  201. package/src/services/module-loading/host-module-loader-registry.ts +15 -0
  202. package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
  203. package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
  204. package/src/services/module-loading/node-bootstrap-plugin.ts +297 -0
  205. package/src/services/module-loading/page-module-import.service.test.ts +504 -0
  206. package/src/services/module-loading/page-module-import.service.ts +252 -0
  207. package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
  208. package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
  209. package/src/services/module-loading/source-module-support.ts +19 -0
  210. package/src/services/runtime-state/dev-graph.service.ts +217 -0
  211. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
  212. package/src/services/runtime-state/runtime-specifier-registry.service.ts +96 -0
  213. package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
  214. package/src/services/validation/schema-validation-service.test.ts +223 -0
  215. package/src/services/validation/schema-validation-service.ts +204 -0
  216. package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
  217. package/src/static-site-generator/static-site-generator.test.ts +316 -0
  218. package/src/static-site-generator/static-site-generator.ts +462 -0
  219. package/src/types/internal-types.ts +242 -0
  220. package/src/types/public-types.ts +1443 -0
  221. package/src/utils/deep-merge.test.ts +114 -0
  222. package/src/utils/deep-merge.ts +47 -0
  223. package/src/utils/hash.ts +5 -0
  224. package/src/utils/html-escaping.ts +9 -0
  225. package/src/utils/invariant.test.ts +22 -0
  226. package/src/utils/invariant.ts +15 -0
  227. package/src/utils/locals-utils.ts +37 -0
  228. package/src/utils/parse-cli-args.test.ts +69 -0
  229. package/src/utils/parse-cli-args.ts +105 -0
  230. package/src/utils/path-utils.module.ts +14 -0
  231. package/src/utils/path-utils.test.ts +15 -0
  232. package/src/utils/resolve-work-dir.ts +45 -0
  233. package/src/utils/runtime.ts +44 -0
  234. package/src/utils/server-utils.module.ts +67 -0
  235. package/src/utils/server-utils.test.ts +38 -0
  236. package/src/watchers/project-watcher.integration.test.ts +337 -0
  237. package/src/watchers/project-watcher.test-helpers.ts +41 -0
  238. package/src/watchers/project-watcher.test.ts +768 -0
  239. package/src/watchers/project-watcher.ts +357 -0
  240. package/CHANGELOG.md +0 -51
  241. package/src/adapters/abstract/application-adapter.d.ts +0 -194
  242. package/src/adapters/abstract/application-adapter.js +0 -121
  243. package/src/adapters/abstract/router-adapter.d.ts +0 -26
  244. package/src/adapters/abstract/router-adapter.js +0 -5
  245. package/src/adapters/abstract/server-adapter.d.ts +0 -69
  246. package/src/adapters/abstract/server-adapter.js +0 -15
  247. package/src/adapters/bun/client-bridge.d.ts +0 -34
  248. package/src/adapters/bun/client-bridge.js +0 -48
  249. package/src/adapters/bun/create-app.d.ts +0 -52
  250. package/src/adapters/bun/create-app.js +0 -116
  251. package/src/adapters/bun/hmr-manager.d.ts +0 -143
  252. package/src/adapters/bun/hmr-manager.js +0 -333
  253. package/src/adapters/bun/index.d.ts +0 -2
  254. package/src/adapters/bun/index.js +0 -8
  255. package/src/adapters/bun/server-adapter.d.ts +0 -155
  256. package/src/adapters/bun/server-adapter.js +0 -374
  257. package/src/adapters/bun/server-lifecycle.d.ts +0 -63
  258. package/src/adapters/bun/server-lifecycle.js +0 -92
  259. package/src/adapters/create-app.d.ts +0 -20
  260. package/src/adapters/create-app.js +0 -66
  261. package/src/adapters/index.d.ts +0 -2
  262. package/src/adapters/index.js +0 -8
  263. package/src/adapters/node/create-app.d.ts +0 -18
  264. package/src/adapters/node/create-app.js +0 -149
  265. package/src/adapters/node/node-client-bridge.d.ts +0 -26
  266. package/src/adapters/node/node-client-bridge.js +0 -66
  267. package/src/adapters/node/node-hmr-manager.d.ts +0 -133
  268. package/src/adapters/node/node-hmr-manager.js +0 -311
  269. package/src/adapters/node/server-adapter.d.ts +0 -161
  270. package/src/adapters/node/server-adapter.js +0 -359
  271. package/src/adapters/node/static-content-server.d.ts +0 -60
  272. package/src/adapters/node/static-content-server.js +0 -194
  273. package/src/adapters/shared/api-response.d.ts +0 -52
  274. package/src/adapters/shared/api-response.js +0 -96
  275. package/src/adapters/shared/application-adapter.d.ts +0 -18
  276. package/src/adapters/shared/application-adapter.js +0 -90
  277. package/src/adapters/shared/define-api-handler.d.ts +0 -25
  278. package/src/adapters/shared/define-api-handler.js +0 -15
  279. package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
  280. package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
  281. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
  282. package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
  283. package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
  284. package/src/adapters/shared/fs-server-response-factory.js +0 -97
  285. package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
  286. package/src/adapters/shared/fs-server-response-matcher.js +0 -147
  287. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
  288. package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
  289. package/src/adapters/shared/hmr-html-response.d.ts +0 -22
  290. package/src/adapters/shared/hmr-html-response.js +0 -32
  291. package/src/adapters/shared/render-context.d.ts +0 -15
  292. package/src/adapters/shared/render-context.js +0 -72
  293. package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
  294. package/src/adapters/shared/runtime-bootstrap.js +0 -43
  295. package/src/adapters/shared/server-adapter.d.ts +0 -97
  296. package/src/adapters/shared/server-adapter.js +0 -390
  297. package/src/adapters/shared/server-route-handler.d.ts +0 -89
  298. package/src/adapters/shared/server-route-handler.js +0 -111
  299. package/src/adapters/shared/server-static-builder.d.ts +0 -70
  300. package/src/adapters/shared/server-static-builder.js +0 -100
  301. package/src/build/build-adapter.d.ts +0 -239
  302. package/src/build/build-adapter.js +0 -642
  303. package/src/build/build-manifest.d.ts +0 -27
  304. package/src/build/build-manifest.js +0 -30
  305. package/src/build/build-types.d.ts +0 -57
  306. package/src/build/build-types.js +0 -0
  307. package/src/build/dev-build-coordinator.d.ts +0 -72
  308. package/src/build/dev-build-coordinator.js +0 -154
  309. package/src/build/esbuild-build-adapter.d.ts +0 -78
  310. package/src/build/esbuild-build-adapter.js +0 -505
  311. package/src/build/runtime-build-executor.d.ts +0 -14
  312. package/src/build/runtime-build-executor.js +0 -22
  313. package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
  314. package/src/build/runtime-specifier-alias-plugin.js +0 -35
  315. package/src/build/runtime-specifier-aliases.d.ts +0 -5
  316. package/src/build/runtime-specifier-aliases.js +0 -95
  317. package/src/config/config-builder.d.ts +0 -252
  318. package/src/config/config-builder.js +0 -603
  319. package/src/config/constants.js +0 -25
  320. package/src/dev/sc-server.d.ts +0 -30
  321. package/src/dev/sc-server.js +0 -111
  322. package/src/eco/eco.browser.d.ts +0 -2
  323. package/src/eco/eco.browser.js +0 -83
  324. package/src/eco/eco.d.ts +0 -9
  325. package/src/eco/eco.js +0 -85
  326. package/src/eco/eco.types.d.ts +0 -178
  327. package/src/eco/eco.types.js +0 -0
  328. package/src/eco/eco.utils.d.ts +0 -1
  329. package/src/eco/eco.utils.js +0 -10
  330. package/src/eco/global-injector-map.d.ts +0 -16
  331. package/src/eco/global-injector-map.js +0 -80
  332. package/src/eco/lazy-injector-map.d.ts +0 -8
  333. package/src/eco/lazy-injector-map.js +0 -70
  334. package/src/eco/module-dependencies.d.ts +0 -18
  335. package/src/eco/module-dependencies.js +0 -49
  336. package/src/errors/http-error.d.ts +0 -31
  337. package/src/errors/http-error.js +0 -50
  338. package/src/errors/index.js +0 -4
  339. package/src/errors/locals-access-error.d.ts +0 -4
  340. package/src/errors/locals-access-error.js +0 -9
  341. package/src/global/app-logger.d.ts +0 -2
  342. package/src/global/app-logger.js +0 -6
  343. package/src/hmr/client/hmr-runtime.d.ts +0 -5
  344. package/src/hmr/client/hmr-runtime.js +0 -109
  345. package/src/hmr/hmr-strategy.d.ts +0 -162
  346. package/src/hmr/hmr-strategy.js +0 -44
  347. package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
  348. package/src/hmr/hmr.postcss.test.e2e.js +0 -31
  349. package/src/hmr/hmr.test.e2e.d.ts +0 -1
  350. package/src/hmr/hmr.test.e2e.js +0 -43
  351. package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
  352. package/src/hmr/strategies/default-hmr-strategy.js +0 -34
  353. package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
  354. package/src/hmr/strategies/js-hmr-strategy.js +0 -178
  355. package/src/index.browser.d.ts +0 -3
  356. package/src/index.browser.js +0 -4
  357. package/src/index.d.ts +0 -6
  358. package/src/index.js +0 -21
  359. package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
  360. package/src/integrations/ghtml/ghtml-renderer.js +0 -63
  361. package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
  362. package/src/integrations/ghtml/ghtml.constants.js +0 -4
  363. package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
  364. package/src/integrations/ghtml/ghtml.plugin.js +0 -20
  365. package/src/plugins/alias-resolver-plugin.d.ts +0 -2
  366. package/src/plugins/alias-resolver-plugin.js +0 -53
  367. package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
  368. package/src/plugins/eco-component-meta-plugin.js +0 -163
  369. package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
  370. package/src/plugins/foreign-jsx-override-plugin.js +0 -35
  371. package/src/plugins/integration-plugin.d.ts +0 -219
  372. package/src/plugins/integration-plugin.js +0 -196
  373. package/src/plugins/processor.d.ts +0 -95
  374. package/src/plugins/processor.js +0 -136
  375. package/src/plugins/runtime-capability.js +0 -0
  376. package/src/plugins/source-transform.d.ts +0 -46
  377. package/src/plugins/source-transform.js +0 -71
  378. package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
  379. package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
  380. package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
  381. package/src/route-renderer/orchestration/component-render-context.js +0 -147
  382. package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -554
  383. package/src/route-renderer/orchestration/integration-renderer.js +0 -957
  384. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
  385. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
  386. package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
  387. package/src/route-renderer/orchestration/render-execution.service.js +0 -106
  388. package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -46
  389. package/src/route-renderer/orchestration/render-output.utils.js +0 -65
  390. package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
  391. package/src/route-renderer/orchestration/render-preparation.service.js +0 -341
  392. package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
  393. package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
  394. package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
  395. package/src/route-renderer/orchestration/template-serialization.js +0 -45
  396. package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
  397. package/src/route-renderer/page-loading/dependency-resolver.js +0 -444
  398. package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
  399. package/src/route-renderer/page-loading/page-module-loader.js +0 -127
  400. package/src/route-renderer/route-renderer.d.ts +0 -67
  401. package/src/route-renderer/route-renderer.js +0 -103
  402. package/src/router/client/link-intent.js +0 -34
  403. package/src/router/client/link-intent.test.browser.d.ts +0 -1
  404. package/src/router/client/link-intent.test.browser.js +0 -43
  405. package/src/router/client/navigation-coordinator.d.ts +0 -149
  406. package/src/router/client/navigation-coordinator.js +0 -215
  407. package/src/router/server/fs-router-scanner.d.ts +0 -41
  408. package/src/router/server/fs-router-scanner.js +0 -161
  409. package/src/router/server/fs-router.d.ts +0 -26
  410. package/src/router/server/fs-router.js +0 -100
  411. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -120
  412. package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -331
  413. package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
  414. package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
  415. package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -89
  416. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  417. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +0 -55
  418. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -48
  419. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
  420. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
  421. package/src/services/assets/asset-processing-service/index.d.ts +0 -5
  422. package/src/services/assets/asset-processing-service/index.js +0 -5
  423. package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
  424. package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
  425. package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
  426. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
  427. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -64
  428. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -17
  429. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -72
  430. package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
  431. package/src/services/assets/asset-processing-service/processors/index.js +0 -5
  432. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -5
  433. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -57
  434. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
  435. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -88
  436. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  437. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -75
  438. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -5
  439. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
  440. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
  441. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -66
  442. package/src/services/assets/browser-bundle.service.d.ts +0 -32
  443. package/src/services/assets/browser-bundle.service.js +0 -33
  444. package/src/services/cache/cache.types.d.ts +0 -107
  445. package/src/services/cache/cache.types.js +0 -0
  446. package/src/services/cache/index.d.ts +0 -7
  447. package/src/services/cache/index.js +0 -7
  448. package/src/services/cache/memory-cache-store.d.ts +0 -42
  449. package/src/services/cache/memory-cache-store.js +0 -98
  450. package/src/services/cache/page-cache-service.d.ts +0 -70
  451. package/src/services/cache/page-cache-service.js +0 -152
  452. package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
  453. package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
  454. package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
  455. package/src/services/html/html-rewriter-provider.service.js +0 -68
  456. package/src/services/html/html-transformer.service.d.ts +0 -77
  457. package/src/services/html/html-transformer.service.js +0 -215
  458. package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
  459. package/src/services/invalidation/development-invalidation.service.js +0 -190
  460. package/src/services/module-loading/app-module-loader.service.d.ts +0 -28
  461. package/src/services/module-loading/app-module-loader.service.js +0 -35
  462. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
  463. package/src/services/module-loading/app-server-module-transpiler.service.js +0 -109
  464. package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
  465. package/src/services/module-loading/host-module-loader-registry.js +0 -15
  466. package/src/services/module-loading/module-loading-types.js +0 -0
  467. package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
  468. package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
  469. package/src/services/module-loading/page-module-import.service.d.ts +0 -76
  470. package/src/services/module-loading/page-module-import.service.js +0 -173
  471. package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -72
  472. package/src/services/module-loading/server-module-transpiler.service.js +0 -64
  473. package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
  474. package/src/services/runtime-state/dev-graph.service.js +0 -162
  475. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
  476. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
  477. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
  478. package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
  479. package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
  480. package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
  481. package/src/services/validation/schema-validation-service.d.ts +0 -122
  482. package/src/services/validation/schema-validation-service.js +0 -101
  483. package/src/services/validation/standard-schema.types.js +0 -0
  484. package/src/static-site-generator/static-site-generator.d.ts +0 -104
  485. package/src/static-site-generator/static-site-generator.js +0 -338
  486. package/src/types/internal-types.d.ts +0 -231
  487. package/src/types/internal-types.js +0 -0
  488. package/src/types/public-types.d.ts +0 -1219
  489. package/src/types/public-types.js +0 -0
  490. package/src/utils/deep-merge.d.ts +0 -14
  491. package/src/utils/deep-merge.js +0 -32
  492. package/src/utils/hash.d.ts +0 -1
  493. package/src/utils/hash.js +0 -7
  494. package/src/utils/html-escaping.d.ts +0 -7
  495. package/src/utils/html-escaping.js +0 -6
  496. package/src/utils/html.js +0 -4
  497. package/src/utils/invariant.d.ts +0 -5
  498. package/src/utils/invariant.js +0 -11
  499. package/src/utils/locals-utils.d.ts +0 -15
  500. package/src/utils/locals-utils.js +0 -24
  501. package/src/utils/parse-cli-args.d.ts +0 -27
  502. package/src/utils/parse-cli-args.js +0 -62
  503. package/src/utils/path-utils.module.d.ts +0 -5
  504. package/src/utils/path-utils.module.js +0 -14
  505. package/src/utils/resolve-work-dir.d.ts +0 -11
  506. package/src/utils/resolve-work-dir.js +0 -31
  507. package/src/utils/runtime.d.ts +0 -11
  508. package/src/utils/runtime.js +0 -40
  509. package/src/utils/server-utils.module.d.ts +0 -19
  510. package/src/utils/server-utils.module.js +0 -56
  511. package/src/watchers/project-watcher.d.ts +0 -136
  512. package/src/watchers/project-watcher.js +0 -275
  513. package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
  514. package/src/watchers/project-watcher.test-helpers.js +0 -52
  515. /package/src/utils/{html.d.ts → html.ts} +0 -0
@@ -0,0 +1,768 @@
1
+ import { describe, expect, test, beforeEach, afterEach, vi } from 'vitest';
2
+ import path from 'node:path';
3
+ import chokidar from 'chokidar';
4
+ import { fileSystem } from '@ecopages/file-system';
5
+ import { ProjectWatcher } from './project-watcher';
6
+ import type { EcoPagesAppConfig, IHmrManager } from '../types/internal-types.ts';
7
+ import type { ClientBridge } from '../adapters/bun/client-bridge.ts';
8
+ import { ConfigBuilder } from '../config/config-builder.ts';
9
+ import { InMemoryDevGraphService, setAppDevGraphService } from '../services/runtime-state/dev-graph.service.ts';
10
+ import { createMockHmrManager, createMockBridge } from './project-watcher.test-helpers.ts';
11
+
12
+ const createMockConfig = async (rootDir = '/test/project'): Promise<EcoPagesAppConfig> => {
13
+ return await new ConfigBuilder().setRootDir(rootDir).build();
14
+ };
15
+
16
+ describe('ProjectWatcher', () => {
17
+ let watcher: ProjectWatcher;
18
+ let Config: EcoPagesAppConfig;
19
+ let HmrManager: IHmrManager;
20
+ let Bridge: ClientBridge;
21
+ let RefreshCallback: any;
22
+
23
+ beforeEach(async () => {
24
+ Config = await createMockConfig();
25
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
26
+ HmrManager = createMockHmrManager();
27
+ Bridge = createMockBridge();
28
+ RefreshCallback = vi.fn(async () => {});
29
+
30
+ watcher = new ProjectWatcher({
31
+ config: Config,
32
+ refreshRouterRoutesCallback: RefreshCallback,
33
+ hmrManager: HmrManager,
34
+ bridge: Bridge,
35
+ });
36
+ });
37
+
38
+ afterEach(() => {
39
+ vi.restoreAllMocks();
40
+ });
41
+
42
+ describe('constructor', () => {
43
+ test('should initialize with provided config', () => {
44
+ expect(watcher).toBeDefined();
45
+ });
46
+
47
+ test('should bind callback methods correctly', () => {
48
+ expect(typeof watcher.triggerRouterRefresh).toBe('function');
49
+ expect(typeof watcher.handleError).toBe('function');
50
+ });
51
+ });
52
+
53
+ describe('triggerRouterRefresh', () => {
54
+ test('should call refresh callback for page directory changes', async () => {
55
+ const pagePath = path.join(Config.absolutePaths.pagesDir, 'index.tsx');
56
+ await watcher.triggerRouterRefresh(pagePath);
57
+
58
+ expect(RefreshCallback).toHaveBeenCalled();
59
+ });
60
+
61
+ test('should not call refresh callback for stylesheet assets inside the pages directory', async () => {
62
+ const stylesheetPath = path.join(Config.absolutePaths.pagesDir, 'index.css');
63
+ await watcher.triggerRouterRefresh(stylesheetPath);
64
+
65
+ expect(RefreshCallback).not.toHaveBeenCalled();
66
+ });
67
+
68
+ test('should not call refresh callback for non-page directory changes', async () => {
69
+ const nonPagePath = '/test/project/src/components/Button.tsx';
70
+ await watcher.triggerRouterRefresh(nonPagePath);
71
+
72
+ expect(RefreshCallback).not.toHaveBeenCalled();
73
+ });
74
+ });
75
+
76
+ describe('handleError', () => {
77
+ test('should broadcast error message and log', () => {
78
+ const error = new Error('Test error');
79
+ watcher.handleError(error);
80
+
81
+ expect(HmrManager.broadcast).toHaveBeenCalledWith({
82
+ type: 'error',
83
+ message: 'Test error',
84
+ });
85
+ });
86
+
87
+ test('should handle non-Error objects', () => {
88
+ watcher.handleError('string error');
89
+
90
+ expect(HmrManager.broadcast).not.toHaveBeenCalled();
91
+ });
92
+ });
93
+ });
94
+
95
+ describe('ProjectWatcher - File Change Handling', () => {
96
+ let watcher: ProjectWatcher;
97
+ let Config: EcoPagesAppConfig;
98
+ let HmrManager: IHmrManager;
99
+ let Bridge: ClientBridge;
100
+ let RefreshCallback: any;
101
+
102
+ beforeEach(async () => {
103
+ Config = await createMockConfig();
104
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
105
+ HmrManager = createMockHmrManager();
106
+ Bridge = createMockBridge();
107
+ RefreshCallback = vi.fn(async () => {});
108
+
109
+ watcher = new ProjectWatcher({
110
+ config: Config as EcoPagesAppConfig,
111
+ refreshRouterRoutesCallback: RefreshCallback,
112
+ hmrManager: HmrManager,
113
+ bridge: Bridge,
114
+ });
115
+ });
116
+
117
+ afterEach(() => {
118
+ vi.restoreAllMocks();
119
+ });
120
+
121
+ test('should only invalidate server modules for route and server source changes', async () => {
122
+ const pageFilePath = path.join(Config.absolutePaths.pagesDir, 'about.tsx');
123
+ const cssFilePath = path.join(Config.absolutePaths.srcDir, 'styles', 'main.css');
124
+ const serverInvalidationState = Config.runtime?.serverInvalidationState as InMemoryDevGraphService;
125
+
126
+ Config.processors.set('css', {
127
+ getWatchConfig: vi.fn(() => ({
128
+ paths: ['/test/project/src'],
129
+ extensions: ['.css'],
130
+ })),
131
+ getAssetCapabilities: vi.fn(() => [{ kind: 'stylesheet', extensions: ['*.css'] }]),
132
+ canProcessAsset: vi.fn(
133
+ (kind: string, filepath?: string) => kind === 'stylesheet' && filepath?.endsWith('.css'),
134
+ ),
135
+ matchesFileFilter: vi.fn((filepath: string) => filepath.endsWith('.css')),
136
+ } as never);
137
+
138
+ expect(serverInvalidationState.getServerInvalidationVersion()).toBe(0);
139
+
140
+ await (watcher as any).handleFileChange(pageFilePath);
141
+ expect(serverInvalidationState.getServerInvalidationVersion()).toBe(1);
142
+
143
+ await (watcher as any).handleFileChange(cssFilePath);
144
+ expect(serverInvalidationState.getServerInvalidationVersion()).toBe(1);
145
+ });
146
+
147
+ describe('public directory files', () => {
148
+ test('should handle public file changes with single-file copy', async () => {
149
+ const publicFilePath = path.join(Config.absolutePaths.publicDir, 'favicon.ico');
150
+
151
+ await (watcher as any).handleFileChange(publicFilePath);
152
+
153
+ expect(Bridge.reload).toHaveBeenCalled();
154
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
155
+ });
156
+
157
+ test('should handle public file in subdirectory', async () => {
158
+ const publicFilePath = path.join(Config.absolutePaths.publicDir, 'images', 'logo.png');
159
+
160
+ await (watcher as any).handleFileChange(publicFilePath);
161
+
162
+ expect(Bridge.reload).toHaveBeenCalled();
163
+ });
164
+
165
+ test('should not call uncacheModules for public files', async () => {
166
+ const publicFilePath = path.join(Config.absolutePaths.publicDir, 'robots.txt');
167
+
168
+ await (watcher as any).handleFileChange(publicFilePath);
169
+
170
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
171
+ });
172
+ });
173
+
174
+ describe('page files', () => {
175
+ test('should refresh router for page file changes', async () => {
176
+ const pageFilePath = path.join(Config.absolutePaths.pagesDir, 'about.tsx');
177
+
178
+ await (watcher as any).handleFileChange(pageFilePath);
179
+
180
+ expect(RefreshCallback).toHaveBeenCalled();
181
+ });
182
+
183
+ test('should call HMR manager for page file changes', async () => {
184
+ const pageFilePath = path.join(Config.absolutePaths.pagesDir, 'contact.tsx');
185
+
186
+ await (watcher as any).handleFileChange(pageFilePath);
187
+
188
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(pageFilePath));
189
+ });
190
+
191
+ test('should await route refresh before delegating page file changes to HMR', async () => {
192
+ const pageFilePath = path.join(Config.absolutePaths.pagesDir, 'contact.tsx');
193
+ let releaseRefresh!: () => void;
194
+ const refreshGate = new Promise<void>((resolve) => {
195
+ releaseRefresh = resolve;
196
+ });
197
+ const asyncRefreshCallback = vi.fn(async () => {
198
+ await refreshGate;
199
+ });
200
+
201
+ watcher = new ProjectWatcher({
202
+ config: Config as EcoPagesAppConfig,
203
+ refreshRouterRoutesCallback: asyncRefreshCallback,
204
+ hmrManager: HmrManager,
205
+ bridge: Bridge,
206
+ });
207
+
208
+ const pendingChange = (watcher as any).handleFileChange(pageFilePath, 'add');
209
+ await Promise.resolve();
210
+
211
+ expect(asyncRefreshCallback).toHaveBeenCalledTimes(1);
212
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
213
+
214
+ releaseRefresh();
215
+ await pendingChange;
216
+
217
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(pageFilePath));
218
+ });
219
+
220
+ test('should not refresh router for stylesheet changes inside the pages directory', async () => {
221
+ const pageCssPath = path.join(Config.absolutePaths.pagesDir, 'index.css');
222
+
223
+ await (watcher as any).handleFileChange(pageCssPath);
224
+
225
+ expect(RefreshCallback).not.toHaveBeenCalled();
226
+ });
227
+
228
+ test('should ignore duplicate save events for the same page within the debounce window', async () => {
229
+ const pageFilePath = path.join(Config.absolutePaths.pagesDir, 'contact.tsx');
230
+ const nowSpy = vi.spyOn(Date, 'now');
231
+ nowSpy.mockReturnValueOnce(1000).mockReturnValueOnce(1050);
232
+
233
+ await (watcher as any).handleFileChange(pageFilePath);
234
+ await (watcher as any).handleFileChange(pageFilePath);
235
+
236
+ expect(HmrManager.handleFileChange).toHaveBeenCalledTimes(1);
237
+ expect(RefreshCallback).toHaveBeenCalledTimes(1);
238
+ });
239
+ });
240
+
241
+ describe('include files', () => {
242
+ test('should reload for include template changes', async () => {
243
+ const includeFilePath = path.join(Config.absolutePaths.includesDir, 'seo.kita.tsx');
244
+
245
+ await (watcher as any).handleFileChange(includeFilePath);
246
+
247
+ expect(Bridge.reload).toHaveBeenCalledTimes(1);
248
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
249
+ expect(RefreshCallback).not.toHaveBeenCalled();
250
+ });
251
+
252
+ test('should notify processors before reloading include template changes', async () => {
253
+ const onChange = vi.fn(async () => {});
254
+ const Processor = {
255
+ getWatchConfig: vi.fn(() => ({
256
+ paths: ['/test/project/src'],
257
+ extensions: ['.css', '.tsx'],
258
+ onChange,
259
+ })),
260
+ getAssetCapabilities: vi.fn(() => [{ kind: 'stylesheet', extensions: ['*.css'] }]),
261
+ canProcessAsset: vi.fn((kind: string, filepath?: string) => {
262
+ return kind === 'stylesheet' && filepath?.endsWith('.css');
263
+ }),
264
+ matchesFileFilter: vi.fn((filepath: string) => filepath.endsWith('.css')),
265
+ };
266
+ Config.processors.set('css', Processor as any);
267
+
268
+ const includeFilePath = path.join(Config.absolutePaths.includesDir, 'seo.kita.tsx');
269
+
270
+ await (watcher as any).handleFileChange(includeFilePath);
271
+
272
+ expect(onChange).toHaveBeenCalledWith({ path: path.resolve(includeFilePath), bridge: Bridge });
273
+ expect(Bridge.reload).toHaveBeenCalledTimes(1);
274
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
275
+ });
276
+ });
277
+
278
+ describe('additionalWatchPaths', () => {
279
+ test('should reload for files matching additionalWatchPaths pattern', async () => {
280
+ Config.additionalWatchPaths = ['**/*.config.ts'];
281
+ const configFilePath = '/test/project/app.config.ts';
282
+
283
+ await (watcher as any).handleFileChange(configFilePath);
284
+
285
+ expect(Bridge.reload).toHaveBeenCalled();
286
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
287
+ });
288
+
289
+ test('should reload for exact path matches', async () => {
290
+ const exactPath = '/test/project/tailwind.config.ts';
291
+ Config.additionalWatchPaths = [exactPath];
292
+
293
+ await (watcher as any).handleFileChange(exactPath);
294
+
295
+ expect(Bridge.reload).toHaveBeenCalled();
296
+ });
297
+
298
+ test('should not reload for non-matching paths', async () => {
299
+ Config.additionalWatchPaths = ['**/*.config.ts'];
300
+ const nonMatchingPath = '/test/project/src/components/Button.tsx';
301
+
302
+ await (watcher as any).handleFileChange(nonMatchingPath);
303
+
304
+ expect(HmrManager.handleFileChange).toHaveBeenCalled();
305
+ });
306
+ });
307
+
308
+ describe('processor-handled files', () => {
309
+ test('should skip HMR for processor-handled extensions', async () => {
310
+ const onChange = vi.fn(async () => {});
311
+ const Processor = {
312
+ getWatchConfig: vi.fn(() => ({
313
+ paths: ['/test/project/src'],
314
+ extensions: ['.css', '.scss'],
315
+ onChange,
316
+ })),
317
+ };
318
+ Config.processors.set('css', Processor as any);
319
+
320
+ const cssFilePath = '/test/project/src/styles/main.css';
321
+
322
+ await (watcher as any).handleFileChange(cssFilePath);
323
+
324
+ expect(onChange).toHaveBeenCalledWith({ path: path.resolve(cssFilePath), bridge: Bridge });
325
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
326
+ });
327
+
328
+ test('should process files through HMR when not handled by processor', async () => {
329
+ const Processor = {
330
+ getWatchConfig: vi.fn(() => ({
331
+ paths: ['/test/project/src'],
332
+ extensions: ['.css'],
333
+ })),
334
+ };
335
+ Config.processors.set('css', Processor as any);
336
+
337
+ const jsFilePath = '/test/project/src/app.js';
338
+
339
+ await (watcher as any).handleFileChange(jsFilePath);
340
+
341
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(jsFilePath));
342
+ });
343
+
344
+ test('should keep TSX changes in HMR when a processor only handles stylesheet assets', async () => {
345
+ const onChange = vi.fn(async () => {});
346
+ const Processor = {
347
+ getWatchConfig: vi.fn(() => ({
348
+ paths: ['/test/project/src'],
349
+ extensions: ['.css', '.tsx'],
350
+ onChange,
351
+ })),
352
+ getAssetCapabilities: vi.fn(() => [{ kind: 'stylesheet', extensions: ['*.css'] }]),
353
+ canProcessAsset: vi.fn((kind: string, filepath?: string) => {
354
+ return kind === 'stylesheet' && filepath?.endsWith('.css');
355
+ }),
356
+ matchesFileFilter: vi.fn((filepath: string) => filepath.endsWith('.css')),
357
+ };
358
+ Config.processors.set('css', Processor as any);
359
+
360
+ const tsxFilePath = '/test/project/src/components/Button.tsx';
361
+
362
+ await (watcher as any).handleFileChange(tsxFilePath);
363
+
364
+ expect(onChange).toHaveBeenCalledWith({ path: path.resolve(tsxFilePath), bridge: Bridge });
365
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(tsxFilePath));
366
+ });
367
+
368
+ test('should route TSX through HMR even when no specific strategy matches', async () => {
369
+ const onChange = vi.fn(async () => {});
370
+ const Processor = {
371
+ getWatchConfig: vi.fn(() => ({
372
+ paths: ['/test/project/src'],
373
+ extensions: ['.css', '.tsx'],
374
+ onChange,
375
+ })),
376
+ getAssetCapabilities: vi.fn(() => [{ kind: 'stylesheet', extensions: ['*.css'] }]),
377
+ canProcessAsset: vi.fn((kind: string, filepath?: string) => {
378
+ return kind === 'stylesheet' && filepath?.endsWith('.css');
379
+ }),
380
+ matchesFileFilter: vi.fn((filepath: string) => filepath.endsWith('.css')),
381
+ };
382
+ Config.processors.set('css', Processor as any);
383
+
384
+ const tsxFilePath = '/test/project/src/components/Button.tsx';
385
+
386
+ await (watcher as any).handleFileChange(tsxFilePath);
387
+
388
+ expect(onChange).toHaveBeenCalled();
389
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(tsxFilePath));
390
+ expect(HmrManager.broadcast).not.toHaveBeenCalledWith({ type: 'layout-update' });
391
+ });
392
+
393
+ test('should handle processor without watchConfig', async () => {
394
+ const Processor = {
395
+ getWatchConfig: vi.fn(() => null),
396
+ };
397
+ Config.processors.set('no-watch', Processor as any);
398
+
399
+ const filePath = '/test/project/src/app.js';
400
+
401
+ await (watcher as any).handleFileChange(filePath);
402
+
403
+ expect(HmrManager.handleFileChange).toHaveBeenCalled();
404
+ });
405
+ });
406
+
407
+ describe('error handling', () => {
408
+ test('should handle errors during file change processing', async () => {
409
+ HmrManager.handleFileChange = vi.fn(async () => {
410
+ throw new Error('HMR error');
411
+ });
412
+
413
+ const filePath = '/test/project/src/app.js';
414
+
415
+ await (watcher as any).handleFileChange(filePath);
416
+
417
+ expect(Bridge.error).toHaveBeenCalledWith('HMR error');
418
+ });
419
+
420
+ test('should continue processing after error', async () => {
421
+ HmrManager.handleFileChange = vi.fn(async () => {
422
+ throw new Error('Processing failed');
423
+ });
424
+
425
+ const filePath = '/test/project/src/app.js';
426
+
427
+ await (watcher as any).handleFileChange(filePath);
428
+
429
+ expect(Bridge.error).toHaveBeenCalledWith('Processing failed');
430
+ });
431
+ });
432
+ });
433
+
434
+ describe('ProjectWatcher - Priority Rules', () => {
435
+ let watcher: ProjectWatcher;
436
+ let Config: EcoPagesAppConfig;
437
+ let HmrManager: IHmrManager;
438
+ let Bridge: ClientBridge;
439
+
440
+ beforeEach(async () => {
441
+ Config = await createMockConfig();
442
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
443
+ HmrManager = createMockHmrManager();
444
+ Bridge = createMockBridge();
445
+
446
+ watcher = new ProjectWatcher({
447
+ config: Config as EcoPagesAppConfig,
448
+ refreshRouterRoutesCallback: vi.fn(async () => {}),
449
+ hmrManager: HmrManager,
450
+ bridge: Bridge,
451
+ });
452
+ });
453
+
454
+ afterEach(() => {
455
+ vi.restoreAllMocks();
456
+ });
457
+
458
+ test('should prioritize public dir over additionalWatchPaths', async () => {
459
+ Config.additionalWatchPaths = ['**/*'];
460
+ const publicFilePath = path.join(Config.absolutePaths.publicDir, 'icon.png');
461
+
462
+ await (watcher as any).handleFileChange(publicFilePath);
463
+
464
+ expect(Bridge.reload).toHaveBeenCalledTimes(1);
465
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
466
+ });
467
+
468
+ test('should prioritize additionalWatchPaths over processors', async () => {
469
+ Config.additionalWatchPaths = ['**/*.config.ts'];
470
+ const Processor = {
471
+ getWatchConfig: vi.fn(() => ({
472
+ paths: ['/test/project'],
473
+ extensions: ['.ts'],
474
+ })),
475
+ };
476
+ Config.processors.set('ts', Processor as any);
477
+
478
+ const configFilePath = '/test/project/app.config.ts';
479
+
480
+ await (watcher as any).handleFileChange(configFilePath);
481
+
482
+ expect(Bridge.reload).toHaveBeenCalled();
483
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
484
+ });
485
+
486
+ test('should prioritize processors over HMR strategies', async () => {
487
+ const Processor = {
488
+ getWatchConfig: vi.fn(() => ({
489
+ paths: ['/test/project/src'],
490
+ extensions: ['.mdx'],
491
+ })),
492
+ };
493
+ Config.processors.set('mdx', Processor as any);
494
+
495
+ const mdxFilePath = '/test/project/src/content.mdx';
496
+
497
+ await (watcher as any).handleFileChange(mdxFilePath);
498
+
499
+ expect(HmrManager.handleFileChange).not.toHaveBeenCalled();
500
+ });
501
+
502
+ test('should use HMR as final fallback', async () => {
503
+ const regularFilePath = '/test/project/src/components/Button.tsx';
504
+
505
+ await (watcher as any).handleFileChange(regularFilePath);
506
+
507
+ expect(HmrManager.handleFileChange).toHaveBeenCalled();
508
+ });
509
+
510
+ test('should notify processor for dependency file before proceeding to HMR', async () => {
511
+ const onChange = vi.fn(async () => {});
512
+ const Processor = {
513
+ getWatchConfig: vi.fn(() => ({
514
+ paths: ['/test/project/src'],
515
+ extensions: ['.css', '.tsx'],
516
+ onChange,
517
+ })),
518
+ getAssetCapabilities: vi.fn(() => [{ kind: 'stylesheet', extensions: ['*.css'] }]),
519
+ canProcessAsset: vi.fn((kind: string, filepath?: string) => {
520
+ return kind === 'stylesheet' && filepath?.endsWith('.css');
521
+ }),
522
+ matchesFileFilter: vi.fn((filepath: string) => filepath.endsWith('.css')),
523
+ };
524
+ Config.processors.set('css', Processor as any);
525
+
526
+ const tsxFilePath = '/test/project/src/components/Button.tsx';
527
+
528
+ await (watcher as any).handleFileChange(tsxFilePath);
529
+
530
+ expect(onChange).toHaveBeenCalledWith({ path: path.resolve(tsxFilePath), bridge: Bridge });
531
+ expect(HmrManager.handleFileChange).toHaveBeenCalledWith(path.resolve(tsxFilePath));
532
+ });
533
+ });
534
+
535
+ describe('ProjectWatcher - Helper Methods', () => {
536
+ let watcher: ProjectWatcher;
537
+ let Config: EcoPagesAppConfig;
538
+
539
+ beforeEach(async () => {
540
+ Config = await createMockConfig();
541
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
542
+ watcher = new ProjectWatcher({
543
+ config: Config as EcoPagesAppConfig,
544
+ refreshRouterRoutesCallback: vi.fn(async () => {}),
545
+ hmrManager: createMockHmrManager(),
546
+ bridge: createMockBridge(),
547
+ });
548
+ });
549
+
550
+ describe('isPublicDirFile', () => {
551
+ test('should return true for files in public directory', () => {
552
+ const publicFile = path.join(Config.absolutePaths.publicDir, 'favicon.ico');
553
+ const result = (watcher as any).isPublicDirFile(publicFile);
554
+ expect(result).toBe(true);
555
+ });
556
+
557
+ test('should return false for files outside public directory', () => {
558
+ const srcFile = path.join(Config.absolutePaths.srcDir, 'app.tsx');
559
+ const result = (watcher as any).isPublicDirFile(srcFile);
560
+ expect(result).toBe(false);
561
+ });
562
+ });
563
+
564
+ describe('matchesAdditionalWatchPaths', () => {
565
+ test('should match wildcard patterns', () => {
566
+ Config.additionalWatchPaths = ['**/*.config.ts'];
567
+ const result = (watcher as any).matchesAdditionalWatchPaths('/test/app.config.ts');
568
+ expect(result).toBe(true);
569
+ });
570
+
571
+ test('should match exact paths', () => {
572
+ const exactPath = '/test/project/tailwind.config.ts';
573
+ Config.additionalWatchPaths = [exactPath];
574
+ const result = (watcher as any).matchesAdditionalWatchPaths(exactPath);
575
+ expect(result).toBe(true);
576
+ });
577
+
578
+ test('should return false when no patterns match', () => {
579
+ Config.additionalWatchPaths = ['**/*.config.ts'];
580
+ const result = (watcher as any).matchesAdditionalWatchPaths('/test/app.tsx');
581
+ expect(result).toBe(false);
582
+ });
583
+
584
+ test('should return false when additionalWatchPaths is empty', () => {
585
+ Config.additionalWatchPaths = [];
586
+ const result = (watcher as any).matchesAdditionalWatchPaths('/test/app.tsx');
587
+ expect(result).toBe(false);
588
+ });
589
+ });
590
+
591
+ describe('isIncludeSourceFile', () => {
592
+ test('should return true for files in includes directory', () => {
593
+ const includeFile = path.join(Config.absolutePaths.includesDir, 'seo.kita.tsx');
594
+ const result = (watcher as any).isIncludeSourceFile(includeFile);
595
+ expect(result).toBe(true);
596
+ });
597
+
598
+ test('should return false for files outside includes directory', () => {
599
+ const srcFile = path.join(Config.absolutePaths.srcDir, 'components', 'Button.tsx');
600
+ const result = (watcher as any).isIncludeSourceFile(srcFile);
601
+ expect(result).toBe(false);
602
+ });
603
+ });
604
+
605
+ describe('isHandledByProcessor', () => {
606
+ test('should return true when file extension matches processor', () => {
607
+ const Processor = {
608
+ getWatchConfig: vi.fn(() => ({
609
+ paths: ['/test/project/src'],
610
+ extensions: ['.css', '.scss'],
611
+ })),
612
+ };
613
+ Config.processors.set('css', Processor as any);
614
+
615
+ const result = (watcher as any).isHandledByProcessor('/test/styles/main.css');
616
+ expect(result).toBe(true);
617
+ });
618
+
619
+ test('should return false when no processor handles the extension', () => {
620
+ const Processor = {
621
+ getWatchConfig: vi.fn(() => ({
622
+ paths: ['/test/project/src'],
623
+ extensions: ['.css'],
624
+ })),
625
+ };
626
+ Config.processors.set('css', Processor as any);
627
+
628
+ const result = (watcher as any).isHandledByProcessor('/test/app.tsx');
629
+ expect(result).toBe(false);
630
+ });
631
+
632
+ test('should handle processor without watchConfig', () => {
633
+ const Processor = {
634
+ getWatchConfig: vi.fn(() => null),
635
+ };
636
+ Config.processors.set('no-watch', Processor as any);
637
+
638
+ const result = (watcher as any).isHandledByProcessor('/test/app.tsx');
639
+ expect(result).toBe(false);
640
+ });
641
+
642
+ test('should handle processor with empty extensions array', () => {
643
+ const Processor = {
644
+ getWatchConfig: vi.fn(() => ({
645
+ paths: ['/test/project/src'],
646
+ extensions: [],
647
+ })),
648
+ };
649
+ Config.processors.set('empty', Processor as any);
650
+
651
+ const result = (watcher as any).isHandledByProcessor('/test/app.tsx');
652
+ expect(result).toBe(false);
653
+ });
654
+ });
655
+ });
656
+
657
+ describe('ProjectWatcher - Watch Subscriptions', () => {
658
+ afterEach(() => {
659
+ vi.restoreAllMocks();
660
+ });
661
+
662
+ test('should watch includes and src directories alongside pages and processor paths', async () => {
663
+ const Config = await createMockConfig();
664
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
665
+ const HmrManager = createMockHmrManager();
666
+ const Bridge = createMockBridge();
667
+ vi.spyOn(fileSystem, 'exists').mockImplementation((targetPath) =>
668
+ [Config.absolutePaths.includesDir, Config.absolutePaths.srcDir, Config.absolutePaths.pagesDir].includes(
669
+ String(targetPath),
670
+ ),
671
+ );
672
+ const watcherHandle = {
673
+ add: vi.fn(),
674
+ on: vi.fn().mockReturnThis(),
675
+ close: vi.fn(),
676
+ };
677
+ const chokidarWatch = vi.fn(() => watcherHandle);
678
+
679
+ vi.spyOn(chokidar, 'watch').mockImplementation(chokidarWatch as never);
680
+
681
+ Config.processors.set('css', {
682
+ getWatchConfig: vi.fn(() => ({
683
+ paths: ['/test/project/custom-watch'],
684
+ extensions: ['.css'],
685
+ })),
686
+ } as never);
687
+
688
+ const watcher = new ProjectWatcher({
689
+ config: Config,
690
+ refreshRouterRoutesCallback: vi.fn(async () => {}),
691
+ hmrManager: HmrManager,
692
+ bridge: Bridge,
693
+ });
694
+
695
+ await watcher.createWatcherSubscription();
696
+
697
+ expect(chokidarWatch).toHaveBeenCalledWith(
698
+ expect.arrayContaining([
699
+ '/test/project/custom-watch',
700
+ Config.absolutePaths.includesDir,
701
+ Config.absolutePaths.srcDir,
702
+ Config.absolutePaths.pagesDir,
703
+ ]),
704
+ expect.any(Object),
705
+ );
706
+ expect(watcherHandle.add).not.toHaveBeenCalled();
707
+ });
708
+
709
+ test('should attach chokidar handlers only once when watcher subscription is requested twice', async () => {
710
+ const Config = await createMockConfig();
711
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
712
+ const HmrManager = createMockHmrManager();
713
+ const Bridge = createMockBridge();
714
+ const watcherHandle = {
715
+ add: vi.fn(),
716
+ on: vi.fn().mockReturnThis(),
717
+ close: vi.fn(),
718
+ };
719
+
720
+ vi.spyOn(chokidar, 'watch').mockImplementation(() => watcherHandle as never);
721
+
722
+ const watcher = new ProjectWatcher({
723
+ config: Config,
724
+ refreshRouterRoutesCallback: vi.fn(async () => {}),
725
+ hmrManager: HmrManager,
726
+ bridge: Bridge,
727
+ });
728
+
729
+ await watcher.createWatcherSubscription();
730
+ await watcher.createWatcherSubscription();
731
+
732
+ expect(chokidar.watch).toHaveBeenCalledTimes(1);
733
+ expect(watcherHandle.on).toHaveBeenCalledTimes(6);
734
+ });
735
+
736
+ test('should refresh routes once for added page files', async () => {
737
+ const Config = await createMockConfig();
738
+ setAppDevGraphService(Config, new InMemoryDevGraphService());
739
+ const HmrManager = createMockHmrManager();
740
+ const Bridge = createMockBridge();
741
+ const eventHandlers = new Map<string, (path: string) => void>();
742
+ const refreshRouterRoutesCallback = vi.fn(async () => {});
743
+ const watcherHandle = {
744
+ add: vi.fn(),
745
+ on: vi.fn((event: string, handler: (path: string) => void) => {
746
+ eventHandlers.set(event, handler);
747
+ return watcherHandle;
748
+ }),
749
+ close: vi.fn(),
750
+ };
751
+
752
+ vi.spyOn(chokidar, 'watch').mockImplementation(() => watcherHandle as never);
753
+
754
+ const watcher = new ProjectWatcher({
755
+ config: Config,
756
+ refreshRouterRoutesCallback,
757
+ hmrManager: HmrManager,
758
+ bridge: Bridge,
759
+ });
760
+
761
+ await watcher.createWatcherSubscription();
762
+
763
+ eventHandlers.get('add')?.(path.join(Config.absolutePaths.pagesDir, 'new-page.tsx'));
764
+ await (watcher as any).changeQueue;
765
+
766
+ expect(refreshRouterRoutesCallback).toHaveBeenCalledTimes(1);
767
+ });
768
+ });