@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,596 @@
1
+ import path from 'node:path';
2
+ import { readFileSync } from 'node:fs';
3
+ import type {
4
+ DependencyLazyTrigger,
5
+ EcoComponent,
6
+ EcoComponentScriptEntry,
7
+ LazyTriggerRule,
8
+ ResolvedLazyTrigger,
9
+ } from '../public-types.ts';
10
+ import type { EcoPagesAppConfig } from '../internal-types.ts';
11
+ import type {
12
+ AssetDefinition,
13
+ AssetProcessingService,
14
+ ProcessedAsset,
15
+ } from '../services/asset-processing-service/index.ts';
16
+ import { rapidhash } from '../utils/hash.ts';
17
+ import { AssetFactory } from '../services/asset-processing-service/index.ts';
18
+ import { normalizeModuleDeclarations } from '../eco/module-dependencies.ts';
19
+ import { parseSync } from 'oxc-parser';
20
+
21
+ export const DEPENDENCY_ERRORS = {
22
+ INVALID_STYLESHEET_ENTRY: 'Invalid stylesheet dependency entry: expected src or content',
23
+ INVALID_SCRIPT_ENTRY: 'Invalid script dependency entry: expected src or content',
24
+ LAZY_SCRIPT_MISSING_SRC: 'Lazy script dependency entry in dependencies.scripts requires a src value',
25
+ } as const;
26
+
27
+ /**
28
+ * Parses a component's source file with oxc-parser and collects all
29
+ * `ecopages:` virtual module imports, preserving their named specifiers.
30
+ * Type-only imports (`import type`) are skipped since they don't exist at runtime.
31
+ *
32
+ * @example
33
+ * // import { foo } from 'ecopages:images' → { from: 'ecopages:images', imports: ['foo'] }
34
+ */
35
+ function extractEcopagesVirtualImports(file: string): Array<{ from: string; imports: string[] | undefined }> {
36
+ let source: string;
37
+ try {
38
+ source = readFileSync(file, 'utf-8');
39
+ } catch {
40
+ return [];
41
+ }
42
+
43
+ let result;
44
+ try {
45
+ result = parseSync(file, source, { sourceType: 'module' });
46
+ } catch {
47
+ return [];
48
+ }
49
+
50
+ const found = new Map<string, Set<string> | null>();
51
+
52
+ for (const node of result.program.body ?? []) {
53
+ if (node.type !== 'ImportDeclaration') continue;
54
+ if (node.importKind === 'type') continue;
55
+ const specifier: string = node.source?.value ?? '';
56
+ if (!specifier.startsWith('ecopages:')) continue;
57
+
58
+ if (found.get(specifier) === null) {
59
+ continue;
60
+ }
61
+
62
+ const namedImports: string[] = [];
63
+ for (const spec of node.specifiers ?? []) {
64
+ if (spec.type === 'ImportSpecifier') {
65
+ const importedName =
66
+ spec.imported?.type === 'Identifier'
67
+ ? spec.imported.name
68
+ : spec.imported?.type === 'Literal'
69
+ ? spec.imported.value
70
+ : spec.local?.name;
71
+ namedImports.push(importedName);
72
+ }
73
+ }
74
+
75
+ if (namedImports.length === 0) {
76
+ found.set(specifier, null);
77
+ continue;
78
+ }
79
+
80
+ const existing = found.get(specifier);
81
+ if (!existing) {
82
+ found.set(specifier, new Set(namedImports));
83
+ continue;
84
+ }
85
+
86
+ for (const imported of namedImports) {
87
+ existing.add(imported);
88
+ }
89
+ }
90
+
91
+ return Array.from(found.entries()).map(([from, importsSet]) => ({
92
+ from,
93
+ imports: importsSet ? Array.from(importsSet) : undefined,
94
+ }));
95
+ }
96
+
97
+ function createModuleScriptName(from: string, imports: string[] | undefined): string {
98
+ const normalizedImports = imports ? [...imports].sort().join(',') : '*';
99
+ const hash = rapidhash(`${from}|${normalizedImports}`).toString(16);
100
+ return `module-${hash}`;
101
+ }
102
+
103
+ function createNamedImportModuleSource(from: string, imports: string[]): string {
104
+ const namedImports = imports.join(', ');
105
+ return `export { ${namedImports} } from '${from}';`;
106
+ }
107
+
108
+ function createNamespaceImportModuleSource(from: string): string {
109
+ return `export * from '${from}';`;
110
+ }
111
+
112
+ function resolveDependencyPath(componentDir: string, pathUrl: string): string {
113
+ return path.join(componentDir, pathUrl);
114
+ }
115
+
116
+ function isDependencyEntryObject(entry: string | EcoComponentScriptEntry): entry is EcoComponentScriptEntry {
117
+ return typeof entry === 'object' && entry !== null;
118
+ }
119
+
120
+ function getDependencyEntrySrc(entry: string | EcoComponentScriptEntry): string | undefined {
121
+ return isDependencyEntryObject(entry) ? entry.src : entry;
122
+ }
123
+
124
+ /**
125
+ * Reads inline dependency content from an entry object.
126
+ * Returns `undefined` for string entries.
127
+ */
128
+ function getDependencyEntryContent(entry: string | EcoComponentScriptEntry): string | undefined {
129
+ return isDependencyEntryObject(entry) ? entry.content : undefined;
130
+ }
131
+
132
+ /**
133
+ * Reads optional HTML tag attributes from an entry object.
134
+ * Returns `undefined` for string entries.
135
+ */
136
+ function getDependencyEntryAttributes(entry: string | EcoComponentScriptEntry): Record<string, string> | undefined {
137
+ return isDependencyEntryObject(entry) ? entry.attributes : undefined;
138
+ }
139
+
140
+ /**
141
+ * Resolves the `src` field from a dependency entry and throws when absent.
142
+ *
143
+ * Used for entry forms where a file-backed source is mandatory
144
+ * (for example certain lazy script code paths).
145
+ */
146
+ function getDependencyEntrySrcOrThrow(entry: string | EcoComponentScriptEntry, errorMessage: string): string {
147
+ const src = getDependencyEntrySrc(entry);
148
+ if (!src) {
149
+ throw new Error(errorMessage);
150
+ }
151
+
152
+ return src;
153
+ }
154
+
155
+ /**
156
+ * Creates a stable grouping key for a lazy trigger.
157
+ *
158
+ * This key is used to bucket lazy scripts that share the same trigger so
159
+ * each group can be emitted into its own scripts-injector wrapper.
160
+ */
161
+ function getLazyTriggerKey(lazy: DependencyLazyTrigger): string {
162
+ if ('on:idle' in lazy) {
163
+ return 'on:idle';
164
+ }
165
+
166
+ if ('on:interaction' in lazy) {
167
+ return `on:interaction:${lazy['on:interaction']}`;
168
+ }
169
+
170
+ if ('on:visible' in lazy) {
171
+ const value = lazy['on:visible'];
172
+ return `on:visible:${value === true ? 'true' : value}`;
173
+ }
174
+
175
+ return JSON.stringify(lazy);
176
+ }
177
+
178
+ function resolveLazyScripts(appConfig: EcoPagesAppConfig, componentDir: string, scripts: string[]): string {
179
+ const getSafeFileName = (filepath: string): string => {
180
+ const EXTENSIONS_TO_JS = ['ts', 'tsx'];
181
+ const safe = filepath.replace(new RegExp(`\\.(${EXTENSIONS_TO_JS.join('|')})$`), '.js');
182
+ return safe.startsWith('./') ? safe.slice(2) : safe;
183
+ };
184
+
185
+ const baseDir = componentDir.split(appConfig.srcDir)[1] ?? '';
186
+ const resolvedPaths = scripts.map((script) => {
187
+ const relativePath = [AssetFactory.RESOLVED_ASSETS_DIR, baseDir, getSafeFileName(script)]
188
+ .filter(Boolean)
189
+ .join('/')
190
+ .replace(/\/+/g, '/');
191
+
192
+ return `/${relativePath.replace(/^\/+/, '')}`;
193
+ });
194
+
195
+ return resolvedPaths.join(',');
196
+ }
197
+
198
+ type ResolvedLazyGroup = { lazy: DependencyLazyTrigger; scripts: string[] };
199
+
200
+ /**
201
+ * Derives a deterministic `triggerId` and the full set of `LazyTriggerRule` entries
202
+ * for a component's lazy script groups.
203
+ *
204
+ * The trigger ID is a hex digest of `rapidhash(componentFile + ':' + sortedUrls)`,
205
+ * making it stable across renders as long as the source file path and resolved
206
+ * script URLs do not change. Sorting the URLs before hashing ensures the ID is
207
+ * independent of declaration order in the component config.
208
+ *
209
+ * Scripts are received as plain arrays (already deduplicated by the caller) to
210
+ * avoid a CSV round-trip that would otherwise require `split(',')` here.
211
+ */
212
+ function buildResolvedLazyTriggers(
213
+ config: NonNullable<EcoComponent['config']>,
214
+ groups: ResolvedLazyGroup[],
215
+ ): ResolvedLazyTrigger[] {
216
+ if (groups.length === 0) return [];
217
+
218
+ const componentFile = config.__eco?.file ?? '';
219
+ const sortedUrls = groups
220
+ .flatMap((group) => group.scripts)
221
+ .sort()
222
+ .join(',');
223
+ const triggerId = `eco-trigger-${rapidhash(`${componentFile}:${sortedUrls}`).toString(16)}`;
224
+
225
+ const rules: LazyTriggerRule[] = groups.map((group) => {
226
+ const { scripts, lazy } = group;
227
+
228
+ if ('on:idle' in lazy) {
229
+ return { 'on:idle': { scripts } };
230
+ }
231
+ if ('on:interaction' in lazy) {
232
+ return { 'on:interaction': { value: lazy['on:interaction'], scripts } };
233
+ }
234
+ if ('on:visible' in lazy) {
235
+ const visibleSelector = lazy['on:visible'];
236
+ if (visibleSelector === true) return { 'on:visible': { scripts } };
237
+ return { 'on:visible': { value: String(visibleSelector), scripts } };
238
+ }
239
+ throw new Error(`Unknown lazy trigger kind: ${JSON.stringify(lazy)}`);
240
+ });
241
+
242
+ return [{ triggerId, rules }];
243
+ }
244
+
245
+ export class DependencyResolverService {
246
+ constructor(
247
+ private appConfig: EcoPagesAppConfig,
248
+ private assetProcessingService: AssetProcessingService,
249
+ ) {}
250
+
251
+ resolveDependencyPath(componentDir: string, pathUrl: string): string {
252
+ return resolveDependencyPath(componentDir, pathUrl);
253
+ }
254
+
255
+ /**
256
+ * Maps lazy script source entries to deterministic fallback public URLs
257
+ * used when bundling output URLs are unavailable.
258
+ */
259
+ resolveLazyScripts(componentDir: string, scripts: string[]): string {
260
+ return resolveLazyScripts(this.appConfig, componentDir, scripts);
261
+ }
262
+
263
+ /**
264
+ * Collects and processes component dependencies (styles, scripts, modules, lazy scripts).
265
+ * Lazy dependencies are always resolved into global-injector trigger maps.
266
+ */
267
+ async processComponentDependencies(
268
+ components: (EcoComponent | Partial<EcoComponent>)[],
269
+ integrationName: string,
270
+ ): Promise<ProcessedAsset[]> {
271
+ if (!this.assetProcessingService?.processDependencies) return [];
272
+ const dependencies: AssetDefinition[] = [];
273
+ type LazyScriptRef = {
274
+ lazyKey: string;
275
+ fallbackUrl?: string;
276
+ };
277
+ type LazyGroup = {
278
+ lazy: DependencyLazyTrigger;
279
+ scripts: LazyScriptRef[];
280
+ };
281
+ const lazyScriptsByConfig = new Map<NonNullable<EcoComponent['config']>, Map<string, LazyGroup>>();
282
+ const lazyDependencyKeys = new Set<string>();
283
+
284
+ for (const component of components) {
285
+ const componentFile = component.config?.__eco?.file;
286
+ if (!componentFile) continue;
287
+
288
+ const stylesheetDependencyKeys = new Set<string>();
289
+ const scriptDependencyKeys = new Set<string>();
290
+ const modulesMap = new Map<string, Set<string> | null>();
291
+
292
+ const collect = (config: EcoComponent['config']) => {
293
+ if (!config) return;
294
+
295
+ const file = config.__eco?.file;
296
+ if (!file) return;
297
+ const dir = path.dirname(file);
298
+ const dependenciesConfig = config.dependencies;
299
+
300
+ const registerLazyScript = ({
301
+ lazy,
302
+ lazyKey,
303
+ fallbackUrl,
304
+ }: {
305
+ lazy: DependencyLazyTrigger;
306
+ lazyKey: string;
307
+ fallbackUrl?: string;
308
+ }) => {
309
+ let grouped = lazyScriptsByConfig.get(config);
310
+ if (!grouped) {
311
+ grouped = new Map<string, LazyGroup>();
312
+ lazyScriptsByConfig.set(config, grouped);
313
+ }
314
+
315
+ const triggerKey = getLazyTriggerKey(lazy);
316
+ const existing = grouped.get(triggerKey) ?? { lazy, scripts: [] };
317
+ existing.scripts.push({ lazyKey, fallbackUrl });
318
+ grouped.set(triggerKey, existing);
319
+ };
320
+
321
+ if (dependenciesConfig?.stylesheets) {
322
+ for (const style of dependenciesConfig.stylesheets) {
323
+ const content = getDependencyEntryContent(style);
324
+ const src = getDependencyEntrySrc(style);
325
+ const attributes = getDependencyEntryAttributes(style);
326
+
327
+ if (content) {
328
+ const depKey = `style:content:${content}:${JSON.stringify(attributes ?? {})}`;
329
+ if (stylesheetDependencyKeys.has(depKey)) {
330
+ continue;
331
+ }
332
+
333
+ stylesheetDependencyKeys.add(depKey);
334
+ dependencies.push(
335
+ AssetFactory.createContentStylesheet({
336
+ content,
337
+ position: 'head',
338
+ attributes,
339
+ }),
340
+ );
341
+ continue;
342
+ }
343
+
344
+ if (!src) {
345
+ throw new Error(DEPENDENCY_ERRORS.INVALID_STYLESHEET_ENTRY);
346
+ }
347
+
348
+ const resolvedPath = resolveDependencyPath(dir, src);
349
+ const depKey = `style:file:${resolvedPath}:${JSON.stringify(attributes ?? {})}`;
350
+ if (stylesheetDependencyKeys.has(depKey)) {
351
+ continue;
352
+ }
353
+
354
+ stylesheetDependencyKeys.add(depKey);
355
+ dependencies.push(
356
+ AssetFactory.createFileStylesheet({
357
+ filepath: resolvedPath,
358
+ position: 'head',
359
+ attributes: {
360
+ rel: 'stylesheet',
361
+ ...attributes,
362
+ },
363
+ }),
364
+ );
365
+ }
366
+ }
367
+
368
+ if (dependenciesConfig?.scripts) {
369
+ for (const script of dependenciesConfig.scripts) {
370
+ if (isDependencyEntryObject(script) && script.lazy) {
371
+ continue;
372
+ }
373
+
374
+ const content = getDependencyEntryContent(script);
375
+ const src = getDependencyEntrySrc(script);
376
+ const attributes = getDependencyEntryAttributes(script);
377
+
378
+ if (content) {
379
+ const depKey = `script:content:${content}:${JSON.stringify(attributes ?? {})}`;
380
+ if (scriptDependencyKeys.has(depKey)) {
381
+ continue;
382
+ }
383
+
384
+ scriptDependencyKeys.add(depKey);
385
+ dependencies.push(
386
+ AssetFactory.createContentScript({
387
+ position: 'head',
388
+ content,
389
+ attributes: {
390
+ type: 'module',
391
+ defer: '',
392
+ ...attributes,
393
+ },
394
+ }),
395
+ );
396
+ continue;
397
+ }
398
+
399
+ if (!src) {
400
+ throw new Error(DEPENDENCY_ERRORS.INVALID_SCRIPT_ENTRY);
401
+ }
402
+
403
+ const resolvedPath = resolveDependencyPath(dir, src);
404
+ const depKey = `script:file:${resolvedPath}:${JSON.stringify(attributes ?? {})}`;
405
+ if (scriptDependencyKeys.has(depKey)) {
406
+ continue;
407
+ }
408
+
409
+ scriptDependencyKeys.add(depKey);
410
+ dependencies.push(
411
+ AssetFactory.createFileScript({
412
+ filepath: resolvedPath,
413
+ position: 'head',
414
+ attributes: {
415
+ type: 'module',
416
+ defer: '',
417
+ ...attributes,
418
+ },
419
+ }),
420
+ );
421
+ }
422
+ }
423
+
424
+ const normalizedModules = normalizeModuleDeclarations(dependenciesConfig?.modules);
425
+
426
+ for (const declaration of normalizedModules) {
427
+ const existing = modulesMap.get(declaration.from);
428
+ if (!declaration.imports || declaration.imports.length === 0) {
429
+ modulesMap.set(declaration.from, null);
430
+ continue;
431
+ }
432
+
433
+ if (existing === null) {
434
+ continue;
435
+ }
436
+
437
+ const merged = existing ?? new Set<string>();
438
+ for (const imported of declaration.imports) {
439
+ merged.add(imported);
440
+ }
441
+ modulesMap.set(declaration.from, merged);
442
+ }
443
+
444
+ for (const [index, scriptEntry] of (dependenciesConfig?.scripts ?? []).entries()) {
445
+ if (!isDependencyEntryObject(scriptEntry) || !scriptEntry.lazy) {
446
+ continue;
447
+ }
448
+
449
+ const lazy = scriptEntry.lazy;
450
+ const content = scriptEntry.content;
451
+ const src = scriptEntry.src;
452
+ const attributes = scriptEntry.attributes;
453
+
454
+ if (content) {
455
+ const lazyKey = `lazy:${rapidhash(`${file}:entry:${index}:${content}`).toString(16)}`;
456
+ const depKey = `lazy:entry:content:${getLazyTriggerKey(lazy)}:${content}:${JSON.stringify(attributes ?? {})}`;
457
+ if (!lazyDependencyKeys.has(depKey)) {
458
+ lazyDependencyKeys.add(depKey);
459
+ dependencies.push(
460
+ AssetFactory.createContentScript({
461
+ position: 'head',
462
+ content,
463
+ excludeFromHtml: true,
464
+ attributes: {
465
+ type: 'module',
466
+ defer: '',
467
+ 'data-eco-lazy-key': lazyKey,
468
+ ...attributes,
469
+ },
470
+ }),
471
+ );
472
+ }
473
+
474
+ registerLazyScript({
475
+ lazy,
476
+ lazyKey,
477
+ });
478
+ continue;
479
+ }
480
+
481
+ const script =
482
+ src ?? getDependencyEntrySrcOrThrow(scriptEntry, DEPENDENCY_ERRORS.LAZY_SCRIPT_MISSING_SRC);
483
+ const resolvedPath = this.resolveDependencyPath(dir, script);
484
+ const fallbackUrl = this.resolveLazyScripts(dir, [script]).split(',')[0] ?? '';
485
+ const lazyKey = `lazy:${rapidhash(`${resolvedPath}:${getLazyTriggerKey(lazy)}`).toString(16)}`;
486
+ const depKey = `lazy:entry:file:${getLazyTriggerKey(lazy)}:${resolvedPath}:${JSON.stringify(attributes ?? {})}`;
487
+
488
+ if (!lazyDependencyKeys.has(depKey)) {
489
+ lazyDependencyKeys.add(depKey);
490
+ dependencies.push(
491
+ AssetFactory.createFileScript({
492
+ filepath: resolvedPath,
493
+ position: 'head',
494
+ excludeFromHtml: true,
495
+ attributes: {
496
+ type: 'module',
497
+ defer: '',
498
+ 'data-eco-lazy-key': lazyKey,
499
+ ...attributes,
500
+ },
501
+ }),
502
+ );
503
+ }
504
+
505
+ registerLazyScript({
506
+ lazy,
507
+ lazyKey,
508
+ fallbackUrl,
509
+ });
510
+ }
511
+
512
+ if (dependenciesConfig?.components) {
513
+ for (const nestedComponent of dependenciesConfig.components) {
514
+ if (nestedComponent.config) {
515
+ collect(nestedComponent.config);
516
+ }
517
+ }
518
+ }
519
+
520
+ // Auto-detect `ecopages:` virtual module imports from the component source file
521
+ const autoVirtualImports = extractEcopagesVirtualImports(file);
522
+ for (const { from, imports } of autoVirtualImports) {
523
+ const existing = modulesMap.get(from);
524
+ if (!imports || imports.length === 0) {
525
+ if (existing !== null) modulesMap.set(from, null);
526
+ continue;
527
+ }
528
+ if (existing === null) continue;
529
+ const merged = existing ?? new Set<string>();
530
+ for (const imported of imports) merged.add(imported);
531
+ modulesMap.set(from, merged);
532
+ }
533
+ };
534
+
535
+ collect(component.config);
536
+
537
+ dependencies.push(
538
+ ...Array.from(modulesMap.entries()).map(([from, importsSet]) => {
539
+ const imports = importsSet ? Array.from(importsSet) : undefined;
540
+ return AssetFactory.createContentScript({
541
+ position: 'head',
542
+ name: createModuleScriptName(from, imports),
543
+ content:
544
+ imports && imports.length > 0
545
+ ? createNamedImportModuleSource(from, imports)
546
+ : createNamespaceImportModuleSource(from),
547
+ attributes: {
548
+ type: 'module',
549
+ defer: '',
550
+ },
551
+ });
552
+ }),
553
+ );
554
+ }
555
+
556
+ const hasLazyDependencies = dependencies.some((dep) => dep.kind === 'script' && dep.excludeFromHtml === true);
557
+
558
+ const processedDependencies = await this.assetProcessingService.processDependencies(
559
+ dependencies,
560
+ integrationName,
561
+ );
562
+ const lazyKeyToOutputUrl = new Map<string, string>();
563
+
564
+ for (const dependency of processedDependencies) {
565
+ if (dependency.kind === 'script' && dependency.srcUrl) {
566
+ const lazyKey = dependency.attributes?.['data-eco-lazy-key'];
567
+ if (lazyKey) {
568
+ lazyKeyToOutputUrl.set(lazyKey, dependency.srcUrl);
569
+ }
570
+ }
571
+ }
572
+
573
+ for (const [config, lazyGroupsMap] of lazyScriptsByConfig.entries()) {
574
+ const rawGroups: ResolvedLazyGroup[] = [];
575
+
576
+ for (const group of lazyGroupsMap.values()) {
577
+ const resolvedUrls = group.scripts
578
+ .map(({ lazyKey, fallbackUrl }) => lazyKeyToOutputUrl.get(lazyKey) ?? fallbackUrl)
579
+ .filter((url): url is string => Boolean(url && url.length > 0));
580
+
581
+ if (resolvedUrls.length === 0) {
582
+ continue;
583
+ }
584
+
585
+ rawGroups.push({ lazy: group.lazy, scripts: Array.from(new Set(resolvedUrls)) });
586
+ }
587
+
588
+ if (hasLazyDependencies) {
589
+ config._resolvedLazyTriggers = buildResolvedLazyTriggers(config, rawGroups);
590
+ config._resolvedLazyScripts = undefined;
591
+ }
592
+ }
593
+
594
+ return processedDependencies;
595
+ }
596
+ }
@@ -0,0 +1,40 @@
1
+ import type { ProcessedAsset } from '../services/asset-processing-service/index.js';
2
+ /**
3
+ * Encapsulates HTML mutation and processed-asset normalization helpers used in
4
+ * the final stages of route rendering.
5
+ */
6
+ export declare class HtmlPostProcessingService {
7
+ /**
8
+ * Applies attributes to the first element immediately inside the document
9
+ * `<body>`.
10
+ *
11
+ * @param html Full HTML document.
12
+ * @param attributes Attribute map to inject.
13
+ * @returns Updated HTML document.
14
+ */
15
+ applyAttributesToFirstBodyElement(html: string, attributes: Record<string, string>): string;
16
+ /**
17
+ * Applies attributes to the first top-level element in an HTML fragment.
18
+ *
19
+ * @param html HTML fragment.
20
+ * @param attributes Attribute map to inject.
21
+ * @returns Updated HTML fragment.
22
+ */
23
+ applyAttributesToFirstElement(html: string, attributes: Record<string, string>): string;
24
+ /**
25
+ * Deduplicates processed assets using a stable composite key while preserving
26
+ * first-seen order.
27
+ *
28
+ * @param assets Candidate processed assets.
29
+ * @returns Deduplicated processed asset list.
30
+ */
31
+ dedupeProcessedAssets(assets: ProcessedAsset[]): ProcessedAsset[];
32
+ /**
33
+ * Converts an attribute map into an HTML attribute string, skipping empty
34
+ * keys and values.
35
+ *
36
+ * @param attributes Attribute map to serialize.
37
+ * @returns Serialized attributes prefixed with spaces.
38
+ */
39
+ private buildAttributeString;
40
+ }