@ecopages/core 0.2.0-alpha.4 → 0.2.0-alpha.41

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 (454) hide show
  1. package/README.md +268 -14
  2. package/package.json +92 -110
  3. package/src/adapters/README.md +39 -0
  4. package/src/adapters/abstract/application-adapter.d.ts +28 -2
  5. package/src/adapters/abstract/application-adapter.js +14 -2
  6. package/src/adapters/abstract/router-adapter.d.ts +1 -1
  7. package/src/adapters/abstract/server-adapter.d.ts +2 -2
  8. package/src/adapters/bun/client-bridge.d.ts +1 -1
  9. package/src/adapters/bun/create-app.d.ts +12 -12
  10. package/src/adapters/bun/create-app.js +64 -41
  11. package/src/adapters/bun/hmr-manager.d.ts +30 -68
  12. package/src/adapters/bun/hmr-manager.js +30 -212
  13. package/src/adapters/bun/index.d.ts +2 -3
  14. package/src/adapters/bun/index.js +3 -3
  15. package/src/adapters/bun/runtime-host.d.ts +52 -0
  16. package/src/adapters/bun/runtime-host.js +56 -0
  17. package/src/adapters/bun/server-adapter.d.ts +93 -32
  18. package/src/adapters/bun/server-adapter.js +166 -89
  19. package/src/adapters/bun/static-preview-host.d.ts +28 -0
  20. package/src/adapters/bun/static-preview-host.js +45 -0
  21. package/src/{create-app.d.ts → adapters/create-app.d.ts} +9 -6
  22. package/src/{create-app.js → adapters/create-app.js} +4 -4
  23. package/src/adapters/index.d.ts +2 -6
  24. package/src/adapters/index.js +2 -8
  25. package/src/adapters/node/create-app.d.ts +15 -12
  26. package/src/adapters/node/create-app.js +34 -85
  27. package/src/adapters/node/http-request-bridge.d.ts +57 -0
  28. package/src/adapters/node/http-request-bridge.js +118 -0
  29. package/src/adapters/node/node-client-bridge.d.ts +1 -1
  30. package/src/adapters/node/node-hmr-manager.d.ts +38 -49
  31. package/src/adapters/node/node-hmr-manager.js +31 -219
  32. package/src/adapters/node/runtime-host.d.ts +57 -0
  33. package/src/adapters/node/runtime-host.js +92 -0
  34. package/src/adapters/node/server-adapter-dependencies.d.ts +19 -0
  35. package/src/adapters/node/server-adapter-dependencies.js +18 -0
  36. package/src/adapters/node/server-adapter.d.ts +20 -72
  37. package/src/adapters/node/server-adapter.js +99 -203
  38. package/src/adapters/node/static-content-server.d.ts +37 -1
  39. package/src/adapters/node/static-content-server.js +29 -1
  40. package/src/adapters/node/static-preview-host.d.ts +55 -0
  41. package/src/adapters/node/static-preview-host.js +68 -0
  42. package/src/adapters/shared/application-adapter.d.ts +1 -1
  43. package/src/{define-api-handler.d.ts → adapters/shared/define-api-handler.d.ts} +1 -1
  44. package/src/adapters/shared/explicit-static-render-preparation.d.ts +25 -0
  45. package/src/adapters/shared/explicit-static-render-preparation.js +26 -0
  46. package/src/adapters/shared/explicit-static-route-matcher.d.ts +7 -4
  47. package/src/adapters/shared/explicit-static-route-matcher.js +14 -13
  48. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +7 -10
  49. package/src/adapters/shared/file-route-middleware-pipeline.js +3 -11
  50. package/src/adapters/shared/fs-server-response-factory.d.ts +14 -10
  51. package/src/adapters/shared/fs-server-response-factory.js +11 -27
  52. package/src/adapters/shared/fs-server-response-matcher.d.ts +20 -16
  53. package/src/adapters/shared/fs-server-response-matcher.js +76 -45
  54. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +55 -0
  55. package/src/adapters/shared/hmr-entrypoint-registrar.js +87 -0
  56. package/src/adapters/shared/hmr-html-response.d.ts +22 -0
  57. package/src/adapters/shared/hmr-html-response.js +33 -0
  58. package/src/adapters/shared/render-context.d.ts +5 -3
  59. package/src/adapters/shared/render-context.js +27 -3
  60. package/src/adapters/shared/runtime-app-bootstrap.d.ts +26 -0
  61. package/src/adapters/shared/runtime-app-bootstrap.js +46 -0
  62. package/src/adapters/shared/runtime-host.d.ts +12 -0
  63. package/src/adapters/shared/server-adapter.d.ts +33 -12
  64. package/src/adapters/shared/server-adapter.js +215 -132
  65. package/src/adapters/shared/server-route-handler.d.ts +5 -5
  66. package/src/adapters/shared/server-route-handler.js +7 -16
  67. package/src/adapters/shared/server-static-builder.d.ts +41 -8
  68. package/src/adapters/shared/server-static-builder.js +65 -11
  69. package/src/adapters/shared/shared-hmr-manager.d.ts +59 -0
  70. package/src/adapters/shared/shared-hmr-manager.js +240 -0
  71. package/src/adapters/shared/static-preview-host.d.ts +10 -0
  72. package/src/build/README.md +107 -0
  73. package/src/build/build-adapter.d.ts +176 -3
  74. package/src/build/build-adapter.js +619 -16
  75. package/src/build/build-manifest.d.ts +27 -0
  76. package/src/build/build-manifest.js +30 -0
  77. package/src/build/dev-build-coordinator.d.ts +72 -0
  78. package/src/build/dev-build-coordinator.js +154 -0
  79. package/src/build/esbuild-build-adapter.d.ts +16 -6
  80. package/src/build/esbuild-build-adapter.js +202 -74
  81. package/src/build/runtime-build-executor.d.ts +14 -0
  82. package/src/build/runtime-build-executor.js +22 -0
  83. package/src/build/runtime-specifier-alias-plugin.d.ts +15 -0
  84. package/src/build/runtime-specifier-alias-plugin.js +35 -0
  85. package/src/build/runtime-specifier-aliases.d.ts +5 -0
  86. package/src/build/runtime-specifier-aliases.js +95 -0
  87. package/src/config/README.md +36 -0
  88. package/src/config/config-builder.d.ts +54 -29
  89. package/src/config/config-builder.js +256 -50
  90. package/src/{constants.d.ts → config/constants.d.ts} +13 -0
  91. package/src/{constants.js → config/constants.js} +4 -0
  92. package/src/declarations.d.ts +19 -14
  93. package/src/dev/host-runtime.d.ts +10 -0
  94. package/src/dev/host-runtime.js +24 -0
  95. package/src/dev/sc-server.d.ts +1 -1
  96. package/src/dev/sc-server.js +1 -1
  97. package/src/eco/README.md +70 -16
  98. package/src/eco/eco.browser.d.ts +2 -0
  99. package/src/eco/eco.browser.js +88 -0
  100. package/src/eco/eco.js +63 -54
  101. package/src/eco/eco.types.d.ts +69 -6
  102. package/src/eco/eco.utils.d.ts +1 -40
  103. package/src/eco/eco.utils.js +5 -35
  104. package/src/eco/global-injector-map.d.ts +3 -3
  105. package/src/eco/global-injector-map.js +2 -2
  106. package/src/eco/lazy-injector-map.d.ts +2 -2
  107. package/src/errors/index.d.ts +1 -0
  108. package/src/errors/index.js +3 -1
  109. package/src/hmr/README.md +26 -0
  110. package/src/hmr/client/hmr-runtime.d.ts +1 -6
  111. package/src/hmr/client/hmr-runtime.js +38 -7
  112. package/src/hmr/hmr-strategy.d.ts +16 -13
  113. package/src/hmr/hmr-strategy.js +22 -7
  114. package/src/hmr/hmr.postcss.test.e2e.d.ts +1 -0
  115. package/src/hmr/hmr.postcss.test.e2e.js +31 -0
  116. package/src/hmr/hmr.test.e2e.js +26 -33
  117. package/src/hmr/strategies/default-hmr-strategy.d.ts +2 -2
  118. package/src/hmr/strategies/default-hmr-strategy.js +1 -1
  119. package/src/hmr/strategies/js-hmr-strategy.d.ts +46 -48
  120. package/src/hmr/strategies/js-hmr-strategy.js +60 -75
  121. package/src/index.browser.d.ts +2 -2
  122. package/src/index.browser.js +1 -1
  123. package/src/index.d.ts +4 -3
  124. package/src/index.js +16 -5
  125. package/src/integrations/ghtml/ghtml-renderer.d.ts +3 -2
  126. package/src/integrations/ghtml/ghtml-renderer.js +27 -30
  127. package/src/integrations/ghtml/ghtml.constants.d.ts +1 -0
  128. package/src/integrations/ghtml/ghtml.constants.js +4 -0
  129. package/src/integrations/ghtml/ghtml.plugin.d.ts +2 -6
  130. package/src/integrations/ghtml/ghtml.plugin.js +3 -4
  131. package/src/plugins/README.md +35 -0
  132. package/src/plugins/alias-resolver-plugin.d.ts +1 -0
  133. package/src/plugins/alias-resolver-plugin.js +27 -5
  134. package/src/plugins/eco-component-meta-plugin.d.ts +14 -1
  135. package/src/plugins/eco-component-meta-plugin.js +42 -24
  136. package/src/plugins/foreign-jsx-override-plugin.d.ts +33 -0
  137. package/src/plugins/foreign-jsx-override-plugin.js +41 -0
  138. package/src/plugins/integration-plugin.d.ts +147 -29
  139. package/src/plugins/integration-plugin.js +103 -14
  140. package/src/plugins/processor.d.ts +17 -2
  141. package/src/plugins/processor.js +22 -3
  142. package/src/plugins/runtime-capability.d.ts +9 -0
  143. package/src/plugins/source-transform.d.ts +46 -0
  144. package/src/plugins/source-transform.js +71 -0
  145. package/src/route-renderer/GRAPH.md +83 -325
  146. package/src/route-renderer/README.md +73 -90
  147. package/src/route-renderer/orchestration/component-render-context.d.ts +97 -0
  148. package/src/route-renderer/orchestration/component-render-context.js +147 -0
  149. package/src/route-renderer/orchestration/declared-ownership-graph.d.ts +18 -0
  150. package/src/route-renderer/orchestration/declared-ownership-graph.js +34 -0
  151. package/src/route-renderer/orchestration/foreign-subtree-execution.service.d.ts +110 -0
  152. package/src/route-renderer/orchestration/foreign-subtree-execution.service.js +233 -0
  153. package/src/route-renderer/orchestration/integration-renderer.d.ts +534 -0
  154. package/src/route-renderer/orchestration/integration-renderer.js +986 -0
  155. package/src/route-renderer/orchestration/ownership-planning.service.d.ts +24 -0
  156. package/src/route-renderer/orchestration/ownership-planning.service.js +63 -0
  157. package/src/route-renderer/orchestration/ownership-validation.service.d.ts +29 -0
  158. package/src/route-renderer/orchestration/ownership-validation.service.js +53 -0
  159. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +3 -0
  160. package/src/route-renderer/orchestration/processed-asset-dedupe.js +27 -0
  161. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.d.ts +91 -0
  162. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.js +170 -0
  163. package/src/route-renderer/orchestration/render-output.utils.d.ts +66 -0
  164. package/src/route-renderer/orchestration/render-output.utils.js +171 -0
  165. package/src/route-renderer/orchestration/route-render-orchestrator.d.ts +134 -0
  166. package/src/route-renderer/orchestration/route-render-orchestrator.js +442 -0
  167. package/src/route-renderer/orchestration/template-serialization.d.ts +38 -0
  168. package/src/route-renderer/orchestration/template-serialization.js +45 -0
  169. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +37 -0
  170. package/src/route-renderer/page-loading/component-dependency-collection.js +132 -0
  171. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +24 -0
  172. package/src/route-renderer/page-loading/declared-asset-collection.js +106 -0
  173. package/src/route-renderer/{dependency-resolver.d.ts → page-loading/dependency-resolver.d.ts} +15 -4
  174. package/src/route-renderer/page-loading/dependency-resolver.js +115 -0
  175. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +11 -0
  176. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +57 -0
  177. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +45 -0
  178. package/src/route-renderer/page-loading/lazy-entry-collection.js +105 -0
  179. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +19 -0
  180. package/src/route-renderer/page-loading/lazy-trigger-planning.js +40 -0
  181. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +5 -0
  182. package/src/route-renderer/page-loading/module-declaration-aggregation.js +33 -0
  183. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +3 -0
  184. package/src/route-renderer/page-loading/module-declaration-scripts.js +18 -0
  185. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +13 -0
  186. package/src/route-renderer/page-loading/page-dependency-bundling.js +137 -0
  187. package/src/route-renderer/page-loading/page-module-loader.d.ts +90 -0
  188. package/src/route-renderer/{page-module-loader.js → page-loading/page-module-loader.js} +39 -14
  189. package/src/route-renderer/route-renderer.d.ts +57 -14
  190. package/src/route-renderer/route-renderer.js +30 -18
  191. package/src/router/README.md +94 -0
  192. package/src/router/client/link-intent.d.ts +53 -0
  193. package/src/router/client/link-intent.js +34 -0
  194. package/src/router/client/link-intent.test.browser.d.ts +1 -0
  195. package/src/router/client/link-intent.test.browser.js +43 -0
  196. package/src/router/client/navigation-coordinator.d.ts +169 -0
  197. package/src/router/client/navigation-coordinator.js +215 -0
  198. package/src/router/server/route-registry.d.ts +78 -0
  199. package/src/router/server/route-registry.js +262 -0
  200. package/src/services/README.md +28 -0
  201. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +3 -0
  202. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +56 -0
  203. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +103 -0
  204. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset-processing.service.js +124 -89
  205. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.d.ts +1 -1
  206. package/src/services/{asset-processing-service → assets/asset-processing-service}/asset.factory.js +2 -2
  207. package/src/services/{asset-processing-service → assets/asset-processing-service}/assets.types.d.ts +16 -1
  208. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  209. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +55 -0
  210. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +49 -0
  211. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +20 -0
  212. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +41 -0
  213. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +30 -0
  214. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +65 -0
  215. package/src/services/assets/asset-processing-service/index.d.ts +6 -0
  216. package/src/services/assets/asset-processing-service/index.js +6 -0
  217. package/src/services/assets/asset-processing-service/page-package.d.ts +6 -0
  218. package/src/services/assets/asset-processing-service/page-package.js +80 -0
  219. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.d.ts +2 -2
  220. package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.d.ts +2 -2
  221. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.d.ts +1 -1
  222. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/base/base-processor.js +11 -5
  223. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +22 -0
  224. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +136 -0
  225. package/src/services/assets/asset-processing-service/processors/index.d.ts +5 -0
  226. package/src/services/assets/asset-processing-service/processors/index.js +5 -0
  227. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/content-script.processor.d.ts +3 -2
  228. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +119 -0
  229. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.d.ts +4 -3
  230. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/script/file-script.processor.js +28 -7
  231. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +42 -0
  232. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +126 -0
  233. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/content-stylesheet.processor.d.ts +5 -2
  234. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +59 -0
  235. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.d.ts +2 -2
  236. package/src/services/{asset-processing-service → assets/asset-processing-service}/processors/stylesheet/file-stylesheet.processor.js +9 -3
  237. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +18 -0
  238. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +45 -0
  239. package/src/services/assets/browser-bundle.service.d.ts +73 -0
  240. package/src/services/assets/browser-bundle.service.js +41 -0
  241. package/src/services/{page-request-cache-coordinator.service.d.ts → cache/page-request-cache-coordinator.service.d.ts} +2 -2
  242. package/src/services/{page-request-cache-coordinator.service.js → cache/page-request-cache-coordinator.service.js} +3 -1
  243. package/src/services/html/html-rewriter-provider.service.d.ts +40 -0
  244. package/src/services/html/html-rewriter-provider.service.js +68 -0
  245. package/src/services/html/html-transformer.service.d.ts +96 -0
  246. package/src/services/html/html-transformer.service.js +287 -0
  247. package/src/services/invalidation/development-invalidation.service.d.ts +74 -0
  248. package/src/services/invalidation/development-invalidation.service.js +190 -0
  249. package/src/services/module-loading/app-module-loader.service.d.ts +7 -0
  250. package/src/services/module-loading/app-module-loader.service.js +0 -0
  251. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +24 -0
  252. package/src/services/module-loading/app-server-module-transpiler.service.js +151 -0
  253. package/src/services/module-loading/host-module-loader-registry.d.ts +4 -0
  254. package/src/services/module-loading/host-module-loader-registry.js +15 -0
  255. package/src/services/module-loading/module-loading-types.d.ts +2 -0
  256. package/src/services/module-loading/module-loading-types.js +0 -0
  257. package/src/services/module-loading/node-bootstrap-plugin.d.ts +42 -0
  258. package/src/services/module-loading/node-bootstrap-plugin.js +297 -0
  259. package/src/services/module-loading/page-module-import.service.d.ts +95 -0
  260. package/src/services/module-loading/page-module-import.service.js +191 -0
  261. package/src/services/module-loading/server-module-transpiler.service.d.ts +63 -0
  262. package/src/services/module-loading/server-module-transpiler.service.js +64 -0
  263. package/src/services/module-loading/source-module-support.d.ts +5 -0
  264. package/src/services/module-loading/source-module-support.js +8 -0
  265. package/src/services/runtime-state/dev-graph.service.d.ts +118 -0
  266. package/src/services/runtime-state/dev-graph.service.js +162 -0
  267. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +41 -0
  268. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +85 -0
  269. package/src/services/runtime-state/server-invalidation-state.service.d.ts +26 -0
  270. package/src/services/runtime-state/server-invalidation-state.service.js +35 -0
  271. package/src/services/{schema-validation-service.d.ts → validation/schema-validation-service.d.ts} +1 -1
  272. package/src/static-site-generator/README.md +26 -0
  273. package/src/static-site-generator/static-site-generator.d.ts +67 -20
  274. package/src/static-site-generator/static-site-generator.js +182 -138
  275. package/src/{internal-types.d.ts → types/internal-types.d.ts} +62 -30
  276. package/src/types/internal-types.js +0 -0
  277. package/src/{public-types.d.ts → types/public-types.d.ts} +239 -35
  278. package/src/types/public-types.js +0 -0
  279. package/src/utils/html-escaping.d.ts +7 -0
  280. package/src/utils/html-escaping.js +6 -0
  281. package/src/utils/locals-utils.d.ts +1 -1
  282. package/src/utils/parse-cli-args.d.ts +4 -1
  283. package/src/utils/parse-cli-args.js +16 -1
  284. package/src/utils/resolve-work-dir.d.ts +11 -0
  285. package/src/utils/resolve-work-dir.js +31 -0
  286. package/src/watchers/project-watcher.d.ts +40 -29
  287. package/src/watchers/project-watcher.js +126 -116
  288. package/src/watchers/project-watcher.test-helpers.d.ts +2 -2
  289. package/src/watchers/project-watcher.test-helpers.js +6 -6
  290. package/CHANGELOG.md +0 -91
  291. package/src/adapters/abstract/application-adapter.ts +0 -337
  292. package/src/adapters/abstract/router-adapter.ts +0 -30
  293. package/src/adapters/abstract/server-adapter.ts +0 -79
  294. package/src/adapters/bun/client-bridge.ts +0 -62
  295. package/src/adapters/bun/create-app.ts +0 -189
  296. package/src/adapters/bun/define-api-handler.d.ts +0 -61
  297. package/src/adapters/bun/define-api-handler.ts +0 -114
  298. package/src/adapters/bun/hmr-manager.ts +0 -296
  299. package/src/adapters/bun/index.ts +0 -3
  300. package/src/adapters/bun/server-adapter.ts +0 -492
  301. package/src/adapters/bun/server-lifecycle.d.ts +0 -52
  302. package/src/adapters/bun/server-lifecycle.js +0 -120
  303. package/src/adapters/bun/server-lifecycle.ts +0 -154
  304. package/src/adapters/index.ts +0 -6
  305. package/src/adapters/node/create-app.ts +0 -179
  306. package/src/adapters/node/index.d.ts +0 -4
  307. package/src/adapters/node/index.js +0 -8
  308. package/src/adapters/node/index.ts +0 -9
  309. package/src/adapters/node/node-client-bridge.ts +0 -79
  310. package/src/adapters/node/node-hmr-manager.ts +0 -289
  311. package/src/adapters/node/server-adapter.ts +0 -561
  312. package/src/adapters/node/static-content-server.ts +0 -203
  313. package/src/adapters/shared/api-response.ts +0 -104
  314. package/src/adapters/shared/application-adapter.ts +0 -199
  315. package/src/adapters/shared/explicit-static-route-matcher.ts +0 -134
  316. package/src/adapters/shared/file-route-middleware-pipeline.ts +0 -123
  317. package/src/adapters/shared/fs-server-response-factory.ts +0 -118
  318. package/src/adapters/shared/fs-server-response-matcher.ts +0 -198
  319. package/src/adapters/shared/render-context.ts +0 -105
  320. package/src/adapters/shared/server-adapter.ts +0 -442
  321. package/src/adapters/shared/server-route-handler.ts +0 -166
  322. package/src/adapters/shared/server-static-builder.ts +0 -82
  323. package/src/build/build-adapter.ts +0 -132
  324. package/src/build/build-types.ts +0 -83
  325. package/src/build/esbuild-build-adapter.ts +0 -510
  326. package/src/config/config-builder.ts +0 -474
  327. package/src/constants.ts +0 -39
  328. package/src/create-app.ts +0 -87
  329. package/src/define-api-handler.js +0 -15
  330. package/src/define-api-handler.ts +0 -66
  331. package/src/dev/sc-server.ts +0 -143
  332. package/src/eco/component-render-context.d.ts +0 -105
  333. package/src/eco/component-render-context.js +0 -77
  334. package/src/eco/component-render-context.ts +0 -202
  335. package/src/eco/eco.ts +0 -221
  336. package/src/eco/eco.types.ts +0 -202
  337. package/src/eco/eco.utils.ts +0 -89
  338. package/src/eco/global-injector-map.ts +0 -112
  339. package/src/eco/lazy-injector-map.ts +0 -120
  340. package/src/eco/module-dependencies.ts +0 -75
  341. package/src/errors/http-error.ts +0 -72
  342. package/src/errors/index.ts +0 -2
  343. package/src/errors/locals-access-error.ts +0 -7
  344. package/src/global/app-logger.ts +0 -4
  345. package/src/hmr/client/hmr-runtime.ts +0 -121
  346. package/src/hmr/hmr-strategy.ts +0 -172
  347. package/src/hmr/hmr.test.e2e.ts +0 -75
  348. package/src/hmr/strategies/default-hmr-strategy.ts +0 -60
  349. package/src/hmr/strategies/js-hmr-strategy.ts +0 -320
  350. package/src/index.browser.ts +0 -3
  351. package/src/index.ts +0 -5
  352. package/src/integrations/ghtml/ghtml-renderer.ts +0 -93
  353. package/src/integrations/ghtml/ghtml.plugin.ts +0 -32
  354. package/src/internal-types.ts +0 -212
  355. package/src/plugins/alias-resolver-plugin.ts +0 -45
  356. package/src/plugins/eco-component-meta-plugin.ts +0 -474
  357. package/src/plugins/integration-plugin.ts +0 -184
  358. package/src/plugins/processor.ts +0 -220
  359. package/src/public-types.ts +0 -1260
  360. package/src/route-renderer/component-graph-executor.d.ts +0 -32
  361. package/src/route-renderer/component-graph-executor.js +0 -31
  362. package/src/route-renderer/component-graph-executor.ts +0 -84
  363. package/src/route-renderer/component-graph.d.ts +0 -42
  364. package/src/route-renderer/component-graph.js +0 -72
  365. package/src/route-renderer/component-graph.ts +0 -159
  366. package/src/route-renderer/component-marker.d.ts +0 -52
  367. package/src/route-renderer/component-marker.js +0 -46
  368. package/src/route-renderer/component-marker.ts +0 -117
  369. package/src/route-renderer/dependency-resolver.js +0 -428
  370. package/src/route-renderer/dependency-resolver.ts +0 -596
  371. package/src/route-renderer/html-post-processing.service.d.ts +0 -40
  372. package/src/route-renderer/html-post-processing.service.js +0 -86
  373. package/src/route-renderer/html-post-processing.service.ts +0 -103
  374. package/src/route-renderer/integration-renderer.d.ts +0 -339
  375. package/src/route-renderer/integration-renderer.js +0 -526
  376. package/src/route-renderer/integration-renderer.ts +0 -696
  377. package/src/route-renderer/marker-graph-resolver.d.ts +0 -76
  378. package/src/route-renderer/marker-graph-resolver.js +0 -93
  379. package/src/route-renderer/marker-graph-resolver.ts +0 -153
  380. package/src/route-renderer/page-module-loader.d.ts +0 -61
  381. package/src/route-renderer/page-module-loader.ts +0 -153
  382. package/src/route-renderer/render-execution.service.d.ts +0 -69
  383. package/src/route-renderer/render-execution.service.js +0 -91
  384. package/src/route-renderer/render-execution.service.ts +0 -158
  385. package/src/route-renderer/render-preparation.service.d.ts +0 -112
  386. package/src/route-renderer/render-preparation.service.js +0 -243
  387. package/src/route-renderer/render-preparation.service.ts +0 -358
  388. package/src/route-renderer/route-renderer.ts +0 -80
  389. package/src/router/fs-router-scanner.d.ts +0 -41
  390. package/src/router/fs-router-scanner.js +0 -155
  391. package/src/router/fs-router-scanner.ts +0 -217
  392. package/src/router/fs-router.d.ts +0 -26
  393. package/src/router/fs-router.js +0 -100
  394. package/src/router/fs-router.ts +0 -122
  395. package/src/services/asset-processing-service/asset-processing.service.d.ts +0 -41
  396. package/src/services/asset-processing-service/asset-processing.service.ts +0 -306
  397. package/src/services/asset-processing-service/asset.factory.ts +0 -105
  398. package/src/services/asset-processing-service/assets.types.ts +0 -112
  399. package/src/services/asset-processing-service/index.d.ts +0 -3
  400. package/src/services/asset-processing-service/index.js +0 -3
  401. package/src/services/asset-processing-service/index.ts +0 -3
  402. package/src/services/asset-processing-service/processor.interface.ts +0 -27
  403. package/src/services/asset-processing-service/processor.registry.ts +0 -18
  404. package/src/services/asset-processing-service/processors/base/base-processor.ts +0 -76
  405. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +0 -16
  406. package/src/services/asset-processing-service/processors/base/base-script-processor.js +0 -80
  407. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +0 -105
  408. package/src/services/asset-processing-service/processors/index.d.ts +0 -5
  409. package/src/services/asset-processing-service/processors/index.js +0 -5
  410. package/src/services/asset-processing-service/processors/index.ts +0 -5
  411. package/src/services/asset-processing-service/processors/script/content-script.processor.js +0 -57
  412. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +0 -66
  413. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +0 -88
  414. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  415. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +0 -74
  416. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +0 -84
  417. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -25
  418. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +0 -27
  419. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +0 -77
  420. package/src/services/cache/cache.types.ts +0 -126
  421. package/src/services/cache/index.ts +0 -18
  422. package/src/services/cache/memory-cache-store.ts +0 -130
  423. package/src/services/cache/page-cache-service.ts +0 -202
  424. package/src/services/html-transformer.service.d.ts +0 -50
  425. package/src/services/html-transformer.service.js +0 -163
  426. package/src/services/html-transformer.service.ts +0 -217
  427. package/src/services/page-module-import.service.d.ts +0 -37
  428. package/src/services/page-module-import.service.js +0 -88
  429. package/src/services/page-module-import.service.ts +0 -129
  430. package/src/services/page-request-cache-coordinator.service.ts +0 -128
  431. package/src/services/schema-validation-service.ts +0 -204
  432. package/src/services/validation/standard-schema.types.ts +0 -68
  433. package/src/static-site-generator/static-site-generator.ts +0 -359
  434. package/src/utils/css.d.ts +0 -1
  435. package/src/utils/css.js +0 -7
  436. package/src/utils/css.ts +0 -5
  437. package/src/utils/deep-merge.ts +0 -47
  438. package/src/utils/hash.ts +0 -5
  439. package/src/utils/html.ts +0 -1
  440. package/src/utils/invariant.ts +0 -15
  441. package/src/utils/locals-utils.ts +0 -37
  442. package/src/utils/parse-cli-args.ts +0 -83
  443. package/src/utils/path-utils.module.ts +0 -14
  444. package/src/utils/runtime.ts +0 -44
  445. package/src/utils/server-utils.module.ts +0 -67
  446. package/src/watchers/project-watcher.test-helpers.ts +0 -41
  447. package/src/watchers/project-watcher.ts +0 -344
  448. /package/src/adapters/{bun → shared}/define-api-handler.js +0 -0
  449. /package/src/{internal-types.js → adapters/shared/runtime-host.js} +0 -0
  450. /package/src/{public-types.js → adapters/shared/static-preview-host.js} +0 -0
  451. /package/src/{services/asset-processing-service/assets.types.js → plugins/runtime-capability.js} +0 -0
  452. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.interface.js +0 -0
  453. /package/src/services/{asset-processing-service → assets/asset-processing-service}/processor.registry.js +0 -0
  454. /package/src/services/{schema-validation-service.js → validation/schema-validation-service.js} +0 -0
@@ -0,0 +1,137 @@
1
+ import { existsSync, readFileSync } from "node:fs";
2
+ import { AssetFactory } from "../../services/assets/asset-processing-service/index.js";
3
+ import { rapidhash } from "../../utils/hash.js";
4
+ function hasOnlyExpectedAttributes(attributes, expected) {
5
+ const keys = Object.keys(attributes ?? {});
6
+ const expectedKeys = Object.keys(expected);
7
+ if (keys.length !== expectedKeys.length) {
8
+ return false;
9
+ }
10
+ return expectedKeys.every((key) => attributes?.[key] === expected[key]);
11
+ }
12
+ function isSafeBundledStylesheetReference(reference) {
13
+ return reference.startsWith("/") || reference.startsWith("data:") || reference.startsWith("http://") || reference.startsWith("https://") || reference.startsWith("#");
14
+ }
15
+ function normalizeCssReferenceToken(token) {
16
+ return token.trim().replace(/^['"]|['"]$/g, "");
17
+ }
18
+ function isSafeBundledStylesheetContent(content) {
19
+ for (const match of content.matchAll(/@import\s+(?:url\()?['"]?([^'")\s]+)['"]?\)?/g)) {
20
+ const reference = normalizeCssReferenceToken(match[1] ?? "");
21
+ if (reference && !isSafeBundledStylesheetReference(reference)) {
22
+ return false;
23
+ }
24
+ }
25
+ for (const match of content.matchAll(/url\(([^)]+)\)/g)) {
26
+ const reference = normalizeCssReferenceToken(match[1] ?? "");
27
+ if (reference && !isSafeBundledStylesheetReference(reference)) {
28
+ return false;
29
+ }
30
+ }
31
+ return true;
32
+ }
33
+ function isBundleableFileStylesheetAsset(dependency) {
34
+ return dependency.kind === "stylesheet" && dependency.source === "file";
35
+ }
36
+ function isBundleableFileScriptAsset(dependency) {
37
+ return dependency.kind === "script" && dependency.source === "file";
38
+ }
39
+ function shouldBundlePageDependencies(integrationName) {
40
+ return integrationName === "react" || integrationName === "ecopages-jsx";
41
+ }
42
+ function createPageDependencyPackagingPlan(dependencies, integrationName) {
43
+ const shouldBundleDependencies = process.env.NODE_ENV === "production";
44
+ const bundleableStyles = dependencies.filter((dependency) => {
45
+ if (!shouldBundleDependencies) {
46
+ return false;
47
+ }
48
+ if (dependency.kind !== "stylesheet" || dependency.inline || dependency.position === "body") {
49
+ return false;
50
+ }
51
+ if (dependency.packageRole || !hasOnlyExpectedAttributes(dependency.attributes, { rel: "stylesheet" })) {
52
+ return false;
53
+ }
54
+ if (dependency.source === "content") {
55
+ return isSafeBundledStylesheetContent(dependency.content);
56
+ }
57
+ if (!existsSync(dependency.filepath)) {
58
+ return false;
59
+ }
60
+ return isSafeBundledStylesheetContent(readFileSync(dependency.filepath, "utf8"));
61
+ }).filter(isBundleableFileStylesheetAsset);
62
+ const bundleableScripts = dependencies.filter((dependency) => {
63
+ if (!shouldBundleDependencies) {
64
+ return false;
65
+ }
66
+ return dependency.kind === "script" && dependency.source === "file" && !dependency.inline && !dependency.excludeFromHtml && dependency.position !== "body" && !dependency.packageRole && dependency.bundle !== false && hasOnlyExpectedAttributes(dependency.attributes, { type: "module", defer: "" }) && existsSync(dependency.filepath);
67
+ }).filter(isBundleableFileScriptAsset);
68
+ const bundledStylesheet = bundleableStyles.length > 1 ? AssetFactory.createContentStylesheet({
69
+ content: bundleableStyles.map((dependency) => readFileSync(dependency.filepath, "utf8")).join("\n"),
70
+ position: "head",
71
+ attributes: { rel: "stylesheet" },
72
+ packageRole: "page-style",
73
+ bundledSourceFilepaths: bundleableStyles.map((dependency) => dependency.filepath)
74
+ }) : void 0;
75
+ const pageScriptImports = [...new Set(bundleableScripts.map((dependency) => dependency.filepath))];
76
+ const shouldBundlePageScript = pageScriptImports.length > 0 && bundleableScripts.length > 1;
77
+ const bundledScript = shouldBundlePageScript ? AssetFactory.createContentScript({
78
+ name: `${integrationName}-page-${rapidhash(pageScriptImports.join("|")).toString(16)}`,
79
+ content: pageScriptImports.map((filepath) => `import ${JSON.stringify(filepath)};`).join("\n"),
80
+ position: "head",
81
+ attributes: { type: "module", defer: "" },
82
+ packageRole: "page-script",
83
+ bundledSourceFilepaths: pageScriptImports
84
+ }) : void 0;
85
+ if (!bundledStylesheet && !bundledScript) {
86
+ return void 0;
87
+ }
88
+ return {
89
+ bundledStylesheet,
90
+ bundledScript,
91
+ bundleableStyleFilepaths: new Set(bundleableStyles.map((dependency) => dependency.filepath)),
92
+ bundleableScriptFilepaths: new Set(pageScriptImports)
93
+ };
94
+ }
95
+ function applyPageDependencyPackagingPlan(dependencies, plan) {
96
+ const unifiedDependencies = [];
97
+ let insertedStylesheet = false;
98
+ let insertedScript = false;
99
+ for (const dependency of dependencies) {
100
+ if (plan.bundledStylesheet && dependency.kind === "stylesheet" && dependency.source === "file" && plan.bundleableStyleFilepaths.has(dependency.filepath)) {
101
+ if (!insertedStylesheet) {
102
+ unifiedDependencies.push(plan.bundledStylesheet);
103
+ insertedStylesheet = true;
104
+ }
105
+ continue;
106
+ }
107
+ if (plan.bundledScript && dependency.kind === "script" && dependency.source === "file" && plan.bundleableScriptFilepaths.has(dependency.filepath)) {
108
+ if (!insertedScript) {
109
+ unifiedDependencies.push(plan.bundledScript);
110
+ insertedScript = true;
111
+ }
112
+ continue;
113
+ }
114
+ unifiedDependencies.push(dependency);
115
+ }
116
+ if (plan.bundledScript && !insertedScript) {
117
+ unifiedDependencies.push(plan.bundledScript);
118
+ }
119
+ if (plan.bundledStylesheet && !insertedStylesheet) {
120
+ unifiedDependencies.push(plan.bundledStylesheet);
121
+ }
122
+ return unifiedDependencies;
123
+ }
124
+ function packagePageDependencies(dependencies, integrationName) {
125
+ if (!shouldBundlePageDependencies(integrationName)) {
126
+ return dependencies;
127
+ }
128
+ const plan = createPageDependencyPackagingPlan(dependencies, integrationName);
129
+ if (!plan) {
130
+ return dependencies;
131
+ }
132
+ return applyPageDependencyPackagingPlan(dependencies, plan);
133
+ }
134
+ export {
135
+ packagePageDependencies,
136
+ shouldBundlePageDependencies
137
+ };
@@ -0,0 +1,90 @@
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
+ cacheScope?: string;
33
+ }): Promise<EcoPageFile>;
34
+ /**
35
+ * Executes the page's static-props hook with Ecopages runtime context.
36
+ *
37
+ * @remarks
38
+ * Pages without a static-props hook still return a normalized empty props
39
+ * object so downstream render preparation does not branch on hook presence.
40
+ */
41
+ getStaticPropsForPage(options: {
42
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
43
+ params?: RouteRendererOptions['params'];
44
+ }): Promise<{
45
+ props: Record<string, unknown>;
46
+ metadata?: PageMetadataProps;
47
+ }>;
48
+ /**
49
+ * Builds the final page metadata object for one render request.
50
+ *
51
+ * @remarks
52
+ * App-level default metadata forms the baseline, then page-level metadata is
53
+ * overlaid so route-specific fields win without dropping global defaults.
54
+ */
55
+ getMetadataPropsForPage(options: {
56
+ getMetadata: GetMetadata | undefined;
57
+ context: GetMetadataContext;
58
+ }): Promise<PageMetadataProps>;
59
+ /**
60
+ * Loads a page module and normalizes integration-facing exports.
61
+ * When both component static methods and module exports exist, component statics win.
62
+ */
63
+ resolvePageModule(options: {
64
+ file: string;
65
+ importPageFileFn?: (file: string) => Promise<EcoPageFile>;
66
+ }): Promise<{
67
+ Page: EcoPageFile['default'] | EcoPageComponent<any>;
68
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
69
+ getMetadata?: GetMetadata;
70
+ integrationSpecificProps: Record<string, unknown>;
71
+ }>;
72
+ /**
73
+ * Resolves the page data needed by the render pipeline.
74
+ *
75
+ * @remarks
76
+ * Static props are resolved first because page metadata may depend on those
77
+ * props. This preserves the same ordering whether data hooks are declared as
78
+ * component statics or module exports.
79
+ */
80
+ resolvePageData(options: {
81
+ pageModule: {
82
+ getStaticProps?: GetStaticProps<Record<string, unknown>>;
83
+ getMetadata?: GetMetadata;
84
+ };
85
+ routeOptions: RouteRendererOptions;
86
+ }): Promise<{
87
+ props: Record<string, unknown>;
88
+ metadata: PageMetadataProps;
89
+ }>;
90
+ }
@@ -1,22 +1,36 @@
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,
33
+ cacheScope: options?.cacheScope,
20
34
  transpileErrorMessage: (details) => `Error transpiling page file: ${details}`,
21
35
  noOutputMessage: (targetFilePath) => `No transpiled output generated for page: ${targetFilePath}`
22
36
  });
@@ -25,8 +39,11 @@ class PageModuleLoaderService {
25
39
  }
26
40
  }
27
41
  /**
28
- * Executes `getStaticProps` with Ecopages runtime context.
29
- * Returns an empty props object when no static props function is defined.
42
+ * Executes the page's static-props hook with Ecopages runtime context.
43
+ *
44
+ * @remarks
45
+ * Pages without a static-props hook still return a normalized empty props
46
+ * object so downstream render preparation does not branch on hook presence.
30
47
  */
31
48
  async getStaticPropsForPage(options) {
32
49
  const { getStaticProps, params } = options;
@@ -42,8 +59,11 @@ class PageModuleLoaderService {
42
59
  };
43
60
  }
44
61
  /**
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.
62
+ * Builds the final page metadata object for one render request.
63
+ *
64
+ * @remarks
65
+ * App-level default metadata forms the baseline, then page-level metadata is
66
+ * overlaid so route-specific fields win without dropping global defaults.
47
67
  */
48
68
  async getMetadataPropsForPage(options) {
49
69
  const { getMetadata, context } = options;
@@ -79,7 +99,12 @@ class PageModuleLoaderService {
79
99
  };
80
100
  }
81
101
  /**
82
- * Resolves render-time page data in order: static props first, then metadata derived from those props.
102
+ * Resolves the page data needed by the render pipeline.
103
+ *
104
+ * @remarks
105
+ * Static props are resolved first because page metadata may depend on those
106
+ * props. This preserves the same ordering whether data hooks are declared as
107
+ * component statics or module exports.
83
108
  */
84
109
  async resolvePageData(options) {
85
110
  const { props } = await this.getStaticPropsForPage({
@@ -1,26 +1,69 @@
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>;
1
+ import type { EcoPagesAppConfig } from '../types/internal-types.js';
2
+ import type { AnyIntegrationPlugin } from '../plugins/integration-plugin.js';
3
+ import type { IntegrationRenderer } from './orchestration/integration-renderer.js';
4
+ /**
5
+ * Narrow route-render contract exposed to higher-level routing code.
6
+ *
7
+ * @remarks
8
+ * Higher-level routing code only needs request execution and page-module
9
+ * loading. Returning this narrowed shape avoids a dedicated wrapper class while
10
+ * still keeping callers off the full integration renderer surface.
11
+ */
12
+ export type PageRouteRenderer = Pick<IntegrationRenderer<unknown>, 'execute' | 'loadPageModule'>;
13
+ /**
14
+ * Narrow explicit-view render contract exposed to static route handling.
15
+ *
16
+ * @remarks
17
+ * Explicit static routes only need `renderToResponse()`, so the factory can
18
+ * hide the broader integration renderer surface there as well.
19
+ */
20
+ export type ExplicitViewRenderer = Pick<IntegrationRenderer<unknown>, 'renderToResponse'>;
21
+ export interface PageRendererResolver {
22
+ getPageRenderer(filePath: string): PageRouteRenderer;
9
23
  }
24
+ export interface ExplicitViewRendererResolver {
25
+ getExplicitViewRenderer(integrationName: string): ExplicitViewRenderer | null;
26
+ }
27
+ /**
28
+ * Combined renderer-factory contract used by static generation.
29
+ */
30
+ export type StaticGenerationRendererResolver = PageRendererResolver & ExplicitViewRendererResolver;
31
+ /**
32
+ * Selects and caches integration renderers for route files and explicit views.
33
+ *
34
+ * @remarks
35
+ * The factory owns the policy that maps a route file or explicit integration
36
+ * name to one initialized integration renderer. Renderer instances are cached by
37
+ * integration name so repeated requests do not rebuild renderer state.
38
+ */
10
39
  export declare class RouteRendererFactory {
11
40
  private appConfig;
12
41
  runtimeOrigin: string;
42
+ private rendererModules?;
13
43
  private rendererCache;
14
- constructor({ appConfig, runtimeOrigin }: {
44
+ /**
45
+ * Creates the route-renderer factory for one app/runtime instance.
46
+ */
47
+ constructor({ appConfig, rendererModules, runtimeOrigin, }: {
15
48
  appConfig: EcoPagesAppConfig;
49
+ rendererModules?: unknown;
16
50
  runtimeOrigin: string;
17
51
  });
18
- createRenderer(filePath: string): RouteRenderer;
19
52
  /**
20
- * Get an integration renderer by its name.
21
- * Used for explicit routing where views specify their integration via __eco.integration.
53
+ * Returns a route renderer for the supplied route file.
54
+ */
55
+ getPageRenderer(filePath: string): PageRouteRenderer;
56
+ /**
57
+ * Returns a renderer for an explicit view integration.
58
+ */
59
+ getExplicitViewRenderer(integrationName: string): ExplicitViewRenderer | null;
60
+ /**
61
+ * Resolves the integration plugin that owns a given route file.
62
+ */
63
+ getIntegrationPlugin(filePath: string): AnyIntegrationPlugin;
64
+ /**
65
+ * Returns the cached renderer engine for the file's owning integration,
66
+ * creating it on first use.
22
67
  */
23
- getRendererByIntegration(integrationName: string): IntegrationRenderer | null;
24
- getIntegrationPlugin(filePath: string): IntegrationPlugin;
25
68
  private getRouteRendererEngine;
26
69
  }
@@ -1,32 +1,34 @@
1
1
  import { invariant } from "../utils/invariant.js";
2
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
3
  class RouteRendererFactory {
13
4
  appConfig;
14
5
  runtimeOrigin;
6
+ rendererModules;
15
7
  rendererCache = /* @__PURE__ */ new Map();
16
- constructor({ appConfig, runtimeOrigin }) {
8
+ /**
9
+ * Creates the route-renderer factory for one app/runtime instance.
10
+ */
11
+ constructor({
12
+ appConfig,
13
+ rendererModules,
14
+ runtimeOrigin
15
+ }) {
17
16
  this.appConfig = appConfig;
17
+ this.rendererModules = rendererModules;
18
18
  this.runtimeOrigin = runtimeOrigin;
19
19
  }
20
- createRenderer(filePath) {
20
+ /**
21
+ * Returns a route renderer for the supplied route file.
22
+ */
23
+ getPageRenderer(filePath) {
21
24
  const integrationRenderer = this.getRouteRendererEngine(filePath);
22
25
  invariant(!!integrationRenderer, `No integration renderer found for file: ${filePath}`);
23
- return new RouteRenderer(integrationRenderer);
26
+ return integrationRenderer;
24
27
  }
25
28
  /**
26
- * Get an integration renderer by its name.
27
- * Used for explicit routing where views specify their integration via __eco.integration.
29
+ * Returns a renderer for an explicit view integration.
28
30
  */
29
- getRendererByIntegration(integrationName) {
31
+ getExplicitViewRenderer(integrationName) {
30
32
  const integrationPlugin = this.appConfig.integrations.find((plugin) => plugin.name === integrationName);
31
33
  if (!integrationPlugin) {
32
34
  return null;
@@ -35,10 +37,15 @@ class RouteRendererFactory {
35
37
  if (cached) {
36
38
  return cached;
37
39
  }
38
- const renderer = integrationPlugin.initializeRenderer();
40
+ const renderer = integrationPlugin.initializeRenderer({
41
+ rendererModules: this.rendererModules
42
+ });
39
43
  this.rendererCache.set(integrationName, renderer);
40
44
  return renderer;
41
45
  }
46
+ /**
47
+ * Resolves the integration plugin that owns a given route file.
48
+ */
42
49
  getIntegrationPlugin(filePath) {
43
50
  const templateExtension = PathUtils.getEcoTemplateExtension(filePath);
44
51
  const isIntegrationPlugin = (plugin) => {
@@ -51,18 +58,23 @@ class RouteRendererFactory {
51
58
  );
52
59
  return integrationPlugin;
53
60
  }
61
+ /**
62
+ * Returns the cached renderer engine for the file's owning integration,
63
+ * creating it on first use.
64
+ */
54
65
  getRouteRendererEngine(filePath) {
55
66
  const integrationPlugin = this.getIntegrationPlugin(filePath);
56
67
  const cached = this.rendererCache.get(integrationPlugin.name);
57
68
  if (cached) {
58
69
  return cached;
59
70
  }
60
- const renderer = integrationPlugin.initializeRenderer();
71
+ const renderer = integrationPlugin.initializeRenderer({
72
+ rendererModules: this.rendererModules
73
+ });
61
74
  this.rendererCache.set(integrationPlugin.name, renderer);
62
75
  return renderer;
63
76
  }
64
77
  }
65
78
  export {
66
- RouteRenderer,
67
79
  RouteRendererFactory
68
80
  };
@@ -0,0 +1,94 @@
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 discovery and classification (`exact`, `dynamic`, `catch-all`)
12
+ - matching incoming request URLs to canonical template routes
13
+ - server-side static-path expansion for dynamic template 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 discovery and matching
22
+ │ └── route-registry.ts # Owns template routes, request matching, static expansion, and reload
23
+ └── client/ # Browser-side navigation coordination
24
+ ├── navigation-coordinator.ts # Singleton runtime coordinator
25
+ └── link-intent.ts # Shared anchor detection and intent recovery helpers
26
+ ```
27
+
28
+ ## `server/`
29
+
30
+ ### `RouteRegistry`
31
+
32
+ Owns the canonical set of filesystem-discovered template routes for one application.
33
+
34
+ File patterns determine route kind:
35
+
36
+ | Pattern | Kind | Example |
37
+ | --------------- | ----------- | ----------------- |
38
+ | `page.tsx` | `exact` | `/about` |
39
+ | `[slug].tsx` | `dynamic` | `/blog/[slug]` |
40
+ | `[...slug].tsx` | `catch-all` | `/docs/[...slug]` |
41
+
42
+ The registry stores canonical template routes only. It compiles request-time matching metadata during `init()` and `reload()`, but it does not execute `staticPaths()` during discovery.
43
+
44
+ Build-time static expansion is a separate operation. The registry invokes `staticPaths()` lazily through an injected page-module adapter and returns concrete static path expansions for the static generator.
45
+
46
+ The public interface is intentionally small:
47
+
48
+ - `templateRoutes` — ordered readonly template routes
49
+ - `init()` / `reload()` — rebuild discovery state and match metadata
50
+ - `matchRequest(requestUrl)` — request-time matching result with requested pathname, matched template route, params, and query
51
+ - `listStaticPathExpansions()` — build-time expansion for dynamic routes
52
+ - `listStaticGenerationRoutes()` — build-time route planning for static generation across exact and expanded routes
53
+
54
+ Match priority:
55
+
56
+ 1. `exact` — the pathname must equal the route pathname exactly.
57
+ 2. `dynamic` — the clean (bracket-stripped) prefix must appear in the pathname, and the segment counts must match.
58
+ 3. `catch-all` — the clean prefix must appear in the pathname.
59
+
60
+ ## `client/`
61
+
62
+ ### Navigation Coordinator (`navigation-coordinator.ts`)
63
+
64
+ A singleton browser-side runtime stored on `window.__ECO_PAGES__.navigation`.
65
+
66
+ Access it with:
67
+
68
+ ```ts
69
+ import { getEcoNavigationRuntime } from '@ecopages/core/router/navigation-coordinator';
70
+ const runtime = getEcoNavigationRuntime();
71
+ ```
72
+
73
+ The coordinator is framework-agnostic. Browser runtimes (e.g. `browser-router`, `react-router`) register themselves and the coordinator arbitrates:
74
+
75
+ - **Ownership** — which runtime currently drives SPA navigation (`claimOwnership`, `releaseOwnership`, `setOwner`).
76
+ - **Document owner marker** — an HTML attribute (`data-eco-document-owner`) written into rendered markup so the coordinator can `adoptDocumentOwner` on the incoming page.
77
+ - **Navigation transactions** — each navigation begins a transaction with an `AbortSignal`; superseded navigations are automatically cancelled.
78
+ - **Cross-runtime handoff** — `requestHandoff` passes a pre-fetched `Document` to the target runtime without tearing down the current page prematurely.
79
+ - **Reload** — `reloadCurrentPage` delegates to whichever runtime currently owns the document.
80
+ - **Events** — `subscribe` lets runtimes react to `owner-change` and `registration-change` events.
81
+
82
+ ### Link Intent (`link-intent.ts`)
83
+
84
+ Shared helpers for locating anchors and recovering stale navigation intent.
85
+
86
+ - `getAnchorFromNavigationEvent(event, linkSelector)` — finds the nearest matching anchor in the event's composed path, including across Shadow DOM boundaries.
87
+ - `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).
88
+
89
+ ## Relationship To Rendering
90
+
91
+ The router layer answers **which route should run**.
92
+ The route-renderer layer answers **how that route gets rendered**.
93
+
94
+ Keeping those seams separate avoids mixing route ownership, module loading, and component orchestration into one service.
@@ -0,0 +1,53 @@
1
+ /**
2
+ * Shared client-side navigation intent helpers.
3
+ *
4
+ * Browser runtimes use the same low-level mechanics to:
5
+ * - locate the anchor associated with a click, pointer, or hover event,
6
+ * - persist the last valid pointer or hover target while a navigation is in flight,
7
+ * - recover the final intended href when the DOM changes before the click lands.
8
+ *
9
+ * Keeping those mechanics here lets browser-router and react-router share one
10
+ * implementation while preserving their router-specific interception rules.
11
+ *
12
+ * @module
13
+ */
14
+ /**
15
+ * Timestamped navigation intent captured from pointer or hover activity.
16
+ */
17
+ export type EcoPendingNavigationIntent = {
18
+ href: string;
19
+ timestamp: number;
20
+ };
21
+ /**
22
+ * Finds the nearest matching anchor within an event's composed path.
23
+ *
24
+ * This works across Shadow DOM boundaries, which is required for delegated
25
+ * navigation handling when links are rendered inside custom elements.
26
+ *
27
+ * @param event - Pointer or mouse event being inspected.
28
+ * @param linkSelector - Selector that identifies navigable anchors.
29
+ * @returns The matched anchor element, or `null` when no matching anchor exists.
30
+ */
31
+ export declare function getAnchorFromNavigationEvent(
32
+ event: MouseEvent | PointerEvent,
33
+ linkSelector: string,
34
+ ): HTMLAnchorElement | null;
35
+ /**
36
+ * Resolves a previously captured intent while a navigation is still in flight.
37
+ *
38
+ * Pending intents expire quickly because they are only meant to bridge the gap
39
+ * between pointer or hover capture and the later click event when the DOM or
40
+ * active runtime changes during a rapid navigation sequence.
41
+ *
42
+ * @param intent - Previously captured pointer or hover intent.
43
+ * @param hasInFlightNavigation - Whether a router navigation is still active.
44
+ * @param now - Current monotonic timestamp, usually from `performance.now()`.
45
+ * @param maxAgeMs - Maximum allowed age for the recovered intent.
46
+ * @returns The intended href when still valid, otherwise `null`.
47
+ */
48
+ export declare function recoverPendingNavigationHref(
49
+ intent: EcoPendingNavigationIntent | null,
50
+ hasInFlightNavigation: boolean,
51
+ now: number,
52
+ maxAgeMs?: number,
53
+ ): string | null;
@@ -0,0 +1,34 @@
1
+ function getAnchorFromNavigationEvent(event, linkSelector) {
2
+ const eventTarget = event.target;
3
+ if (eventTarget instanceof HTMLAnchorElement && eventTarget.matches(linkSelector)) {
4
+ return eventTarget;
5
+ }
6
+ if (eventTarget instanceof Element) {
7
+ const closestAnchor = eventTarget.closest(linkSelector);
8
+ if (closestAnchor instanceof HTMLAnchorElement) {
9
+ return closestAnchor;
10
+ }
11
+ }
12
+ if (eventTarget instanceof Text) {
13
+ const parentAnchor = eventTarget.parentElement?.closest(linkSelector);
14
+ if (parentAnchor instanceof HTMLAnchorElement) {
15
+ return parentAnchor;
16
+ }
17
+ }
18
+ return event.composedPath().find(
19
+ (target) => target instanceof HTMLAnchorElement && target.matches(linkSelector)
20
+ );
21
+ }
22
+ function recoverPendingNavigationHref(intent, hasInFlightNavigation, now, maxAgeMs = 1e3) {
23
+ if (!intent || !hasInFlightNavigation) {
24
+ return null;
25
+ }
26
+ if (now - intent.timestamp > maxAgeMs) {
27
+ return null;
28
+ }
29
+ return intent.href;
30
+ }
31
+ export {
32
+ getAnchorFromNavigationEvent,
33
+ recoverPendingNavigationHref
34
+ };
@@ -0,0 +1 @@
1
+ export {};