@ecopages/core 0.2.0-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (342) hide show
  1. package/CHANGELOG.md +89 -0
  2. package/LICENSE +21 -0
  3. package/README.md +32 -0
  4. package/package.json +279 -0
  5. package/src/adapters/abstract/application-adapter.d.ts +168 -0
  6. package/src/adapters/abstract/application-adapter.js +109 -0
  7. package/src/adapters/abstract/application-adapter.ts +337 -0
  8. package/src/adapters/abstract/router-adapter.d.ts +26 -0
  9. package/src/adapters/abstract/router-adapter.js +5 -0
  10. package/src/adapters/abstract/router-adapter.ts +30 -0
  11. package/src/adapters/abstract/server-adapter.d.ts +69 -0
  12. package/src/adapters/abstract/server-adapter.js +15 -0
  13. package/src/adapters/abstract/server-adapter.ts +79 -0
  14. package/src/adapters/bun/client-bridge.d.ts +34 -0
  15. package/src/adapters/bun/client-bridge.js +48 -0
  16. package/src/adapters/bun/client-bridge.ts +62 -0
  17. package/src/adapters/bun/create-app.d.ts +60 -0
  18. package/src/adapters/bun/create-app.js +117 -0
  19. package/src/adapters/bun/create-app.ts +189 -0
  20. package/src/adapters/bun/define-api-handler.d.ts +61 -0
  21. package/src/adapters/bun/define-api-handler.js +15 -0
  22. package/src/adapters/bun/define-api-handler.ts +114 -0
  23. package/src/adapters/bun/hmr-manager.d.ts +84 -0
  24. package/src/adapters/bun/hmr-manager.js +227 -0
  25. package/src/adapters/bun/hmr-manager.ts +281 -0
  26. package/src/adapters/bun/index.d.ts +3 -0
  27. package/src/adapters/bun/index.js +8 -0
  28. package/src/adapters/bun/index.ts +3 -0
  29. package/src/adapters/bun/server-adapter.d.ts +155 -0
  30. package/src/adapters/bun/server-adapter.js +368 -0
  31. package/src/adapters/bun/server-adapter.ts +492 -0
  32. package/src/adapters/bun/server-lifecycle.d.ts +52 -0
  33. package/src/adapters/bun/server-lifecycle.js +120 -0
  34. package/src/adapters/bun/server-lifecycle.ts +154 -0
  35. package/src/adapters/index.d.ts +6 -0
  36. package/src/adapters/index.js +14 -0
  37. package/src/adapters/index.ts +6 -0
  38. package/src/adapters/node/create-app.d.ts +21 -0
  39. package/src/adapters/node/create-app.js +143 -0
  40. package/src/adapters/node/create-app.ts +179 -0
  41. package/src/adapters/node/index.d.ts +4 -0
  42. package/src/adapters/node/index.js +8 -0
  43. package/src/adapters/node/index.ts +9 -0
  44. package/src/adapters/node/node-client-bridge.d.ts +26 -0
  45. package/src/adapters/node/node-client-bridge.js +66 -0
  46. package/src/adapters/node/node-client-bridge.ts +79 -0
  47. package/src/adapters/node/node-hmr-manager.d.ts +62 -0
  48. package/src/adapters/node/node-hmr-manager.js +221 -0
  49. package/src/adapters/node/node-hmr-manager.ts +271 -0
  50. package/src/adapters/node/server-adapter.d.ts +190 -0
  51. package/src/adapters/node/server-adapter.js +420 -0
  52. package/src/adapters/node/server-adapter.ts +561 -0
  53. package/src/adapters/node/static-content-server.d.ts +24 -0
  54. package/src/adapters/node/static-content-server.js +166 -0
  55. package/src/adapters/node/static-content-server.ts +203 -0
  56. package/src/adapters/shared/api-response.d.ts +52 -0
  57. package/src/adapters/shared/api-response.js +96 -0
  58. package/src/adapters/shared/api-response.ts +104 -0
  59. package/src/adapters/shared/application-adapter.d.ts +18 -0
  60. package/src/adapters/shared/application-adapter.js +90 -0
  61. package/src/adapters/shared/application-adapter.ts +199 -0
  62. package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
  63. package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
  64. package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
  65. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
  66. package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
  67. package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
  68. package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
  69. package/src/adapters/shared/fs-server-response-factory.js +97 -0
  70. package/src/adapters/shared/fs-server-response-factory.ts +118 -0
  71. package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
  72. package/src/adapters/shared/fs-server-response-matcher.js +155 -0
  73. package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
  74. package/src/adapters/shared/render-context.d.ts +14 -0
  75. package/src/adapters/shared/render-context.js +69 -0
  76. package/src/adapters/shared/render-context.ts +105 -0
  77. package/src/adapters/shared/server-adapter.d.ts +87 -0
  78. package/src/adapters/shared/server-adapter.js +353 -0
  79. package/src/adapters/shared/server-adapter.ts +442 -0
  80. package/src/adapters/shared/server-route-handler.d.ts +89 -0
  81. package/src/adapters/shared/server-route-handler.js +120 -0
  82. package/src/adapters/shared/server-route-handler.ts +166 -0
  83. package/src/adapters/shared/server-static-builder.d.ts +38 -0
  84. package/src/adapters/shared/server-static-builder.js +46 -0
  85. package/src/adapters/shared/server-static-builder.ts +82 -0
  86. package/src/build/build-adapter.d.ts +74 -0
  87. package/src/build/build-adapter.js +54 -0
  88. package/src/build/build-adapter.ts +132 -0
  89. package/src/build/build-types.d.ts +57 -0
  90. package/src/build/build-types.js +0 -0
  91. package/src/build/build-types.ts +83 -0
  92. package/src/build/esbuild-build-adapter.d.ts +69 -0
  93. package/src/build/esbuild-build-adapter.js +390 -0
  94. package/src/build/esbuild-build-adapter.ts +510 -0
  95. package/src/config/config-builder.d.ts +227 -0
  96. package/src/config/config-builder.js +392 -0
  97. package/src/config/config-builder.ts +474 -0
  98. package/src/constants.d.ts +32 -0
  99. package/src/constants.js +21 -0
  100. package/src/constants.ts +39 -0
  101. package/src/create-app.d.ts +17 -0
  102. package/src/create-app.js +66 -0
  103. package/src/create-app.ts +87 -0
  104. package/src/declarations.d.ts +26 -0
  105. package/src/define-api-handler.d.ts +25 -0
  106. package/src/define-api-handler.js +15 -0
  107. package/src/define-api-handler.ts +66 -0
  108. package/src/dev/sc-server.d.ts +30 -0
  109. package/src/dev/sc-server.js +111 -0
  110. package/src/dev/sc-server.ts +143 -0
  111. package/src/eco/README.md +636 -0
  112. package/src/eco/component-render-context.d.ts +105 -0
  113. package/src/eco/component-render-context.js +77 -0
  114. package/src/eco/component-render-context.ts +202 -0
  115. package/src/eco/eco.d.ts +9 -0
  116. package/src/eco/eco.js +110 -0
  117. package/src/eco/eco.ts +221 -0
  118. package/src/eco/eco.types.d.ts +170 -0
  119. package/src/eco/eco.types.js +0 -0
  120. package/src/eco/eco.types.ts +202 -0
  121. package/src/eco/eco.utils.d.ts +40 -0
  122. package/src/eco/eco.utils.js +40 -0
  123. package/src/eco/eco.utils.ts +89 -0
  124. package/src/eco/global-injector-map.d.ts +16 -0
  125. package/src/eco/global-injector-map.js +80 -0
  126. package/src/eco/global-injector-map.ts +112 -0
  127. package/src/eco/lazy-injector-map.d.ts +8 -0
  128. package/src/eco/lazy-injector-map.js +70 -0
  129. package/src/eco/lazy-injector-map.ts +120 -0
  130. package/src/eco/module-dependencies.d.ts +18 -0
  131. package/src/eco/module-dependencies.js +49 -0
  132. package/src/eco/module-dependencies.ts +75 -0
  133. package/src/env.d.ts +20 -0
  134. package/src/errors/http-error.d.ts +31 -0
  135. package/src/errors/http-error.js +50 -0
  136. package/src/errors/http-error.ts +72 -0
  137. package/src/errors/index.d.ts +2 -0
  138. package/src/errors/index.js +4 -0
  139. package/src/errors/index.ts +2 -0
  140. package/src/errors/locals-access-error.d.ts +4 -0
  141. package/src/errors/locals-access-error.js +9 -0
  142. package/src/errors/locals-access-error.ts +7 -0
  143. package/src/global/app-logger.d.ts +2 -0
  144. package/src/global/app-logger.js +6 -0
  145. package/src/global/app-logger.ts +4 -0
  146. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-have-HMR-script-injected-in-page-1.png +0 -0
  147. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  148. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  149. package/src/hmr/client/hmr-runtime.d.ts +10 -0
  150. package/src/hmr/client/hmr-runtime.js +86 -0
  151. package/src/hmr/client/hmr-runtime.ts +121 -0
  152. package/src/hmr/hmr-strategy.d.ts +159 -0
  153. package/src/hmr/hmr-strategy.js +29 -0
  154. package/src/hmr/hmr-strategy.ts +172 -0
  155. package/src/hmr/hmr.test.e2e.d.ts +1 -0
  156. package/src/hmr/hmr.test.e2e.js +50 -0
  157. package/src/hmr/hmr.test.e2e.ts +75 -0
  158. package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
  159. package/src/hmr/strategies/default-hmr-strategy.js +34 -0
  160. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  161. package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
  162. package/src/hmr/strategies/js-hmr-strategy.js +179 -0
  163. package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
  164. package/src/index.browser.d.ts +3 -0
  165. package/src/index.browser.js +4 -0
  166. package/src/index.browser.ts +3 -0
  167. package/src/index.d.ts +5 -0
  168. package/src/index.js +10 -0
  169. package/src/index.ts +5 -0
  170. package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
  171. package/src/integrations/ghtml/ghtml-renderer.js +60 -0
  172. package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
  173. package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
  174. package/src/integrations/ghtml/ghtml.plugin.js +21 -0
  175. package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
  176. package/src/internal-types.d.ts +200 -0
  177. package/src/internal-types.js +0 -0
  178. package/src/internal-types.ts +212 -0
  179. package/src/plugins/alias-resolver-plugin.d.ts +2 -0
  180. package/src/plugins/alias-resolver-plugin.js +39 -0
  181. package/src/plugins/alias-resolver-plugin.ts +45 -0
  182. package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
  183. package/src/plugins/eco-component-meta-plugin.js +157 -0
  184. package/src/plugins/eco-component-meta-plugin.ts +474 -0
  185. package/src/plugins/integration-plugin.d.ts +102 -0
  186. package/src/plugins/integration-plugin.js +100 -0
  187. package/src/plugins/integration-plugin.ts +184 -0
  188. package/src/plugins/processor.d.ts +82 -0
  189. package/src/plugins/processor.js +122 -0
  190. package/src/plugins/processor.ts +220 -0
  191. package/src/public-types.d.ts +1094 -0
  192. package/src/public-types.js +0 -0
  193. package/src/public-types.ts +1255 -0
  194. package/src/route-renderer/GRAPH.md +387 -0
  195. package/src/route-renderer/README.md +135 -0
  196. package/src/route-renderer/component-graph-executor.d.ts +32 -0
  197. package/src/route-renderer/component-graph-executor.js +31 -0
  198. package/src/route-renderer/component-graph-executor.ts +84 -0
  199. package/src/route-renderer/component-graph.d.ts +42 -0
  200. package/src/route-renderer/component-graph.js +72 -0
  201. package/src/route-renderer/component-graph.ts +159 -0
  202. package/src/route-renderer/component-marker.d.ts +52 -0
  203. package/src/route-renderer/component-marker.js +46 -0
  204. package/src/route-renderer/component-marker.ts +117 -0
  205. package/src/route-renderer/dependency-resolver.d.ts +24 -0
  206. package/src/route-renderer/dependency-resolver.js +428 -0
  207. package/src/route-renderer/dependency-resolver.ts +596 -0
  208. package/src/route-renderer/html-post-processing.service.d.ts +40 -0
  209. package/src/route-renderer/html-post-processing.service.js +86 -0
  210. package/src/route-renderer/html-post-processing.service.ts +103 -0
  211. package/src/route-renderer/integration-renderer.d.ts +339 -0
  212. package/src/route-renderer/integration-renderer.js +526 -0
  213. package/src/route-renderer/integration-renderer.ts +696 -0
  214. package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
  215. package/src/route-renderer/marker-graph-resolver.js +93 -0
  216. package/src/route-renderer/marker-graph-resolver.ts +153 -0
  217. package/src/route-renderer/page-module-loader.d.ts +61 -0
  218. package/src/route-renderer/page-module-loader.js +102 -0
  219. package/src/route-renderer/page-module-loader.ts +153 -0
  220. package/src/route-renderer/render-execution.service.d.ts +69 -0
  221. package/src/route-renderer/render-execution.service.js +91 -0
  222. package/src/route-renderer/render-execution.service.ts +158 -0
  223. package/src/route-renderer/render-preparation.service.d.ts +112 -0
  224. package/src/route-renderer/render-preparation.service.js +243 -0
  225. package/src/route-renderer/render-preparation.service.ts +358 -0
  226. package/src/route-renderer/route-renderer.d.ts +26 -0
  227. package/src/route-renderer/route-renderer.js +68 -0
  228. package/src/route-renderer/route-renderer.ts +80 -0
  229. package/src/router/fs-router-scanner.d.ts +41 -0
  230. package/src/router/fs-router-scanner.js +155 -0
  231. package/src/router/fs-router-scanner.ts +217 -0
  232. package/src/router/fs-router.d.ts +26 -0
  233. package/src/router/fs-router.js +100 -0
  234. package/src/router/fs-router.ts +122 -0
  235. package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
  236. package/src/services/asset-processing-service/asset-processing.service.js +250 -0
  237. package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
  238. package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
  239. package/src/services/asset-processing-service/asset.factory.js +82 -0
  240. package/src/services/asset-processing-service/asset.factory.ts +105 -0
  241. package/src/services/asset-processing-service/assets.types.d.ts +88 -0
  242. package/src/services/asset-processing-service/assets.types.js +0 -0
  243. package/src/services/asset-processing-service/assets.types.ts +112 -0
  244. package/src/services/asset-processing-service/index.d.ts +3 -0
  245. package/src/services/asset-processing-service/index.js +3 -0
  246. package/src/services/asset-processing-service/index.ts +3 -0
  247. package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
  248. package/src/services/asset-processing-service/processor.interface.js +6 -0
  249. package/src/services/asset-processing-service/processor.interface.ts +27 -0
  250. package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
  251. package/src/services/asset-processing-service/processor.registry.js +15 -0
  252. package/src/services/asset-processing-service/processor.registry.ts +18 -0
  253. package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
  254. package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
  255. package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
  256. package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
  257. package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
  258. package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
  259. package/src/services/asset-processing-service/processors/index.d.ts +5 -0
  260. package/src/services/asset-processing-service/processors/index.js +5 -0
  261. package/src/services/asset-processing-service/processors/index.ts +5 -0
  262. package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
  263. package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
  264. package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
  265. package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
  266. package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
  267. package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
  268. package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
  269. package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
  270. package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
  271. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
  272. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
  273. package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
  274. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
  275. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
  276. package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
  277. package/src/services/cache/cache.types.d.ts +107 -0
  278. package/src/services/cache/cache.types.js +0 -0
  279. package/src/services/cache/cache.types.ts +126 -0
  280. package/src/services/cache/index.d.ts +7 -0
  281. package/src/services/cache/index.js +7 -0
  282. package/src/services/cache/index.ts +18 -0
  283. package/src/services/cache/memory-cache-store.d.ts +42 -0
  284. package/src/services/cache/memory-cache-store.js +98 -0
  285. package/src/services/cache/memory-cache-store.ts +130 -0
  286. package/src/services/cache/page-cache-service.d.ts +70 -0
  287. package/src/services/cache/page-cache-service.js +152 -0
  288. package/src/services/cache/page-cache-service.ts +202 -0
  289. package/src/services/html-transformer.service.d.ts +50 -0
  290. package/src/services/html-transformer.service.js +163 -0
  291. package/src/services/html-transformer.service.ts +217 -0
  292. package/src/services/page-module-import.service.d.ts +37 -0
  293. package/src/services/page-module-import.service.js +88 -0
  294. package/src/services/page-module-import.service.ts +129 -0
  295. package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
  296. package/src/services/page-request-cache-coordinator.service.js +107 -0
  297. package/src/services/page-request-cache-coordinator.service.ts +128 -0
  298. package/src/services/schema-validation-service.d.ts +122 -0
  299. package/src/services/schema-validation-service.js +101 -0
  300. package/src/services/schema-validation-service.ts +204 -0
  301. package/src/services/validation/standard-schema.types.d.ts +65 -0
  302. package/src/services/validation/standard-schema.types.js +0 -0
  303. package/src/services/validation/standard-schema.types.ts +68 -0
  304. package/src/static-site-generator/static-site-generator.d.ts +57 -0
  305. package/src/static-site-generator/static-site-generator.js +272 -0
  306. package/src/static-site-generator/static-site-generator.ts +359 -0
  307. package/src/utils/css.d.ts +1 -0
  308. package/src/utils/css.js +7 -0
  309. package/src/utils/css.ts +5 -0
  310. package/src/utils/deep-merge.d.ts +14 -0
  311. package/src/utils/deep-merge.js +32 -0
  312. package/src/utils/deep-merge.ts +47 -0
  313. package/src/utils/hash.d.ts +1 -0
  314. package/src/utils/hash.js +7 -0
  315. package/src/utils/hash.ts +5 -0
  316. package/src/utils/html.d.ts +1 -0
  317. package/src/utils/html.js +4 -0
  318. package/src/utils/html.ts +1 -0
  319. package/src/utils/invariant.d.ts +5 -0
  320. package/src/utils/invariant.js +11 -0
  321. package/src/utils/invariant.ts +15 -0
  322. package/src/utils/locals-utils.d.ts +15 -0
  323. package/src/utils/locals-utils.js +24 -0
  324. package/src/utils/locals-utils.ts +37 -0
  325. package/src/utils/parse-cli-args.d.ts +24 -0
  326. package/src/utils/parse-cli-args.js +47 -0
  327. package/src/utils/parse-cli-args.ts +83 -0
  328. package/src/utils/path-utils.module.d.ts +5 -0
  329. package/src/utils/path-utils.module.js +14 -0
  330. package/src/utils/path-utils.module.ts +14 -0
  331. package/src/utils/runtime.d.ts +11 -0
  332. package/src/utils/runtime.js +40 -0
  333. package/src/utils/runtime.ts +44 -0
  334. package/src/utils/server-utils.module.d.ts +19 -0
  335. package/src/utils/server-utils.module.js +56 -0
  336. package/src/utils/server-utils.module.ts +67 -0
  337. package/src/watchers/project-watcher.d.ts +120 -0
  338. package/src/watchers/project-watcher.js +238 -0
  339. package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
  340. package/src/watchers/project-watcher.test-helpers.js +51 -0
  341. package/src/watchers/project-watcher.test-helpers.ts +40 -0
  342. package/src/watchers/project-watcher.ts +306 -0
@@ -0,0 +1,166 @@
1
+ import type { IHmrManager } from '../../public-types';
2
+ import type { FSRouter } from '../../router/fs-router';
3
+ import type { ExplicitStaticRouteMatcher } from './explicit-static-route-matcher';
4
+ import type { FileSystemResponseMatcher } from './fs-server-response-matcher';
5
+ import { appLogger } from '../../global/app-logger';
6
+ import { HttpError } from '../../errors/http-error';
7
+
8
+ /**
9
+ * Configuration parameters for ServerRouteHandler.
10
+ */
11
+ export interface ServerRouteHandlerParams {
12
+ /** File system router for matching request URLs to route handlers. */
13
+ router: FSRouter;
14
+ /** Matcher for handling file system route responses. */
15
+ fileSystemResponseMatcher: FileSystemResponseMatcher;
16
+ /** Optional matcher for explicit static routes like processed images or sitemaps. */
17
+ explicitStaticRouteMatcher?: ExplicitStaticRouteMatcher;
18
+ /** Enable watch mode for development. */
19
+ watch?: boolean;
20
+ /** HMR manager for broadcasting hot module reload events in development. */
21
+ hmrManager?: IHmrManager;
22
+ }
23
+
24
+ /**
25
+ * Handles HTTP requests and routing for the server.
26
+ *
27
+ * This class manages the request routing flow with a priority-based approach:
28
+ * 1. Explicit static routes (highest priority - intentional mappings like /sitemap.xml)
29
+ * 2. File-based routes without extensions (application routes)
30
+ * 3. Static file fallback (lowest priority - disk serving)
31
+ *
32
+ * In development mode, it also injects HMR scripts into HTML responses.
33
+ */
34
+ export class ServerRouteHandler {
35
+ private readonly router: FSRouter;
36
+ private readonly fileSystemResponseMatcher: FileSystemResponseMatcher;
37
+ private readonly explicitStaticRouteMatcher?: ExplicitStaticRouteMatcher;
38
+ private readonly watch: boolean;
39
+ private readonly hmrManager?: IHmrManager;
40
+
41
+ /**
42
+ * Creates a new ServerRouteHandler instance.
43
+ *
44
+ * @param params - Configuration parameters
45
+ */
46
+ constructor({
47
+ router,
48
+ fileSystemResponseMatcher,
49
+ explicitStaticRouteMatcher,
50
+ watch = false,
51
+ hmrManager,
52
+ }: ServerRouteHandlerParams) {
53
+ this.router = router;
54
+ this.fileSystemResponseMatcher = fileSystemResponseMatcher;
55
+ this.explicitStaticRouteMatcher = explicitStaticRouteMatcher;
56
+ this.watch = watch;
57
+ this.hmrManager = hmrManager;
58
+ }
59
+
60
+ /**
61
+ * Determines if HMR script should be injected.
62
+ *
63
+ * @returns true if in watch mode and HMR manager is enabled
64
+ */
65
+ shouldInjectHmrScript(): boolean {
66
+ return this.watch && this.hmrManager?.isEnabled() === true;
67
+ }
68
+
69
+ /**
70
+ * Checks if a response contains HTML content.
71
+ *
72
+ * @param response - The HTTP response to check
73
+ * @returns true if Content-Type header starts with 'text/html'
74
+ */
75
+ isHtmlResponse(response: Response): boolean {
76
+ const contentType = response.headers.get('Content-Type');
77
+ return contentType !== null && contentType.startsWith('text/html');
78
+ }
79
+
80
+ /**
81
+ * Handles HTTP requests from the router adapter.
82
+ *
83
+ * Priority-based routing flow:
84
+ * 1. Check explicit static routes first (e.g., /sitemap.xml, /image.webp from plugins)
85
+ * 2. Match file-based routes without extensions (application routes)
86
+ * 3. Fall back to static file serving from disk
87
+ *
88
+ * @param request - The incoming HTTP request
89
+ * @returns HTTP response, potentially with injected HMR script in dev mode
90
+ */
91
+ async handleResponse(request: Request): Promise<Response> {
92
+ const pathname = new URL(request.url).pathname;
93
+
94
+ const explicitMatch = this.explicitStaticRouteMatcher?.match(request.url);
95
+
96
+ if (explicitMatch) {
97
+ const response = await this.explicitStaticRouteMatcher!.handleMatch(explicitMatch);
98
+ return this.maybeInjectHmrScript(response);
99
+ }
100
+
101
+ const fsMatch = !pathname.includes('.') && this.router.match(request.url);
102
+
103
+ const response = await (fsMatch
104
+ ? this.fileSystemResponseMatcher.handleMatch(fsMatch, request)
105
+ : this.handleNoMatch(request));
106
+
107
+ return this.maybeInjectHmrScript(response);
108
+ }
109
+
110
+ /**
111
+ * Injects HMR script into HTML responses in development mode.
112
+ *
113
+ * The script is inserted before the closing </html> tag to enable
114
+ * hot module reloading without full page refreshes.
115
+ *
116
+ * @param response - The HTTP response to potentially modify
117
+ * @returns Original response or modified response with HMR script
118
+ */
119
+ private async maybeInjectHmrScript(response: Response): Promise<Response> {
120
+ if (this.shouldInjectHmrScript() && this.isHtmlResponse(response)) {
121
+ const html = await response.text();
122
+
123
+ const hmrScript = `<script type="module">import '/_hmr_runtime.js';</script>`;
124
+
125
+ const updatedHtml = html.replace(/<\/html>/i, `${hmrScript}</html>`);
126
+
127
+ const headers = new Headers(response.headers);
128
+ headers.delete('Content-Length');
129
+
130
+ return new Response(updatedHtml, {
131
+ status: response.status,
132
+ statusText: response.statusText,
133
+ headers,
134
+ });
135
+ }
136
+
137
+ return response;
138
+ }
139
+
140
+ /**
141
+ * Handles requests that do not match any routes.
142
+ *
143
+ * This is the final fallback that attempts to serve static files from disk.
144
+ * If the requested path corresponds to an HTML file or no file is found,
145
+ * a custom 404 response is returned.
146
+ *
147
+ * @param request - The HTTP request to handle
148
+ * @returns Response from file system or error response
149
+ * @throws HttpError for standard HTTP errors
150
+ */
151
+ async handleNoMatch(request: Request): Promise<Response> {
152
+ try {
153
+ const pathname = new URL(request.url).pathname;
154
+ return await this.fileSystemResponseMatcher.handleNoMatch(pathname);
155
+ } catch (error) {
156
+ if (error instanceof HttpError) {
157
+ return error.toResponse();
158
+ }
159
+ if (error instanceof Error) {
160
+ this.hmrManager?.broadcast({ type: 'error', message: error.message });
161
+ appLogger.error('Error handling no match:', error);
162
+ }
163
+ return new Response('Internal Server Error', { status: 500 });
164
+ }
165
+ }
166
+ }
@@ -0,0 +1,38 @@
1
+ import type { EcoPagesAppConfig } from '../../internal-types';
2
+ import type { StaticRoute } from '../../public-types';
3
+ import type { RouteRendererFactory } from '../../route-renderer/route-renderer';
4
+ import type { FSRouter } from '../../router/fs-router';
5
+ import type { StaticSiteGenerator } from '../../static-site-generator/static-site-generator';
6
+ export interface StaticBuildOptions {
7
+ preview?: boolean;
8
+ }
9
+ export interface ServeOptions {
10
+ hostname?: string;
11
+ port?: number | string;
12
+ }
13
+ export interface ServerStaticBuilderParams {
14
+ appConfig: EcoPagesAppConfig;
15
+ staticSiteGenerator: StaticSiteGenerator;
16
+ serveOptions: ServeOptions;
17
+ }
18
+ /**
19
+ * Handles static site generation and previews.
20
+ */
21
+ export declare class ServerStaticBuilder {
22
+ private readonly appConfig;
23
+ private readonly staticSiteGenerator;
24
+ private readonly serveOptions;
25
+ constructor({ appConfig, staticSiteGenerator, serveOptions }: ServerStaticBuilderParams);
26
+ /**
27
+ * Generates a static build of the site for deployment.
28
+ * @param options.preview - If true, starts a preview server after build
29
+ * @param dependencies.router - The initialized router
30
+ * @param dependencies.routeRendererFactory - The route renderer factory
31
+ * @param dependencies.staticRoutes - Explicit static routes registered via app.static()
32
+ */
33
+ build(options: StaticBuildOptions | undefined, dependencies: {
34
+ router: FSRouter;
35
+ routeRendererFactory: RouteRendererFactory;
36
+ staticRoutes?: StaticRoute[];
37
+ }): Promise<void>;
38
+ }
@@ -0,0 +1,46 @@
1
+ import { StaticContentServer } from "../../dev/sc-server";
2
+ import { appLogger } from "../../global/app-logger";
3
+ class ServerStaticBuilder {
4
+ appConfig;
5
+ staticSiteGenerator;
6
+ serveOptions;
7
+ constructor({ appConfig, staticSiteGenerator, serveOptions }) {
8
+ this.appConfig = appConfig;
9
+ this.staticSiteGenerator = staticSiteGenerator;
10
+ this.serveOptions = serveOptions;
11
+ }
12
+ /**
13
+ * Generates a static build of the site for deployment.
14
+ * @param options.preview - If true, starts a preview server after build
15
+ * @param dependencies.router - The initialized router
16
+ * @param dependencies.routeRendererFactory - The route renderer factory
17
+ * @param dependencies.staticRoutes - Explicit static routes registered via app.static()
18
+ */
19
+ async build(options, dependencies) {
20
+ const { preview = false } = options ?? {};
21
+ const baseUrl = `http://${this.serveOptions.hostname || "localhost"}:${this.serveOptions.port || 3e3}`;
22
+ await this.staticSiteGenerator.run({
23
+ router: dependencies.router,
24
+ baseUrl,
25
+ routeRendererFactory: dependencies.routeRendererFactory,
26
+ staticRoutes: dependencies.staticRoutes
27
+ });
28
+ if (!preview) {
29
+ appLogger.info("Build completed");
30
+ return;
31
+ }
32
+ const previewPort = this.serveOptions.port || 3e3;
33
+ const { server } = StaticContentServer.createServer({
34
+ appConfig: this.appConfig,
35
+ options: { port: Number(previewPort) }
36
+ });
37
+ if (server) {
38
+ appLogger.info(`Preview running at http://localhost:${server.port}`);
39
+ } else {
40
+ appLogger.error("Failed to start preview server");
41
+ }
42
+ }
43
+ }
44
+ export {
45
+ ServerStaticBuilder
46
+ };
@@ -0,0 +1,82 @@
1
+ import { StaticContentServer } from '../../dev/sc-server';
2
+ import { appLogger } from '../../global/app-logger';
3
+ import type { EcoPagesAppConfig } from '../../internal-types';
4
+ import type { StaticRoute } from '../../public-types';
5
+ import type { RouteRendererFactory } from '../../route-renderer/route-renderer';
6
+ import type { FSRouter } from '../../router/fs-router';
7
+ import type { StaticSiteGenerator } from '../../static-site-generator/static-site-generator';
8
+
9
+ export interface StaticBuildOptions {
10
+ preview?: boolean;
11
+ }
12
+
13
+ export interface ServeOptions {
14
+ hostname?: string;
15
+ port?: number | string;
16
+ }
17
+
18
+ export interface ServerStaticBuilderParams {
19
+ appConfig: EcoPagesAppConfig;
20
+ staticSiteGenerator: StaticSiteGenerator;
21
+ serveOptions: ServeOptions;
22
+ }
23
+
24
+ /**
25
+ * Handles static site generation and previews.
26
+ */
27
+ export class ServerStaticBuilder {
28
+ private readonly appConfig: EcoPagesAppConfig;
29
+ private readonly staticSiteGenerator: StaticSiteGenerator;
30
+ private readonly serveOptions: ServeOptions;
31
+
32
+ constructor({ appConfig, staticSiteGenerator, serveOptions }: ServerStaticBuilderParams) {
33
+ this.appConfig = appConfig;
34
+ this.staticSiteGenerator = staticSiteGenerator;
35
+ this.serveOptions = serveOptions;
36
+ }
37
+
38
+ /**
39
+ * Generates a static build of the site for deployment.
40
+ * @param options.preview - If true, starts a preview server after build
41
+ * @param dependencies.router - The initialized router
42
+ * @param dependencies.routeRendererFactory - The route renderer factory
43
+ * @param dependencies.staticRoutes - Explicit static routes registered via app.static()
44
+ */
45
+ async build(
46
+ options: StaticBuildOptions | undefined,
47
+ dependencies: {
48
+ router: FSRouter;
49
+ routeRendererFactory: RouteRendererFactory;
50
+ staticRoutes?: StaticRoute[];
51
+ },
52
+ ): Promise<void> {
53
+ const { preview = false } = options ?? {};
54
+
55
+ const baseUrl = `http://${this.serveOptions.hostname || 'localhost'}:${this.serveOptions.port || 3000}`;
56
+
57
+ await this.staticSiteGenerator.run({
58
+ router: dependencies.router,
59
+ baseUrl,
60
+ routeRendererFactory: dependencies.routeRendererFactory,
61
+ staticRoutes: dependencies.staticRoutes,
62
+ });
63
+
64
+ if (!preview) {
65
+ appLogger.info('Build completed');
66
+ return;
67
+ }
68
+
69
+ const previewPort = this.serveOptions.port || 3000;
70
+
71
+ const { server } = StaticContentServer.createServer({
72
+ appConfig: this.appConfig,
73
+ options: { port: Number(previewPort) },
74
+ });
75
+
76
+ if (server) {
77
+ appLogger.info(`Preview running at http://localhost:${server.port}`);
78
+ } else {
79
+ appLogger.error('Failed to start preview server');
80
+ }
81
+ }
82
+ }
@@ -0,0 +1,74 @@
1
+ import type { EcoBuildPlugin } from './build-types.js';
2
+ export { EsbuildBuildAdapter } from './esbuild-build-adapter.js';
3
+ export interface BuildLog {
4
+ message: string;
5
+ }
6
+ export interface BuildOutput {
7
+ path: string;
8
+ }
9
+ /**
10
+ * Dependency graph metadata produced by a build backend.
11
+ *
12
+ * @remarks
13
+ * This structure is runtime-neutral at the type level, but current population
14
+ * is Node/esbuild-only. Bun-backed builds may omit this metadata.
15
+ */
16
+ export interface BuildDependencyGraph {
17
+ /**
18
+ * Normalized absolute entrypoint path mapped to all normalized absolute
19
+ * source inputs that contributed to that entrypoint output.
20
+ */
21
+ entrypoints: Record<string, string[]>;
22
+ }
23
+ export interface BuildResult {
24
+ success: boolean;
25
+ logs: BuildLog[];
26
+ outputs: BuildOutput[];
27
+ /**
28
+ * Optional build dependency metadata for selective invalidation.
29
+ *
30
+ * @remarks
31
+ * This is currently filled by the Node/esbuild adapter. Other runtimes should
32
+ * treat missing graph data as a valid state and fall back deterministically.
33
+ */
34
+ dependencyGraph?: BuildDependencyGraph;
35
+ }
36
+ export interface BuildOptions {
37
+ entrypoints: string[];
38
+ outdir?: string;
39
+ naming?: string;
40
+ minify?: boolean;
41
+ treeshaking?: boolean;
42
+ target?: string;
43
+ format?: string;
44
+ sourcemap?: string;
45
+ splitting?: boolean;
46
+ root?: string;
47
+ bundle?: boolean;
48
+ externalPackages?: boolean;
49
+ external?: string[];
50
+ plugins?: EcoBuildPlugin[];
51
+ [key: string]: unknown;
52
+ }
53
+ export type BuildTranspileProfile = 'browser-script' | 'hmr-runtime' | 'hmr-entrypoint';
54
+ export interface BuildTranspileOptions {
55
+ target: string;
56
+ format: string;
57
+ sourcemap: string;
58
+ }
59
+ export interface BuildAdapter {
60
+ build(options: BuildOptions): Promise<BuildResult>;
61
+ resolve(importPath: string, rootDir: string): string;
62
+ registerPlugin(plugin: EcoBuildPlugin): void;
63
+ getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions;
64
+ }
65
+ /**
66
+ * @deprecated Use EsbuildBuildAdapter instead. Bun native build is missing some dependency graph features.
67
+ */
68
+ export declare class BunBuildAdapter implements BuildAdapter {
69
+ build(options: BuildOptions): Promise<BuildResult>;
70
+ resolve(importPath: string, rootDir: string): string;
71
+ registerPlugin(plugin: EcoBuildPlugin): void;
72
+ getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions;
73
+ }
74
+ export declare const defaultBuildAdapter: BuildAdapter;
@@ -0,0 +1,54 @@
1
+ import { EsbuildBuildAdapter } from "./esbuild-build-adapter.js";
2
+ import { getRequiredBunRuntime } from "../utils/runtime.js";
3
+ import { EsbuildBuildAdapter as EsbuildBuildAdapter2 } from "./esbuild-build-adapter.js";
4
+ function transpileProfileToOptions(profile) {
5
+ switch (profile) {
6
+ case "browser-script":
7
+ return {
8
+ target: "browser",
9
+ format: "esm",
10
+ sourcemap: "none"
11
+ };
12
+ case "hmr-runtime":
13
+ return {
14
+ target: "browser",
15
+ format: "esm",
16
+ sourcemap: "none"
17
+ };
18
+ case "hmr-entrypoint":
19
+ return {
20
+ target: "browser",
21
+ format: "esm",
22
+ sourcemap: "none"
23
+ };
24
+ }
25
+ }
26
+ class BunBuildAdapter {
27
+ async build(options) {
28
+ const bun = getRequiredBunRuntime();
29
+ const result = await bun.build({
30
+ ...options,
31
+ plugins: options.plugins
32
+ });
33
+ return {
34
+ success: result.success,
35
+ logs: result.logs.map((log) => ({ message: log.message })),
36
+ outputs: result.outputs.map((output) => ({ path: output.path }))
37
+ };
38
+ }
39
+ resolve(importPath, rootDir) {
40
+ return getRequiredBunRuntime().resolveSync(importPath, rootDir);
41
+ }
42
+ registerPlugin(plugin) {
43
+ getRequiredBunRuntime().plugin(plugin);
44
+ }
45
+ getTranspileOptions(profile) {
46
+ return transpileProfileToOptions(profile);
47
+ }
48
+ }
49
+ const defaultBuildAdapter = new EsbuildBuildAdapter();
50
+ export {
51
+ BunBuildAdapter,
52
+ EsbuildBuildAdapter2 as EsbuildBuildAdapter,
53
+ defaultBuildAdapter
54
+ };
@@ -0,0 +1,132 @@
1
+ import type { BunPlugin } from 'bun';
2
+ import type { EcoBuildPlugin } from './build-types.ts';
3
+ import { EsbuildBuildAdapter } from './esbuild-build-adapter.ts';
4
+ import { getRequiredBunRuntime } from '../utils/runtime.ts';
5
+
6
+ export { EsbuildBuildAdapter } from './esbuild-build-adapter.ts';
7
+
8
+ export interface BuildLog {
9
+ message: string;
10
+ }
11
+
12
+ export interface BuildOutput {
13
+ path: string;
14
+ }
15
+
16
+ /**
17
+ * Dependency graph metadata produced by a build backend.
18
+ *
19
+ * @remarks
20
+ * This structure is runtime-neutral at the type level, but current population
21
+ * is Node/esbuild-only. Bun-backed builds may omit this metadata.
22
+ */
23
+ export interface BuildDependencyGraph {
24
+ /**
25
+ * Normalized absolute entrypoint path mapped to all normalized absolute
26
+ * source inputs that contributed to that entrypoint output.
27
+ */
28
+ entrypoints: Record<string, string[]>;
29
+ }
30
+
31
+ export interface BuildResult {
32
+ success: boolean;
33
+ logs: BuildLog[];
34
+ outputs: BuildOutput[];
35
+ /**
36
+ * Optional build dependency metadata for selective invalidation.
37
+ *
38
+ * @remarks
39
+ * This is currently filled by the Node/esbuild adapter. Other runtimes should
40
+ * treat missing graph data as a valid state and fall back deterministically.
41
+ */
42
+ dependencyGraph?: BuildDependencyGraph;
43
+ }
44
+
45
+ export interface BuildOptions {
46
+ entrypoints: string[];
47
+ outdir?: string;
48
+ naming?: string;
49
+ minify?: boolean;
50
+ treeshaking?: boolean;
51
+ target?: string;
52
+ format?: string;
53
+ sourcemap?: string;
54
+ splitting?: boolean;
55
+ root?: string;
56
+ bundle?: boolean;
57
+ externalPackages?: boolean;
58
+ external?: string[];
59
+ plugins?: EcoBuildPlugin[];
60
+ [key: string]: unknown;
61
+ }
62
+
63
+ export type BuildTranspileProfile = 'browser-script' | 'hmr-runtime' | 'hmr-entrypoint';
64
+
65
+ export interface BuildTranspileOptions {
66
+ target: string;
67
+ format: string;
68
+ sourcemap: string;
69
+ }
70
+
71
+ export interface BuildAdapter {
72
+ build(options: BuildOptions): Promise<BuildResult>;
73
+ resolve(importPath: string, rootDir: string): string;
74
+ registerPlugin(plugin: EcoBuildPlugin): void;
75
+ getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions;
76
+ }
77
+
78
+ function transpileProfileToOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
79
+ switch (profile) {
80
+ case 'browser-script':
81
+ return {
82
+ target: 'browser',
83
+ format: 'esm',
84
+ sourcemap: 'none',
85
+ };
86
+ case 'hmr-runtime':
87
+ return {
88
+ target: 'browser',
89
+ format: 'esm',
90
+ sourcemap: 'none',
91
+ };
92
+ case 'hmr-entrypoint':
93
+ return {
94
+ target: 'browser',
95
+ format: 'esm',
96
+ sourcemap: 'none',
97
+ };
98
+ }
99
+ }
100
+
101
+ /**
102
+ * @deprecated Use EsbuildBuildAdapter instead. Bun native build is missing some dependency graph features.
103
+ */
104
+ export class BunBuildAdapter implements BuildAdapter {
105
+ async build(options: BuildOptions): Promise<BuildResult> {
106
+ const bun = getRequiredBunRuntime();
107
+ const result = await bun.build({
108
+ ...options,
109
+ plugins: options.plugins as BunPlugin[] | undefined,
110
+ } as Bun.BuildConfig);
111
+
112
+ return {
113
+ success: result.success,
114
+ logs: result.logs.map((log) => ({ message: log.message })),
115
+ outputs: result.outputs.map((output) => ({ path: output.path })),
116
+ };
117
+ }
118
+
119
+ resolve(importPath: string, rootDir: string): string {
120
+ return getRequiredBunRuntime().resolveSync(importPath, rootDir);
121
+ }
122
+
123
+ registerPlugin(plugin: EcoBuildPlugin): void {
124
+ getRequiredBunRuntime().plugin(plugin as BunPlugin);
125
+ }
126
+
127
+ getTranspileOptions(profile: BuildTranspileProfile): BuildTranspileOptions {
128
+ return transpileProfileToOptions(profile);
129
+ }
130
+ }
131
+
132
+ export const defaultBuildAdapter: BuildAdapter = new EsbuildBuildAdapter();
@@ -0,0 +1,57 @@
1
+ /**
2
+ * Arguments passed to a build plugin `onResolve` callback.
3
+ */
4
+ export type EcoBuildOnResolveArgs = {
5
+ path: string;
6
+ importer?: string;
7
+ namespace?: string;
8
+ };
9
+ /**
10
+ * Result returned by a build plugin `onResolve` callback.
11
+ */
12
+ export type EcoBuildOnResolveResult = {
13
+ path?: string;
14
+ namespace?: string;
15
+ external?: boolean;
16
+ };
17
+ /**
18
+ * Arguments passed to a build plugin `onLoad` callback.
19
+ */
20
+ export type EcoBuildOnLoadArgs = {
21
+ path: string;
22
+ namespace?: string;
23
+ };
24
+ /**
25
+ * Loader kinds supported across Bun/esbuild bridge points.
26
+ */
27
+ export type EcoBuildLoader = 'base64' | 'binary' | 'copy' | 'css' | 'dataurl' | 'empty' | 'file' | 'global-css' | 'js' | 'json' | 'jsx' | 'local-css' | 'object' | 'text' | 'ts' | 'tsx';
28
+ /**
29
+ * Result returned by a build plugin `onLoad` callback.
30
+ */
31
+ export type EcoBuildOnLoadResult = {
32
+ contents?: string | Uint8Array;
33
+ loader?: EcoBuildLoader;
34
+ exports?: Record<string, unknown>;
35
+ resolveDir?: string;
36
+ };
37
+ /**
38
+ * Shared plugin builder contract used by Ecopages across Bun and esbuild.
39
+ */
40
+ export interface EcoBuildPluginBuilder {
41
+ onResolve(options: {
42
+ filter: RegExp;
43
+ namespace?: string;
44
+ }, callback: (args: EcoBuildOnResolveArgs) => EcoBuildOnResolveResult | undefined | Promise<EcoBuildOnResolveResult | undefined>): void;
45
+ onLoad(options: {
46
+ filter: RegExp;
47
+ namespace?: string;
48
+ }, callback: (args: EcoBuildOnLoadArgs) => EcoBuildOnLoadResult | undefined | Promise<EcoBuildOnLoadResult | undefined>): void;
49
+ module(specifier: string, callback: () => EcoBuildOnLoadResult | Promise<EcoBuildOnLoadResult>): void;
50
+ }
51
+ /**
52
+ * Runtime-agnostic build plugin contract consumed by Ecopages processors/loaders.
53
+ */
54
+ export type EcoBuildPlugin = {
55
+ name: string;
56
+ setup: (build: EcoBuildPluginBuilder) => void | Promise<void>;
57
+ };
File without changes