@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,442 @@
1
+ import path from 'node:path';
2
+ import { AbstractServerAdapter } from '../abstract/server-adapter.ts';
3
+ import type { ServerAdapterOptions, ServerAdapterResult } from '../abstract/server-adapter.ts';
4
+ import { RouteRendererFactory } from '../../route-renderer/route-renderer.ts';
5
+ import { FSRouter } from '../../router/fs-router.ts';
6
+ import { FSRouterScanner } from '../../router/fs-router-scanner.ts';
7
+ import { MemoryCacheStore } from '../../services/cache/memory-cache-store.ts';
8
+ import { PageCacheService } from '../../services/cache/page-cache-service.ts';
9
+ import { SchemaValidationService } from '../../services/schema-validation-service.ts';
10
+ import { StaticSiteGenerator } from '../../static-site-generator/static-site-generator.ts';
11
+ import { ServerStaticBuilder } from './server-static-builder.ts';
12
+ import { ExplicitStaticRouteMatcher } from './explicit-static-route-matcher.ts';
13
+ import { FileSystemServerResponseFactory } from './fs-server-response-factory.ts';
14
+ import { FileSystemResponseMatcher } from './fs-server-response-matcher.ts';
15
+ import { ServerRouteHandler } from './server-route-handler.ts';
16
+ import { createRenderContext } from './render-context.ts';
17
+ import { createRequire } from '../../utils/locals-utils.ts';
18
+ import { HttpError } from '../../errors/http-error.ts';
19
+ import { ApiResponseBuilder } from './api-response.ts';
20
+ import { appLogger } from '../../global/app-logger.ts';
21
+ import type {
22
+ ApiHandler,
23
+ ApiHandlerContext,
24
+ CacheInvalidator,
25
+ ErrorHandler,
26
+ RenderContext,
27
+ StaticRoute,
28
+ } from '../../public-types.ts';
29
+
30
+ export abstract class SharedServerAdapter<
31
+ TOptions extends ServerAdapterOptions,
32
+ TResult extends ServerAdapterResult,
33
+ > extends AbstractServerAdapter<TOptions, TResult> {
34
+ protected router!: FSRouter;
35
+ protected fileSystemResponseMatcher!: FileSystemResponseMatcher;
36
+ protected routeRendererFactory!: RouteRendererFactory;
37
+ protected routeHandler!: ServerRouteHandler;
38
+ protected staticSiteGenerator!: StaticSiteGenerator;
39
+ protected staticBuilder!: ServerStaticBuilder;
40
+ protected readonly schemaValidator = new SchemaValidationService();
41
+
42
+ /**
43
+ * Scans the filesystem and dynamically constructs the universal router map.
44
+ *
45
+ * This process runs identically across both Bun and Node wrappers. It analyzes the configured pages
46
+ * directory, building a map of all available UI routes and API endpoints.
47
+ * The resulting `FSRouter` instance becomes the central nervous system for mapping WinterCG incoming
48
+ * Web Requests (`Request`) to their corresponding internal execution paths.
49
+ */
50
+ protected async initSharedRouter(): Promise<void> {
51
+ const scanner = new FSRouterScanner({
52
+ dir: path.join(this.appConfig.rootDir, this.appConfig.srcDir, this.appConfig.pagesDir),
53
+ appConfig: this.appConfig,
54
+ origin: this.runtimeOrigin,
55
+ templatesExt: this.appConfig.templatesExt,
56
+ options: {
57
+ buildMode: !this.options?.watch,
58
+ },
59
+ });
60
+
61
+ this.router = new FSRouter({
62
+ origin: this.runtimeOrigin,
63
+ assetPrefix: path.join(this.appConfig.rootDir, this.appConfig.distDir),
64
+ scanner,
65
+ });
66
+
67
+ await this.router.init();
68
+ }
69
+
70
+ /**
71
+ * Sets up the unified rendering pipeline and response matching chain.
72
+ *
73
+ * It bridges several sub-systems together so that when an incoming request is received, the adapter knows:
74
+ * 1. How to render React/Lit pages via `RouteRendererFactory`
75
+ * 2. How to match logical routes to physical filesystem artifacts via `FileSystemResponseMatcher`
76
+ * 3. Whether to serve the response from the embedded `PageCacheService` or generate it fresh on the fly.
77
+ *
78
+ * Because `HmrManager` implementations rely heavily on runtime-specific WebSocket APIs (e.g. Bun.serve Websockets vs Node WS),
79
+ * we leave it untyped (`any`) here at the common denominator core.
80
+ *
81
+ * @param staticRoutes - A map of explicitly served static assets.
82
+ * @param hmrManager - The runtime-specific Hot Module Replacement orchestrator (if watching).
83
+ */
84
+ protected configureSharedResponseHandlers(staticRoutes: StaticRoute[], hmrManager?: any): void {
85
+ this.routeRendererFactory = new RouteRendererFactory({
86
+ appConfig: this.appConfig,
87
+ runtimeOrigin: this.runtimeOrigin,
88
+ });
89
+
90
+ const fileSystemResponseFactory = new FileSystemServerResponseFactory({
91
+ appConfig: this.appConfig,
92
+ routeRendererFactory: this.routeRendererFactory,
93
+ options: {
94
+ watchMode: !!this.options?.watch,
95
+ },
96
+ });
97
+
98
+ const cacheConfig = this.appConfig.cache;
99
+ const isCacheEnabled = cacheConfig?.enabled ?? !this.options?.watch;
100
+ let cacheService: PageCacheService | null = null;
101
+
102
+ if (isCacheEnabled) {
103
+ const store =
104
+ cacheConfig?.store === 'memory' || !cacheConfig?.store
105
+ ? new MemoryCacheStore({ maxEntries: cacheConfig?.maxEntries })
106
+ : cacheConfig.store;
107
+ cacheService = new PageCacheService({ store, enabled: true });
108
+ }
109
+
110
+ this.fileSystemResponseMatcher = new FileSystemResponseMatcher({
111
+ router: this.router,
112
+ routeRendererFactory: this.routeRendererFactory,
113
+ fileSystemResponseFactory,
114
+ cacheService,
115
+ defaultCacheStrategy: cacheConfig?.defaultStrategy ?? 'static',
116
+ });
117
+
118
+ const explicitStaticRouteMatcher =
119
+ staticRoutes.length > 0
120
+ ? new ExplicitStaticRouteMatcher({
121
+ appConfig: this.appConfig,
122
+ routeRendererFactory: this.routeRendererFactory,
123
+ staticRoutes,
124
+ })
125
+ : undefined;
126
+
127
+ this.routeHandler = new ServerRouteHandler({
128
+ router: this.router,
129
+ fileSystemResponseMatcher: this.fileSystemResponseMatcher,
130
+ explicitStaticRouteMatcher,
131
+ watch: !!this.options?.watch,
132
+ hmrManager,
133
+ });
134
+ }
135
+
136
+ protected getCacheService(): CacheInvalidator | null {
137
+ return this.fileSystemResponseMatcher?.getCacheService() ?? null;
138
+ }
139
+
140
+ protected getRenderContext(): RenderContext {
141
+ return createRenderContext({ integrations: this.appConfig.integrations });
142
+ }
143
+
144
+ /**
145
+ * Executes an Application Programming Interface (API) handler in an environment-agnostic manner.
146
+ *
147
+ * API routes in Ecopages are universally written using standard WinterCG `Request` and `Response` objects.
148
+ * This execution pipeline takes the raw `Request`, extracts its dynamic segments, runs our high-speed JSON schema
149
+ * validator against the body/query/headers, and triggers the developer's middleware chain sequentially.
150
+ *
151
+ * If the execution throws an error, it is gracefully caught, logged, and mutated into a standardized Http error payload,
152
+ * ensuring the consuming client receives a parsable response even upon internal catastrophic failure.
153
+ *
154
+ * @param request - The incoming Web standard `Request`.
155
+ * @param params - The extracted dynamic URL parameters (e.g., `{ id: '123' }`).
156
+ * @param routeConfig - The user-defined API handler object containing their business logic (`handler`) and `middleware`.
157
+ * @param serverInstance - Untyped reference to the underlying native server instance (BunServer/NodeServer) for potential escape hatches.
158
+ * @param errorHandler - Optional global error trap defined in project configuration.
159
+ * @returns The resulting Web standard `Response` constructed by the user's handler.
160
+ */
161
+ protected async executeApiHandler(
162
+ request: Request,
163
+ params: Record<string, string | string[]>,
164
+ routeConfig: ApiHandler,
165
+ serverInstance: any,
166
+ errorHandler?: ErrorHandler,
167
+ ): Promise<Response> {
168
+ let context: ApiHandlerContext<Request, any> | undefined;
169
+
170
+ try {
171
+ const middleware = routeConfig.middleware || [];
172
+ const schema = routeConfig.schema;
173
+ const locals: Record<string, unknown> = {};
174
+
175
+ const normalizedParams = Object.fromEntries(
176
+ Object.entries(params).map(([key, value]) => [key, Array.isArray(value) ? value.join('/') : value]),
177
+ );
178
+
179
+ context = {
180
+ request,
181
+ params: normalizedParams,
182
+ response: new ApiResponseBuilder(),
183
+ server: serverInstance,
184
+ locals,
185
+ require: createRequire((): Record<string, unknown> => locals),
186
+ services: {
187
+ cache: this.getCacheService(),
188
+ },
189
+ ...this.getRenderContext(),
190
+ };
191
+
192
+ if (schema) {
193
+ const url = new URL(request.url);
194
+ const queryParams = Object.fromEntries(url.searchParams);
195
+ const headers = Object.fromEntries(request.headers);
196
+
197
+ let body: unknown;
198
+ if (schema.body) {
199
+ try {
200
+ const contentType = request.headers.get('Content-Type') || '';
201
+ if (contentType.includes('application/json')) body = await request.clone().json();
202
+ else if (contentType.includes('text/plain')) body = await request.clone().text();
203
+ } catch {
204
+ return context.response.status(400).json({ error: 'Invalid request body' });
205
+ }
206
+ }
207
+
208
+ const validationResult = await this.schemaValidator.validateRequest(
209
+ { body, query: queryParams, headers, params: normalizedParams },
210
+ schema,
211
+ );
212
+
213
+ if (!validationResult.success) {
214
+ return context.response.status(400).json({
215
+ error: 'Validation failed',
216
+ issues: validationResult.errors,
217
+ });
218
+ }
219
+
220
+ const validated = validationResult.data!;
221
+ if (validated.body !== undefined) context.body = validated.body;
222
+ if (validated.query !== undefined) context.query = validated.query;
223
+ if (validated.headers !== undefined) context.headers = validated.headers;
224
+ if (validated.params !== undefined) context.params = validated.params as Record<string, string>;
225
+ }
226
+
227
+ if (middleware.length === 0) {
228
+ return await routeConfig.handler(context);
229
+ }
230
+
231
+ let index = 0;
232
+ const executeNext = async (): Promise<Response> => {
233
+ if (index < middleware.length) {
234
+ const currentMiddleware = middleware[index++];
235
+ return await currentMiddleware(context!, executeNext);
236
+ }
237
+ return await routeConfig.handler(context!);
238
+ };
239
+
240
+ return await executeNext();
241
+ } catch (error) {
242
+ if (error instanceof Response) return error;
243
+
244
+ if (errorHandler) {
245
+ try {
246
+ if (!context) {
247
+ const locals: Record<string, unknown> = {};
248
+ context = {
249
+ request,
250
+ params: Object.fromEntries(
251
+ Object.entries(params).map(([key, value]) => [
252
+ key,
253
+ Array.isArray(value) ? value.join('/') : value,
254
+ ]),
255
+ ),
256
+ response: new ApiResponseBuilder(),
257
+ server: serverInstance,
258
+ locals,
259
+ require: createRequire((): Record<string, unknown> => locals),
260
+ services: { cache: this.getCacheService() },
261
+ ...this.getRenderContext(),
262
+ };
263
+ }
264
+ return await errorHandler(error, context);
265
+ } catch (handlerError) {
266
+ appLogger.error(`[ecopages] Error in custom error handler: ${handlerError}`);
267
+ }
268
+ }
269
+
270
+ if (error instanceof HttpError) return error.toResponse();
271
+ appLogger.error(`[ecopages] Error handling API request: ${error}`);
272
+ return new Response('Internal Server Error', { status: 500 });
273
+ }
274
+ }
275
+
276
+ private normalizePath(pathname: string): string {
277
+ if (pathname.length > 1 && pathname.endsWith('/')) {
278
+ return pathname.slice(0, -1);
279
+ }
280
+
281
+ return pathname;
282
+ }
283
+
284
+ private matchApiPath(pattern: string, pathname: string): Record<string, string | string[]> | null {
285
+ const normalizedPattern = this.normalizePath(pattern);
286
+ const normalizedPathname = this.normalizePath(pathname);
287
+
288
+ const patternSegments = normalizedPattern.split('/').filter(Boolean);
289
+ const pathSegments = normalizedPathname.split('/').filter(Boolean);
290
+ const params: Record<string, string | string[]> = {};
291
+
292
+ let patternIndex = 0;
293
+ let pathIndex = 0;
294
+
295
+ while (patternIndex < patternSegments.length && pathIndex < pathSegments.length) {
296
+ const patternSegment = patternSegments[patternIndex];
297
+ const pathSegment = pathSegments[pathIndex];
298
+
299
+ if (patternSegment === '*') {
300
+ return params;
301
+ }
302
+
303
+ if (patternSegment.startsWith('[...') && patternSegment.endsWith(']')) {
304
+ const paramName = patternSegment.slice(4, -1);
305
+ params[paramName] = pathSegments.slice(pathIndex);
306
+ return params;
307
+ }
308
+
309
+ if (patternSegment.startsWith(':')) {
310
+ params[patternSegment.slice(1)] = pathSegment;
311
+ patternIndex++;
312
+ pathIndex++;
313
+ continue;
314
+ }
315
+
316
+ if (patternSegment.startsWith('[') && patternSegment.endsWith(']')) {
317
+ params[patternSegment.slice(1, -1)] = pathSegment;
318
+ patternIndex++;
319
+ pathIndex++;
320
+ continue;
321
+ }
322
+
323
+ if (patternSegment !== pathSegment) {
324
+ return null;
325
+ }
326
+
327
+ patternIndex++;
328
+ pathIndex++;
329
+ }
330
+
331
+ if (patternIndex < patternSegments.length) {
332
+ const remaining = patternSegments.slice(patternIndex);
333
+ const catchAll = remaining[0];
334
+
335
+ if (
336
+ remaining.length === 1 &&
337
+ (catchAll === '*' || (catchAll.startsWith('[...') && catchAll.endsWith(']')))
338
+ ) {
339
+ if (catchAll.startsWith('[...')) {
340
+ const paramName = catchAll.slice(4, -1);
341
+ params[paramName] = [];
342
+ }
343
+ return params;
344
+ }
345
+
346
+ return null;
347
+ }
348
+
349
+ if (pathIndex < pathSegments.length) {
350
+ return null;
351
+ }
352
+
353
+ return params;
354
+ }
355
+
356
+ private getApiPathScore(pattern: string): number {
357
+ const segments = this.normalizePath(pattern).split('/').filter(Boolean);
358
+ let score = 0;
359
+ for (const segment of segments) {
360
+ if (segment === '*' || (segment.startsWith('[...') && segment.endsWith(']'))) {
361
+ score += 10;
362
+ } else if (segment.startsWith(':') || (segment.startsWith('[') && segment.endsWith(']'))) {
363
+ score += 50;
364
+ } else {
365
+ score += 100;
366
+ }
367
+ }
368
+ return score;
369
+ }
370
+
371
+ protected matchApiHandler(
372
+ request: Request,
373
+ apiHandlers: ApiHandler[],
374
+ ): { routeConfig: ApiHandler; params: Record<string, string | string[]> } | null {
375
+ const pathname = new URL(request.url).pathname;
376
+ const method = request.method.toUpperCase();
377
+
378
+ const sortedHandlers = [...apiHandlers].sort((a, b) => {
379
+ return this.getApiPathScore(b.path) - this.getApiPathScore(a.path);
380
+ });
381
+
382
+ for (const routeConfig of sortedHandlers) {
383
+ const routeMethod = (routeConfig.method || 'GET').toUpperCase();
384
+ if (routeMethod !== method) {
385
+ continue;
386
+ }
387
+
388
+ const params = this.matchApiPath(routeConfig.path, pathname);
389
+ if (params) {
390
+ return { routeConfig, params };
391
+ }
392
+ }
393
+
394
+ return null;
395
+ }
396
+
397
+ /**
398
+ * Universally processes an incoming WinterCG Web standard Request.
399
+ *
400
+ * 1. Resolves static Hot Module Replacement runtime blobs if development.
401
+ * 2. Checks if the incoming request matches any parsed API route schemas.
402
+ * - Routes through `executeApiHandler` which performs strict validation.
403
+ * 3. Falls through to standard `ServerRouteHandler` for React/Lit filesystem pages.
404
+ *
405
+ * Both Bun and Node bindings fall back to this exact function once they have mapped their
406
+ * native HTTP objects into Web Standard Requests.
407
+ */
408
+ public async handleSharedRequest(
409
+ request: Request,
410
+ context: {
411
+ apiHandlers: ApiHandler[];
412
+ errorHandler?: ErrorHandler;
413
+ serverInstance?: any;
414
+ hmrManager?: any;
415
+ },
416
+ ): Promise<Response> {
417
+ const url = new URL(request.url);
418
+
419
+ if (url.pathname === '/_hmr_runtime.js' && context.hmrManager) {
420
+ const runtimePath = context.hmrManager.getRuntimePath();
421
+ const fileSystem = (await import('@ecopages/file-system')).fileSystem;
422
+ if (fileSystem.exists(runtimePath)) {
423
+ return new Response(fileSystem.readFileAsBuffer(runtimePath) as BodyInit, {
424
+ headers: { 'Content-Type': 'application/javascript' },
425
+ });
426
+ }
427
+ }
428
+
429
+ const apiMatch = this.matchApiHandler(request, context.apiHandlers);
430
+ if (apiMatch) {
431
+ return this.executeApiHandler(
432
+ request,
433
+ apiMatch.params,
434
+ apiMatch.routeConfig,
435
+ context.serverInstance,
436
+ context.errorHandler,
437
+ );
438
+ }
439
+
440
+ return this.routeHandler.handleResponse(request);
441
+ }
442
+ }
@@ -0,0 +1,89 @@
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
+ /**
6
+ * Configuration parameters for ServerRouteHandler.
7
+ */
8
+ export interface ServerRouteHandlerParams {
9
+ /** File system router for matching request URLs to route handlers. */
10
+ router: FSRouter;
11
+ /** Matcher for handling file system route responses. */
12
+ fileSystemResponseMatcher: FileSystemResponseMatcher;
13
+ /** Optional matcher for explicit static routes like processed images or sitemaps. */
14
+ explicitStaticRouteMatcher?: ExplicitStaticRouteMatcher;
15
+ /** Enable watch mode for development. */
16
+ watch?: boolean;
17
+ /** HMR manager for broadcasting hot module reload events in development. */
18
+ hmrManager?: IHmrManager;
19
+ }
20
+ /**
21
+ * Handles HTTP requests and routing for the server.
22
+ *
23
+ * This class manages the request routing flow with a priority-based approach:
24
+ * 1. Explicit static routes (highest priority - intentional mappings like /sitemap.xml)
25
+ * 2. File-based routes without extensions (application routes)
26
+ * 3. Static file fallback (lowest priority - disk serving)
27
+ *
28
+ * In development mode, it also injects HMR scripts into HTML responses.
29
+ */
30
+ export declare class ServerRouteHandler {
31
+ private readonly router;
32
+ private readonly fileSystemResponseMatcher;
33
+ private readonly explicitStaticRouteMatcher?;
34
+ private readonly watch;
35
+ private readonly hmrManager?;
36
+ /**
37
+ * Creates a new ServerRouteHandler instance.
38
+ *
39
+ * @param params - Configuration parameters
40
+ */
41
+ constructor({ router, fileSystemResponseMatcher, explicitStaticRouteMatcher, watch, hmrManager, }: ServerRouteHandlerParams);
42
+ /**
43
+ * Determines if HMR script should be injected.
44
+ *
45
+ * @returns true if in watch mode and HMR manager is enabled
46
+ */
47
+ shouldInjectHmrScript(): boolean;
48
+ /**
49
+ * Checks if a response contains HTML content.
50
+ *
51
+ * @param response - The HTTP response to check
52
+ * @returns true if Content-Type header starts with 'text/html'
53
+ */
54
+ isHtmlResponse(response: Response): boolean;
55
+ /**
56
+ * Handles HTTP requests from the router adapter.
57
+ *
58
+ * Priority-based routing flow:
59
+ * 1. Check explicit static routes first (e.g., /sitemap.xml, /image.webp from plugins)
60
+ * 2. Match file-based routes without extensions (application routes)
61
+ * 3. Fall back to static file serving from disk
62
+ *
63
+ * @param request - The incoming HTTP request
64
+ * @returns HTTP response, potentially with injected HMR script in dev mode
65
+ */
66
+ handleResponse(request: Request): Promise<Response>;
67
+ /**
68
+ * Injects HMR script into HTML responses in development mode.
69
+ *
70
+ * The script is inserted before the closing </html> tag to enable
71
+ * hot module reloading without full page refreshes.
72
+ *
73
+ * @param response - The HTTP response to potentially modify
74
+ * @returns Original response or modified response with HMR script
75
+ */
76
+ private maybeInjectHmrScript;
77
+ /**
78
+ * Handles requests that do not match any routes.
79
+ *
80
+ * This is the final fallback that attempts to serve static files from disk.
81
+ * If the requested path corresponds to an HTML file or no file is found,
82
+ * a custom 404 response is returned.
83
+ *
84
+ * @param request - The HTTP request to handle
85
+ * @returns Response from file system or error response
86
+ * @throws HttpError for standard HTTP errors
87
+ */
88
+ handleNoMatch(request: Request): Promise<Response>;
89
+ }
@@ -0,0 +1,120 @@
1
+ import { appLogger } from "../../global/app-logger";
2
+ import { HttpError } from "../../errors/http-error";
3
+ class ServerRouteHandler {
4
+ router;
5
+ fileSystemResponseMatcher;
6
+ explicitStaticRouteMatcher;
7
+ watch;
8
+ hmrManager;
9
+ /**
10
+ * Creates a new ServerRouteHandler instance.
11
+ *
12
+ * @param params - Configuration parameters
13
+ */
14
+ constructor({
15
+ router,
16
+ fileSystemResponseMatcher,
17
+ explicitStaticRouteMatcher,
18
+ watch = false,
19
+ hmrManager
20
+ }) {
21
+ this.router = router;
22
+ this.fileSystemResponseMatcher = fileSystemResponseMatcher;
23
+ this.explicitStaticRouteMatcher = explicitStaticRouteMatcher;
24
+ this.watch = watch;
25
+ this.hmrManager = hmrManager;
26
+ }
27
+ /**
28
+ * Determines if HMR script should be injected.
29
+ *
30
+ * @returns true if in watch mode and HMR manager is enabled
31
+ */
32
+ shouldInjectHmrScript() {
33
+ return this.watch && this.hmrManager?.isEnabled() === true;
34
+ }
35
+ /**
36
+ * Checks if a response contains HTML content.
37
+ *
38
+ * @param response - The HTTP response to check
39
+ * @returns true if Content-Type header starts with 'text/html'
40
+ */
41
+ isHtmlResponse(response) {
42
+ const contentType = response.headers.get("Content-Type");
43
+ return contentType !== null && contentType.startsWith("text/html");
44
+ }
45
+ /**
46
+ * Handles HTTP requests from the router adapter.
47
+ *
48
+ * Priority-based routing flow:
49
+ * 1. Check explicit static routes first (e.g., /sitemap.xml, /image.webp from plugins)
50
+ * 2. Match file-based routes without extensions (application routes)
51
+ * 3. Fall back to static file serving from disk
52
+ *
53
+ * @param request - The incoming HTTP request
54
+ * @returns HTTP response, potentially with injected HMR script in dev mode
55
+ */
56
+ async handleResponse(request) {
57
+ const pathname = new URL(request.url).pathname;
58
+ const explicitMatch = this.explicitStaticRouteMatcher?.match(request.url);
59
+ if (explicitMatch) {
60
+ const response2 = await this.explicitStaticRouteMatcher.handleMatch(explicitMatch);
61
+ return this.maybeInjectHmrScript(response2);
62
+ }
63
+ const fsMatch = !pathname.includes(".") && this.router.match(request.url);
64
+ const response = await (fsMatch ? this.fileSystemResponseMatcher.handleMatch(fsMatch, request) : this.handleNoMatch(request));
65
+ return this.maybeInjectHmrScript(response);
66
+ }
67
+ /**
68
+ * Injects HMR script into HTML responses in development mode.
69
+ *
70
+ * The script is inserted before the closing </html> tag to enable
71
+ * hot module reloading without full page refreshes.
72
+ *
73
+ * @param response - The HTTP response to potentially modify
74
+ * @returns Original response or modified response with HMR script
75
+ */
76
+ async maybeInjectHmrScript(response) {
77
+ if (this.shouldInjectHmrScript() && this.isHtmlResponse(response)) {
78
+ const html = await response.text();
79
+ const hmrScript = `<script type="module">import '/_hmr_runtime.js';<\/script>`;
80
+ const updatedHtml = html.replace(/<\/html>/i, `${hmrScript}</html>`);
81
+ const headers = new Headers(response.headers);
82
+ headers.delete("Content-Length");
83
+ return new Response(updatedHtml, {
84
+ status: response.status,
85
+ statusText: response.statusText,
86
+ headers
87
+ });
88
+ }
89
+ return response;
90
+ }
91
+ /**
92
+ * Handles requests that do not match any routes.
93
+ *
94
+ * This is the final fallback that attempts to serve static files from disk.
95
+ * If the requested path corresponds to an HTML file or no file is found,
96
+ * a custom 404 response is returned.
97
+ *
98
+ * @param request - The HTTP request to handle
99
+ * @returns Response from file system or error response
100
+ * @throws HttpError for standard HTTP errors
101
+ */
102
+ async handleNoMatch(request) {
103
+ try {
104
+ const pathname = new URL(request.url).pathname;
105
+ return await this.fileSystemResponseMatcher.handleNoMatch(pathname);
106
+ } catch (error) {
107
+ if (error instanceof HttpError) {
108
+ return error.toResponse();
109
+ }
110
+ if (error instanceof Error) {
111
+ this.hmrManager?.broadcast({ type: "error", message: error.message });
112
+ appLogger.error("Error handling no match:", error);
113
+ }
114
+ return new Response("Internal Server Error", { status: 500 });
115
+ }
116
+ }
117
+ }
118
+ export {
119
+ ServerRouteHandler
120
+ };