@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,510 @@
1
+ import type {
2
+ Loader as EsbuildLoader,
3
+ OnLoadResult as EsbuildOnLoadResult,
4
+ OnResolveResult as EsbuildOnResolveResult,
5
+ Plugin as EsbuildPlugin,
6
+ } from 'esbuild';
7
+ import path from 'node:path';
8
+ import { createRequire } from 'node:module';
9
+ import { fileSystem } from '@ecopages/file-system';
10
+ import type {
11
+ EcoBuildOnLoadResult,
12
+ EcoBuildPlugin,
13
+ EcoBuildPluginBuilder,
14
+ EcoBuildOnResolveResult,
15
+ } from './build-types.ts';
16
+ import type {
17
+ BuildAdapter,
18
+ BuildDependencyGraph,
19
+ BuildLog,
20
+ BuildOptions,
21
+ BuildResult,
22
+ BuildTranspileOptions,
23
+ BuildTranspileProfile,
24
+ } from './build-adapter.ts';
25
+
26
+ /**
27
+ * Provides common transpile output defaults shared across build profiles.
28
+ */
29
+ function transpileProfileToOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
30
+ switch (profile) {
31
+ case 'browser-script':
32
+ return {
33
+ target: 'browser',
34
+ format: 'esm',
35
+ sourcemap: 'none',
36
+ };
37
+ case 'hmr-runtime':
38
+ return {
39
+ target: 'browser',
40
+ format: 'esm',
41
+ sourcemap: 'none',
42
+ };
43
+ case 'hmr-entrypoint':
44
+ return {
45
+ target: 'browser',
46
+ format: 'esm',
47
+ sourcemap: 'none',
48
+ };
49
+ }
50
+ }
51
+
52
+ /**
53
+ * Node build adapter backed by esbuild.
54
+ *
55
+ * This adapter keeps Ecopages build plugin compatibility (`onResolve`, `onLoad`,
56
+ * and `module`) while delegating bundling and TypeScript/decorator transforms to esbuild.
57
+ */
58
+ export class EsbuildBuildAdapter implements BuildAdapter {
59
+ private registeredPlugins: EcoBuildPlugin[] = [];
60
+
61
+ private escapeRegExp(value: string): string {
62
+ return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
63
+ }
64
+
65
+ private getPluginsForBuild(additionalPlugins?: EcoBuildPlugin[]): EcoBuildPlugin[] {
66
+ const merged = [...(additionalPlugins ?? []), ...this.registeredPlugins];
67
+ const byName = new Map<string, EcoBuildPlugin>();
68
+
69
+ for (const plugin of merged) {
70
+ if (!byName.has(plugin.name)) {
71
+ byName.set(plugin.name, plugin);
72
+ }
73
+ }
74
+
75
+ return Array.from(byName.values());
76
+ }
77
+
78
+ private normalizeEsbuildLoader(loader: unknown): EsbuildLoader | undefined {
79
+ switch (loader) {
80
+ case 'base64':
81
+ case 'binary':
82
+ case 'copy':
83
+ case 'css':
84
+ case 'dataurl':
85
+ case 'empty':
86
+ case 'file':
87
+ case 'global-css':
88
+ case 'js':
89
+ case 'json':
90
+ case 'jsx':
91
+ case 'local-css':
92
+ case 'text':
93
+ case 'ts':
94
+ case 'tsx':
95
+ return loader as EsbuildLoader;
96
+ default:
97
+ return undefined;
98
+ }
99
+ }
100
+
101
+ private inferEsbuildLoaderFromPath(filePath: string): EsbuildLoader {
102
+ const extension = path.extname(filePath).toLowerCase();
103
+
104
+ switch (extension) {
105
+ case '.ts':
106
+ return 'ts';
107
+ case '.tsx':
108
+ return 'tsx';
109
+ case '.jsx':
110
+ return 'jsx';
111
+ case '.json':
112
+ return 'json';
113
+ case '.css':
114
+ return 'css';
115
+ default:
116
+ return 'js';
117
+ }
118
+ }
119
+
120
+ private convertLoadResultToModuleSource(result: unknown): string | undefined {
121
+ if (!result || typeof result !== 'object') {
122
+ return undefined;
123
+ }
124
+
125
+ const candidate = result as {
126
+ contents?: string;
127
+ loader?: unknown;
128
+ exports?: Record<string, unknown>;
129
+ };
130
+
131
+ if (typeof candidate.contents === 'string') {
132
+ return candidate.contents;
133
+ }
134
+
135
+ if (candidate.loader === 'object' && candidate.exports && typeof candidate.exports === 'object') {
136
+ const entries = Object.entries(candidate.exports)
137
+ .map(([key, value]) =>
138
+ key === 'default'
139
+ ? `export default ${JSON.stringify(value)};`
140
+ : `export const ${key} = ${JSON.stringify(value)};`,
141
+ )
142
+ .join('\n');
143
+
144
+ return entries;
145
+ }
146
+
147
+ return undefined;
148
+ }
149
+
150
+ private convertPluginOnLoadResult(args: { path: string }, result: unknown): EsbuildOnLoadResult | undefined {
151
+ if (!result || typeof result !== 'object') {
152
+ return undefined;
153
+ }
154
+
155
+ const candidate = result as EcoBuildOnLoadResult;
156
+
157
+ const sourceFromExports =
158
+ candidate.loader === 'object' && candidate.exports && typeof candidate.exports === 'object'
159
+ ? this.convertLoadResultToModuleSource(candidate)
160
+ : undefined;
161
+
162
+ if (sourceFromExports) {
163
+ return {
164
+ contents: sourceFromExports,
165
+ loader: 'js',
166
+ resolveDir: path.dirname(args.path),
167
+ };
168
+ }
169
+
170
+ if (typeof candidate.contents === 'string' || candidate.contents instanceof Uint8Array) {
171
+ return {
172
+ contents: candidate.contents,
173
+ loader: this.normalizeEsbuildLoader(candidate.loader) ?? this.inferEsbuildLoaderFromPath(args.path),
174
+ resolveDir: typeof candidate.resolveDir === 'string' ? candidate.resolveDir : path.dirname(args.path),
175
+ };
176
+ }
177
+
178
+ return undefined;
179
+ }
180
+
181
+ private resolvePluginPath(value: string, args: { importer: string }, contextRoot: string): string {
182
+ if (path.isAbsolute(value)) {
183
+ return value;
184
+ }
185
+
186
+ if (value.startsWith('.') || value.startsWith('..')) {
187
+ const baseDir = args.importer ? path.dirname(args.importer) : contextRoot;
188
+ return path.resolve(baseDir, value);
189
+ }
190
+
191
+ return value;
192
+ }
193
+
194
+ /**
195
+ * Creates an esbuild plugin bridge compatible with the existing Ecopages
196
+ * plugin API shape.
197
+ *
198
+ * **Plugin ordering is semantically significant.**
199
+ *
200
+ * Esbuild applies `onResolve` and `onLoad` hooks in the order they are
201
+ * registered: the first handler whose filter matches wins for `onResolve`,
202
+ * and the first handler that returns a non-`undefined` result wins for
203
+ * `onLoad`. Because we call `plugin.setup(bridge)` sequentially here, the
204
+ * position of each plugin in the `plugins` array determines its priority:
205
+ *
206
+ * - **Index 0** has the highest priority (its hooks run first).
207
+ * - **Last index** has the lowest priority (its hooks only run if no earlier
208
+ * plugin claimed the path).
209
+ *
210
+ * When adding new integrations or processors, ensure security-critical plugins
211
+ * (e.g. `ecopages-client-graph-boundary`) are placed **before** general-purpose
212
+ * loaders in the array so they always get first refusal on every source file.
213
+ *
214
+ * There is currently no priority system or validation — correct ordering is
215
+ * the caller's responsibility.
216
+ */
217
+ private createEcoPluginBridge(plugins: EcoBuildPlugin[], contextRoot: string): EsbuildPlugin {
218
+ return {
219
+ name: 'ecopages-plugin-bridge',
220
+ setup: async (build) => {
221
+ let moduleCounter = 0;
222
+
223
+ const bridge: EcoBuildPluginBuilder = {
224
+ onResolve: (options: { filter: RegExp; namespace?: string }, callback): void => {
225
+ build.onResolve(options, async (args) => {
226
+ const result = await callback({
227
+ path: args.path,
228
+ importer: args.importer,
229
+ namespace: args.namespace,
230
+ });
231
+
232
+ if (!result || typeof result !== 'object') {
233
+ return undefined;
234
+ }
235
+
236
+ const candidate = result as EcoBuildOnResolveResult;
237
+
238
+ const resolveResult: EsbuildOnResolveResult = {};
239
+
240
+ if (typeof candidate.path === 'string') {
241
+ resolveResult.path = this.resolvePluginPath(candidate.path, args, contextRoot);
242
+ }
243
+
244
+ if (typeof candidate.namespace === 'string') {
245
+ resolveResult.namespace = candidate.namespace;
246
+ }
247
+
248
+ if (typeof candidate.external === 'boolean') {
249
+ resolveResult.external = candidate.external;
250
+ }
251
+
252
+ return Object.keys(resolveResult).length > 0 ? resolveResult : undefined;
253
+ });
254
+ },
255
+ onLoad: (options: { filter: RegExp; namespace?: string }, callback): void => {
256
+ build.onLoad(options, async (args) => {
257
+ const result = await callback({
258
+ path: args.path,
259
+ namespace: args.namespace,
260
+ });
261
+
262
+ return this.convertPluginOnLoadResult(args, result);
263
+ });
264
+ },
265
+ module: (specifier: string, callback): void => {
266
+ const namespace = `ecopages-module-${moduleCounter}`;
267
+ moduleCounter += 1;
268
+ const filter = new RegExp(`^${this.escapeRegExp(specifier)}$`);
269
+
270
+ build.onResolve({ filter }, () => ({
271
+ path: specifier,
272
+ namespace,
273
+ }));
274
+
275
+ build.onLoad({ filter, namespace }, async (args) => {
276
+ const result = await callback();
277
+ return this.convertPluginOnLoadResult(args, result);
278
+ });
279
+ },
280
+ };
281
+
282
+ for (const plugin of plugins) {
283
+ await plugin.setup(bridge);
284
+ }
285
+ },
286
+ };
287
+ }
288
+
289
+ private async loadEsbuildModule(): Promise<typeof import('esbuild')> {
290
+ const esbuildModule = (await import('esbuild')) as typeof import('esbuild');
291
+
292
+ if (typeof esbuildModule.build !== 'function') {
293
+ throw new Error('esbuild is not available. Install esbuild to use Node bundling.');
294
+ }
295
+
296
+ return esbuildModule;
297
+ }
298
+
299
+ private mapEsbuildSourcemap(value: string | undefined): false | 'linked' | 'inline' | 'external' | 'both' {
300
+ switch (value) {
301
+ case 'none':
302
+ return false;
303
+ case 'inline':
304
+ return 'inline';
305
+ case 'both':
306
+ return 'both';
307
+ case 'external':
308
+ return 'external';
309
+ default:
310
+ return 'linked';
311
+ }
312
+ }
313
+
314
+ private mapEsbuildFormat(value: string | undefined): 'esm' | 'cjs' | 'iife' {
315
+ switch (value) {
316
+ case 'cjs':
317
+ return 'cjs';
318
+ case 'iife':
319
+ return 'iife';
320
+ default:
321
+ return 'esm';
322
+ }
323
+ }
324
+
325
+ private hasTemplateTokens(value: string | undefined): boolean {
326
+ return typeof value === 'string' && /\[[^\]]+\]/.test(value);
327
+ }
328
+
329
+ private toEntryNamePattern(value: string | undefined): string {
330
+ if (!value) {
331
+ return '[name]';
332
+ }
333
+
334
+ const pattern = value.replaceAll(/\.?\[ext\]/g, '');
335
+ return pattern.length > 0 ? pattern : '[name]';
336
+ }
337
+
338
+ private normalizeMetafilePath(value: string, contextRoot: string): string {
339
+ if (path.isAbsolute(value)) {
340
+ return path.normalize(value);
341
+ }
342
+
343
+ return path.normalize(path.resolve(contextRoot, value));
344
+ }
345
+
346
+ private extractDependencyGraph(
347
+ metafile: {
348
+ outputs: Record<string, { entryPoint?: string; inputs?: Record<string, unknown> }>;
349
+ },
350
+ contextRoot: string,
351
+ ): BuildDependencyGraph {
352
+ const entrypoints = new Map<string, Set<string>>();
353
+
354
+ for (const outputMeta of Object.values(metafile.outputs)) {
355
+ if (!outputMeta.entryPoint) {
356
+ continue;
357
+ }
358
+
359
+ const entrypointPath = this.normalizeMetafilePath(outputMeta.entryPoint, contextRoot);
360
+ const dependencies = entrypoints.get(entrypointPath) ?? new Set<string>();
361
+ dependencies.add(entrypointPath);
362
+
363
+ for (const inputPath of Object.keys(outputMeta.inputs ?? {})) {
364
+ dependencies.add(this.normalizeMetafilePath(inputPath, contextRoot));
365
+ }
366
+
367
+ entrypoints.set(entrypointPath, dependencies);
368
+ }
369
+
370
+ return {
371
+ entrypoints: Object.fromEntries(
372
+ Array.from(entrypoints.entries(), ([entrypointPath, dependencies]) => [
373
+ entrypointPath,
374
+ Array.from(dependencies),
375
+ ]),
376
+ ),
377
+ };
378
+ }
379
+
380
+ /**
381
+ * Normalizes esbuild errors into Ecopages `BuildLog` entries.
382
+ */
383
+ private toBuildLogs(error: unknown): BuildLog[] {
384
+ if (error && typeof error === 'object') {
385
+ const candidate = error as {
386
+ errors?: Array<{ text?: string; location?: { file?: string; line?: number; column?: number } }>;
387
+ message?: string;
388
+ };
389
+
390
+ if (Array.isArray(candidate.errors) && candidate.errors.length > 0) {
391
+ return candidate.errors.map((entry) => {
392
+ const locationPrefix = entry.location?.file
393
+ ? `${entry.location.file}:${entry.location.line ?? 0}:${entry.location.column ?? 0} `
394
+ : '';
395
+
396
+ return {
397
+ message: `${locationPrefix}${entry.text ?? 'Unknown esbuild error'}`,
398
+ };
399
+ });
400
+ }
401
+
402
+ if (typeof candidate.message === 'string') {
403
+ return [{ message: candidate.message }];
404
+ }
405
+ }
406
+
407
+ return [{ message: 'Unknown esbuild error' }];
408
+ }
409
+
410
+ /**
411
+ * Bundles entrypoints using esbuild for Node runtime builds.
412
+ */
413
+ async build(options: BuildOptions): Promise<BuildResult> {
414
+ const esbuild = await this.loadEsbuildModule();
415
+ const contextRoot = options.root ? path.resolve(options.root) : process.cwd();
416
+ const outdir = path.resolve(options.outdir ?? '.eco/assets');
417
+ const tsconfigPath = path.join(contextRoot, 'tsconfig.json');
418
+ const tsconfigExists = fileSystem.exists(tsconfigPath);
419
+
420
+ const plugins = this.getPluginsForBuild(options.plugins);
421
+ const esbuildPlugins: EsbuildPlugin[] = [
422
+ ...(plugins.length > 0 ? [this.createEcoPluginBridge(plugins, contextRoot)] : []),
423
+ ];
424
+ const transpileTarget = 'es2022';
425
+
426
+ const usesTemplatedNaming = this.hasTemplateTokens(options.naming);
427
+ const outfile = options.naming && !usesTemplatedNaming ? path.join(outdir, options.naming) : undefined;
428
+ if (outfile) {
429
+ fileSystem.ensureDir(path.dirname(outfile));
430
+ }
431
+
432
+ const outputOptions = outfile
433
+ ? { outfile }
434
+ : {
435
+ outdir,
436
+ entryNames: usesTemplatedNaming ? this.toEntryNamePattern(options.naming) : '[name]',
437
+ chunkNames: '[name]-[hash]',
438
+ assetNames: '[name]-[hash]',
439
+ };
440
+
441
+ try {
442
+ const result = await esbuild.build({
443
+ absWorkingDir: contextRoot,
444
+ entryPoints: options.entrypoints,
445
+ bundle: options.bundle ?? true,
446
+ ...outputOptions,
447
+ format: this.mapEsbuildFormat(options.format),
448
+ platform: (options.target === 'browser' ? 'browser' : 'node') as 'browser' | 'node',
449
+ sourcemap: this.mapEsbuildSourcemap(options.sourcemap),
450
+ splitting: outfile ? false : !!options.splitting,
451
+ minify: !!options.minify,
452
+ ...(typeof options.treeshaking === 'boolean' ? { treeShaking: options.treeshaking } : {}),
453
+ external: options.external,
454
+ ...(options.target !== 'browser' && options.externalPackages !== false
455
+ ? { packages: 'external' as const }
456
+ : {}),
457
+ target: transpileTarget,
458
+ metafile: true,
459
+ write: true,
460
+ plugins: esbuildPlugins,
461
+ jsx: 'automatic',
462
+ tsconfig: tsconfigExists ? tsconfigPath : undefined,
463
+ logLevel: 'silent',
464
+ });
465
+
466
+ const outputs = Object.keys(result.metafile.outputs).map((outputPath) => ({
467
+ path: path.isAbsolute(outputPath) ? outputPath : path.join(contextRoot, outputPath),
468
+ }));
469
+ const logs = result.warnings.map((warning) => ({ message: warning.text }));
470
+ const dependencyGraph = this.extractDependencyGraph(result.metafile, contextRoot);
471
+
472
+ return {
473
+ success: true,
474
+ logs,
475
+ outputs,
476
+ dependencyGraph,
477
+ };
478
+ } catch (error) {
479
+ return {
480
+ success: false,
481
+ logs: this.toBuildLogs(error),
482
+ outputs: [],
483
+ };
484
+ }
485
+ }
486
+
487
+ /**
488
+ * Resolves module specifiers from a project root.
489
+ */
490
+ resolve(importPath: string, rootDir: string): string {
491
+ const localRequire = createRequire(path.join(rootDir, 'package.json'));
492
+ return localRequire.resolve(importPath);
493
+ }
494
+
495
+ /**
496
+ * Registers a build plugin once by plugin name.
497
+ */
498
+ registerPlugin(plugin: EcoBuildPlugin): void {
499
+ if (!this.registeredPlugins.find((registered) => registered.name === plugin.name)) {
500
+ this.registeredPlugins.push(plugin);
501
+ }
502
+ }
503
+
504
+ /**
505
+ * Returns transpile defaults for a known transpile profile.
506
+ */
507
+ getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
508
+ return transpileProfileToOptions(profile);
509
+ }
510
+ }