@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,358 @@
1
+ import { createRequire } from 'node:module';
2
+ import type { EcoPagesAppConfig } from '../internal-types.ts';
3
+ import type {
4
+ ComponentRenderResult,
5
+ EcoComponent,
6
+ EcoPageComponent,
7
+ EcoPageFile,
8
+ EcoPagesElement,
9
+ GetMetadata,
10
+ GetStaticProps,
11
+ HtmlTemplateProps,
12
+ IntegrationRendererRenderOptions,
13
+ PageProps,
14
+ PageMetadataProps,
15
+ ResolvedLazyTrigger,
16
+ RouteRendererOptions,
17
+ } from '../public-types.ts';
18
+ import {
19
+ type AssetProcessingService,
20
+ AssetFactory,
21
+ type ProcessedAsset,
22
+ } from '../services/asset-processing-service/index.ts';
23
+ import { buildGlobalInjectorBootstrapContent, buildGlobalInjectorMapScript } from '../eco/global-injector-map.ts';
24
+ import { runWithComponentRenderContext, type ComponentRenderBoundaryContext } from '../eco/component-render-context.ts';
25
+
26
+ const coreRequire = createRequire(import.meta.url);
27
+
28
+ type ResolvedPageModule = {
29
+ Page: EcoPageFile['default'] | EcoPageComponent<any>;
30
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
31
+ getMetadata?: GetMetadata;
32
+ integrationSpecificProps: Record<string, unknown>;
33
+ };
34
+
35
+ export interface RenderPreparationCallbacks {
36
+ resolvePageModule(file: string): Promise<ResolvedPageModule>;
37
+ getHtmlTemplate(): Promise<EcoComponent<HtmlTemplateProps>>;
38
+ resolvePageData(
39
+ pageModule: {
40
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
41
+ getMetadata?: GetMetadata;
42
+ },
43
+ routeOptions: RouteRendererOptions,
44
+ ): Promise<{ props: Record<string, unknown>; metadata: PageMetadataProps }>;
45
+ resolveDependencies(components: (EcoComponent | Partial<EcoComponent>)[]): Promise<ProcessedAsset[]>;
46
+ buildRouteRenderAssets(file: string): Promise<ProcessedAsset[]> | undefined;
47
+ shouldRenderPageComponent(input: {
48
+ Page: EcoComponent;
49
+ Layout?: EcoComponent;
50
+ options: RouteRendererOptions;
51
+ }): boolean;
52
+ renderPageComponent(input: {
53
+ component: EcoComponent;
54
+ props: Record<string, unknown>;
55
+ }): Promise<ComponentRenderResult>;
56
+ /**
57
+ * Returns the boundary policy context that should be active while rendering
58
+ * page-root component output during preparation.
59
+ */
60
+ getComponentRenderBoundaryContext(): ComponentRenderBoundaryContext;
61
+ setProcessedDependencies(dependencies: ProcessedAsset[]): void;
62
+ dedupeProcessedAssets(assets: ProcessedAsset[]): ProcessedAsset[];
63
+ createPageLocalsProxy(filePath: string): RouteRendererOptions['locals'];
64
+ }
65
+
66
+ /**
67
+ * Prepares the normalized render inputs consumed by `IntegrationRenderer.execute()`.
68
+ *
69
+ * This service owns the orchestration that happens before the main HTML render:
70
+ * page module resolution, data loading, dependency aggregation, page-root
71
+ * component artifact capture, lazy trigger bootstrap generation, and request
72
+ * locals policy.
73
+ */
74
+ export class RenderPreparationService {
75
+ constructor(
76
+ private appConfig: EcoPagesAppConfig,
77
+ private assetProcessingService: AssetProcessingService,
78
+ ) {}
79
+
80
+ /**
81
+ * Builds the final render options object used by the integration-specific
82
+ * renderer.
83
+ *
84
+ * The returned object contains normalized page data, processed dependency
85
+ * state, component render artifacts, and the locals contract expected by the
86
+ * rest of the pipeline.
87
+ *
88
+ * @typeParam C Integration render output element type.
89
+ * @param routeOptions Route-level render inputs.
90
+ * @param currentIntegrationName Active integration name for this preparation pass.
91
+ * @param callbacks Renderer-specific hooks used during preparation.
92
+ * @returns Normalized render options.
93
+ */
94
+ async prepare<C = EcoPagesElement>(
95
+ routeOptions: RouteRendererOptions,
96
+ currentIntegrationName: string,
97
+ callbacks: RenderPreparationCallbacks,
98
+ ): Promise<IntegrationRendererRenderOptions<C>> {
99
+ const pageModule = await callbacks.resolvePageModule(routeOptions.file);
100
+ const { Page, integrationSpecificProps } = pageModule;
101
+ const HtmlTemplate = await callbacks.getHtmlTemplate();
102
+ const { props, metadata } = await callbacks.resolvePageData(pageModule, routeOptions);
103
+ const Layout = Page.config?.layout;
104
+
105
+ const componentsToResolve = Layout ? [HtmlTemplate, Layout, Page] : [HtmlTemplate, Page];
106
+ const resolvedDependencies = await callbacks.resolveDependencies(componentsToResolve);
107
+ const usedIntegrationDependencies = this.collectUsedIntegrationDependencies(
108
+ componentsToResolve,
109
+ currentIntegrationName,
110
+ );
111
+ const pageDeps = (await callbacks.buildRouteRenderAssets(routeOptions.file)) || [];
112
+ const allDependencies = [...resolvedDependencies, ...usedIntegrationDependencies, ...pageDeps];
113
+
114
+ let componentRender: ComponentRenderResult | undefined;
115
+ if (callbacks.shouldRenderPageComponent({ Page, Layout, options: routeOptions })) {
116
+ componentRender = await this.renderPageRoot({
117
+ currentIntegrationName,
118
+ Page: Page as EcoComponent,
119
+ props,
120
+ routeOptions,
121
+ callbacks,
122
+ });
123
+
124
+ if (componentRender.assets?.length) {
125
+ allDependencies.push(...componentRender.assets);
126
+ }
127
+ }
128
+
129
+ const triggers = this.collectResolvedTriggers(componentsToResolve);
130
+ if (triggers.length > 0) {
131
+ const globalAssets = await this.buildGlobalInjectorAssets(triggers, currentIntegrationName);
132
+ allDependencies.push(...globalAssets);
133
+ }
134
+
135
+ callbacks.setProcessedDependencies(callbacks.dedupeProcessedAssets(allDependencies));
136
+
137
+ const pageProps = {
138
+ ...props,
139
+ params: routeOptions.params || {},
140
+ query: routeOptions.query || {},
141
+ };
142
+
143
+ const cacheStrategy = (Page as EcoPageComponent<any>).cache;
144
+ const defaultCacheStrategy = this.appConfig.cache?.defaultStrategy ?? 'static';
145
+ const effectiveCacheStrategy = cacheStrategy ?? defaultCacheStrategy;
146
+ const localsAvailable = effectiveCacheStrategy === 'dynamic' && routeOptions.locals !== undefined;
147
+
148
+ const pageLocals = localsAvailable ? routeOptions.locals! : callbacks.createPageLocalsProxy(routeOptions.file);
149
+
150
+ const locals = localsAvailable ? routeOptions.locals : undefined;
151
+ const preparedOptions: IntegrationRendererRenderOptions<C> = {
152
+ ...routeOptions,
153
+ resolvedDependencies,
154
+ componentRender,
155
+ HtmlTemplate: HtmlTemplate as EcoComponent<HtmlTemplateProps, C>,
156
+ Layout,
157
+ props,
158
+ Page: Page as EcoComponent<PageProps, C>,
159
+ metadata,
160
+ params: routeOptions.params || {},
161
+ query: routeOptions.query || {},
162
+ pageProps,
163
+ locals,
164
+ pageLocals,
165
+ cacheStrategy,
166
+ };
167
+
168
+ return {
169
+ ...integrationSpecificProps,
170
+ ...preparedOptions,
171
+ };
172
+ }
173
+
174
+ /**
175
+ * Collects resolved lazy trigger metadata from the component tree.
176
+ *
177
+ * Traversal is depth-first and deduplicated by component identity so shared
178
+ * component dependencies do not emit duplicate trigger sets.
179
+ *
180
+ * @param components Root component set.
181
+ * @param seen Internal visited set for shared graphs.
182
+ * @returns All resolved lazy triggers reachable from the root set.
183
+ */
184
+ private collectResolvedTriggers(
185
+ components: (EcoComponent | Partial<EcoComponent>)[],
186
+ seen = new Set<object>(),
187
+ ): ResolvedLazyTrigger[] {
188
+ const triggers: ResolvedLazyTrigger[] = [];
189
+ for (const comp of components) {
190
+ const ecoComp = comp as EcoComponent;
191
+ if (seen.has(ecoComp)) {
192
+ continue;
193
+ }
194
+ seen.add(ecoComp);
195
+ const ownTriggers = ecoComp.config?._resolvedLazyTriggers;
196
+ if (ownTriggers?.length) {
197
+ triggers.push(...ownTriggers);
198
+ }
199
+ const nested = ecoComp.config?.dependencies?.components;
200
+ if (nested?.length) {
201
+ triggers.push(...this.collectResolvedTriggers(nested, seen));
202
+ }
203
+ }
204
+ return triggers;
205
+ }
206
+
207
+ /**
208
+ * Collects global integration dependencies used by nested components belonging
209
+ * to integrations other than the current renderer.
210
+ *
211
+ * @param components Root component set.
212
+ * @returns Processed integration dependencies contributed by nested integrations.
213
+ */
214
+ private collectUsedIntegrationDependencies(
215
+ components: (EcoComponent | Partial<EcoComponent>)[],
216
+ currentIntegrationName: string,
217
+ ): ProcessedAsset[] {
218
+ const integrationNames = this.collectIntegrationNames(components);
219
+ const dependencies: ProcessedAsset[] = [];
220
+
221
+ for (const integrationName of integrationNames) {
222
+ if (integrationName === currentIntegrationName) {
223
+ continue;
224
+ }
225
+
226
+ const integrationPlugin = this.appConfig.integrations.find(
227
+ (integration) => integration.name === integrationName,
228
+ );
229
+ if (!integrationPlugin || typeof integrationPlugin.getResolvedIntegrationDependencies !== 'function') {
230
+ continue;
231
+ }
232
+
233
+ dependencies.push(...integrationPlugin.getResolvedIntegrationDependencies());
234
+ }
235
+
236
+ return dependencies;
237
+ }
238
+
239
+ /**
240
+ * Discovers integration names referenced by the component dependency graph.
241
+ *
242
+ * @param components Root component set.
243
+ * @param seen Internal visited set for shared graphs.
244
+ * @returns Set of integration names found in the graph.
245
+ */
246
+ private collectIntegrationNames(
247
+ components: (EcoComponent | Partial<EcoComponent>)[],
248
+ seen = new Set<object>(),
249
+ ): Set<string> {
250
+ const integrationNames = new Set<string>();
251
+
252
+ for (const comp of components) {
253
+ const ecoComp = comp as EcoComponent;
254
+ if (seen.has(ecoComp)) {
255
+ continue;
256
+ }
257
+ seen.add(ecoComp);
258
+
259
+ const integrationName = ecoComp.config?.integration ?? ecoComp.config?.__eco?.integration;
260
+ if (integrationName) {
261
+ integrationNames.add(integrationName);
262
+ }
263
+
264
+ const nested = ecoComp.config?.dependencies?.components;
265
+ if (nested?.length) {
266
+ const nestedNames = this.collectIntegrationNames(nested, seen);
267
+ for (const nestedName of nestedNames) {
268
+ integrationNames.add(nestedName);
269
+ }
270
+ }
271
+ }
272
+
273
+ return integrationNames;
274
+ }
275
+
276
+ /**
277
+ * Renders the page root through the component-level render contract so any
278
+ * integration-specific assets and root attributes are available before the main
279
+ * document render.
280
+ *
281
+ * @param input Page root render inputs.
282
+ * @returns Structured component render result.
283
+ */
284
+ private async renderPageRoot(input: {
285
+ currentIntegrationName: string;
286
+ Page: EcoComponent;
287
+ props: Record<string, unknown>;
288
+ routeOptions: RouteRendererOptions;
289
+ callbacks: RenderPreparationCallbacks;
290
+ }): Promise<ComponentRenderResult> {
291
+ const execution = await runWithComponentRenderContext(
292
+ {
293
+ currentIntegration: input.currentIntegrationName,
294
+ boundaryContext: input.callbacks.getComponentRenderBoundaryContext(),
295
+ },
296
+ async () =>
297
+ input.callbacks.renderPageComponent({
298
+ component: input.Page,
299
+ props: {
300
+ ...input.props,
301
+ params: input.routeOptions.params || {},
302
+ query: input.routeOptions.query || {},
303
+ },
304
+ }),
305
+ );
306
+
307
+ return execution.value;
308
+ }
309
+
310
+ /**
311
+ * Builds the runtime assets needed to bootstrap global lazy trigger execution.
312
+ *
313
+ * @param triggers Fully resolved lazy trigger definitions.
314
+ * @returns Processed assets that should be merged into the final dependency set.
315
+ */
316
+ private async buildGlobalInjectorAssets(
317
+ triggers: ResolvedLazyTrigger[],
318
+ currentIntegrationName: string,
319
+ ): Promise<ProcessedAsset[]> {
320
+ const globalInjectorImportPath = coreRequire.resolve('@ecopages/scripts-injector/global');
321
+ const globalInjectorRuntimeAsset = AssetFactory.createNodeModuleScript({
322
+ position: 'head',
323
+ name: 'ecopages-scripts-injector-global',
324
+ importPath: globalInjectorImportPath,
325
+ excludeFromHtml: true,
326
+ });
327
+
328
+ const [globalInjectorRuntimeProcessed] = await this.assetProcessingService.processDependencies(
329
+ [globalInjectorRuntimeAsset],
330
+ currentIntegrationName,
331
+ );
332
+
333
+ const globalInjectorModuleUrl = globalInjectorRuntimeProcessed?.srcUrl;
334
+ if (!globalInjectorModuleUrl) {
335
+ throw new Error('[ecopages] Failed to resolve global injector runtime asset URL.');
336
+ }
337
+
338
+ const mapScript = AssetFactory.createInlineContentScript({
339
+ position: 'head',
340
+ name: 'ecopages-global-injector-map',
341
+ content: buildGlobalInjectorMapScript(triggers),
342
+ attributes: { type: 'ecopages/global-injector-map' },
343
+ bundle: false,
344
+ });
345
+ const bootstrapScript = AssetFactory.createContentScript({
346
+ position: 'head',
347
+ name: 'ecopages-global-injector-bootstrap',
348
+ content: buildGlobalInjectorBootstrapContent(globalInjectorModuleUrl),
349
+ attributes: { type: 'module' },
350
+ bundle: false,
351
+ });
352
+
353
+ return this.assetProcessingService.processDependencies(
354
+ [mapScript, bootstrapScript, globalInjectorRuntimeAsset],
355
+ currentIntegrationName,
356
+ );
357
+ }
358
+ }
@@ -0,0 +1,26 @@
1
+ import type { EcoPagesAppConfig } from '../internal-types.js';
2
+ import type { IntegrationPlugin } from '../plugins/integration-plugin.js';
3
+ import type { RouteRenderResult, RouteRendererOptions } from '../public-types.js';
4
+ import type { IntegrationRenderer } from './integration-renderer.js';
5
+ export declare class RouteRenderer {
6
+ private renderer;
7
+ constructor(renderer: IntegrationRenderer);
8
+ createRoute(options: RouteRendererOptions): Promise<RouteRenderResult>;
9
+ }
10
+ export declare class RouteRendererFactory {
11
+ private appConfig;
12
+ runtimeOrigin: string;
13
+ private rendererCache;
14
+ constructor({ appConfig, runtimeOrigin }: {
15
+ appConfig: EcoPagesAppConfig;
16
+ runtimeOrigin: string;
17
+ });
18
+ createRenderer(filePath: string): RouteRenderer;
19
+ /**
20
+ * Get an integration renderer by its name.
21
+ * Used for explicit routing where views specify their integration via __eco.integration.
22
+ */
23
+ getRendererByIntegration(integrationName: string): IntegrationRenderer | null;
24
+ getIntegrationPlugin(filePath: string): IntegrationPlugin;
25
+ private getRouteRendererEngine;
26
+ }
@@ -0,0 +1,68 @@
1
+ import { invariant } from "../utils/invariant.js";
2
+ import { PathUtils } from "../utils/path-utils.module.js";
3
+ class RouteRenderer {
4
+ renderer;
5
+ constructor(renderer) {
6
+ this.renderer = renderer;
7
+ }
8
+ async createRoute(options) {
9
+ return this.renderer.execute(options);
10
+ }
11
+ }
12
+ class RouteRendererFactory {
13
+ appConfig;
14
+ runtimeOrigin;
15
+ rendererCache = /* @__PURE__ */ new Map();
16
+ constructor({ appConfig, runtimeOrigin }) {
17
+ this.appConfig = appConfig;
18
+ this.runtimeOrigin = runtimeOrigin;
19
+ }
20
+ createRenderer(filePath) {
21
+ const integrationRenderer = this.getRouteRendererEngine(filePath);
22
+ invariant(!!integrationRenderer, `No integration renderer found for file: ${filePath}`);
23
+ return new RouteRenderer(integrationRenderer);
24
+ }
25
+ /**
26
+ * Get an integration renderer by its name.
27
+ * Used for explicit routing where views specify their integration via __eco.integration.
28
+ */
29
+ getRendererByIntegration(integrationName) {
30
+ const integrationPlugin = this.appConfig.integrations.find((plugin) => plugin.name === integrationName);
31
+ if (!integrationPlugin) {
32
+ return null;
33
+ }
34
+ const cached = this.rendererCache.get(integrationName);
35
+ if (cached) {
36
+ return cached;
37
+ }
38
+ const renderer = integrationPlugin.initializeRenderer();
39
+ this.rendererCache.set(integrationName, renderer);
40
+ return renderer;
41
+ }
42
+ getIntegrationPlugin(filePath) {
43
+ const templateExtension = PathUtils.getEcoTemplateExtension(filePath);
44
+ const isIntegrationPlugin = (plugin) => {
45
+ return plugin.extensions.some((extension) => templateExtension === extension);
46
+ };
47
+ const integrationPlugin = this.appConfig.integrations.find(isIntegrationPlugin);
48
+ invariant(
49
+ !!integrationPlugin,
50
+ `No integration plugin found for template extension: ${templateExtension}, file: ${filePath}`
51
+ );
52
+ return integrationPlugin;
53
+ }
54
+ getRouteRendererEngine(filePath) {
55
+ const integrationPlugin = this.getIntegrationPlugin(filePath);
56
+ const cached = this.rendererCache.get(integrationPlugin.name);
57
+ if (cached) {
58
+ return cached;
59
+ }
60
+ const renderer = integrationPlugin.initializeRenderer();
61
+ this.rendererCache.set(integrationPlugin.name, renderer);
62
+ return renderer;
63
+ }
64
+ }
65
+ export {
66
+ RouteRenderer,
67
+ RouteRendererFactory
68
+ };
@@ -0,0 +1,80 @@
1
+ import type { EcoPagesAppConfig } from '../internal-types.ts';
2
+ import type { IntegrationPlugin } from '../plugins/integration-plugin.ts';
3
+ import type { RouteRenderResult, RouteRendererOptions } from '../public-types.ts';
4
+ import { invariant } from '../utils/invariant.ts';
5
+ import { PathUtils } from '../utils/path-utils.module.ts';
6
+ import type { IntegrationRenderer } from './integration-renderer.ts';
7
+
8
+ export class RouteRenderer {
9
+ private renderer: IntegrationRenderer;
10
+
11
+ constructor(renderer: IntegrationRenderer) {
12
+ this.renderer = renderer;
13
+ }
14
+
15
+ async createRoute(options: RouteRendererOptions): Promise<RouteRenderResult> {
16
+ return this.renderer.execute(options);
17
+ }
18
+ }
19
+
20
+ export class RouteRendererFactory {
21
+ private appConfig: EcoPagesAppConfig;
22
+ runtimeOrigin: string;
23
+ private rendererCache = new Map<string, IntegrationRenderer>();
24
+
25
+ constructor({ appConfig, runtimeOrigin }: { appConfig: EcoPagesAppConfig; runtimeOrigin: string }) {
26
+ this.appConfig = appConfig;
27
+ this.runtimeOrigin = runtimeOrigin;
28
+ }
29
+
30
+ createRenderer(filePath: string): RouteRenderer {
31
+ const integrationRenderer = this.getRouteRendererEngine(filePath);
32
+ invariant(!!integrationRenderer, `No integration renderer found for file: ${filePath}`);
33
+ return new RouteRenderer(integrationRenderer);
34
+ }
35
+
36
+ /**
37
+ * Get an integration renderer by its name.
38
+ * Used for explicit routing where views specify their integration via __eco.integration.
39
+ */
40
+ getRendererByIntegration(integrationName: string): IntegrationRenderer | null {
41
+ const integrationPlugin = this.appConfig.integrations.find((plugin) => plugin.name === integrationName);
42
+ if (!integrationPlugin) {
43
+ return null;
44
+ }
45
+
46
+ const cached = this.rendererCache.get(integrationName);
47
+ if (cached) {
48
+ return cached;
49
+ }
50
+
51
+ const renderer = integrationPlugin.initializeRenderer();
52
+ this.rendererCache.set(integrationName, renderer);
53
+ return renderer;
54
+ }
55
+
56
+ getIntegrationPlugin(filePath: string): IntegrationPlugin {
57
+ const templateExtension = PathUtils.getEcoTemplateExtension(filePath);
58
+ const isIntegrationPlugin = (plugin: IntegrationPlugin): boolean => {
59
+ return plugin.extensions.some((extension) => templateExtension === extension);
60
+ };
61
+ const integrationPlugin = this.appConfig.integrations.find(isIntegrationPlugin);
62
+ invariant(
63
+ !!integrationPlugin,
64
+ `No integration plugin found for template extension: ${templateExtension}, file: ${filePath}`,
65
+ );
66
+ return integrationPlugin as IntegrationPlugin;
67
+ }
68
+
69
+ private getRouteRendererEngine(filePath: string): IntegrationRenderer {
70
+ const integrationPlugin = this.getIntegrationPlugin(filePath);
71
+ const cached = this.rendererCache.get(integrationPlugin.name);
72
+ if (cached) {
73
+ return cached;
74
+ }
75
+
76
+ const renderer = integrationPlugin.initializeRenderer();
77
+ this.rendererCache.set(integrationPlugin.name, renderer);
78
+ return renderer;
79
+ }
80
+ }
@@ -0,0 +1,41 @@
1
+ import type { EcoPagesAppConfig, Routes } from '../internal-types.js';
2
+ type FSRouterScannerOptions = {
3
+ buildMode: boolean;
4
+ };
5
+ /**
6
+ * @class FSRouterScanner
7
+ * @description
8
+ * This class is responsible for scanning the file system for routes.
9
+ * It uses the glob package to scan the file system for files with the specified file extensions.
10
+ * It then creates a map of the routes with the pathname as the key.
11
+ * The pathname is the route without the file extension.
12
+ * For example, if the file is "index.tsx", the pathname will be "/index".
13
+ * If the file is "blog/[slug].tsx", the pathname will be "/blog/[slug]".
14
+ * If the file is "blog/[...slug].tsx", the pathname will be "/blog/[...slug]".
15
+ */
16
+ export declare class FSRouterScanner {
17
+ private dir;
18
+ private origin;
19
+ private templatesExt;
20
+ private options;
21
+ private appConfig;
22
+ routes: Routes;
23
+ private pageModuleImportService;
24
+ constructor({ dir, origin, templatesExt, options, appConfig, }: {
25
+ dir: string;
26
+ origin: string;
27
+ templatesExt: string[];
28
+ options: FSRouterScannerOptions;
29
+ appConfig: EcoPagesAppConfig;
30
+ });
31
+ private getRoutePath;
32
+ private getDynamicParamsNames;
33
+ private getStaticPathsFromDynamicRoute;
34
+ private createStaticRoutes;
35
+ private handleDynamicRouteCreation;
36
+ private importPageModule;
37
+ private createRoute;
38
+ private getRouteData;
39
+ scan(): Promise<Routes>;
40
+ }
41
+ export {};