@ecopages/core 0.2.0-alpha.1 → 0.2.0-alpha.11

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 (372) hide show
  1. package/CHANGELOG.md +16 -64
  2. package/README.md +210 -13
  3. package/package.json +88 -62
  4. package/src/adapters/README.md +39 -0
  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 +80 -21
  13. package/src/adapters/bun/hmr-manager.js +163 -56
  14. package/src/adapters/bun/index.d.ts +2 -3
  15. package/src/adapters/bun/index.js +3 -3
  16. package/src/adapters/bun/server-adapter.d.ts +5 -5
  17. package/src/adapters/bun/server-adapter.js +40 -34
  18. package/src/adapters/bun/server-lifecycle.d.ts +28 -17
  19. package/src/adapters/bun/server-lifecycle.js +34 -62
  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 +2 -6
  23. package/src/adapters/index.js +2 -8
  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 +89 -18
  28. package/src/adapters/node/node-hmr-manager.js +180 -89
  29. package/src/adapters/node/server-adapter.d.ts +4 -33
  30. package/src/adapters/node/server-adapter.js +39 -100
  31. package/src/adapters/node/static-content-server.d.ts +37 -1
  32. package/src/adapters/node/static-content-server.js +29 -1
  33. package/src/adapters/shared/application-adapter.d.ts +1 -1
  34. package/src/{define-api-handler.d.ts → 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/explicit-static-route-matcher.js +4 -1
  37. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +1 -1
  38. package/src/adapters/shared/file-route-middleware-pipeline.js +1 -0
  39. package/src/adapters/shared/fs-server-response-factory.d.ts +2 -2
  40. package/src/adapters/shared/fs-server-response-factory.js +1 -1
  41. package/src/adapters/shared/fs-server-response-matcher.d.ts +9 -5
  42. package/src/adapters/shared/fs-server-response-matcher.js +13 -8
  43. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
  44. package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
  45. package/src/adapters/shared/hmr-html-response.d.ts +22 -0
  46. package/src/adapters/shared/hmr-html-response.js +32 -0
  47. package/src/adapters/shared/render-context.d.ts +2 -1
  48. package/src/adapters/shared/render-context.js +6 -3
  49. package/src/adapters/shared/runtime-bootstrap.d.ts +38 -0
  50. package/src/adapters/shared/runtime-bootstrap.js +43 -0
  51. package/src/adapters/shared/server-adapter.d.ts +13 -3
  52. package/src/adapters/shared/server-adapter.js +42 -5
  53. package/src/adapters/shared/server-route-handler.d.ts +4 -4
  54. package/src/adapters/shared/server-route-handler.js +6 -15
  55. package/src/adapters/shared/server-static-builder.d.ts +38 -6
  56. package/src/adapters/shared/server-static-builder.js +64 -10
  57. package/src/build/README.md +107 -0
  58. package/src/build/build-adapter.d.ts +160 -3
  59. package/src/build/build-adapter.js +494 -16
  60. package/src/build/build-manifest.d.ts +27 -0
  61. package/src/build/build-manifest.js +30 -0
  62. package/src/build/dev-build-coordinator.d.ts +72 -0
  63. package/src/build/dev-build-coordinator.js +154 -0
  64. package/src/build/esbuild-build-adapter.d.ts +15 -6
  65. package/src/build/esbuild-build-adapter.js +189 -74
  66. package/src/build/runtime-build-executor.d.ts +14 -0
  67. package/src/build/runtime-build-executor.js +22 -0
  68. package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
  69. package/src/build/runtime-specifier-alias-plugin.js +35 -0
  70. package/src/build/runtime-specifier-aliases.d.ts +5 -0
  71. package/src/build/runtime-specifier-aliases.js +95 -0
  72. package/src/config/README.md +36 -0
  73. package/src/config/config-builder.d.ts +52 -18
  74. package/src/config/config-builder.js +258 -48
  75. package/src/{constants.d.ts → config/constants.d.ts} +13 -0
  76. package/src/{constants.js → config/constants.js} +4 -0
  77. package/src/declarations.d.ts +19 -14
  78. package/src/dev/sc-server.d.ts +1 -1
  79. package/src/dev/sc-server.js +1 -1
  80. package/src/eco/README.md +70 -16
  81. package/src/eco/component-render-context.d.ts +2 -2
  82. package/src/eco/component-render-context.js +33 -16
  83. package/src/eco/eco.browser.d.ts +2 -0
  84. package/src/eco/eco.browser.js +83 -0
  85. package/src/eco/eco.js +91 -16
  86. package/src/eco/eco.types.d.ts +12 -4
  87. package/src/eco/eco.utils.d.ts +1 -1
  88. package/src/eco/global-injector-map.d.ts +1 -1
  89. package/src/eco/lazy-injector-map.d.ts +1 -1
  90. package/src/hmr/README.md +26 -0
  91. package/src/hmr/client/hmr-runtime.d.ts +1 -6
  92. package/src/hmr/client/hmr-runtime.js +30 -7
  93. package/src/hmr/hmr-strategy.d.ts +8 -7
  94. package/src/hmr/hmr-strategy.js +22 -7
  95. package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
  96. package/src/hmr/hmr.postcss.test.e2e.js +31 -0
  97. package/src/hmr/hmr.test.e2e.js +26 -33
  98. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  99. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  100. package/src/hmr/strategies/js-hmr-strategy.d.ts +46 -43
  101. package/src/hmr/strategies/js-hmr-strategy.js +72 -73
  102. package/src/index.browser.d.ts +2 -2
  103. package/src/index.browser.js +1 -1
  104. package/src/index.d.ts +4 -3
  105. package/src/index.js +16 -5
  106. package/src/integrations/ghtml/ghtml-renderer.d.ts +2 -2
  107. package/src/integrations/ghtml/ghtml-renderer.js +3 -1
  108. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -2
  109. package/src/integrations/ghtml/ghtml.plugin.js +2 -2
  110. package/src/plugins/README.md +35 -0
  111. package/src/plugins/alias-resolver-plugin.js +17 -3
  112. package/src/plugins/eco-component-meta-plugin.d.ts +14 -1
  113. package/src/plugins/eco-component-meta-plugin.js +27 -21
  114. package/src/plugins/integration-plugin.d.ts +48 -9
  115. package/src/plugins/integration-plugin.js +39 -2
  116. package/src/plugins/processor.d.ts +15 -2
  117. package/src/plugins/processor.js +16 -2
  118. package/src/plugins/runtime-capability.d.ts +9 -0
  119. package/src/plugins/source-transform.d.ts +46 -0
  120. package/src/plugins/source-transform.js +71 -0
  121. package/src/route-renderer/GRAPH.md +18 -22
  122. package/src/route-renderer/README.md +15 -26
  123. package/src/route-renderer/{component-graph-executor.d.ts → component-graph/component-graph-executor.d.ts} +3 -2
  124. package/src/route-renderer/{component-graph-executor.js → component-graph/component-graph-executor.js} +2 -3
  125. package/src/route-renderer/{component-graph.d.ts → component-graph/component-graph.d.ts} +13 -2
  126. package/src/route-renderer/{component-graph.js → component-graph/component-graph.js} +26 -4
  127. package/src/route-renderer/component-graph/component-reference.d.ts +11 -0
  128. package/src/route-renderer/component-graph/component-reference.js +39 -0
  129. package/src/route-renderer/{marker-graph-resolver.d.ts → component-graph/marker-graph-resolver.d.ts} +9 -6
  130. package/src/route-renderer/{marker-graph-resolver.js → component-graph/marker-graph-resolver.js} +36 -12
  131. package/src/route-renderer/{integration-renderer.d.ts → orchestration/integration-renderer.d.ts} +65 -13
  132. package/src/route-renderer/{integration-renderer.js → orchestration/integration-renderer.js} +206 -30
  133. package/src/route-renderer/{render-execution.service.d.ts → orchestration/render-execution.service.d.ts} +42 -6
  134. package/src/route-renderer/orchestration/render-execution.service.js +191 -0
  135. package/src/route-renderer/{render-preparation.service.d.ts → orchestration/render-preparation.service.d.ts} +13 -4
  136. package/src/route-renderer/{render-preparation.service.js → orchestration/render-preparation.service.js} +95 -5
  137. package/src/route-renderer/{dependency-resolver.d.ts → page-loading/dependency-resolver.d.ts} +15 -4
  138. package/src/route-renderer/{dependency-resolver.js → page-loading/dependency-resolver.js} +18 -4
  139. package/src/route-renderer/page-loading/page-module-loader.d.ts +89 -0
  140. package/src/route-renderer/{page-module-loader.js → page-loading/page-module-loader.js} +38 -14
  141. package/src/route-renderer/route-renderer.d.ts +41 -4
  142. package/src/route-renderer/route-renderer.js +32 -3
  143. package/src/router/README.md +97 -0
  144. package/src/router/client/link-intent.d.ts +53 -0
  145. package/src/router/client/link-intent.js +34 -0
  146. package/src/router/client/link-intent.test.browser.d.ts +1 -0
  147. package/src/router/client/link-intent.test.browser.js +43 -0
  148. package/src/router/client/navigation-coordinator.d.ts +149 -0
  149. package/src/router/client/navigation-coordinator.js +215 -0
  150. package/src/router/{fs-router-scanner.d.ts → server/fs-router-scanner.d.ts} +3 -3
  151. package/src/router/{fs-router-scanner.js → server/fs-router-scanner.js} +8 -7
  152. package/src/router/{fs-router.d.ts → server/fs-router.d.ts} +1 -1
  153. package/src/router/{fs-router.js → server/fs-router.js} +1 -1
  154. package/src/services/README.md +29 -0
  155. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +120 -0
  156. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.js +91 -10
  157. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.d.ts +1 -1
  158. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.js +2 -2
  159. package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.d.ts +2 -1
  160. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +55 -0
  161. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +48 -0
  162. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
  163. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
  164. package/src/services/assets/asset-processing-service/index.d.ts +5 -0
  165. package/src/services/assets/asset-processing-service/index.js +5 -0
  166. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.d.ts +2 -2
  167. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.d.ts +2 -2
  168. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.d.ts +1 -1
  169. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.js +9 -4
  170. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-script-processor.d.ts +5 -4
  171. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-script-processor.js +15 -23
  172. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
  173. package/src/services/assets/asset-processing-service/processors/index.js +5 -0
  174. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.d.ts +2 -2
  175. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.js +1 -1
  176. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.d.ts +4 -3
  177. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.js +16 -4
  178. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.d.ts +3 -3
  179. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/node-module-script.processor.js +6 -5
  180. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.d.ts +2 -2
  181. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.js +1 -1
  182. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  183. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.js +5 -2
  184. package/src/services/assets/browser-bundle.service.d.ts +32 -0
  185. package/src/services/assets/browser-bundle.service.js +33 -0
  186. package/src/services/{page-request-cache-coordinator.service.d.ts → cache/page-request-cache-coordinator.service.d.ts} +2 -2
  187. package/src/services/{page-request-cache-coordinator.service.js → cache/page-request-cache-coordinator.service.js} +3 -1
  188. package/src/services/html/html-rewriter-provider.service.d.ts +37 -0
  189. package/src/services/html/html-rewriter-provider.service.js +68 -0
  190. package/src/services/html/html-transformer.service.d.ts +77 -0
  191. package/src/services/html/html-transformer.service.js +215 -0
  192. package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
  193. package/src/services/invalidation/development-invalidation.service.js +190 -0
  194. package/src/services/module-loading/app-module-loader.service.d.ts +25 -0
  195. package/src/services/module-loading/app-module-loader.service.js +31 -0
  196. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +24 -0
  197. package/src/services/module-loading/app-server-module-transpiler.service.js +109 -0
  198. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  199. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  200. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  201. package/src/services/module-loading/node-bootstrap-plugin.d.ts +22 -0
  202. package/src/services/module-loading/node-bootstrap-plugin.js +179 -0
  203. package/src/services/module-loading/page-module-import.service.d.ts +75 -0
  204. package/src/services/module-loading/page-module-import.service.js +161 -0
  205. package/src/services/module-loading/server-module-transpiler.service.d.ts +72 -0
  206. package/src/services/module-loading/server-module-transpiler.service.js +64 -0
  207. package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
  208. package/src/services/runtime-state/dev-graph.service.js +162 -0
  209. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
  210. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
  211. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +69 -0
  212. package/src/services/runtime-state/runtime-specifier-registry.service.js +37 -0
  213. package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
  214. package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
  215. package/src/services/{schema-validation-service.d.ts → validation/schema-validation-service.d.ts} +1 -1
  216. package/src/static-site-generator/README.md +26 -0
  217. package/src/static-site-generator/static-site-generator.d.ts +55 -3
  218. package/src/static-site-generator/static-site-generator.js +86 -5
  219. package/src/{internal-types.d.ts → types/internal-types.d.ts} +53 -22
  220. package/src/types/internal-types.js +0 -0
  221. package/src/{public-types.d.ts → types/public-types.d.ts} +73 -17
  222. package/src/types/public-types.js +0 -0
  223. package/src/utils/locals-utils.d.ts +1 -1
  224. package/src/utils/parse-cli-args.d.ts +4 -1
  225. package/src/utils/parse-cli-args.js +16 -1
  226. package/src/utils/resolve-work-dir.d.ts +11 -0
  227. package/src/utils/resolve-work-dir.js +31 -0
  228. package/src/watchers/project-watcher.d.ts +40 -24
  229. package/src/watchers/project-watcher.js +105 -68
  230. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  231. package/src/watchers/project-watcher.test-helpers.js +1 -0
  232. package/src/adapters/abstract/application-adapter.ts +0 -337
  233. package/src/adapters/abstract/router-adapter.ts +0 -30
  234. package/src/adapters/abstract/server-adapter.ts +0 -79
  235. package/src/adapters/bun/client-bridge.ts +0 -62
  236. package/src/adapters/bun/create-app.ts +0 -189
  237. package/src/adapters/bun/define-api-handler.d.ts +0 -61
  238. package/src/adapters/bun/define-api-handler.ts +0 -114
  239. package/src/adapters/bun/hmr-manager.ts +0 -281
  240. package/src/adapters/bun/index.ts +0 -3
  241. package/src/adapters/bun/server-adapter.ts +0 -492
  242. package/src/adapters/bun/server-lifecycle.ts +0 -154
  243. package/src/adapters/index.ts +0 -6
  244. package/src/adapters/node/create-app.ts +0 -179
  245. package/src/adapters/node/index.d.ts +0 -4
  246. package/src/adapters/node/index.js +0 -8
  247. package/src/adapters/node/index.ts +0 -9
  248. package/src/adapters/node/node-client-bridge.ts +0 -79
  249. package/src/adapters/node/node-hmr-manager.ts +0 -271
  250. package/src/adapters/node/server-adapter.ts +0 -561
  251. package/src/adapters/node/static-content-server.ts +0 -203
  252. package/src/adapters/shared/api-response.ts +0 -104
  253. package/src/adapters/shared/application-adapter.ts +0 -199
  254. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -134
  255. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -123
  256. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  257. package/src/adapters/shared/fs-server-response-matcher.ts +0 -198
  258. package/src/adapters/shared/render-context.ts +0 -105
  259. package/src/adapters/shared/server-adapter.ts +0 -442
  260. package/src/adapters/shared/server-route-handler.ts +0 -166
  261. package/src/adapters/shared/server-static-builder.ts +0 -82
  262. package/src/build/build-adapter.ts +0 -132
  263. package/src/build/build-types.ts +0 -83
  264. package/src/build/esbuild-build-adapter.ts +0 -510
  265. package/src/config/config-builder.ts +0 -474
  266. package/src/constants.ts +0 -39
  267. package/src/create-app.ts +0 -87
  268. package/src/define-api-handler.js +0 -15
  269. package/src/define-api-handler.ts +0 -66
  270. package/src/dev/sc-server.ts +0 -143
  271. package/src/eco/component-render-context.ts +0 -202
  272. package/src/eco/eco.ts +0 -221
  273. package/src/eco/eco.types.ts +0 -202
  274. package/src/eco/eco.utils.ts +0 -89
  275. package/src/eco/global-injector-map.ts +0 -112
  276. package/src/eco/lazy-injector-map.ts +0 -120
  277. package/src/eco/module-dependencies.ts +0 -75
  278. package/src/errors/http-error.ts +0 -72
  279. package/src/errors/index.ts +0 -2
  280. package/src/errors/locals-access-error.ts +0 -7
  281. package/src/global/app-logger.ts +0 -4
  282. 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
  283. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  284. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  285. package/src/hmr/client/hmr-runtime.ts +0 -121
  286. package/src/hmr/hmr-strategy.ts +0 -172
  287. package/src/hmr/hmr.test.e2e.ts +0 -75
  288. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  289. package/src/hmr/strategies/js-hmr-strategy.ts +0 -308
  290. package/src/index.browser.ts +0 -3
  291. package/src/index.ts +0 -5
  292. package/src/integrations/ghtml/ghtml-renderer.ts +0 -93
  293. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  294. package/src/internal-types.ts +0 -212
  295. package/src/plugins/alias-resolver-plugin.ts +0 -45
  296. package/src/plugins/eco-component-meta-plugin.ts +0 -474
  297. package/src/plugins/integration-plugin.ts +0 -184
  298. package/src/plugins/processor.ts +0 -220
  299. package/src/public-types.ts +0 -1255
  300. package/src/route-renderer/component-graph-executor.ts +0 -84
  301. package/src/route-renderer/component-graph.ts +0 -159
  302. package/src/route-renderer/component-marker.ts +0 -117
  303. package/src/route-renderer/dependency-resolver.ts +0 -596
  304. package/src/route-renderer/html-post-processing.service.d.ts +0 -40
  305. package/src/route-renderer/html-post-processing.service.js +0 -86
  306. package/src/route-renderer/html-post-processing.service.ts +0 -103
  307. package/src/route-renderer/integration-renderer.ts +0 -696
  308. package/src/route-renderer/marker-graph-resolver.ts +0 -153
  309. package/src/route-renderer/page-module-loader.d.ts +0 -61
  310. package/src/route-renderer/page-module-loader.ts +0 -153
  311. package/src/route-renderer/render-execution.service.js +0 -91
  312. package/src/route-renderer/render-execution.service.ts +0 -158
  313. package/src/route-renderer/render-preparation.service.ts +0 -358
  314. package/src/route-renderer/route-renderer.ts +0 -80
  315. package/src/router/fs-router-scanner.ts +0 -217
  316. package/src/router/fs-router.ts +0 -122
  317. package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
  318. package/src/services/asset-processing-service/asset-processing.service.ts +0 -306
  319. package/src/services/asset-processing-service/asset.factory.ts +0 -105
  320. package/src/services/asset-processing-service/assets.types.ts +0 -112
  321. package/src/services/asset-processing-service/index.d.ts +0 -3
  322. package/src/services/asset-processing-service/index.js +0 -3
  323. package/src/services/asset-processing-service/index.ts +0 -3
  324. package/src/services/asset-processing-service/processor.interface.ts +0 -27
  325. package/src/services/asset-processing-service/processor.registry.ts +0 -18
  326. package/src/services/asset-processing-service/processors/base/base-processor.ts +0 -76
  327. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +0 -105
  328. package/src/services/asset-processing-service/processors/index.d.ts +0 -5
  329. package/src/services/asset-processing-service/processors/index.js +0 -5
  330. package/src/services/asset-processing-service/processors/index.ts +0 -5
  331. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  332. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  333. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +0 -84
  334. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  335. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -77
  336. package/src/services/cache/cache.types.ts +0 -126
  337. package/src/services/cache/index.ts +0 -18
  338. package/src/services/cache/memory-cache-store.ts +0 -130
  339. package/src/services/cache/page-cache-service.ts +0 -202
  340. package/src/services/html-transformer.service.d.ts +0 -50
  341. package/src/services/html-transformer.service.js +0 -163
  342. package/src/services/html-transformer.service.ts +0 -217
  343. package/src/services/page-module-import.service.d.ts +0 -37
  344. package/src/services/page-module-import.service.js +0 -88
  345. package/src/services/page-module-import.service.ts +0 -129
  346. package/src/services/page-request-cache-coordinator.service.ts +0 -128
  347. package/src/services/schema-validation-service.ts +0 -204
  348. package/src/services/validation/standard-schema.types.ts +0 -68
  349. package/src/static-site-generator/static-site-generator.ts +0 -359
  350. package/src/utils/css.d.ts +0 -1
  351. package/src/utils/css.js +0 -7
  352. package/src/utils/css.ts +0 -5
  353. package/src/utils/deep-merge.ts +0 -47
  354. package/src/utils/hash.ts +0 -5
  355. package/src/utils/html.ts +0 -1
  356. package/src/utils/invariant.ts +0 -15
  357. package/src/utils/locals-utils.ts +0 -37
  358. package/src/utils/parse-cli-args.ts +0 -83
  359. package/src/utils/path-utils.module.ts +0 -14
  360. package/src/utils/runtime.ts +0 -44
  361. package/src/utils/server-utils.module.ts +0 -67
  362. package/src/watchers/project-watcher.test-helpers.ts +0 -40
  363. package/src/watchers/project-watcher.ts +0 -306
  364. /package/src/adapters/{bun → shared}/define-api-handler.js +0 -0
  365. /package/src/{internal-types.js → plugins/runtime-capability.js} +0 -0
  366. /package/src/route-renderer/{component-marker.d.ts → component-graph/component-marker.d.ts} +0 -0
  367. /package/src/route-renderer/{component-marker.js → component-graph/component-marker.js} +0 -0
  368. /package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.js +0 -0
  369. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.js +0 -0
  370. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.js +0 -0
  371. /package/src/{public-types.js → services/module-loading/module-loading-types.js} +0 -0
  372. /package/src/services/{schema-validation-service.js → validation/schema-validation-service.js} +0 -0
@@ -1,11 +1,23 @@
1
1
  import { createRequire } from "node:module";
2
+ import path from "node:path";
3
+ import { fileURLToPath } from "node:url";
2
4
  import {
3
5
  AssetFactory
4
- } from "../services/asset-processing-service/index.js";
5
- import { buildGlobalInjectorBootstrapContent, buildGlobalInjectorMapScript } from "../eco/global-injector-map.js";
6
- import { runWithComponentRenderContext } from "../eco/component-render-context.js";
7
- const coreRequire = createRequire(import.meta.url);
6
+ } from "../../services/assets/asset-processing-service/index.js";
7
+ import { buildGlobalInjectorBootstrapContent, buildGlobalInjectorMapScript } from "../../eco/global-injector-map.js";
8
+ import {
9
+ runWithComponentRenderContext
10
+ } from "../../eco/component-render-context.js";
8
11
  class RenderPreparationService {
12
+ appConfig;
13
+ assetProcessingService;
14
+ /**
15
+ * Creates the render-preparation orchestrator for one app instance.
16
+ *
17
+ * @remarks
18
+ * The service is app-scoped because it depends on finalized config defaults and
19
+ * the app-owned asset-processing pipeline while remaining renderer-agnostic.
20
+ */
9
21
  constructor(appConfig, assetProcessingService) {
10
22
  this.appConfig = appConfig;
11
23
  this.assetProcessingService = assetProcessingService;
@@ -56,6 +68,10 @@ class RenderPreparationService {
56
68
  const globalAssets = await this.buildGlobalInjectorAssets(triggers, currentIntegrationName);
57
69
  allDependencies.push(...globalAssets);
58
70
  }
71
+ const eagerSsrLazyAssets = await this.buildEagerSsrLazyAssets(componentsToResolve, currentIntegrationName);
72
+ if (eagerSsrLazyAssets.length > 0) {
73
+ allDependencies.push(...eagerSsrLazyAssets);
74
+ }
59
75
  callbacks.setProcessedDependencies(callbacks.dedupeProcessedAssets(allDependencies));
60
76
  const pageProps = {
61
77
  ...props,
@@ -203,7 +219,7 @@ class RenderPreparationService {
203
219
  * @returns Processed assets that should be merged into the final dependency set.
204
220
  */
205
221
  async buildGlobalInjectorAssets(triggers, currentIntegrationName) {
206
- const globalInjectorImportPath = coreRequire.resolve("@ecopages/scripts-injector/global");
222
+ const globalInjectorImportPath = createRequire(import.meta.url).resolve("@ecopages/scripts-injector/global");
207
223
  const globalInjectorRuntimeAsset = AssetFactory.createNodeModuleScript({
208
224
  position: "head",
209
225
  name: "ecopages-scripts-injector-global",
@@ -237,6 +253,80 @@ class RenderPreparationService {
237
253
  currentIntegrationName
238
254
  );
239
255
  }
256
+ async buildEagerSsrLazyAssets(components, currentIntegrationName) {
257
+ const dependencies = this.collectEagerSsrLazyDependencies(components);
258
+ if (dependencies.length === 0) {
259
+ return [];
260
+ }
261
+ return this.assetProcessingService.processDependencies(dependencies, `${currentIntegrationName}:ssr-lazy`);
262
+ }
263
+ collectEagerSsrLazyDependencies(components) {
264
+ const dependencies = [];
265
+ const visitedConfigs = /* @__PURE__ */ new Set();
266
+ const seenKeys = /* @__PURE__ */ new Set();
267
+ const normalizeAttributes = (attributes) => ({
268
+ type: "module",
269
+ defer: "",
270
+ ...attributes ?? {}
271
+ });
272
+ const collect = (config) => {
273
+ if (!config || visitedConfigs.has(config)) {
274
+ return;
275
+ }
276
+ visitedConfigs.add(config);
277
+ const componentFile = config.__eco?.file;
278
+ if (componentFile) {
279
+ const componentDir = path.dirname(componentFile);
280
+ for (const script of config.dependencies?.scripts ?? []) {
281
+ if (typeof script === "string" || !script.lazy || script.ssr !== true) {
282
+ continue;
283
+ }
284
+ const attributes = normalizeAttributes(script.attributes);
285
+ if (script.content) {
286
+ const key2 = `content:${script.content}:${JSON.stringify(attributes)}`;
287
+ if (seenKeys.has(key2)) {
288
+ continue;
289
+ }
290
+ seenKeys.add(key2);
291
+ dependencies.push(
292
+ AssetFactory.createContentScript({
293
+ position: "head",
294
+ content: script.content,
295
+ attributes
296
+ })
297
+ );
298
+ continue;
299
+ }
300
+ if (!script.src) {
301
+ continue;
302
+ }
303
+ const resolvedPath = path.resolve(componentDir, script.src);
304
+ const key = `file:${resolvedPath}:${JSON.stringify(attributes)}`;
305
+ if (seenKeys.has(key)) {
306
+ continue;
307
+ }
308
+ seenKeys.add(key);
309
+ dependencies.push(
310
+ AssetFactory.createFileScript({
311
+ filepath: resolvedPath,
312
+ position: "head",
313
+ attributes
314
+ })
315
+ );
316
+ }
317
+ }
318
+ if (config.layout?.config) {
319
+ collect(config.layout.config);
320
+ }
321
+ for (const nestedComponent of config.dependencies?.components ?? []) {
322
+ collect(nestedComponent?.config);
323
+ }
324
+ };
325
+ for (const component of components) {
326
+ collect(component.config);
327
+ }
328
+ return dependencies;
329
+ }
240
330
  }
241
331
  export {
242
332
  RenderPreparationService
@@ -1,6 +1,6 @@
1
- import type { EcoComponent } from '../public-types.js';
2
- import type { EcoPagesAppConfig } from '../internal-types.js';
3
- import type { AssetProcessingService, ProcessedAsset } from '../services/asset-processing-service/index.js';
1
+ import type { EcoComponent } from '../../types/public-types.js';
2
+ import type { EcoPagesAppConfig } from '../../types/internal-types.js';
3
+ import type { AssetProcessingService, ProcessedAsset } from '../../services/assets/asset-processing-service/index.js';
4
4
  export declare const DEPENDENCY_ERRORS: {
5
5
  readonly INVALID_STYLESHEET_ENTRY: "Invalid stylesheet dependency entry: expected src or content";
6
6
  readonly INVALID_SCRIPT_ENTRY: "Invalid script dependency entry: expected src or content";
@@ -9,7 +9,18 @@ export declare const DEPENDENCY_ERRORS: {
9
9
  export declare class DependencyResolverService {
10
10
  private appConfig;
11
11
  private assetProcessingService;
12
+ /**
13
+ * Creates the dependency resolver used by route and component rendering.
14
+ *
15
+ * @remarks
16
+ * The resolver stays intentionally separate from HTML rendering so component
17
+ * dependency collection, lazy trigger grouping, and processed-asset generation
18
+ * can evolve without changing renderer implementations.
19
+ */
12
20
  constructor(appConfig: EcoPagesAppConfig, assetProcessingService: AssetProcessingService);
21
+ /**
22
+ * Resolves one dependency path relative to the component that declared it.
23
+ */
13
24
  resolveDependencyPath(componentDir: string, pathUrl: string): string;
14
25
  /**
15
26
  * Maps lazy script source entries to deterministic fallback public URLs
@@ -20,5 +31,5 @@ export declare class DependencyResolverService {
20
31
  * Collects and processes component dependencies (styles, scripts, modules, lazy scripts).
21
32
  * Lazy dependencies are always resolved into global-injector trigger maps.
22
33
  */
23
- processComponentDependencies(components: (EcoComponent | Partial<EcoComponent>)[], integrationName: string): Promise<ProcessedAsset[]>;
34
+ processComponentDependencies(components: Array<EcoComponent | Partial<EcoComponent> | undefined | null>, integrationName: string): Promise<ProcessedAsset[]>;
24
35
  }
@@ -1,8 +1,8 @@
1
1
  import path from "node:path";
2
2
  import { readFileSync } from "node:fs";
3
- import { rapidhash } from "../utils/hash.js";
4
- import { AssetFactory } from "../services/asset-processing-service/index.js";
5
- import { normalizeModuleDeclarations } from "../eco/module-dependencies.js";
3
+ import { rapidhash } from "../../utils/hash.js";
4
+ import { AssetFactory } from "../../services/assets/asset-processing-service/index.js";
5
+ import { normalizeModuleDeclarations } from "../../eco/module-dependencies.js";
6
6
  import { parseSync } from "oxc-parser";
7
7
  const DEPENDENCY_ERRORS = {
8
8
  INVALID_STYLESHEET_ENTRY: "Invalid stylesheet dependency entry: expected src or content",
@@ -139,10 +139,23 @@ function buildResolvedLazyTriggers(config, groups) {
139
139
  return [{ triggerId, rules }];
140
140
  }
141
141
  class DependencyResolverService {
142
+ appConfig;
143
+ assetProcessingService;
144
+ /**
145
+ * Creates the dependency resolver used by route and component rendering.
146
+ *
147
+ * @remarks
148
+ * The resolver stays intentionally separate from HTML rendering so component
149
+ * dependency collection, lazy trigger grouping, and processed-asset generation
150
+ * can evolve without changing renderer implementations.
151
+ */
142
152
  constructor(appConfig, assetProcessingService) {
143
153
  this.appConfig = appConfig;
144
154
  this.assetProcessingService = assetProcessingService;
145
155
  }
156
+ /**
157
+ * Resolves one dependency path relative to the component that declared it.
158
+ */
146
159
  resolveDependencyPath(componentDir, pathUrl) {
147
160
  return resolveDependencyPath(componentDir, pathUrl);
148
161
  }
@@ -163,6 +176,7 @@ class DependencyResolverService {
163
176
  const lazyScriptsByConfig = /* @__PURE__ */ new Map();
164
177
  const lazyDependencyKeys = /* @__PURE__ */ new Set();
165
178
  for (const component of components) {
179
+ if (!component) continue;
166
180
  const componentFile = component.config?.__eco?.file;
167
181
  if (!componentFile) continue;
168
182
  const stylesheetDependencyKeys = /* @__PURE__ */ new Set();
@@ -357,7 +371,7 @@ class DependencyResolverService {
357
371
  }
358
372
  if (dependenciesConfig?.components) {
359
373
  for (const nestedComponent of dependenciesConfig.components) {
360
- if (nestedComponent.config) {
374
+ if (nestedComponent?.config) {
361
375
  collect(nestedComponent.config);
362
376
  }
363
377
  }
@@ -0,0 +1,89 @@
1
+ import type { EcoPageFile, GetMetadata, GetMetadataContext, GetStaticProps, PageMetadataProps, RouteRendererOptions, EcoPageComponent } from '../../types/public-types.js';
2
+ import type { EcoPagesAppConfig } from '../../types/internal-types.js';
3
+ /**
4
+ * Loads route page modules and normalizes their data hooks for rendering.
5
+ *
6
+ * @remarks
7
+ * This service keeps the render pipeline from depending directly on raw module
8
+ * imports. It owns the shared server-module transpiler setup, the precedence
9
+ * rules between component statics and module exports, and the normalization of
10
+ * page props and metadata into one renderer-facing shape.
11
+ */
12
+ export declare class PageModuleLoaderService {
13
+ private appModuleLoader;
14
+ private appConfig;
15
+ private runtimeOrigin;
16
+ /**
17
+ * Creates the page-module loader for one app/runtime instance.
18
+ *
19
+ * @param appConfig Finalized app config that owns build and invalidation state.
20
+ * @param runtimeOrigin Runtime origin exposed to page data hooks.
21
+ */
22
+ constructor(appConfig: EcoPagesAppConfig, runtimeOrigin: string);
23
+ /**
24
+ * Imports one page module through the shared server-side module loading path.
25
+ *
26
+ * @remarks
27
+ * The underlying transpiler keeps Bun and Node aligned on one framework-owned
28
+ * loading contract even though the runtime-specific execution transport differs.
29
+ */
30
+ importPageFile(file: string, options?: {
31
+ bypassCache?: boolean;
32
+ }): Promise<EcoPageFile>;
33
+ /**
34
+ * Executes the page's static-props hook with Ecopages runtime context.
35
+ *
36
+ * @remarks
37
+ * Pages without a static-props hook still return a normalized empty props
38
+ * object so downstream render preparation does not branch on hook presence.
39
+ */
40
+ getStaticPropsForPage(options: {
41
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
42
+ params?: RouteRendererOptions['params'];
43
+ }): Promise<{
44
+ props: Record<string, unknown>;
45
+ metadata?: PageMetadataProps;
46
+ }>;
47
+ /**
48
+ * Builds the final page metadata object for one render request.
49
+ *
50
+ * @remarks
51
+ * App-level default metadata forms the baseline, then page-level metadata is
52
+ * overlaid so route-specific fields win without dropping global defaults.
53
+ */
54
+ getMetadataPropsForPage(options: {
55
+ getMetadata: GetMetadata | undefined;
56
+ context: GetMetadataContext;
57
+ }): Promise<PageMetadataProps>;
58
+ /**
59
+ * Loads a page module and normalizes integration-facing exports.
60
+ * When both component static methods and module exports exist, component statics win.
61
+ */
62
+ resolvePageModule(options: {
63
+ file: string;
64
+ importPageFileFn?: (file: string) => Promise<EcoPageFile>;
65
+ }): Promise<{
66
+ Page: EcoPageFile['default'] | EcoPageComponent<any>;
67
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
68
+ getMetadata?: GetMetadata;
69
+ integrationSpecificProps: Record<string, unknown>;
70
+ }>;
71
+ /**
72
+ * Resolves the page data needed by the render pipeline.
73
+ *
74
+ * @remarks
75
+ * Static props are resolved first because page metadata may depend on those
76
+ * props. This preserves the same ordering whether data hooks are declared as
77
+ * component statics or module exports.
78
+ */
79
+ resolvePageData(options: {
80
+ pageModule: {
81
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
82
+ getMetadata?: GetMetadata;
83
+ };
84
+ routeOptions: RouteRendererOptions;
85
+ }): Promise<{
86
+ props: Record<string, unknown>;
87
+ metadata: PageMetadataProps;
88
+ }>;
89
+ }
@@ -1,22 +1,35 @@
1
- import { invariant } from "../utils/invariant.js";
2
- import { PageModuleImportService } from "../services/page-module-import.service.js";
1
+ import { invariant } from "../../utils/invariant.js";
2
+ import { getAppModuleLoader } from "../../services/module-loading/app-server-module-transpiler.service.js";
3
+ import { resolveInternalExecutionDir } from "../../utils/resolve-work-dir.js";
3
4
  class PageModuleLoaderService {
5
+ appModuleLoader;
6
+ appConfig;
7
+ runtimeOrigin;
8
+ /**
9
+ * Creates the page-module loader for one app/runtime instance.
10
+ *
11
+ * @param appConfig Finalized app config that owns build and invalidation state.
12
+ * @param runtimeOrigin Runtime origin exposed to page data hooks.
13
+ */
4
14
  constructor(appConfig, runtimeOrigin) {
5
15
  this.appConfig = appConfig;
6
16
  this.runtimeOrigin = runtimeOrigin;
7
- this.pageModuleImportService = new PageModuleImportService();
17
+ this.appModuleLoader = getAppModuleLoader(appConfig);
8
18
  }
9
- pageModuleImportService;
10
19
  /**
11
- * Imports a page module from source.
12
- * Uses direct dynamic import in Bun and transpile+import fallback for other runtimes.
20
+ * Imports one page module through the shared server-side module loading path.
21
+ *
22
+ * @remarks
23
+ * The underlying transpiler keeps Bun and Node aligned on one framework-owned
24
+ * loading contract even though the runtime-specific execution transport differs.
13
25
  */
14
- async importPageFile(file) {
26
+ async importPageFile(file, options) {
15
27
  try {
16
- return await this.pageModuleImportService.importModule({
28
+ return await this.appModuleLoader.importModule({
17
29
  filePath: file,
18
30
  rootDir: this.appConfig.rootDir,
19
- outdir: `${this.appConfig.absolutePaths.distDir}/.server-modules`,
31
+ outdir: `${resolveInternalExecutionDir(this.appConfig)}/.server-modules`,
32
+ bypassCache: options?.bypassCache,
20
33
  transpileErrorMessage: (details) => `Error transpiling page file: ${details}`,
21
34
  noOutputMessage: (targetFilePath) => `No transpiled output generated for page: ${targetFilePath}`
22
35
  });
@@ -25,8 +38,11 @@ class PageModuleLoaderService {
25
38
  }
26
39
  }
27
40
  /**
28
- * Executes `getStaticProps` with Ecopages runtime context.
29
- * Returns an empty props object when no static props function is defined.
41
+ * Executes the page's static-props hook with Ecopages runtime context.
42
+ *
43
+ * @remarks
44
+ * Pages without a static-props hook still return a normalized empty props
45
+ * object so downstream render preparation does not branch on hook presence.
30
46
  */
31
47
  async getStaticPropsForPage(options) {
32
48
  const { getStaticProps, params } = options;
@@ -42,8 +58,11 @@ class PageModuleLoaderService {
42
58
  };
43
59
  }
44
60
  /**
45
- * Builds final page metadata using app-level defaults as a baseline.
46
- * If `getMetadata` exists, its result overlays defaults so page-level fields take precedence.
61
+ * Builds the final page metadata object for one render request.
62
+ *
63
+ * @remarks
64
+ * App-level default metadata forms the baseline, then page-level metadata is
65
+ * overlaid so route-specific fields win without dropping global defaults.
47
66
  */
48
67
  async getMetadataPropsForPage(options) {
49
68
  const { getMetadata, context } = options;
@@ -79,7 +98,12 @@ class PageModuleLoaderService {
79
98
  };
80
99
  }
81
100
  /**
82
- * Resolves render-time page data in order: static props first, then metadata derived from those props.
101
+ * Resolves the page data needed by the render pipeline.
102
+ *
103
+ * @remarks
104
+ * Static props are resolved first because page metadata may depend on those
105
+ * props. This preserves the same ordering whether data hooks are declared as
106
+ * component statics or module exports.
83
107
  */
84
108
  async resolvePageData(options) {
85
109
  const { props } = await this.getStaticPropsForPage({
@@ -1,26 +1,63 @@
1
- import type { EcoPagesAppConfig } from '../internal-types.js';
1
+ import type { EcoPagesAppConfig } from '../types/internal-types.js';
2
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';
3
+ import type { RouteRenderResult, RouteRendererOptions } from '../types/public-types.js';
4
+ import type { IntegrationRenderer } from './orchestration/integration-renderer.js';
5
+ /**
6
+ * Thin wrapper around one initialized integration renderer.
7
+ *
8
+ * @remarks
9
+ * This type exists so higher-level routing code can ask for a route renderer
10
+ * without depending on the full integration plugin lifecycle. It delegates all
11
+ * real work to the integration-specific renderer selected by the factory.
12
+ */
5
13
  export declare class RouteRenderer {
6
14
  private renderer;
15
+ /**
16
+ * Creates a route renderer bound to one integration renderer instance.
17
+ */
7
18
  constructor(renderer: IntegrationRenderer);
19
+ /**
20
+ * Executes the render pipeline for one matched route.
21
+ */
8
22
  createRoute(options: RouteRendererOptions): Promise<RouteRenderResult>;
9
23
  }
24
+ /**
25
+ * Selects and caches integration renderers for route files and explicit views.
26
+ *
27
+ * @remarks
28
+ * The factory owns the policy that maps a route file or explicit integration
29
+ * name to one initialized integration renderer. Renderer instances are cached by
30
+ * integration name so repeated requests do not rebuild renderer state.
31
+ */
10
32
  export declare class RouteRendererFactory {
11
33
  private appConfig;
12
34
  runtimeOrigin: string;
35
+ private rendererModules?;
13
36
  private rendererCache;
14
- constructor({ appConfig, runtimeOrigin }: {
37
+ /**
38
+ * Creates the route-renderer factory for one app/runtime instance.
39
+ */
40
+ constructor({ appConfig, rendererModules, runtimeOrigin, }: {
15
41
  appConfig: EcoPagesAppConfig;
42
+ rendererModules?: unknown;
16
43
  runtimeOrigin: string;
17
44
  });
45
+ /**
46
+ * Returns a route renderer for the supplied route file.
47
+ */
18
48
  createRenderer(filePath: string): RouteRenderer;
19
49
  /**
20
50
  * Get an integration renderer by its name.
21
51
  * Used for explicit routing where views specify their integration via __eco.integration.
22
52
  */
23
53
  getRendererByIntegration(integrationName: string): IntegrationRenderer | null;
54
+ /**
55
+ * Resolves the integration plugin that owns a given route file.
56
+ */
24
57
  getIntegrationPlugin(filePath: string): IntegrationPlugin;
58
+ /**
59
+ * Returns the cached renderer engine for the file's owning integration,
60
+ * creating it on first use.
61
+ */
25
62
  private getRouteRendererEngine;
26
63
  }
@@ -2,9 +2,15 @@ import { invariant } from "../utils/invariant.js";
2
2
  import { PathUtils } from "../utils/path-utils.module.js";
3
3
  class RouteRenderer {
4
4
  renderer;
5
+ /**
6
+ * Creates a route renderer bound to one integration renderer instance.
7
+ */
5
8
  constructor(renderer) {
6
9
  this.renderer = renderer;
7
10
  }
11
+ /**
12
+ * Executes the render pipeline for one matched route.
13
+ */
8
14
  async createRoute(options) {
9
15
  return this.renderer.execute(options);
10
16
  }
@@ -12,11 +18,23 @@ class RouteRenderer {
12
18
  class RouteRendererFactory {
13
19
  appConfig;
14
20
  runtimeOrigin;
21
+ rendererModules;
15
22
  rendererCache = /* @__PURE__ */ new Map();
16
- constructor({ appConfig, runtimeOrigin }) {
23
+ /**
24
+ * Creates the route-renderer factory for one app/runtime instance.
25
+ */
26
+ constructor({
27
+ appConfig,
28
+ rendererModules,
29
+ runtimeOrigin
30
+ }) {
17
31
  this.appConfig = appConfig;
32
+ this.rendererModules = rendererModules;
18
33
  this.runtimeOrigin = runtimeOrigin;
19
34
  }
35
+ /**
36
+ * Returns a route renderer for the supplied route file.
37
+ */
20
38
  createRenderer(filePath) {
21
39
  const integrationRenderer = this.getRouteRendererEngine(filePath);
22
40
  invariant(!!integrationRenderer, `No integration renderer found for file: ${filePath}`);
@@ -35,10 +53,15 @@ class RouteRendererFactory {
35
53
  if (cached) {
36
54
  return cached;
37
55
  }
38
- const renderer = integrationPlugin.initializeRenderer();
56
+ const renderer = integrationPlugin.initializeRenderer({
57
+ rendererModules: this.rendererModules
58
+ });
39
59
  this.rendererCache.set(integrationName, renderer);
40
60
  return renderer;
41
61
  }
62
+ /**
63
+ * Resolves the integration plugin that owns a given route file.
64
+ */
42
65
  getIntegrationPlugin(filePath) {
43
66
  const templateExtension = PathUtils.getEcoTemplateExtension(filePath);
44
67
  const isIntegrationPlugin = (plugin) => {
@@ -51,13 +74,19 @@ class RouteRendererFactory {
51
74
  );
52
75
  return integrationPlugin;
53
76
  }
77
+ /**
78
+ * Returns the cached renderer engine for the file's owning integration,
79
+ * creating it on first use.
80
+ */
54
81
  getRouteRendererEngine(filePath) {
55
82
  const integrationPlugin = this.getIntegrationPlugin(filePath);
56
83
  const cached = this.rendererCache.get(integrationPlugin.name);
57
84
  if (cached) {
58
85
  return cached;
59
86
  }
60
- const renderer = integrationPlugin.initializeRenderer();
87
+ const renderer = integrationPlugin.initializeRenderer({
88
+ rendererModules: this.rendererModules
89
+ });
61
90
  this.rendererCache.set(integrationPlugin.name, renderer);
62
91
  return renderer;
63
92
  }
@@ -0,0 +1,97 @@
1
+ # Router Layer
2
+
3
+ This directory contains route discovery, matching, and browser-side navigation infrastructure.
4
+
5
+ ## Purpose
6
+
7
+ The router layer determines what route is being handled and how the client runtime coordinates navigation.
8
+
9
+ It is responsible for:
10
+
11
+ - filesystem route scanning and classification (`exact`, `dynamic`, `catch-all`)
12
+ - matching incoming request URLs to discovered routes
13
+ - server-side static-path expansion for dynamic routes
14
+ - client-side navigation ownership and cross-runtime handoff
15
+ - keeping route discovery separate from rendering execution
16
+
17
+ ## Directory Structure
18
+
19
+ ```
20
+ router/
21
+ ├── server/ # Server-side route scanning and matching
22
+ │ ├── fs-router-scanner.ts # Scans the filesystem and classifies routes
23
+ │ └── fs-router.ts # Matches request URLs to discovered routes
24
+ └── client/ # Browser-side navigation coordination
25
+ ├── navigation-coordinator.ts # Singleton runtime coordinator
26
+ └── link-intent.ts # Shared anchor detection and intent recovery helpers
27
+ ```
28
+
29
+ ## `server/`
30
+
31
+ ### `FSRouterScanner`
32
+
33
+ Walks the pages directory and builds a `Routes` map keyed by route pathname.
34
+
35
+ File patterns determine route kind:
36
+
37
+ | Pattern | Kind | Example |
38
+ | --------------- | ----------- | ----------------- |
39
+ | `page.tsx` | `exact` | `/about` |
40
+ | `[slug].tsx` | `dynamic` | `/blog/[slug]` |
41
+ | `[...slug].tsx` | `catch-all` | `/docs/[...slug]` |
42
+
43
+ For `dynamic` routes, the scanner checks whether the page module exports `getStaticPaths`. If present, every returned path is expanded into a concrete `exact`-style route at scan time. In build mode, both `getStaticPaths` and `getStaticProps` are required or an invariant is thrown.
44
+
45
+ Catch-all routes are registered but skipped during static generation with a warning.
46
+
47
+ ### `FSRouter`
48
+
49
+ Holds the scanned `Routes` map and exposes a `match(requestUrl)` method used by adapters.
50
+
51
+ Match priority:
52
+
53
+ 1. `exact` — the pathname must equal the route pathname exactly.
54
+ 2. `dynamic` — the clean (bracket-stripped) prefix must appear in the pathname, and the segment counts must match.
55
+ 3. `catch-all` — the clean prefix must appear in the pathname.
56
+
57
+ Additional helpers:
58
+
59
+ - `getDynamicParams(route, pathname)` — extracts named and spread parameters from a matched dynamic or catch-all route.
60
+ - `getSearchParams(url)` — converts `URLSearchParams` to a plain object.
61
+ - `setOnReload(cb)` / `reload()` — re-scans routes and fires an optional callback, used during development HMR.
62
+
63
+ ## `client/`
64
+
65
+ ### Navigation Coordinator (`navigation-coordinator.ts`)
66
+
67
+ A singleton browser-side runtime stored on `window.__ECO_PAGES__.navigation`.
68
+
69
+ Access it with:
70
+
71
+ ```ts
72
+ import { getEcoNavigationRuntime } from '@ecopages/core/router/client/navigation-coordinator';
73
+ const runtime = getEcoNavigationRuntime();
74
+ ```
75
+
76
+ The coordinator is framework-agnostic. Browser runtimes (e.g. `browser-router`, `react-router`) register themselves and the coordinator arbitrates:
77
+
78
+ - **Ownership** — which runtime currently drives SPA navigation (`claimOwnership`, `releaseOwnership`, `setOwner`).
79
+ - **Document owner marker** — an HTML attribute (`data-eco-document-owner`) written into rendered markup so the coordinator can `adoptDocumentOwner` on the incoming page.
80
+ - **Navigation transactions** — each navigation begins a transaction with an `AbortSignal`; superseded navigations are automatically cancelled.
81
+ - **Cross-runtime handoff** — `requestHandoff` passes a pre-fetched `Document` to the target runtime without tearing down the current page prematurely.
82
+ - **Reload** — `reloadCurrentPage` delegates to whichever runtime currently owns the document.
83
+ - **Events** — `subscribe` lets runtimes react to `owner-change` and `registration-change` events.
84
+
85
+ ### Link Intent (`link-intent.ts`)
86
+
87
+ Shared helpers for locating anchors and recovering stale navigation intent.
88
+
89
+ - `getAnchorFromNavigationEvent(event, linkSelector)` — finds the nearest matching anchor in the event's composed path, including across Shadow DOM boundaries.
90
+ - `recoverPendingNavigationHref(intent, hasInFlightNavigation, now, maxAgeMs?)` — resolves a previously captured pointer or hover target when the DOM changes before the click lands. Intents expire after `maxAgeMs` (default 1000 ms).
91
+
92
+ ## Relationship To Rendering
93
+
94
+ The router layer answers **which route should run**.
95
+ The route-renderer layer answers **how that route gets rendered**.
96
+
97
+ Keeping those seams separate avoids mixing route ownership, module loading, and component orchestration into one service.