@ecopages/core 0.2.0-alpha.9 → 0.2.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 (357) hide show
  1. package/CHANGELOG.md +18 -11
  2. package/README.md +23 -26
  3. package/package.json +76 -58
  4. package/src/adapters/README.md +2 -2
  5. package/src/adapters/abstract/application-adapter.d.ts +28 -2
  6. package/src/adapters/abstract/application-adapter.js +14 -2
  7. package/src/adapters/abstract/router-adapter.d.ts +1 -1
  8. package/src/adapters/abstract/server-adapter.d.ts +2 -2
  9. package/src/adapters/bun/client-bridge.d.ts +1 -1
  10. package/src/adapters/bun/create-app.d.ts +4 -12
  11. package/src/adapters/bun/create-app.js +4 -5
  12. package/src/adapters/bun/hmr-manager.d.ts +4 -4
  13. package/src/adapters/bun/hmr-manager.js +8 -8
  14. package/src/adapters/bun/index.d.ts +1 -1
  15. package/src/adapters/bun/index.js +2 -2
  16. package/src/adapters/bun/server-adapter.d.ts +5 -5
  17. package/src/adapters/bun/server-adapter.js +6 -5
  18. package/src/adapters/bun/server-lifecycle.d.ts +4 -4
  19. package/src/adapters/bun/server-lifecycle.js +2 -2
  20. package/src/{create-app.d.ts → adapters/create-app.d.ts} +9 -6
  21. package/src/{create-app.js → adapters/create-app.js} +4 -4
  22. package/src/adapters/index.d.ts +1 -5
  23. package/src/adapters/index.js +1 -7
  24. package/src/adapters/node/create-app.d.ts +6 -9
  25. package/src/adapters/node/create-app.js +12 -6
  26. package/src/adapters/node/node-client-bridge.d.ts +1 -1
  27. package/src/adapters/node/node-hmr-manager.d.ts +2 -2
  28. package/src/adapters/node/node-hmr-manager.js +3 -3
  29. package/src/adapters/node/server-adapter.d.ts +4 -4
  30. package/src/adapters/node/server-adapter.js +13 -12
  31. package/src/adapters/node/static-content-server.d.ts +1 -1
  32. package/src/adapters/node/static-content-server.js +1 -1
  33. package/src/adapters/shared/application-adapter.d.ts +1 -1
  34. package/src/adapters/shared/define-api-handler.d.ts +1 -1
  35. package/src/adapters/shared/explicit-static-route-matcher.d.ts +2 -2
  36. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +1 -1
  37. package/src/adapters/shared/fs-server-response-factory.d.ts +2 -2
  38. package/src/adapters/shared/fs-server-response-factory.js +1 -1
  39. package/src/adapters/shared/fs-server-response-matcher.d.ts +1 -1
  40. package/src/adapters/shared/hmr-entrypoint-registrar.js +1 -1
  41. package/src/adapters/shared/hmr-html-response.d.ts +1 -1
  42. package/src/adapters/shared/render-context.d.ts +2 -1
  43. package/src/adapters/shared/render-context.js +3 -1
  44. package/src/adapters/shared/runtime-bootstrap.d.ts +1 -1
  45. package/src/adapters/shared/runtime-bootstrap.js +1 -1
  46. package/src/adapters/shared/server-adapter.d.ts +1 -1
  47. package/src/adapters/shared/server-adapter.js +5 -1
  48. package/src/adapters/shared/server-route-handler.d.ts +4 -4
  49. package/src/adapters/shared/server-route-handler.js +3 -3
  50. package/src/adapters/shared/server-static-builder.d.ts +5 -5
  51. package/src/adapters/shared/server-static-builder.js +5 -4
  52. package/src/build/README.md +19 -13
  53. package/src/build/build-adapter.d.ts +67 -20
  54. package/src/build/build-adapter.js +420 -19
  55. package/src/build/dev-build-coordinator.d.ts +10 -12
  56. package/src/build/dev-build-coordinator.js +8 -15
  57. package/src/build/esbuild-build-adapter.d.ts +6 -0
  58. package/src/build/esbuild-build-adapter.js +94 -11
  59. package/src/build/runtime-build-executor.d.ts +5 -4
  60. package/src/build/runtime-build-executor.js +5 -3
  61. package/src/build/runtime-specifier-alias-plugin.js +19 -15
  62. package/src/build/runtime-specifier-aliases.d.ts +5 -0
  63. package/src/build/runtime-specifier-aliases.js +95 -0
  64. package/src/config/README.md +5 -2
  65. package/src/config/config-builder.d.ts +25 -2
  66. package/src/config/config-builder.js +44 -7
  67. package/src/declarations.d.ts +1 -1
  68. package/src/dev/sc-server.d.ts +1 -1
  69. package/src/dev/sc-server.js +1 -1
  70. package/src/eco/eco.browser.d.ts +2 -0
  71. package/src/eco/eco.browser.js +83 -0
  72. package/src/eco/eco.js +19 -48
  73. package/src/eco/eco.types.d.ts +1 -1
  74. package/src/eco/eco.utils.d.ts +1 -40
  75. package/src/eco/eco.utils.js +5 -35
  76. package/src/eco/global-injector-map.d.ts +1 -1
  77. package/src/eco/lazy-injector-map.d.ts +1 -1
  78. package/src/hmr/hmr-strategy.d.ts +16 -13
  79. package/src/hmr/hmr-strategy.js +22 -7
  80. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  81. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  82. package/src/hmr/strategies/js-hmr-strategy.d.ts +2 -2
  83. package/src/hmr/strategies/js-hmr-strategy.js +2 -2
  84. package/src/index.browser.d.ts +2 -2
  85. package/src/index.browser.js +1 -1
  86. package/src/index.d.ts +3 -2
  87. package/src/index.js +15 -4
  88. package/src/integrations/ghtml/ghtml-renderer.d.ts +6 -1
  89. package/src/integrations/ghtml/ghtml-renderer.js +29 -28
  90. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -2
  91. package/src/integrations/ghtml/ghtml.plugin.js +2 -2
  92. package/src/plugins/README.md +1 -0
  93. package/src/plugins/eco-component-meta-plugin.d.ts +12 -1
  94. package/src/plugins/eco-component-meta-plugin.js +26 -20
  95. package/src/plugins/foreign-jsx-override-plugin.d.ts +31 -0
  96. package/src/plugins/foreign-jsx-override-plugin.js +35 -0
  97. package/src/plugins/integration-plugin.d.ts +99 -33
  98. package/src/plugins/integration-plugin.js +68 -21
  99. package/src/plugins/processor.d.ts +2 -2
  100. package/src/plugins/processor.js +2 -2
  101. package/src/plugins/source-transform.d.ts +46 -0
  102. package/src/plugins/source-transform.js +71 -0
  103. package/src/route-renderer/GRAPH.md +54 -84
  104. package/src/route-renderer/README.md +14 -20
  105. package/src/route-renderer/orchestration/component-render-context.d.ts +83 -0
  106. package/src/route-renderer/orchestration/component-render-context.js +147 -0
  107. package/src/route-renderer/orchestration/integration-renderer.d.ts +233 -76
  108. package/src/route-renderer/orchestration/integration-renderer.js +500 -143
  109. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +93 -0
  110. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +155 -0
  111. package/src/route-renderer/orchestration/render-execution.service.d.ts +11 -71
  112. package/src/route-renderer/orchestration/render-execution.service.js +65 -80
  113. package/src/{eco/eco.utils.ts → route-renderer/orchestration/render-output.utils.d.ts} +10 -53
  114. package/src/route-renderer/orchestration/render-output.utils.js +65 -0
  115. package/src/route-renderer/orchestration/render-preparation.service.d.ts +2 -8
  116. package/src/route-renderer/orchestration/render-preparation.service.js +10 -17
  117. package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
  118. package/src/route-renderer/orchestration/template-serialization.js +45 -0
  119. package/src/route-renderer/page-loading/dependency-resolver.d.ts +2 -2
  120. package/src/route-renderer/page-loading/dependency-resolver.js +10 -8
  121. package/src/route-renderer/page-loading/page-module-loader.d.ts +6 -4
  122. package/src/route-renderer/page-loading/page-module-loader.js +7 -5
  123. package/src/route-renderer/route-renderer.d.ts +5 -3
  124. package/src/route-renderer/route-renderer.js +13 -3
  125. package/src/router/README.md +79 -8
  126. package/src/router/client/navigation-coordinator.js +2 -2
  127. package/src/router/server/fs-router-scanner.d.ts +1 -1
  128. package/src/router/server/fs-router-scanner.js +6 -1
  129. package/src/router/server/fs-router.d.ts +1 -1
  130. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +3 -3
  131. package/src/services/assets/asset-processing-service/asset-processing.service.js +6 -6
  132. package/src/services/assets/asset-processing-service/asset.factory.d.ts +1 -1
  133. package/src/services/assets/asset-processing-service/asset.factory.js +2 -2
  134. package/src/services/assets/asset-processing-service/index.d.ts +5 -5
  135. package/src/services/assets/asset-processing-service/index.js +5 -5
  136. package/src/services/assets/asset-processing-service/processor.interface.d.ts +2 -2
  137. package/src/services/assets/asset-processing-service/processor.registry.d.ts +2 -2
  138. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +1 -1
  139. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +1 -1
  140. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +3 -3
  141. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +2 -2
  142. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -5
  143. package/src/services/assets/asset-processing-service/processors/index.js +5 -5
  144. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +2 -2
  145. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +1 -1
  146. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +4 -3
  147. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +15 -3
  148. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +3 -3
  149. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +1 -1
  150. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +2 -2
  151. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +1 -1
  152. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  153. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +1 -1
  154. package/src/services/assets/browser-bundle.service.d.ts +1 -1
  155. package/src/services/assets/browser-bundle.service.js +2 -2
  156. package/src/services/html/html-rewriter-provider.service.js +4 -1
  157. package/src/services/html/html-transformer.service.d.ts +2 -2
  158. package/src/services/html/html-transformer.service.js +4 -10
  159. package/src/services/invalidation/development-invalidation.service.d.ts +1 -1
  160. package/src/services/invalidation/development-invalidation.service.js +1 -0
  161. package/src/services/module-loading/app-module-loader.service.d.ts +25 -0
  162. package/src/services/module-loading/app-module-loader.service.js +31 -0
  163. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +9 -1
  164. package/src/services/module-loading/app-server-module-transpiler.service.js +77 -2
  165. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  166. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  167. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  168. package/src/{adapters/node/bootstrap-dependency-resolver.d.ts → services/module-loading/node-bootstrap-plugin.d.ts} +2 -24
  169. package/src/{adapters/node/bootstrap-dependency-resolver.js → services/module-loading/node-bootstrap-plugin.js} +42 -22
  170. package/src/services/module-loading/page-module-import.service.d.ts +4 -0
  171. package/src/services/module-loading/page-module-import.service.js +38 -9
  172. package/src/services/module-loading/server-module-transpiler.service.d.ts +3 -0
  173. package/src/services/module-loading/server-module-transpiler.service.js +4 -1
  174. package/src/services/runtime-state/dev-graph.service.d.ts +6 -6
  175. package/src/services/runtime-state/dev-graph.service.js +10 -10
  176. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +1 -1
  177. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +1 -1
  178. package/src/services/runtime-state/server-invalidation-state.service.d.ts +1 -1
  179. package/src/static-site-generator/static-site-generator.d.ts +2 -2
  180. package/src/static-site-generator/static-site-generator.js +1 -1
  181. package/src/{internal-types.d.ts → types/internal-types.d.ts} +24 -14
  182. package/src/{public-types.d.ts → types/public-types.d.ts} +30 -14
  183. package/src/types/public-types.js +0 -0
  184. package/src/utils/html-escaping.d.ts +7 -0
  185. package/src/utils/html-escaping.js +6 -0
  186. package/src/utils/locals-utils.d.ts +1 -1
  187. package/src/utils/parse-cli-args.d.ts +4 -1
  188. package/src/utils/parse-cli-args.js +16 -1
  189. package/src/utils/resolve-work-dir.js +1 -1
  190. package/src/watchers/project-watcher.d.ts +4 -4
  191. package/src/watchers/project-watcher.js +4 -10
  192. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  193. package/src/adapters/abstract/application-adapter.ts +0 -337
  194. package/src/adapters/abstract/router-adapter.ts +0 -30
  195. package/src/adapters/abstract/server-adapter.ts +0 -79
  196. package/src/adapters/bun/client-bridge.ts +0 -62
  197. package/src/adapters/bun/create-app.ts +0 -189
  198. package/src/adapters/bun/hmr-manager.ts +0 -409
  199. package/src/adapters/bun/index.ts +0 -2
  200. package/src/adapters/bun/server-adapter.ts +0 -499
  201. package/src/adapters/bun/server-lifecycle.ts +0 -124
  202. package/src/adapters/index.ts +0 -6
  203. package/src/adapters/node/bootstrap-dependency-resolver.ts +0 -301
  204. package/src/adapters/node/create-app.ts +0 -179
  205. package/src/adapters/node/index.d.ts +0 -6
  206. package/src/adapters/node/index.js +0 -11
  207. package/src/adapters/node/index.ts +0 -16
  208. package/src/adapters/node/node-client-bridge.ts +0 -79
  209. package/src/adapters/node/node-hmr-manager.ts +0 -381
  210. package/src/adapters/node/runtime-adapter.d.ts +0 -46
  211. package/src/adapters/node/runtime-adapter.js +0 -306
  212. package/src/adapters/node/runtime-adapter.ts +0 -439
  213. package/src/adapters/node/server-adapter.ts +0 -488
  214. package/src/adapters/node/static-content-server.ts +0 -239
  215. package/src/adapters/node/write-runtime-manifest.d.ts +0 -26
  216. package/src/adapters/node/write-runtime-manifest.js +0 -12
  217. package/src/adapters/node/write-runtime-manifest.ts +0 -38
  218. package/src/adapters/shared/api-response.ts +0 -104
  219. package/src/adapters/shared/application-adapter.ts +0 -199
  220. package/src/adapters/shared/define-api-handler.ts +0 -66
  221. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -140
  222. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -127
  223. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  224. package/src/adapters/shared/fs-server-response-matcher.ts +0 -205
  225. package/src/adapters/shared/hmr-entrypoint-registrar.ts +0 -149
  226. package/src/adapters/shared/hmr-html-response.ts +0 -52
  227. package/src/adapters/shared/render-context.ts +0 -120
  228. package/src/adapters/shared/runtime-bootstrap.ts +0 -79
  229. package/src/adapters/shared/server-adapter.ts +0 -489
  230. package/src/adapters/shared/server-route-handler.ts +0 -153
  231. package/src/adapters/shared/server-static-builder.ts +0 -166
  232. package/src/build/build-adapter.ts +0 -361
  233. package/src/build/build-manifest.ts +0 -54
  234. package/src/build/build-types.ts +0 -83
  235. package/src/build/dev-build-coordinator.ts +0 -221
  236. package/src/build/esbuild-build-adapter.ts +0 -559
  237. package/src/build/runtime-build-executor.ts +0 -34
  238. package/src/build/runtime-specifier-alias-plugin.ts +0 -58
  239. package/src/config/config-builder.ts +0 -706
  240. package/src/constants.ts +0 -54
  241. package/src/create-app.ts +0 -87
  242. package/src/dev/sc-server.ts +0 -143
  243. package/src/eco/component-render-context.d.ts +0 -105
  244. package/src/eco/component-render-context.js +0 -87
  245. package/src/eco/component-render-context.ts +0 -224
  246. package/src/eco/eco.ts +0 -242
  247. package/src/eco/eco.types.ts +0 -221
  248. package/src/eco/global-injector-map.ts +0 -112
  249. package/src/eco/lazy-injector-map.ts +0 -120
  250. package/src/eco/module-dependencies.ts +0 -75
  251. package/src/errors/http-error.ts +0 -72
  252. package/src/errors/index.ts +0 -2
  253. package/src/errors/locals-access-error.ts +0 -7
  254. package/src/global/app-logger.ts +0 -4
  255. package/src/hmr/client/hmr-runtime.ts +0 -152
  256. package/src/hmr/hmr-strategy.ts +0 -172
  257. package/src/hmr/hmr.postcss.test.e2e.ts +0 -41
  258. package/src/hmr/hmr.test.e2e.ts +0 -66
  259. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  260. package/src/hmr/strategies/js-hmr-strategy.ts +0 -320
  261. package/src/index.browser.ts +0 -3
  262. package/src/index.ts +0 -5
  263. package/src/integrations/ghtml/ghtml-renderer.ts +0 -96
  264. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  265. package/src/internal-types.ts +0 -232
  266. package/src/plugins/alias-resolver-plugin.ts +0 -63
  267. package/src/plugins/eco-component-meta-plugin.ts +0 -481
  268. package/src/plugins/integration-plugin.ts +0 -226
  269. package/src/plugins/processor.ts +0 -240
  270. package/src/plugins/runtime-capability.ts +0 -14
  271. package/src/public-types.ts +0 -1317
  272. package/src/route-renderer/component-graph/component-graph-executor.d.ts +0 -32
  273. package/src/route-renderer/component-graph/component-graph-executor.js +0 -31
  274. package/src/route-renderer/component-graph/component-graph-executor.ts +0 -84
  275. package/src/route-renderer/component-graph/component-graph.d.ts +0 -42
  276. package/src/route-renderer/component-graph/component-graph.js +0 -72
  277. package/src/route-renderer/component-graph/component-graph.ts +0 -159
  278. package/src/route-renderer/component-graph/component-marker.d.ts +0 -52
  279. package/src/route-renderer/component-graph/component-marker.js +0 -46
  280. package/src/route-renderer/component-graph/component-marker.ts +0 -117
  281. package/src/route-renderer/component-graph/component-reference.d.ts +0 -10
  282. package/src/route-renderer/component-graph/component-reference.js +0 -19
  283. package/src/route-renderer/component-graph/component-reference.ts +0 -29
  284. package/src/route-renderer/component-graph/marker-graph-resolver.d.ts +0 -77
  285. package/src/route-renderer/component-graph/marker-graph-resolver.js +0 -95
  286. package/src/route-renderer/component-graph/marker-graph-resolver.ts +0 -155
  287. package/src/route-renderer/orchestration/integration-renderer.ts +0 -790
  288. package/src/route-renderer/orchestration/render-execution.service.ts +0 -230
  289. package/src/route-renderer/orchestration/render-preparation.service.ts +0 -476
  290. package/src/route-renderer/page-loading/dependency-resolver.ts +0 -612
  291. package/src/route-renderer/page-loading/page-module-loader.ts +0 -181
  292. package/src/route-renderer/route-renderer.ts +0 -115
  293. package/src/router/client/link-intent.test.browser.ts +0 -51
  294. package/src/router/client/link-intent.ts +0 -92
  295. package/src/router/client/navigation-coordinator.ts +0 -433
  296. package/src/router/server/fs-router-scanner.ts +0 -219
  297. package/src/router/server/fs-router.ts +0 -122
  298. package/src/services/assets/asset-processing-service/asset-processing.service.ts +0 -401
  299. package/src/services/assets/asset-processing-service/asset.factory.ts +0 -105
  300. package/src/services/assets/asset-processing-service/assets.types.ts +0 -113
  301. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +0 -95
  302. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +0 -78
  303. package/src/services/assets/asset-processing-service/index.ts +0 -5
  304. package/src/services/assets/asset-processing-service/processor.interface.ts +0 -27
  305. package/src/services/assets/asset-processing-service/processor.registry.ts +0 -18
  306. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +0 -82
  307. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +0 -95
  308. package/src/services/assets/asset-processing-service/processors/index.ts +0 -5
  309. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  310. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  311. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +0 -85
  312. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  313. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -80
  314. package/src/services/assets/browser-bundle.service.ts +0 -53
  315. package/src/services/cache/cache.types.ts +0 -126
  316. package/src/services/cache/index.ts +0 -18
  317. package/src/services/cache/memory-cache-store.ts +0 -130
  318. package/src/services/cache/page-cache-service.ts +0 -202
  319. package/src/services/cache/page-request-cache-coordinator.service.ts +0 -131
  320. package/src/services/html/html-rewriter-provider.service.ts +0 -103
  321. package/src/services/html/html-transformer.service.ts +0 -279
  322. package/src/services/invalidation/development-invalidation.service.ts +0 -261
  323. package/src/services/module-loading/app-server-module-transpiler.service.ts +0 -52
  324. package/src/services/module-loading/page-module-import.service.ts +0 -200
  325. package/src/services/module-loading/server-loader.service.d.ts +0 -96
  326. package/src/services/module-loading/server-loader.service.js +0 -32
  327. package/src/services/module-loading/server-loader.service.ts +0 -130
  328. package/src/services/module-loading/server-module-transpiler.service.ts +0 -105
  329. package/src/services/runtime-manifest/node-runtime-manifest.service.d.ts +0 -35
  330. package/src/services/runtime-manifest/node-runtime-manifest.service.js +0 -60
  331. package/src/services/runtime-manifest/node-runtime-manifest.service.ts +0 -101
  332. package/src/services/runtime-state/dev-graph.service.ts +0 -217
  333. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +0 -136
  334. package/src/services/runtime-state/runtime-specifier-registry.service.ts +0 -96
  335. package/src/services/runtime-state/server-invalidation-state.service.ts +0 -68
  336. package/src/services/validation/schema-validation-service.ts +0 -204
  337. package/src/services/validation/standard-schema.types.ts +0 -68
  338. package/src/static-site-generator/static-site-generator.ts +0 -462
  339. package/src/utils/css.d.ts +0 -1
  340. package/src/utils/css.js +0 -7
  341. package/src/utils/css.ts +0 -5
  342. package/src/utils/deep-merge.ts +0 -47
  343. package/src/utils/hash.ts +0 -5
  344. package/src/utils/html.ts +0 -1
  345. package/src/utils/invariant.ts +0 -15
  346. package/src/utils/locals-utils.ts +0 -37
  347. package/src/utils/parse-cli-args.ts +0 -83
  348. package/src/utils/path-utils.module.ts +0 -14
  349. package/src/utils/resolve-work-dir.ts +0 -45
  350. package/src/utils/runtime.ts +0 -44
  351. package/src/utils/server-utils.module.ts +0 -67
  352. package/src/watchers/project-watcher.test-helpers.ts +0 -41
  353. package/src/watchers/project-watcher.ts +0 -363
  354. /package/src/{constants.d.ts → config/constants.d.ts} +0 -0
  355. /package/src/{constants.js → config/constants.js} +0 -0
  356. /package/src/{internal-types.js → services/module-loading/module-loading-types.js} +0 -0
  357. /package/src/{public-types.js → types/internal-types.js} +0 -0
@@ -1,559 +0,0 @@
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 { pathToFileURL } from 'node:url';
10
- import { fileSystem } from '@ecopages/file-system';
11
- import type {
12
- EcoBuildOnLoadResult,
13
- EcoBuildPlugin,
14
- EcoBuildPluginBuilder,
15
- EcoBuildOnResolveResult,
16
- } from './build-types.ts';
17
- import type {
18
- BuildAdapter,
19
- BuildDependencyGraph,
20
- BuildLog,
21
- BuildOptions,
22
- BuildResult,
23
- BuildTranspileOptions,
24
- BuildTranspileProfile,
25
- } from './build-adapter.ts';
26
-
27
- const esbuildRequire = createRequire(import.meta.url);
28
-
29
- /**
30
- * Provides common transpile output defaults shared across build profiles.
31
- */
32
- function transpileProfileToOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
33
- switch (profile) {
34
- case 'browser-script':
35
- return {
36
- target: 'browser',
37
- format: 'esm',
38
- sourcemap: 'none',
39
- };
40
- case 'hmr-runtime':
41
- return {
42
- target: 'browser',
43
- format: 'esm',
44
- sourcemap: 'none',
45
- };
46
- case 'hmr-entrypoint':
47
- return {
48
- target: 'browser',
49
- format: 'esm',
50
- sourcemap: 'none',
51
- };
52
- }
53
- }
54
-
55
- /**
56
- * Node build adapter backed by esbuild.
57
- *
58
- * This adapter keeps Ecopages build plugin compatibility (`onResolve`, `onLoad`,
59
- * and `module`) while delegating bundling and TypeScript/decorator transforms to esbuild.
60
- */
61
- export class EsbuildBuildAdapter implements BuildAdapter {
62
- private escapeRegExp(value: string): string {
63
- return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
64
- }
65
-
66
- private getPluginsForBuild(additionalPlugins?: EcoBuildPlugin[]): EcoBuildPlugin[] {
67
- const byName = new Map<string, EcoBuildPlugin>();
68
-
69
- for (const plugin of additionalPlugins ?? []) {
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(moduleGeneration = 0): Promise<typeof import('esbuild')> {
290
- const importedModule = await import('esbuild');
291
- const esbuildModule = (
292
- typeof (importedModule as typeof import('esbuild')).build === 'function'
293
- ? importedModule
294
- : ((importedModule as { default?: typeof import('esbuild') }).default ?? importedModule)
295
- ) as typeof import('esbuild');
296
-
297
- if (moduleGeneration > 0 && !this.isMockedEsbuildModule(esbuildModule)) {
298
- const freshModule = await import(
299
- `${pathToFileURL(esbuildRequire.resolve('esbuild')).href}?ecopages_esbuild=${moduleGeneration}`
300
- );
301
- const normalizedFreshModule = (
302
- typeof (freshModule as typeof import('esbuild')).build === 'function'
303
- ? freshModule
304
- : ((freshModule as { default?: typeof import('esbuild') }).default ?? freshModule)
305
- ) as typeof import('esbuild');
306
-
307
- if (typeof normalizedFreshModule.build === 'function') {
308
- return normalizedFreshModule;
309
- }
310
- }
311
-
312
- if (typeof esbuildModule.build !== 'function') {
313
- throw new Error('esbuild is not available. Install esbuild to use Node bundling.');
314
- }
315
-
316
- return esbuildModule;
317
- }
318
-
319
- private isMockedEsbuildModule(esbuildModule: typeof import('esbuild')): boolean {
320
- const build = esbuildModule.build as { mock?: unknown } | undefined;
321
- const stop = esbuildModule.stop as { mock?: unknown } | undefined;
322
- return typeof build?.mock === 'object' || typeof stop?.mock === 'object';
323
- }
324
-
325
- /**
326
- * Detects the subset of runtime faults that indicate esbuild's worker
327
- * protocol is corrupted rather than a normal build error.
328
- */
329
- isEsbuildProtocolError(error: unknown): boolean {
330
- if (!(error instanceof Error)) {
331
- return false;
332
- }
333
-
334
- return ['Unexpected end of JSON input', 'Unexpected EOF', 'parseJSON', 'buildResponseToResult'].some(
335
- (fragment) => error.message.includes(fragment) || error.stack?.includes(fragment),
336
- );
337
- }
338
-
339
- async stopEsbuildService(moduleGeneration = 0): Promise<void> {
340
- const esbuild = await this.loadEsbuildModule(moduleGeneration);
341
- if (typeof esbuild.stop === 'function') {
342
- esbuild.stop();
343
- }
344
- }
345
-
346
- async buildOrThrow(options: BuildOptions, moduleGeneration = 0): Promise<BuildResult> {
347
- const esbuild = await this.loadEsbuildModule(moduleGeneration);
348
- const contextRoot = options.root ? path.resolve(options.root) : process.cwd();
349
- const outdir = path.resolve(options.outdir ?? 'dist/assets');
350
- const tsconfigPath = path.join(contextRoot, 'tsconfig.json');
351
- const tsconfigExists = fileSystem.exists(tsconfigPath);
352
-
353
- const plugins = this.getPluginsForBuild(options.plugins);
354
- const esbuildPlugins: EsbuildPlugin[] = [
355
- ...(plugins.length > 0 ? [this.createEcoPluginBridge(plugins, contextRoot)] : []),
356
- ];
357
- const transpileTarget = 'es2022';
358
-
359
- const usesTemplatedNaming = this.hasTemplateTokens(options.naming);
360
- const outfile = options.naming && !usesTemplatedNaming ? path.join(outdir, options.naming) : undefined;
361
- if (outfile) {
362
- fileSystem.ensureDir(path.dirname(outfile));
363
- }
364
-
365
- const outputOptions = outfile
366
- ? { outfile }
367
- : {
368
- outdir,
369
- ...(options.outbase ? { outbase: path.resolve(options.outbase) } : {}),
370
- entryNames: usesTemplatedNaming ? this.toEntryNamePattern(options.naming) : '[name]',
371
- chunkNames: '[name]-[hash]',
372
- assetNames: '[name]-[hash]',
373
- };
374
-
375
- const result = await esbuild.build({
376
- absWorkingDir: contextRoot,
377
- entryPoints: options.entrypoints,
378
- bundle: options.bundle ?? true,
379
- ...outputOptions,
380
- ...(options.conditions ? { conditions: options.conditions } : {}),
381
- ...(options.define ? { define: options.define } : {}),
382
- format: this.mapEsbuildFormat(options.format),
383
- platform: (options.target === 'browser' ? 'browser' : 'node') as 'browser' | 'node',
384
- sourcemap: this.mapEsbuildSourcemap(options.sourcemap),
385
- splitting: outfile ? false : !!options.splitting,
386
- minify: !!options.minify,
387
- ...(typeof options.treeshaking === 'boolean' ? { treeShaking: options.treeshaking } : {}),
388
- external: options.external,
389
- ...(options.target !== 'browser' && options.externalPackages !== false
390
- ? { packages: 'external' as const }
391
- : {}),
392
- target: transpileTarget,
393
- metafile: true,
394
- write: true,
395
- plugins: esbuildPlugins,
396
- jsx: 'automatic',
397
- tsconfig: tsconfigExists ? tsconfigPath : undefined,
398
- logLevel: 'silent',
399
- });
400
-
401
- const outputs = Object.keys(result.metafile.outputs).map((outputPath) => ({
402
- path: path.isAbsolute(outputPath) ? outputPath : path.join(contextRoot, outputPath),
403
- }));
404
- const logs = result.warnings.map((warning) => ({ message: warning.text }));
405
- const dependencyGraph = this.extractDependencyGraph(result.metafile, contextRoot);
406
-
407
- return {
408
- success: true,
409
- logs,
410
- outputs,
411
- dependencyGraph,
412
- };
413
- }
414
-
415
- private mapEsbuildSourcemap(value: string | undefined): false | 'linked' | 'inline' | 'external' | 'both' {
416
- switch (value) {
417
- case 'none':
418
- return false;
419
- case 'inline':
420
- return 'inline';
421
- case 'both':
422
- return 'both';
423
- case 'external':
424
- return 'external';
425
- default:
426
- return 'linked';
427
- }
428
- }
429
-
430
- private mapEsbuildFormat(value: string | undefined): 'esm' | 'cjs' | 'iife' {
431
- switch (value) {
432
- case 'cjs':
433
- return 'cjs';
434
- case 'iife':
435
- return 'iife';
436
- default:
437
- return 'esm';
438
- }
439
- }
440
-
441
- private hasTemplateTokens(value: string | undefined): boolean {
442
- return typeof value === 'string' && /\[[^\]]+\]/.test(value);
443
- }
444
-
445
- private toEntryNamePattern(value: string | undefined): string {
446
- if (!value) {
447
- return '[name]';
448
- }
449
-
450
- const pattern = value.replaceAll(/\.?\[ext\]/g, '');
451
- return pattern.length > 0 ? pattern : '[name]';
452
- }
453
-
454
- private normalizeMetafilePath(value: string, contextRoot: string): string {
455
- if (path.isAbsolute(value)) {
456
- return path.normalize(value);
457
- }
458
-
459
- return path.normalize(path.resolve(contextRoot, value));
460
- }
461
-
462
- private extractDependencyGraph(
463
- metafile: {
464
- outputs: Record<string, { entryPoint?: string; inputs?: Record<string, unknown> }>;
465
- },
466
- contextRoot: string,
467
- ): BuildDependencyGraph {
468
- const entrypoints = new Map<string, Set<string>>();
469
-
470
- for (const outputMeta of Object.values(metafile.outputs)) {
471
- if (!outputMeta.entryPoint) {
472
- continue;
473
- }
474
-
475
- const entrypointPath = this.normalizeMetafilePath(outputMeta.entryPoint, contextRoot);
476
- const dependencies = entrypoints.get(entrypointPath) ?? new Set<string>();
477
- dependencies.add(entrypointPath);
478
-
479
- for (const inputPath of Object.keys(outputMeta.inputs ?? {})) {
480
- dependencies.add(this.normalizeMetafilePath(inputPath, contextRoot));
481
- }
482
-
483
- entrypoints.set(entrypointPath, dependencies);
484
- }
485
-
486
- return {
487
- entrypoints: Object.fromEntries(
488
- Array.from(entrypoints.entries(), ([entrypointPath, dependencies]) => [
489
- entrypointPath,
490
- Array.from(dependencies),
491
- ]),
492
- ),
493
- };
494
- }
495
-
496
- /**
497
- * Normalizes esbuild errors into Ecopages `BuildLog` entries.
498
- */
499
- private toBuildLogs(error: unknown): BuildLog[] {
500
- if (error && typeof error === 'object') {
501
- const candidate = error as {
502
- errors?: Array<{ text?: string; location?: { file?: string; line?: number; column?: number } }>;
503
- message?: string;
504
- };
505
-
506
- if (Array.isArray(candidate.errors) && candidate.errors.length > 0) {
507
- return candidate.errors.map((entry) => {
508
- const locationPrefix = entry.location?.file
509
- ? `${entry.location.file}:${entry.location.line ?? 0}:${entry.location.column ?? 0} `
510
- : '';
511
-
512
- return {
513
- message: `${locationPrefix}${entry.text ?? 'Unknown esbuild error'}`,
514
- };
515
- });
516
- }
517
-
518
- if (typeof candidate.message === 'string') {
519
- return [{ message: candidate.message }];
520
- }
521
- }
522
-
523
- return [{ message: 'Unknown esbuild error' }];
524
- }
525
-
526
- createFailureResult(error: unknown): BuildResult {
527
- return {
528
- success: false,
529
- logs: this.toBuildLogs(error),
530
- outputs: [],
531
- };
532
- }
533
-
534
- /**
535
- * Bundles entrypoints using esbuild for Node runtime builds.
536
- */
537
- async build(options: BuildOptions): Promise<BuildResult> {
538
- try {
539
- return await this.buildOrThrow(options);
540
- } catch (error) {
541
- return this.createFailureResult(error);
542
- }
543
- }
544
-
545
- /**
546
- * Resolves module specifiers from a project root.
547
- */
548
- resolve(importPath: string, rootDir: string): string {
549
- const localRequire = createRequire(path.join(rootDir, 'package.json'));
550
- return localRequire.resolve(importPath);
551
- }
552
-
553
- /**
554
- * Returns transpile defaults for a known transpile profile.
555
- */
556
- getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
557
- return transpileProfileToOptions(profile);
558
- }
559
- }
@@ -1,34 +0,0 @@
1
- import type { EcoPagesAppConfig } from '../internal-types.ts';
2
- import {
3
- getAppBuildAdapter,
4
- getAppBuildExecutor,
5
- getAppServerBuildPlugins,
6
- setAppBuildExecutor,
7
- type BuildExecutor,
8
- } from './build-adapter.ts';
9
- import { createOrReuseAppBuildExecutor } from './dev-build-coordinator.ts';
10
-
11
- /**
12
- * Installs the app-owned runtime build executor for one app instance.
13
- *
14
- * @remarks
15
- * This is the single coordinator boundary for runtime executor ownership across
16
- * adapters. It preserves an existing development coordinator when available and
17
- * always applies app server build plugins through one shared path.
18
- */
19
- export function installAppRuntimeBuildExecutor(
20
- appConfig: EcoPagesAppConfig,
21
- options: {
22
- development: boolean;
23
- },
24
- ): BuildExecutor {
25
- const buildExecutor = createOrReuseAppBuildExecutor({
26
- development: options.development,
27
- adapter: getAppBuildAdapter(appConfig),
28
- currentExecutor: getAppBuildExecutor(appConfig),
29
- getPlugins: () => getAppServerBuildPlugins(appConfig),
30
- });
31
-
32
- setAppBuildExecutor(appConfig, buildExecutor);
33
- return buildExecutor;
34
- }
@@ -1,58 +0,0 @@
1
- import type { EcoBuildPlugin } from './build-types.ts';
2
-
3
- type RuntimeSpecifierMap = ReadonlyMap<string, string> | Record<string, string>;
4
-
5
- /**
6
- * Normalizes runtime specifier input into a read-only map shape.
7
- */
8
- function toRuntimeSpecifierMap(specifierMap: RuntimeSpecifierMap): ReadonlyMap<string, string> {
9
- return specifierMap instanceof Map ? specifierMap : new Map(Object.entries(specifierMap));
10
- }
11
-
12
- /**
13
- * Escapes a literal specifier for inclusion in a regular expression.
14
- */
15
- function escapeRegExp(value: string): string {
16
- return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
17
- }
18
-
19
- /**
20
- * Creates a build plugin that aliases runtime bare specifiers to concrete URLs.
21
- *
22
- * @remarks
23
- * This helper is used when browser-target builds must preserve integration-owned
24
- * runtime specifier semantics while letting the bundler treat the mapped URLs as
25
- * external runtime assets.
26
- */
27
- export function createRuntimeSpecifierAliasPlugin(
28
- specifierMapInput: RuntimeSpecifierMap,
29
- options?: {
30
- name?: string;
31
- external?: boolean;
32
- },
33
- ): EcoBuildPlugin | null {
34
- const specifierMap = toRuntimeSpecifierMap(specifierMapInput);
35
-
36
- if (specifierMap.size === 0) {
37
- return null;
38
- }
39
-
40
- const filter = new RegExp(`^(${Array.from(specifierMap.keys()).map(escapeRegExp).join('|')})$`);
41
-
42
- return {
43
- name: options?.name ?? 'runtime-specifier-alias',
44
- setup(build) {
45
- build.onResolve({ filter }, (args) => {
46
- const mappedPath = specifierMap.get(args.path);
47
- if (!mappedPath) {
48
- return undefined;
49
- }
50
-
51
- return {
52
- path: mappedPath,
53
- external: options?.external ?? true,
54
- };
55
- });
56
- },
57
- };
58
- }