@ecopages/core 0.2.0-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (342) hide show
  1. package/CHANGELOG.md +89 -0
  2. package/LICENSE +21 -0
  3. package/README.md +32 -0
  4. package/package.json +279 -0
  5. package/src/adapters/abstract/application-adapter.d.ts +168 -0
  6. package/src/adapters/abstract/application-adapter.js +109 -0
  7. package/src/adapters/abstract/application-adapter.ts +337 -0
  8. package/src/adapters/abstract/router-adapter.d.ts +26 -0
  9. package/src/adapters/abstract/router-adapter.js +5 -0
  10. package/src/adapters/abstract/router-adapter.ts +30 -0
  11. package/src/adapters/abstract/server-adapter.d.ts +69 -0
  12. package/src/adapters/abstract/server-adapter.js +15 -0
  13. package/src/adapters/abstract/server-adapter.ts +79 -0
  14. package/src/adapters/bun/client-bridge.d.ts +34 -0
  15. package/src/adapters/bun/client-bridge.js +48 -0
  16. package/src/adapters/bun/client-bridge.ts +62 -0
  17. package/src/adapters/bun/create-app.d.ts +60 -0
  18. package/src/adapters/bun/create-app.js +117 -0
  19. package/src/adapters/bun/create-app.ts +189 -0
  20. package/src/adapters/bun/define-api-handler.d.ts +61 -0
  21. package/src/adapters/bun/define-api-handler.js +15 -0
  22. package/src/adapters/bun/define-api-handler.ts +114 -0
  23. package/src/adapters/bun/hmr-manager.d.ts +84 -0
  24. package/src/adapters/bun/hmr-manager.js +227 -0
  25. package/src/adapters/bun/hmr-manager.ts +281 -0
  26. package/src/adapters/bun/index.d.ts +3 -0
  27. package/src/adapters/bun/index.js +8 -0
  28. package/src/adapters/bun/index.ts +3 -0
  29. package/src/adapters/bun/server-adapter.d.ts +155 -0
  30. package/src/adapters/bun/server-adapter.js +368 -0
  31. package/src/adapters/bun/server-adapter.ts +492 -0
  32. package/src/adapters/bun/server-lifecycle.d.ts +52 -0
  33. package/src/adapters/bun/server-lifecycle.js +120 -0
  34. package/src/adapters/bun/server-lifecycle.ts +154 -0
  35. package/src/adapters/index.d.ts +6 -0
  36. package/src/adapters/index.js +14 -0
  37. package/src/adapters/index.ts +6 -0
  38. package/src/adapters/node/create-app.d.ts +21 -0
  39. package/src/adapters/node/create-app.js +143 -0
  40. package/src/adapters/node/create-app.ts +179 -0
  41. package/src/adapters/node/index.d.ts +4 -0
  42. package/src/adapters/node/index.js +8 -0
  43. package/src/adapters/node/index.ts +9 -0
  44. package/src/adapters/node/node-client-bridge.d.ts +26 -0
  45. package/src/adapters/node/node-client-bridge.js +66 -0
  46. package/src/adapters/node/node-client-bridge.ts +79 -0
  47. package/src/adapters/node/node-hmr-manager.d.ts +62 -0
  48. package/src/adapters/node/node-hmr-manager.js +221 -0
  49. package/src/adapters/node/node-hmr-manager.ts +271 -0
  50. package/src/adapters/node/server-adapter.d.ts +190 -0
  51. package/src/adapters/node/server-adapter.js +420 -0
  52. package/src/adapters/node/server-adapter.ts +561 -0
  53. package/src/adapters/node/static-content-server.d.ts +24 -0
  54. package/src/adapters/node/static-content-server.js +166 -0
  55. package/src/adapters/node/static-content-server.ts +203 -0
  56. package/src/adapters/shared/api-response.d.ts +52 -0
  57. package/src/adapters/shared/api-response.js +96 -0
  58. package/src/adapters/shared/api-response.ts +104 -0
  59. package/src/adapters/shared/application-adapter.d.ts +18 -0
  60. package/src/adapters/shared/application-adapter.js +90 -0
  61. package/src/adapters/shared/application-adapter.ts +199 -0
  62. package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
  63. package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
  64. package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
  65. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
  66. package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
  67. package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
  68. package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
  69. package/src/adapters/shared/fs-server-response-factory.js +97 -0
  70. package/src/adapters/shared/fs-server-response-factory.ts +118 -0
  71. package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
  72. package/src/adapters/shared/fs-server-response-matcher.js +155 -0
  73. package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
  74. package/src/adapters/shared/render-context.d.ts +14 -0
  75. package/src/adapters/shared/render-context.js +69 -0
  76. package/src/adapters/shared/render-context.ts +105 -0
  77. package/src/adapters/shared/server-adapter.d.ts +87 -0
  78. package/src/adapters/shared/server-adapter.js +353 -0
  79. package/src/adapters/shared/server-adapter.ts +442 -0
  80. package/src/adapters/shared/server-route-handler.d.ts +89 -0
  81. package/src/adapters/shared/server-route-handler.js +120 -0
  82. package/src/adapters/shared/server-route-handler.ts +166 -0
  83. package/src/adapters/shared/server-static-builder.d.ts +38 -0
  84. package/src/adapters/shared/server-static-builder.js +46 -0
  85. package/src/adapters/shared/server-static-builder.ts +82 -0
  86. package/src/build/build-adapter.d.ts +74 -0
  87. package/src/build/build-adapter.js +54 -0
  88. package/src/build/build-adapter.ts +132 -0
  89. package/src/build/build-types.d.ts +57 -0
  90. package/src/build/build-types.js +0 -0
  91. package/src/build/build-types.ts +83 -0
  92. package/src/build/esbuild-build-adapter.d.ts +69 -0
  93. package/src/build/esbuild-build-adapter.js +390 -0
  94. package/src/build/esbuild-build-adapter.ts +510 -0
  95. package/src/config/config-builder.d.ts +227 -0
  96. package/src/config/config-builder.js +392 -0
  97. package/src/config/config-builder.ts +474 -0
  98. package/src/constants.d.ts +32 -0
  99. package/src/constants.js +21 -0
  100. package/src/constants.ts +39 -0
  101. package/src/create-app.d.ts +17 -0
  102. package/src/create-app.js +66 -0
  103. package/src/create-app.ts +87 -0
  104. package/src/declarations.d.ts +26 -0
  105. package/src/define-api-handler.d.ts +25 -0
  106. package/src/define-api-handler.js +15 -0
  107. package/src/define-api-handler.ts +66 -0
  108. package/src/dev/sc-server.d.ts +30 -0
  109. package/src/dev/sc-server.js +111 -0
  110. package/src/dev/sc-server.ts +143 -0
  111. package/src/eco/README.md +636 -0
  112. package/src/eco/component-render-context.d.ts +105 -0
  113. package/src/eco/component-render-context.js +77 -0
  114. package/src/eco/component-render-context.ts +202 -0
  115. package/src/eco/eco.d.ts +9 -0
  116. package/src/eco/eco.js +110 -0
  117. package/src/eco/eco.ts +221 -0
  118. package/src/eco/eco.types.d.ts +170 -0
  119. package/src/eco/eco.types.js +0 -0
  120. package/src/eco/eco.types.ts +202 -0
  121. package/src/eco/eco.utils.d.ts +40 -0
  122. package/src/eco/eco.utils.js +40 -0
  123. package/src/eco/eco.utils.ts +89 -0
  124. package/src/eco/global-injector-map.d.ts +16 -0
  125. package/src/eco/global-injector-map.js +80 -0
  126. package/src/eco/global-injector-map.ts +112 -0
  127. package/src/eco/lazy-injector-map.d.ts +8 -0
  128. package/src/eco/lazy-injector-map.js +70 -0
  129. package/src/eco/lazy-injector-map.ts +120 -0
  130. package/src/eco/module-dependencies.d.ts +18 -0
  131. package/src/eco/module-dependencies.js +49 -0
  132. package/src/eco/module-dependencies.ts +75 -0
  133. package/src/env.d.ts +20 -0
  134. package/src/errors/http-error.d.ts +31 -0
  135. package/src/errors/http-error.js +50 -0
  136. package/src/errors/http-error.ts +72 -0
  137. package/src/errors/index.d.ts +2 -0
  138. package/src/errors/index.js +4 -0
  139. package/src/errors/index.ts +2 -0
  140. package/src/errors/locals-access-error.d.ts +4 -0
  141. package/src/errors/locals-access-error.js +9 -0
  142. package/src/errors/locals-access-error.ts +7 -0
  143. package/src/global/app-logger.d.ts +2 -0
  144. package/src/global/app-logger.js +6 -0
  145. package/src/global/app-logger.ts +4 -0
  146. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
  147. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  148. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  149. package/src/hmr/client/hmr-runtime.d.ts +10 -0
  150. package/src/hmr/client/hmr-runtime.js +86 -0
  151. package/src/hmr/client/hmr-runtime.ts +121 -0
  152. package/src/hmr/hmr-strategy.d.ts +159 -0
  153. package/src/hmr/hmr-strategy.js +29 -0
  154. package/src/hmr/hmr-strategy.ts +172 -0
  155. package/src/hmr/hmr.test.e2e.d.ts +1 -0
  156. package/src/hmr/hmr.test.e2e.js +50 -0
  157. package/src/hmr/hmr.test.e2e.ts +75 -0
  158. package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
  159. package/src/hmr/strategies/default-hmr-strategy.js +34 -0
  160. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  161. package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
  162. package/src/hmr/strategies/js-hmr-strategy.js +179 -0
  163. package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
  164. package/src/index.browser.d.ts +3 -0
  165. package/src/index.browser.js +4 -0
  166. package/src/index.browser.ts +3 -0
  167. package/src/index.d.ts +5 -0
  168. package/src/index.js +10 -0
  169. package/src/index.ts +5 -0
  170. package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
  171. package/src/integrations/ghtml/ghtml-renderer.js +60 -0
  172. package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
  173. package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
  174. package/src/integrations/ghtml/ghtml.plugin.js +21 -0
  175. package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
  176. package/src/internal-types.d.ts +200 -0
  177. package/src/internal-types.js +0 -0
  178. package/src/internal-types.ts +212 -0
  179. package/src/plugins/alias-resolver-plugin.d.ts +2 -0
  180. package/src/plugins/alias-resolver-plugin.js +39 -0
  181. package/src/plugins/alias-resolver-plugin.ts +45 -0
  182. package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
  183. package/src/plugins/eco-component-meta-plugin.js +157 -0
  184. package/src/plugins/eco-component-meta-plugin.ts +474 -0
  185. package/src/plugins/integration-plugin.d.ts +102 -0
  186. package/src/plugins/integration-plugin.js +100 -0
  187. package/src/plugins/integration-plugin.ts +184 -0
  188. package/src/plugins/processor.d.ts +82 -0
  189. package/src/plugins/processor.js +122 -0
  190. package/src/plugins/processor.ts +220 -0
  191. package/src/public-types.d.ts +1094 -0
  192. package/src/public-types.js +0 -0
  193. package/src/public-types.ts +1255 -0
  194. package/src/route-renderer/GRAPH.md +387 -0
  195. package/src/route-renderer/README.md +135 -0
  196. package/src/route-renderer/component-graph-executor.d.ts +32 -0
  197. package/src/route-renderer/component-graph-executor.js +31 -0
  198. package/src/route-renderer/component-graph-executor.ts +84 -0
  199. package/src/route-renderer/component-graph.d.ts +42 -0
  200. package/src/route-renderer/component-graph.js +72 -0
  201. package/src/route-renderer/component-graph.ts +159 -0
  202. package/src/route-renderer/component-marker.d.ts +52 -0
  203. package/src/route-renderer/component-marker.js +46 -0
  204. package/src/route-renderer/component-marker.ts +117 -0
  205. package/src/route-renderer/dependency-resolver.d.ts +24 -0
  206. package/src/route-renderer/dependency-resolver.js +428 -0
  207. package/src/route-renderer/dependency-resolver.ts +596 -0
  208. package/src/route-renderer/html-post-processing.service.d.ts +40 -0
  209. package/src/route-renderer/html-post-processing.service.js +86 -0
  210. package/src/route-renderer/html-post-processing.service.ts +103 -0
  211. package/src/route-renderer/integration-renderer.d.ts +339 -0
  212. package/src/route-renderer/integration-renderer.js +526 -0
  213. package/src/route-renderer/integration-renderer.ts +696 -0
  214. package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
  215. package/src/route-renderer/marker-graph-resolver.js +93 -0
  216. package/src/route-renderer/marker-graph-resolver.ts +153 -0
  217. package/src/route-renderer/page-module-loader.d.ts +61 -0
  218. package/src/route-renderer/page-module-loader.js +102 -0
  219. package/src/route-renderer/page-module-loader.ts +153 -0
  220. package/src/route-renderer/render-execution.service.d.ts +69 -0
  221. package/src/route-renderer/render-execution.service.js +91 -0
  222. package/src/route-renderer/render-execution.service.ts +158 -0
  223. package/src/route-renderer/render-preparation.service.d.ts +112 -0
  224. package/src/route-renderer/render-preparation.service.js +243 -0
  225. package/src/route-renderer/render-preparation.service.ts +358 -0
  226. package/src/route-renderer/route-renderer.d.ts +26 -0
  227. package/src/route-renderer/route-renderer.js +68 -0
  228. package/src/route-renderer/route-renderer.ts +80 -0
  229. package/src/router/fs-router-scanner.d.ts +41 -0
  230. package/src/router/fs-router-scanner.js +155 -0
  231. package/src/router/fs-router-scanner.ts +217 -0
  232. package/src/router/fs-router.d.ts +26 -0
  233. package/src/router/fs-router.js +100 -0
  234. package/src/router/fs-router.ts +122 -0
  235. package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
  236. package/src/services/asset-processing-service/asset-processing.service.js +250 -0
  237. package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
  238. package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
  239. package/src/services/asset-processing-service/asset.factory.js +82 -0
  240. package/src/services/asset-processing-service/asset.factory.ts +105 -0
  241. package/src/services/asset-processing-service/assets.types.d.ts +88 -0
  242. package/src/services/asset-processing-service/assets.types.js +0 -0
  243. package/src/services/asset-processing-service/assets.types.ts +112 -0
  244. package/src/services/asset-processing-service/index.d.ts +3 -0
  245. package/src/services/asset-processing-service/index.js +3 -0
  246. package/src/services/asset-processing-service/index.ts +3 -0
  247. package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
  248. package/src/services/asset-processing-service/processor.interface.js +6 -0
  249. package/src/services/asset-processing-service/processor.interface.ts +27 -0
  250. package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
  251. package/src/services/asset-processing-service/processor.registry.js +15 -0
  252. package/src/services/asset-processing-service/processor.registry.ts +18 -0
  253. package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
  254. package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
  255. package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
  256. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
  257. package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
  258. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
  259. package/src/services/asset-processing-service/processors/index.d.ts +5 -0
  260. package/src/services/asset-processing-service/processors/index.js +5 -0
  261. package/src/services/asset-processing-service/processors/index.ts +5 -0
  262. package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
  263. package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
  264. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
  265. package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
  266. package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
  267. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
  268. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
  269. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
  270. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
  271. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
  272. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
  273. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
  274. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
  275. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
  276. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
  277. package/src/services/cache/cache.types.d.ts +107 -0
  278. package/src/services/cache/cache.types.js +0 -0
  279. package/src/services/cache/cache.types.ts +126 -0
  280. package/src/services/cache/index.d.ts +7 -0
  281. package/src/services/cache/index.js +7 -0
  282. package/src/services/cache/index.ts +18 -0
  283. package/src/services/cache/memory-cache-store.d.ts +42 -0
  284. package/src/services/cache/memory-cache-store.js +98 -0
  285. package/src/services/cache/memory-cache-store.ts +130 -0
  286. package/src/services/cache/page-cache-service.d.ts +70 -0
  287. package/src/services/cache/page-cache-service.js +152 -0
  288. package/src/services/cache/page-cache-service.ts +202 -0
  289. package/src/services/html-transformer.service.d.ts +50 -0
  290. package/src/services/html-transformer.service.js +163 -0
  291. package/src/services/html-transformer.service.ts +217 -0
  292. package/src/services/page-module-import.service.d.ts +37 -0
  293. package/src/services/page-module-import.service.js +88 -0
  294. package/src/services/page-module-import.service.ts +129 -0
  295. package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
  296. package/src/services/page-request-cache-coordinator.service.js +107 -0
  297. package/src/services/page-request-cache-coordinator.service.ts +128 -0
  298. package/src/services/schema-validation-service.d.ts +122 -0
  299. package/src/services/schema-validation-service.js +101 -0
  300. package/src/services/schema-validation-service.ts +204 -0
  301. package/src/services/validation/standard-schema.types.d.ts +65 -0
  302. package/src/services/validation/standard-schema.types.js +0 -0
  303. package/src/services/validation/standard-schema.types.ts +68 -0
  304. package/src/static-site-generator/static-site-generator.d.ts +57 -0
  305. package/src/static-site-generator/static-site-generator.js +272 -0
  306. package/src/static-site-generator/static-site-generator.ts +359 -0
  307. package/src/utils/css.d.ts +1 -0
  308. package/src/utils/css.js +7 -0
  309. package/src/utils/css.ts +5 -0
  310. package/src/utils/deep-merge.d.ts +14 -0
  311. package/src/utils/deep-merge.js +32 -0
  312. package/src/utils/deep-merge.ts +47 -0
  313. package/src/utils/hash.d.ts +1 -0
  314. package/src/utils/hash.js +7 -0
  315. package/src/utils/hash.ts +5 -0
  316. package/src/utils/html.d.ts +1 -0
  317. package/src/utils/html.js +4 -0
  318. package/src/utils/html.ts +1 -0
  319. package/src/utils/invariant.d.ts +5 -0
  320. package/src/utils/invariant.js +11 -0
  321. package/src/utils/invariant.ts +15 -0
  322. package/src/utils/locals-utils.d.ts +15 -0
  323. package/src/utils/locals-utils.js +24 -0
  324. package/src/utils/locals-utils.ts +37 -0
  325. package/src/utils/parse-cli-args.d.ts +24 -0
  326. package/src/utils/parse-cli-args.js +47 -0
  327. package/src/utils/parse-cli-args.ts +83 -0
  328. package/src/utils/path-utils.module.d.ts +5 -0
  329. package/src/utils/path-utils.module.js +14 -0
  330. package/src/utils/path-utils.module.ts +14 -0
  331. package/src/utils/runtime.d.ts +11 -0
  332. package/src/utils/runtime.js +40 -0
  333. package/src/utils/runtime.ts +44 -0
  334. package/src/utils/server-utils.module.d.ts +19 -0
  335. package/src/utils/server-utils.module.js +56 -0
  336. package/src/utils/server-utils.module.ts +67 -0
  337. package/src/watchers/project-watcher.d.ts +120 -0
  338. package/src/watchers/project-watcher.js +238 -0
  339. package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
  340. package/src/watchers/project-watcher.test-helpers.js +51 -0
  341. package/src/watchers/project-watcher.test-helpers.ts +40 -0
  342. package/src/watchers/project-watcher.ts +306 -0
@@ -0,0 +1,67 @@
1
+ const ContentTypeMap = new Map<string, string>([
2
+ ['jpg', 'image/jpeg'],
3
+ ['jpeg', 'image/jpeg'],
4
+ ['png', 'image/png'],
5
+ ['gif', 'image/gif'],
6
+ ['bmp', 'image/bmp'],
7
+ ['svg', 'image/svg+xml'],
8
+ ['tiff', 'image/tiff'],
9
+ ['webp', 'image/webp'],
10
+ ['avif', 'image/avif'],
11
+ ['ico', 'image/x-icon'],
12
+ ['mp3', 'audio/mpeg'],
13
+ ['ogg', 'audio/ogg'],
14
+ ['wav', 'audio/wav'],
15
+ ['mp4', 'video/mp4'],
16
+ ['webm', 'video/webm'],
17
+ ['ogv', 'video/ogg'],
18
+ ['mov', 'video/quicktime'],
19
+ ['txt', 'text/plain'],
20
+ ['html', 'text/html'],
21
+ ['css', 'text/css'],
22
+ ['js', 'text/javascript'],
23
+ ['mjs', 'text/javascript'],
24
+ ['json', 'application/json'],
25
+ ['map', 'application/json'],
26
+ ['xml', 'application/xml'],
27
+ ['webmanifest', 'application/manifest+json'],
28
+ ['wasm', 'application/wasm'],
29
+ ['csv', 'text/csv'],
30
+ ['ttf', 'font/ttf'],
31
+ ['woff', 'font/woff'],
32
+ ['woff2', 'font/woff2'],
33
+ ['otf', 'font/otf'],
34
+ ['eot', 'application/vnd.ms-fontobject'],
35
+ ['gz', 'application/x-gzip'],
36
+ ['zip', 'application/zip'],
37
+ ['pdf', 'application/pdf'],
38
+ ['doc', 'application/msword'],
39
+ ]);
40
+
41
+ /**
42
+ * Get the content type of a file based on its extension.
43
+ * @param file - The file name.
44
+ * @returns The content type.
45
+ */
46
+ export const getContentType = (file: string): string => {
47
+ const extension = file.split('.').pop() || 'txt';
48
+ return ContentTypeMap.get(extension) || 'text/plain';
49
+ };
50
+
51
+ /**
52
+ * Check if a file path has a known static file extension.
53
+ * @param file - The file name or path.
54
+ * @returns true if the extension is recognized.
55
+ */
56
+ export const hasKnownExtension = (file: string): boolean => {
57
+ const extension = file.split('.').pop();
58
+ return extension !== undefined && ContentTypeMap.has(extension);
59
+ };
60
+
61
+ /**
62
+ * A module for server utilities.
63
+ */
64
+ export const ServerUtils = {
65
+ getContentType,
66
+ hasKnownExtension,
67
+ };
@@ -0,0 +1,120 @@
1
+ import { type FSWatcher } from 'chokidar';
2
+ import type { EcoPagesAppConfig, IHmrManager, IClientBridge } from '../internal-types.js';
3
+ /**
4
+ * Configuration options for the ProjectWatcher
5
+ * @interface ProjectWatcherConfig
6
+ * @property {EcoPagesAppConfig} config - The application configuration
7
+ * @property {() => void} refreshRouterRoutesCallback - Callback to refresh router routes
8
+ * @property {IHmrManager} hmrManager - The HMR manager instance
9
+ * @property {ClientBridge} bridge - The client bridge instance
10
+ */
11
+ export interface ProjectWatcherConfig {
12
+ config: EcoPagesAppConfig;
13
+ refreshRouterRoutesCallback: () => void;
14
+ hmrManager: IHmrManager;
15
+ bridge: IClientBridge;
16
+ }
17
+ /**
18
+ * ProjectWatcher handles file system changes for hot module replacement (HMR).
19
+ * It uses chokidar to watch for file changes and triggers appropriate actions:
20
+ * - Uncaches modules when files change
21
+ * - Refreshes router routes for page files
22
+ * - Triggers HMR server reload
23
+ * - Handles processor-specific file changes
24
+ *
25
+ * The watcher uses chokidar's built-in debouncing through `awaitWriteFinish`
26
+ * to handle rapid file changes efficiently:
27
+ * - stabilityThreshold: 50ms - Time to wait for writes to stabilize
28
+ * - pollInterval: 50ms - Interval to poll for file changes
29
+ *
30
+ * @class ProjectWatcher
31
+ */
32
+ export declare class ProjectWatcher {
33
+ /**
34
+ * Duplicate identical watcher events within this window are ignored.
35
+ *
36
+ * Some editors or save pipelines emit two near-identical filesystem change
37
+ * notifications for the same file. Ecopages should treat those as one logical
38
+ * update so HMR and route refresh work are not repeated unnecessarily.
39
+ */
40
+ private static readonly duplicateChangeWindowMs;
41
+ private appConfig;
42
+ private refreshRouterRoutesCallback;
43
+ private hmrManager;
44
+ private bridge;
45
+ private watcher;
46
+ private lastHandledChange;
47
+ constructor({ config, refreshRouterRoutesCallback, hmrManager, bridge }: ProjectWatcherConfig);
48
+ /**
49
+ * Uncaches modules in the source directory to ensure fresh imports.
50
+ * This is necessary for hot module replacement to work correctly.
51
+ * @private
52
+ */
53
+ private uncacheModules;
54
+ /**
55
+ * Handles public directory file changes by copying only the changed file.
56
+ * @param filePath - Absolute path of the changed file
57
+ */
58
+ private handlePublicDirFileChange;
59
+ /**
60
+ * Handles file changes by uncaching modules, refreshing routes, and delegating appropriately.
61
+ * Follows 4-rule priority:
62
+ * 0. Public directory match? → copy file and reload
63
+ * 1. additionalWatchPaths match? → reload
64
+ * 2. Processor extension match? → processor handles (skip HMR)
65
+ * 3. Otherwise → HMR strategies
66
+ *
67
+ * Duplicate identical watcher events for the same file are coalesced within a
68
+ * short window before any of the priority rules run.
69
+ * @param rawPath - Path of the changed file
70
+ */
71
+ private handleFileChange;
72
+ /**
73
+ * Checks if a file is in the public directory.
74
+ */
75
+ private isPublicDirFile;
76
+ /**
77
+ * Checks if file path matches any additionalWatchPaths patterns.
78
+ */
79
+ private matchesAdditionalWatchPaths;
80
+ /**
81
+ * Checks if a file is handled by a processor.
82
+ * Processors that declare extensions own those file types.
83
+ */
84
+ private isHandledByProcessor;
85
+ /**
86
+ * Triggers router refresh for page directory changes.
87
+ * This ensures the router is updated when pages are added or removed.
88
+ *
89
+ * @param {string} path - Path of the changed directory
90
+ */
91
+ triggerRouterRefresh(path: string): void;
92
+ /**
93
+ * Handles and logs errors that occur during file watching.
94
+ *
95
+ * @param {unknown} error - The error to handle
96
+ */
97
+ handleError(error: unknown): void;
98
+ /**
99
+ * Processes file changes for specific file extensions.
100
+ * Used by processors to handle their specific file types.
101
+ *
102
+ * @private
103
+ * @param {string} path - Path of the changed file
104
+ * @param {string[]} extensions - File extensions to process
105
+ * @param {(ctx: ProcessorWatchContext) => void} handler - Handler function for the file change
106
+ */
107
+ private shouldProcess;
108
+ /**
109
+ * Creates and configures the file system watcher.
110
+ * This sets up:
111
+ * 1. Processor-specific file watching
112
+ * 2. Page file watching
113
+ * 3. Directory watching
114
+ * 4. Error handling
115
+ *
116
+ * Uses chokidar's built-in debouncing through `awaitWriteFinish` to handle
117
+ * rapid file changes efficiently.
118
+ */
119
+ createWatcherSubscription(): Promise<FSWatcher>;
120
+ }
@@ -0,0 +1,238 @@
1
+ import path from "node:path";
2
+ import chokidar, {} from "chokidar";
3
+ import { fileSystem } from "@ecopages/file-system";
4
+ import { appLogger } from "../global/app-logger.js";
5
+ class ProjectWatcher {
6
+ /**
7
+ * Duplicate identical watcher events within this window are ignored.
8
+ *
9
+ * Some editors or save pipelines emit two near-identical filesystem change
10
+ * notifications for the same file. Ecopages should treat those as one logical
11
+ * update so HMR and route refresh work are not repeated unnecessarily.
12
+ */
13
+ static duplicateChangeWindowMs = 150;
14
+ appConfig;
15
+ refreshRouterRoutesCallback;
16
+ hmrManager;
17
+ bridge;
18
+ watcher = null;
19
+ lastHandledChange = /* @__PURE__ */ new Map();
20
+ constructor({ config, refreshRouterRoutesCallback, hmrManager, bridge }) {
21
+ this.appConfig = config;
22
+ this.refreshRouterRoutesCallback = refreshRouterRoutesCallback;
23
+ this.hmrManager = hmrManager;
24
+ this.bridge = bridge;
25
+ this.triggerRouterRefresh = this.triggerRouterRefresh.bind(this);
26
+ this.handleError = this.handleError.bind(this);
27
+ this.handleFileChange = this.handleFileChange.bind(this);
28
+ }
29
+ /**
30
+ * Uncaches modules in the source directory to ensure fresh imports.
31
+ * This is necessary for hot module replacement to work correctly.
32
+ * @private
33
+ */
34
+ uncacheModules() {
35
+ if (typeof require === "undefined") return;
36
+ const { srcDir, rootDir } = this.appConfig;
37
+ const regex = new RegExp(`${rootDir}/${srcDir}/.*`);
38
+ for (const key in require.cache) {
39
+ if (regex.test(key)) {
40
+ delete require.cache[key];
41
+ }
42
+ }
43
+ }
44
+ /**
45
+ * Handles public directory file changes by copying only the changed file.
46
+ * @param filePath - Absolute path of the changed file
47
+ */
48
+ async handlePublicDirFileChange(filePath) {
49
+ try {
50
+ const relativePath = path.relative(this.appConfig.absolutePaths.publicDir, filePath);
51
+ const destPath = path.join(this.appConfig.absolutePaths.distDir, relativePath);
52
+ if (fileSystem.exists(filePath)) {
53
+ const destDir = path.dirname(destPath);
54
+ fileSystem.ensureDir(destDir);
55
+ await fileSystem.copyFileAsync(filePath, destPath);
56
+ }
57
+ this.bridge.reload();
58
+ } catch (error) {
59
+ appLogger.error(`Failed to copy public file: ${error instanceof Error ? error.message : String(error)}`);
60
+ this.bridge.reload();
61
+ }
62
+ }
63
+ /**
64
+ * Handles file changes by uncaching modules, refreshing routes, and delegating appropriately.
65
+ * Follows 4-rule priority:
66
+ * 0. Public directory match? → copy file and reload
67
+ * 1. additionalWatchPaths match? → reload
68
+ * 2. Processor extension match? → processor handles (skip HMR)
69
+ * 3. Otherwise → HMR strategies
70
+ *
71
+ * Duplicate identical watcher events for the same file are coalesced within a
72
+ * short window before any of the priority rules run.
73
+ * @param rawPath - Path of the changed file
74
+ */
75
+ async handleFileChange(rawPath) {
76
+ const filePath = path.resolve(rawPath);
77
+ const now = Date.now();
78
+ const lastHandledAt = this.lastHandledChange.get(filePath);
79
+ if (lastHandledAt !== void 0 && now - lastHandledAt < ProjectWatcher.duplicateChangeWindowMs) {
80
+ return;
81
+ }
82
+ this.lastHandledChange.set(filePath, now);
83
+ try {
84
+ if (this.isPublicDirFile(filePath)) {
85
+ await this.handlePublicDirFileChange(filePath);
86
+ return;
87
+ }
88
+ this.uncacheModules();
89
+ const isPageFile = filePath.startsWith(this.appConfig.absolutePaths.pagesDir);
90
+ if (isPageFile) {
91
+ this.refreshRouterRoutesCallback();
92
+ }
93
+ if (this.matchesAdditionalWatchPaths(filePath)) {
94
+ this.bridge.reload();
95
+ return;
96
+ }
97
+ if (this.isHandledByProcessor(filePath)) {
98
+ return;
99
+ }
100
+ await this.hmrManager.handleFileChange(filePath);
101
+ } catch (error) {
102
+ if (error instanceof Error) {
103
+ this.bridge.error(error.message);
104
+ this.handleError(error);
105
+ }
106
+ }
107
+ }
108
+ /**
109
+ * Checks if a file is in the public directory.
110
+ */
111
+ isPublicDirFile(filePath) {
112
+ return filePath.startsWith(this.appConfig.absolutePaths.publicDir);
113
+ }
114
+ /**
115
+ * Checks if file path matches any additionalWatchPaths patterns.
116
+ */
117
+ matchesAdditionalWatchPaths(filePath) {
118
+ const patterns = this.appConfig.additionalWatchPaths;
119
+ if (!patterns.length) return false;
120
+ for (const pattern of patterns) {
121
+ if (pattern.includes("*")) {
122
+ const ext = pattern.replace(/\*\*?\/\*/, "");
123
+ if (filePath.endsWith(ext)) return true;
124
+ } else {
125
+ if (filePath.endsWith(pattern) || filePath === path.resolve(pattern)) return true;
126
+ }
127
+ }
128
+ return false;
129
+ }
130
+ /**
131
+ * Checks if a file is handled by a processor.
132
+ * Processors that declare extensions own those file types.
133
+ */
134
+ isHandledByProcessor(filePath) {
135
+ for (const processor of this.appConfig.processors.values()) {
136
+ const watchConfig = processor.getWatchConfig();
137
+ if (!watchConfig) continue;
138
+ const { extensions = [] } = watchConfig;
139
+ if (extensions.length && extensions.some((ext) => filePath.endsWith(ext))) {
140
+ return true;
141
+ }
142
+ }
143
+ return false;
144
+ }
145
+ /**
146
+ * Triggers router refresh for page directory changes.
147
+ * This ensures the router is updated when pages are added or removed.
148
+ *
149
+ * @param {string} path - Path of the changed directory
150
+ */
151
+ triggerRouterRefresh(path2) {
152
+ const isPageDir = path2.startsWith(this.appConfig.absolutePaths.pagesDir);
153
+ if (isPageDir) {
154
+ this.refreshRouterRoutesCallback();
155
+ }
156
+ }
157
+ /**
158
+ * Handles and logs errors that occur during file watching.
159
+ *
160
+ * @param {unknown} error - The error to handle
161
+ */
162
+ handleError(error) {
163
+ if (error instanceof Error) {
164
+ this.hmrManager.broadcast({ type: "error", message: error.message });
165
+ }
166
+ appLogger.error(`Watcher error: ${error}`);
167
+ }
168
+ /**
169
+ * Processes file changes for specific file extensions.
170
+ * Used by processors to handle their specific file types.
171
+ *
172
+ * @private
173
+ * @param {string} path - Path of the changed file
174
+ * @param {string[]} extensions - File extensions to process
175
+ * @param {(ctx: ProcessorWatchContext) => void} handler - Handler function for the file change
176
+ */
177
+ shouldProcess(path2, extensions, handler) {
178
+ if (!extensions.length || extensions.some((ext) => path2.endsWith(ext))) {
179
+ handler({ path: path2, bridge: this.bridge });
180
+ }
181
+ }
182
+ /**
183
+ * Creates and configures the file system watcher.
184
+ * This sets up:
185
+ * 1. Processor-specific file watching
186
+ * 2. Page file watching
187
+ * 3. Directory watching
188
+ * 4. Error handling
189
+ *
190
+ * Uses chokidar's built-in debouncing through `awaitWriteFinish` to handle
191
+ * rapid file changes efficiently.
192
+ */
193
+ async createWatcherSubscription() {
194
+ if (!this.watcher) {
195
+ const processorPaths = [];
196
+ for (const processor of this.appConfig.processors.values()) {
197
+ const watchConfig = processor.getWatchConfig();
198
+ if (!watchConfig) continue;
199
+ processorPaths.push(...watchConfig.paths);
200
+ }
201
+ if (fileSystem.exists(this.appConfig.absolutePaths.pagesDir)) {
202
+ processorPaths.push(this.appConfig.absolutePaths.pagesDir);
203
+ }
204
+ if (fileSystem.exists(this.appConfig.absolutePaths.publicDir)) {
205
+ processorPaths.push(this.appConfig.absolutePaths.publicDir);
206
+ }
207
+ if (this.appConfig.additionalWatchPaths.length) {
208
+ processorPaths.push(...this.appConfig.additionalWatchPaths);
209
+ }
210
+ this.watcher = chokidar.watch(processorPaths, {
211
+ ignoreInitial: true,
212
+ ignorePermissionErrors: true,
213
+ awaitWriteFinish: {
214
+ stabilityThreshold: 50,
215
+ pollInterval: 50
216
+ }
217
+ });
218
+ }
219
+ for (const processor of this.appConfig.processors.values()) {
220
+ const watchConfig = processor.getWatchConfig();
221
+ if (!watchConfig) continue;
222
+ const { extensions = [], onCreate, onChange, onDelete, onError } = watchConfig;
223
+ if (onCreate) this.watcher.on("add", (path2) => this.shouldProcess(path2, extensions, onCreate));
224
+ if (onChange) this.watcher.on("change", (path2) => this.shouldProcess(path2, extensions, onChange));
225
+ if (onDelete) this.watcher.on("unlink", (path2) => this.shouldProcess(path2, extensions, onDelete));
226
+ if (onError) this.watcher.on("error", onError);
227
+ }
228
+ this.watcher.add(this.appConfig.absolutePaths.srcDir);
229
+ this.watcher.on("change", (path2) => this.handleFileChange(path2)).on("add", (path2) => {
230
+ this.handleFileChange(path2);
231
+ this.triggerRouterRefresh(path2);
232
+ }).on("addDir", (path2) => this.triggerRouterRefresh(path2)).on("unlink", (path2) => this.triggerRouterRefresh(path2)).on("unlinkDir", (path2) => this.triggerRouterRefresh(path2)).on("error", (error) => this.handleError(error));
233
+ return this.watcher;
234
+ }
235
+ }
236
+ export {
237
+ ProjectWatcher
238
+ };
@@ -0,0 +1,4 @@
1
+ import type { IHmrManager } from '../internal-types';
2
+ import type { ClientBridge } from '../adapters/bun/client-bridge';
3
+ export declare const createMockHmrManager: () => IHmrManager;
4
+ export declare const createMockBridge: () => ClientBridge;
@@ -0,0 +1,51 @@
1
+ import { vi } from "vitest";
2
+ const createMockHmrManager = () => ({
3
+ handleFileChange: vi.fn(async () => {
4
+ }),
5
+ broadcast: vi.fn(() => {
6
+ }),
7
+ setEnabled: vi.fn(() => {
8
+ }),
9
+ setPlugins: vi.fn(() => {
10
+ }),
11
+ registerEntrypoint: vi.fn(async () => ""),
12
+ registerSpecifierMap: vi.fn(() => {
13
+ }),
14
+ registerStrategy: vi.fn(() => {
15
+ }),
16
+ isEnabled: vi.fn(() => true),
17
+ getOutputUrl: vi.fn(() => void 0),
18
+ getWatchedFiles: vi.fn(() => /* @__PURE__ */ new Map()),
19
+ getSpecifierMap: vi.fn(() => /* @__PURE__ */ new Map()),
20
+ getDistDir: vi.fn(() => ""),
21
+ getPlugins: vi.fn(() => []),
22
+ getDefaultContext: vi.fn(() => ({
23
+ getWatchedFiles: () => /* @__PURE__ */ new Map(),
24
+ getSpecifierMap: () => /* @__PURE__ */ new Map(),
25
+ getDistDir: () => "",
26
+ getPlugins: () => [],
27
+ getSrcDir: () => "",
28
+ getLayoutsDir: () => ""
29
+ }))
30
+ });
31
+ const createMockBridge = () => ({
32
+ reload: vi.fn(() => {
33
+ }),
34
+ error: vi.fn(() => {
35
+ }),
36
+ cssUpdate: vi.fn(() => {
37
+ }),
38
+ update: vi.fn(() => {
39
+ }),
40
+ subscribe: vi.fn(() => {
41
+ }),
42
+ unsubscribe: vi.fn(() => {
43
+ }),
44
+ broadcast: vi.fn(() => {
45
+ }),
46
+ subscriberCount: 0
47
+ });
48
+ export {
49
+ createMockBridge,
50
+ createMockHmrManager
51
+ };
@@ -0,0 +1,40 @@
1
+ import { vi } from 'vitest';
2
+ import type { IHmrManager } from '../internal-types';
3
+ import type { ClientBridge } from '../adapters/bun/client-bridge';
4
+
5
+ export const createMockHmrManager = (): IHmrManager =>
6
+ ({
7
+ handleFileChange: vi.fn(async () => {}),
8
+ broadcast: vi.fn(() => {}),
9
+ setEnabled: vi.fn(() => {}),
10
+ setPlugins: vi.fn(() => {}),
11
+ registerEntrypoint: vi.fn(async () => ''),
12
+ registerSpecifierMap: vi.fn(() => {}),
13
+ registerStrategy: vi.fn(() => {}),
14
+ isEnabled: vi.fn(() => true),
15
+ getOutputUrl: vi.fn(() => undefined),
16
+ getWatchedFiles: vi.fn(() => new Map()),
17
+ getSpecifierMap: vi.fn(() => new Map()),
18
+ getDistDir: vi.fn(() => ''),
19
+ getPlugins: vi.fn(() => []),
20
+ getDefaultContext: vi.fn(() => ({
21
+ getWatchedFiles: () => new Map(),
22
+ getSpecifierMap: () => new Map(),
23
+ getDistDir: () => '',
24
+ getPlugins: () => [],
25
+ getSrcDir: () => '',
26
+ getLayoutsDir: () => '',
27
+ })),
28
+ }) as unknown as IHmrManager;
29
+
30
+ export const createMockBridge = (): ClientBridge =>
31
+ ({
32
+ reload: vi.fn(() => {}),
33
+ error: vi.fn(() => {}),
34
+ cssUpdate: vi.fn(() => {}),
35
+ update: vi.fn(() => {}),
36
+ subscribe: vi.fn(() => {}),
37
+ unsubscribe: vi.fn(() => {}),
38
+ broadcast: vi.fn(() => {}),
39
+ subscriberCount: 0,
40
+ }) as unknown as ClientBridge;