@ecopages/core 0.2.0-alpha.26 → 0.2.0-alpha.27

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 (550) hide show
  1. package/README.md +63 -7
  2. package/package.json +73 -249
  3. package/src/adapters/abstract/application-adapter.test.ts +172 -0
  4. package/src/adapters/abstract/application-adapter.ts +379 -0
  5. package/src/adapters/abstract/router-adapter.ts +30 -0
  6. package/src/adapters/abstract/server-adapter.ts +79 -0
  7. package/src/adapters/bun/client-bridge.ts +62 -0
  8. package/src/adapters/bun/create-app.ts +232 -0
  9. package/src/adapters/bun/hmr-manager.test.ts +265 -0
  10. package/src/adapters/bun/hmr-manager.ts +383 -0
  11. package/src/adapters/bun/index.ts +2 -0
  12. package/src/adapters/bun/server-adapter.ts +526 -0
  13. package/src/adapters/bun/server-lifecycle.ts +124 -0
  14. package/src/adapters/create-app.test.ts +10 -0
  15. package/src/adapters/create-app.ts +91 -0
  16. package/src/adapters/index.ts +2 -0
  17. package/src/adapters/node/create-app.test.ts +53 -0
  18. package/src/adapters/node/create-app.ts +183 -0
  19. package/src/adapters/node/node-client-bridge.test.ts +198 -0
  20. package/src/adapters/node/node-client-bridge.ts +79 -0
  21. package/src/adapters/node/node-hmr-manager.test.ts +320 -0
  22. package/src/adapters/node/node-hmr-manager.ts +355 -0
  23. package/src/adapters/node/server-adapter.ts +502 -0
  24. package/src/adapters/node/static-content-server.test.ts +60 -0
  25. package/src/adapters/node/static-content-server.ts +239 -0
  26. package/src/adapters/shared/api-response.test.ts +97 -0
  27. package/src/adapters/shared/api-response.ts +104 -0
  28. package/src/adapters/shared/application-adapter.ts +199 -0
  29. package/src/adapters/shared/define-api-handler.ts +66 -0
  30. package/src/adapters/shared/explicit-static-render-preparation.ts +58 -0
  31. package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
  32. package/src/adapters/shared/explicit-static-route-matcher.ts +131 -0
  33. package/src/adapters/shared/file-route-middleware-pipeline.test.ts +85 -0
  34. package/src/adapters/shared/file-route-middleware-pipeline.ts +118 -0
  35. package/src/adapters/shared/fs-server-response-factory.test.ts +176 -0
  36. package/src/adapters/shared/fs-server-response-factory.ts +96 -0
  37. package/src/adapters/shared/fs-server-response-matcher.test.ts +311 -0
  38. package/src/adapters/shared/fs-server-response-matcher.ts +240 -0
  39. package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
  40. package/src/adapters/shared/hmr-html-response.ts +52 -0
  41. package/src/adapters/shared/hmr-manager.contract.test.ts +228 -0
  42. package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
  43. package/src/adapters/shared/render-context.test.ts +150 -0
  44. package/src/adapters/shared/render-context.ts +123 -0
  45. package/src/adapters/shared/runtime-bootstrap.ts +79 -0
  46. package/src/adapters/shared/server-adapter.test.ts +130 -0
  47. package/src/adapters/shared/server-adapter.ts +562 -0
  48. package/src/adapters/shared/server-route-handler.test.ts +111 -0
  49. package/src/adapters/shared/server-route-handler.ts +153 -0
  50. package/src/adapters/shared/server-static-builder.test.ts +338 -0
  51. package/src/adapters/shared/server-static-builder.ts +170 -0
  52. package/src/build/build-adapter-serialization.test.ts +281 -0
  53. package/src/build/build-adapter.test.ts +1240 -0
  54. package/src/build/build-adapter.ts +1012 -0
  55. package/src/build/build-manifest.ts +54 -0
  56. package/src/build/build-types.ts +83 -0
  57. package/src/build/dev-build-coordinator.ts +220 -0
  58. package/src/build/esbuild-build-adapter.ts +660 -0
  59. package/src/build/runtime-build-executor.test.ts +81 -0
  60. package/src/build/runtime-build-executor.ts +40 -0
  61. package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
  62. package/src/build/runtime-specifier-alias-plugin.ts +62 -0
  63. package/src/build/runtime-specifier-aliases.ts +135 -0
  64. package/src/config/README.md +1 -1
  65. package/src/config/config-builder.test.ts +442 -0
  66. package/src/config/config-builder.ts +737 -0
  67. package/src/config/config-builder.typecheck.test.ts +96 -0
  68. package/src/config/{constants.d.ts → constants.ts} +22 -13
  69. package/src/dev/host-runtime.ts +34 -0
  70. package/src/dev/sc-server.ts +143 -0
  71. package/src/eco/eco.browser.test.ts +43 -0
  72. package/src/eco/eco.browser.ts +118 -0
  73. package/src/eco/eco.test.ts +654 -0
  74. package/src/eco/eco.ts +205 -0
  75. package/src/eco/eco.types.ts +221 -0
  76. package/src/eco/eco.utils.test.ts +219 -0
  77. package/src/eco/eco.utils.ts +5 -0
  78. package/src/eco/global-injector-map.test.ts +42 -0
  79. package/src/eco/global-injector-map.ts +112 -0
  80. package/src/eco/lazy-injector-map.test.ts +66 -0
  81. package/src/eco/lazy-injector-map.ts +120 -0
  82. package/src/eco/module-dependencies.test.ts +30 -0
  83. package/src/eco/module-dependencies.ts +75 -0
  84. package/src/errors/http-error.test.ts +134 -0
  85. package/src/errors/http-error.ts +72 -0
  86. package/src/errors/index.ts +3 -0
  87. package/src/errors/locals-access-error.ts +7 -0
  88. package/src/global/app-logger.ts +4 -0
  89. package/src/global/utils.test.ts +12 -0
  90. 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
  91. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  92. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  93. package/src/hmr/client/hmr-runtime.ts +162 -0
  94. package/src/hmr/hmr-strategy.test.ts +124 -0
  95. package/src/hmr/hmr-strategy.ts +177 -0
  96. package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
  97. package/src/hmr/hmr.test.e2e.ts +66 -0
  98. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  99. package/src/hmr/strategies/js-hmr-strategy.test.ts +334 -0
  100. package/src/hmr/strategies/js-hmr-strategy.ts +314 -0
  101. package/src/index.browser.ts +3 -0
  102. package/src/index.ts +15 -0
  103. package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
  104. package/src/integrations/ghtml/ghtml-renderer.ts +87 -0
  105. package/src/integrations/ghtml/ghtml.constants.ts +1 -0
  106. package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
  107. package/src/plugins/alias-resolver-plugin.test.ts +41 -0
  108. package/src/plugins/alias-resolver-plugin.ts +63 -0
  109. package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
  110. package/src/plugins/eco-component-meta-plugin.ts +494 -0
  111. package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
  112. package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
  113. package/src/plugins/integration-plugin.test.ts +151 -0
  114. package/src/plugins/integration-plugin.ts +323 -0
  115. package/src/plugins/processor.test.ts +148 -0
  116. package/src/plugins/processor.ts +257 -0
  117. package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
  118. package/src/plugins/source-transform.test.ts +82 -0
  119. package/src/plugins/source-transform.ts +123 -0
  120. package/src/route-renderer/GRAPH.md +81 -289
  121. package/src/route-renderer/README.md +67 -105
  122. package/src/route-renderer/orchestration/component-render-context.ts +325 -0
  123. package/src/route-renderer/orchestration/declared-ownership-graph.ts +62 -0
  124. package/src/route-renderer/orchestration/foreign-subtree-execution.service.ts +383 -0
  125. package/src/route-renderer/orchestration/integration-renderer.test.ts +2085 -0
  126. package/src/route-renderer/orchestration/integration-renderer.ts +1244 -0
  127. package/src/route-renderer/orchestration/ownership-planning.service.ts +97 -0
  128. package/src/route-renderer/orchestration/ownership-validation.service.ts +76 -0
  129. package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
  130. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.test.ts +324 -0
  131. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.ts +294 -0
  132. package/src/route-renderer/orchestration/render-output.utils.ts +310 -0
  133. package/src/route-renderer/orchestration/route-render-orchestrator.prepare-render-options.test.ts +644 -0
  134. package/src/route-renderer/orchestration/route-render-orchestrator.test.ts +265 -0
  135. package/src/route-renderer/orchestration/route-render-orchestrator.ts +592 -0
  136. package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
  137. package/src/route-renderer/orchestration/template-serialization.ts +117 -0
  138. package/src/route-renderer/page-loading/component-dependency-collection.ts +202 -0
  139. package/src/route-renderer/page-loading/declared-asset-collection.ts +153 -0
  140. package/src/route-renderer/page-loading/dependency-resolver.test.ts +761 -0
  141. package/src/route-renderer/page-loading/dependency-resolver.ts +144 -0
  142. package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
  143. package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
  144. package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
  145. package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
  146. package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
  147. package/src/route-renderer/page-loading/page-dependency-bundling.ts +244 -0
  148. package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
  149. package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
  150. package/src/route-renderer/route-renderer.ts +133 -0
  151. package/src/router/README.md +16 -19
  152. package/src/router/client/link-intent.test.browser.ts +51 -0
  153. package/src/router/client/link-intent.ts +92 -0
  154. package/src/router/client/navigation-coordinator.test.ts +237 -0
  155. package/src/router/client/navigation-coordinator.ts +453 -0
  156. package/src/router/server/route-registry.test.ts +176 -0
  157. package/src/router/server/route-registry.ts +382 -0
  158. package/src/services/README.md +1 -2
  159. package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
  160. package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +473 -0
  161. package/src/services/assets/asset-processing-service/asset-processing.service.ts +344 -0
  162. package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
  163. package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
  164. package/src/services/assets/asset-processing-service/assets.types.ts +128 -0
  165. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
  166. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
  167. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
  168. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
  169. package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
  170. package/src/services/assets/asset-processing-service/index.ts +6 -0
  171. package/src/services/assets/asset-processing-service/page-package.test.ts +100 -0
  172. package/src/services/assets/asset-processing-service/page-package.ts +93 -0
  173. package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
  174. package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
  175. package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
  176. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
  177. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +173 -0
  178. package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
  179. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +195 -0
  180. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +137 -0
  181. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
  182. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +116 -0
  183. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
  184. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +89 -0
  185. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
  186. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +72 -0
  187. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +83 -0
  188. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
  189. package/src/services/assets/browser-bundle.service.test.ts +66 -0
  190. package/src/services/assets/browser-bundle.service.ts +109 -0
  191. package/src/services/cache/cache.types.ts +126 -0
  192. package/src/services/cache/index.ts +18 -0
  193. package/src/services/cache/memory-cache-store.test.ts +225 -0
  194. package/src/services/cache/memory-cache-store.ts +130 -0
  195. package/src/services/cache/page-cache-service.test.ts +175 -0
  196. package/src/services/cache/page-cache-service.ts +202 -0
  197. package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
  198. package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
  199. package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
  200. package/src/services/html/html-rewriter-provider.service.ts +104 -0
  201. package/src/services/html/html-transformer.service.test.ts +476 -0
  202. package/src/services/html/html-transformer.service.ts +275 -0
  203. package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
  204. package/src/services/invalidation/development-invalidation.service.ts +262 -0
  205. package/src/services/module-loading/app-module-loader.service.ts +9 -0
  206. package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
  207. package/src/services/module-loading/app-server-module-transpiler.service.ts +141 -0
  208. package/src/services/module-loading/host-module-loader-registry.ts +15 -0
  209. package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
  210. package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
  211. package/src/services/module-loading/node-bootstrap-plugin.ts +311 -0
  212. package/src/services/module-loading/page-module-import.service.test.ts +504 -0
  213. package/src/services/module-loading/page-module-import.service.ts +251 -0
  214. package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
  215. package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
  216. package/src/services/module-loading/source-module-support.ts +19 -0
  217. package/src/services/runtime-state/dev-graph.service.ts +217 -0
  218. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
  219. package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
  220. package/src/services/validation/schema-validation-service.test.ts +223 -0
  221. package/src/services/validation/schema-validation-service.ts +204 -0
  222. package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
  223. package/src/static-site-generator/static-site-generator.test.ts +408 -0
  224. package/src/static-site-generator/static-site-generator.ts +445 -0
  225. package/src/types/internal-types.ts +243 -0
  226. package/src/types/public-types.ts +1459 -0
  227. package/src/utils/deep-merge.test.ts +114 -0
  228. package/src/utils/deep-merge.ts +47 -0
  229. package/src/utils/hash.ts +5 -0
  230. package/src/utils/html-escaping.ts +9 -0
  231. package/src/utils/invariant.test.ts +22 -0
  232. package/src/utils/invariant.ts +15 -0
  233. package/src/utils/locals-utils.ts +37 -0
  234. package/src/utils/parse-cli-args.test.ts +69 -0
  235. package/src/utils/parse-cli-args.ts +105 -0
  236. package/src/utils/path-utils.module.ts +14 -0
  237. package/src/utils/path-utils.test.ts +15 -0
  238. package/src/utils/resolve-work-dir.ts +45 -0
  239. package/src/utils/runtime.ts +44 -0
  240. package/src/utils/server-utils.module.ts +67 -0
  241. package/src/utils/server-utils.test.ts +38 -0
  242. package/src/watchers/project-watcher.integration.test.ts +337 -0
  243. package/src/watchers/project-watcher.test-helpers.ts +42 -0
  244. package/src/watchers/project-watcher.test.ts +768 -0
  245. package/src/watchers/project-watcher.ts +357 -0
  246. package/CHANGELOG.md +0 -66
  247. package/src/adapters/abstract/application-adapter.d.ts +0 -194
  248. package/src/adapters/abstract/application-adapter.js +0 -121
  249. package/src/adapters/abstract/router-adapter.d.ts +0 -26
  250. package/src/adapters/abstract/router-adapter.js +0 -5
  251. package/src/adapters/abstract/server-adapter.d.ts +0 -69
  252. package/src/adapters/abstract/server-adapter.js +0 -15
  253. package/src/adapters/bun/client-bridge.d.ts +0 -34
  254. package/src/adapters/bun/client-bridge.js +0 -48
  255. package/src/adapters/bun/create-app.d.ts +0 -52
  256. package/src/adapters/bun/create-app.js +0 -116
  257. package/src/adapters/bun/hmr-manager.d.ts +0 -143
  258. package/src/adapters/bun/hmr-manager.js +0 -333
  259. package/src/adapters/bun/index.d.ts +0 -2
  260. package/src/adapters/bun/index.js +0 -8
  261. package/src/adapters/bun/server-adapter.d.ts +0 -155
  262. package/src/adapters/bun/server-adapter.js +0 -374
  263. package/src/adapters/bun/server-lifecycle.d.ts +0 -63
  264. package/src/adapters/bun/server-lifecycle.js +0 -92
  265. package/src/adapters/create-app.d.ts +0 -20
  266. package/src/adapters/create-app.js +0 -66
  267. package/src/adapters/index.d.ts +0 -2
  268. package/src/adapters/index.js +0 -8
  269. package/src/adapters/node/create-app.d.ts +0 -18
  270. package/src/adapters/node/create-app.js +0 -149
  271. package/src/adapters/node/node-client-bridge.d.ts +0 -26
  272. package/src/adapters/node/node-client-bridge.js +0 -66
  273. package/src/adapters/node/node-hmr-manager.d.ts +0 -133
  274. package/src/adapters/node/node-hmr-manager.js +0 -311
  275. package/src/adapters/node/server-adapter.d.ts +0 -162
  276. package/src/adapters/node/server-adapter.js +0 -368
  277. package/src/adapters/node/static-content-server.d.ts +0 -60
  278. package/src/adapters/node/static-content-server.js +0 -194
  279. package/src/adapters/shared/api-response.d.ts +0 -52
  280. package/src/adapters/shared/api-response.js +0 -96
  281. package/src/adapters/shared/application-adapter.d.ts +0 -18
  282. package/src/adapters/shared/application-adapter.js +0 -90
  283. package/src/adapters/shared/define-api-handler.d.ts +0 -25
  284. package/src/adapters/shared/define-api-handler.js +0 -15
  285. package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
  286. package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
  287. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
  288. package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
  289. package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
  290. package/src/adapters/shared/fs-server-response-factory.js +0 -97
  291. package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
  292. package/src/adapters/shared/fs-server-response-matcher.js +0 -147
  293. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
  294. package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
  295. package/src/adapters/shared/hmr-html-response.d.ts +0 -22
  296. package/src/adapters/shared/hmr-html-response.js +0 -32
  297. package/src/adapters/shared/render-context.d.ts +0 -15
  298. package/src/adapters/shared/render-context.js +0 -72
  299. package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
  300. package/src/adapters/shared/runtime-bootstrap.js +0 -43
  301. package/src/adapters/shared/server-adapter.d.ts +0 -97
  302. package/src/adapters/shared/server-adapter.js +0 -390
  303. package/src/adapters/shared/server-route-handler.d.ts +0 -89
  304. package/src/adapters/shared/server-route-handler.js +0 -111
  305. package/src/adapters/shared/server-static-builder.d.ts +0 -71
  306. package/src/adapters/shared/server-static-builder.js +0 -100
  307. package/src/build/build-adapter.d.ts +0 -239
  308. package/src/build/build-adapter.js +0 -642
  309. package/src/build/build-manifest.d.ts +0 -27
  310. package/src/build/build-manifest.js +0 -30
  311. package/src/build/build-types.d.ts +0 -57
  312. package/src/build/build-types.js +0 -0
  313. package/src/build/dev-build-coordinator.d.ts +0 -72
  314. package/src/build/dev-build-coordinator.js +0 -154
  315. package/src/build/esbuild-build-adapter.d.ts +0 -78
  316. package/src/build/esbuild-build-adapter.js +0 -505
  317. package/src/build/runtime-build-executor.d.ts +0 -14
  318. package/src/build/runtime-build-executor.js +0 -22
  319. package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
  320. package/src/build/runtime-specifier-alias-plugin.js +0 -35
  321. package/src/build/runtime-specifier-aliases.d.ts +0 -5
  322. package/src/build/runtime-specifier-aliases.js +0 -95
  323. package/src/config/config-builder.d.ts +0 -252
  324. package/src/config/config-builder.js +0 -603
  325. package/src/config/constants.js +0 -25
  326. package/src/dev/sc-server.d.ts +0 -30
  327. package/src/dev/sc-server.js +0 -111
  328. package/src/eco/eco.browser.d.ts +0 -2
  329. package/src/eco/eco.browser.js +0 -83
  330. package/src/eco/eco.d.ts +0 -9
  331. package/src/eco/eco.js +0 -85
  332. package/src/eco/eco.types.d.ts +0 -178
  333. package/src/eco/eco.types.js +0 -0
  334. package/src/eco/eco.utils.d.ts +0 -1
  335. package/src/eco/eco.utils.js +0 -10
  336. package/src/eco/global-injector-map.d.ts +0 -16
  337. package/src/eco/global-injector-map.js +0 -80
  338. package/src/eco/lazy-injector-map.d.ts +0 -8
  339. package/src/eco/lazy-injector-map.js +0 -70
  340. package/src/eco/module-dependencies.d.ts +0 -18
  341. package/src/eco/module-dependencies.js +0 -49
  342. package/src/errors/http-error.d.ts +0 -31
  343. package/src/errors/http-error.js +0 -50
  344. package/src/errors/index.d.ts +0 -2
  345. package/src/errors/index.js +0 -4
  346. package/src/errors/locals-access-error.d.ts +0 -4
  347. package/src/errors/locals-access-error.js +0 -9
  348. package/src/global/app-logger.d.ts +0 -2
  349. package/src/global/app-logger.js +0 -6
  350. package/src/hmr/client/hmr-runtime.d.ts +0 -5
  351. package/src/hmr/client/hmr-runtime.js +0 -117
  352. package/src/hmr/hmr-strategy.d.ts +0 -162
  353. package/src/hmr/hmr-strategy.js +0 -44
  354. package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
  355. package/src/hmr/hmr.postcss.test.e2e.js +0 -31
  356. package/src/hmr/hmr.test.e2e.d.ts +0 -1
  357. package/src/hmr/hmr.test.e2e.js +0 -43
  358. package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
  359. package/src/hmr/strategies/default-hmr-strategy.js +0 -34
  360. package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
  361. package/src/hmr/strategies/js-hmr-strategy.js +0 -178
  362. package/src/index.browser.d.ts +0 -3
  363. package/src/index.browser.js +0 -4
  364. package/src/index.d.ts +0 -6
  365. package/src/index.js +0 -21
  366. package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
  367. package/src/integrations/ghtml/ghtml-renderer.js +0 -63
  368. package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
  369. package/src/integrations/ghtml/ghtml.constants.js +0 -4
  370. package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
  371. package/src/integrations/ghtml/ghtml.plugin.js +0 -20
  372. package/src/plugins/alias-resolver-plugin.d.ts +0 -2
  373. package/src/plugins/alias-resolver-plugin.js +0 -53
  374. package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
  375. package/src/plugins/eco-component-meta-plugin.js +0 -163
  376. package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
  377. package/src/plugins/foreign-jsx-override-plugin.js +0 -35
  378. package/src/plugins/integration-plugin.d.ts +0 -219
  379. package/src/plugins/integration-plugin.js +0 -196
  380. package/src/plugins/processor.d.ts +0 -95
  381. package/src/plugins/processor.js +0 -136
  382. package/src/plugins/runtime-capability.js +0 -0
  383. package/src/plugins/source-transform.d.ts +0 -46
  384. package/src/plugins/source-transform.js +0 -71
  385. package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
  386. package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
  387. package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
  388. package/src/route-renderer/orchestration/component-render-context.js +0 -147
  389. package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -556
  390. package/src/route-renderer/orchestration/integration-renderer.js +0 -932
  391. package/src/route-renderer/orchestration/page-packaging.service.d.ts +0 -16
  392. package/src/route-renderer/orchestration/page-packaging.service.js +0 -66
  393. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +0 -2
  394. package/src/route-renderer/orchestration/processed-asset-dedupe.js +0 -23
  395. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
  396. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
  397. package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
  398. package/src/route-renderer/orchestration/render-execution.service.js +0 -106
  399. package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -66
  400. package/src/route-renderer/orchestration/render-output.utils.js +0 -171
  401. package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
  402. package/src/route-renderer/orchestration/render-preparation.service.js +0 -364
  403. package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
  404. package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
  405. package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
  406. package/src/route-renderer/orchestration/template-serialization.js +0 -45
  407. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +0 -37
  408. package/src/route-renderer/page-loading/component-dependency-collection.js +0 -125
  409. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +0 -24
  410. package/src/route-renderer/page-loading/declared-asset-collection.js +0 -106
  411. package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
  412. package/src/route-renderer/page-loading/dependency-resolver.js +0 -117
  413. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +0 -11
  414. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +0 -57
  415. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +0 -45
  416. package/src/route-renderer/page-loading/lazy-entry-collection.js +0 -105
  417. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +0 -19
  418. package/src/route-renderer/page-loading/lazy-trigger-planning.js +0 -40
  419. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +0 -5
  420. package/src/route-renderer/page-loading/module-declaration-aggregation.js +0 -33
  421. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +0 -3
  422. package/src/route-renderer/page-loading/module-declaration-scripts.js +0 -18
  423. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +0 -13
  424. package/src/route-renderer/page-loading/page-dependency-bundling.js +0 -115
  425. package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
  426. package/src/route-renderer/page-loading/page-module-loader.js +0 -127
  427. package/src/route-renderer/route-renderer.d.ts +0 -67
  428. package/src/route-renderer/route-renderer.js +0 -103
  429. package/src/router/client/link-intent.js +0 -34
  430. package/src/router/client/link-intent.test.browser.d.ts +0 -1
  431. package/src/router/client/link-intent.test.browser.js +0 -43
  432. package/src/router/client/navigation-coordinator.d.ts +0 -169
  433. package/src/router/client/navigation-coordinator.js +0 -215
  434. package/src/router/server/fs-router-scanner.d.ts +0 -41
  435. package/src/router/server/fs-router-scanner.js +0 -161
  436. package/src/router/server/fs-router.d.ts +0 -26
  437. package/src/router/server/fs-router.js +0 -100
  438. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +0 -3
  439. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +0 -56
  440. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -103
  441. package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -285
  442. package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
  443. package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
  444. package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -100
  445. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  446. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +0 -55
  447. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -49
  448. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
  449. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
  450. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +0 -30
  451. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +0 -65
  452. package/src/services/assets/asset-processing-service/index.d.ts +0 -5
  453. package/src/services/assets/asset-processing-service/index.js +0 -5
  454. package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
  455. package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
  456. package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
  457. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
  458. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -65
  459. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -22
  460. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -136
  461. package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
  462. package/src/services/assets/asset-processing-service/processors/index.js +0 -5
  463. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -6
  464. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -116
  465. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
  466. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -91
  467. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  468. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -77
  469. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -8
  470. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -58
  471. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
  472. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -67
  473. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +0 -18
  474. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +0 -45
  475. package/src/services/assets/browser-bundle.service.d.ts +0 -73
  476. package/src/services/assets/browser-bundle.service.js +0 -41
  477. package/src/services/cache/cache.types.d.ts +0 -107
  478. package/src/services/cache/cache.types.js +0 -0
  479. package/src/services/cache/index.d.ts +0 -7
  480. package/src/services/cache/index.js +0 -7
  481. package/src/services/cache/memory-cache-store.d.ts +0 -42
  482. package/src/services/cache/memory-cache-store.js +0 -98
  483. package/src/services/cache/page-cache-service.d.ts +0 -70
  484. package/src/services/cache/page-cache-service.js +0 -152
  485. package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
  486. package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
  487. package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
  488. package/src/services/html/html-rewriter-provider.service.js +0 -68
  489. package/src/services/html/html-transformer.service.d.ts +0 -87
  490. package/src/services/html/html-transformer.service.js +0 -216
  491. package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
  492. package/src/services/invalidation/development-invalidation.service.js +0 -190
  493. package/src/services/module-loading/app-module-loader.service.d.ts +0 -7
  494. package/src/services/module-loading/app-module-loader.service.js +0 -0
  495. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
  496. package/src/services/module-loading/app-server-module-transpiler.service.js +0 -115
  497. package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
  498. package/src/services/module-loading/host-module-loader-registry.js +0 -15
  499. package/src/services/module-loading/module-loading-types.js +0 -0
  500. package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
  501. package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
  502. package/src/services/module-loading/page-module-import.service.d.ts +0 -76
  503. package/src/services/module-loading/page-module-import.service.js +0 -170
  504. package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -63
  505. package/src/services/module-loading/server-module-transpiler.service.js +0 -64
  506. package/src/services/module-loading/source-module-support.d.ts +0 -5
  507. package/src/services/module-loading/source-module-support.js +0 -8
  508. package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
  509. package/src/services/runtime-state/dev-graph.service.js +0 -162
  510. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
  511. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
  512. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
  513. package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
  514. package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
  515. package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
  516. package/src/services/validation/schema-validation-service.d.ts +0 -122
  517. package/src/services/validation/schema-validation-service.js +0 -101
  518. package/src/services/validation/standard-schema.types.js +0 -0
  519. package/src/static-site-generator/static-site-generator.d.ts +0 -105
  520. package/src/static-site-generator/static-site-generator.js +0 -349
  521. package/src/types/internal-types.d.ts +0 -231
  522. package/src/types/internal-types.js +0 -0
  523. package/src/types/public-types.d.ts +0 -1257
  524. package/src/types/public-types.js +0 -0
  525. package/src/utils/deep-merge.d.ts +0 -14
  526. package/src/utils/deep-merge.js +0 -32
  527. package/src/utils/hash.d.ts +0 -1
  528. package/src/utils/hash.js +0 -7
  529. package/src/utils/html-escaping.d.ts +0 -7
  530. package/src/utils/html-escaping.js +0 -6
  531. package/src/utils/html.js +0 -4
  532. package/src/utils/invariant.d.ts +0 -5
  533. package/src/utils/invariant.js +0 -11
  534. package/src/utils/locals-utils.d.ts +0 -15
  535. package/src/utils/locals-utils.js +0 -24
  536. package/src/utils/parse-cli-args.d.ts +0 -27
  537. package/src/utils/parse-cli-args.js +0 -62
  538. package/src/utils/path-utils.module.d.ts +0 -5
  539. package/src/utils/path-utils.module.js +0 -14
  540. package/src/utils/resolve-work-dir.d.ts +0 -11
  541. package/src/utils/resolve-work-dir.js +0 -31
  542. package/src/utils/runtime.d.ts +0 -11
  543. package/src/utils/runtime.js +0 -40
  544. package/src/utils/server-utils.module.d.ts +0 -19
  545. package/src/utils/server-utils.module.js +0 -56
  546. package/src/watchers/project-watcher.d.ts +0 -136
  547. package/src/watchers/project-watcher.js +0 -275
  548. package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
  549. package/src/watchers/project-watcher.test-helpers.js +0 -52
  550. /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
+ });