@ecopages/core 0.2.0-alpha.26 → 0.2.0-alpha.27

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 (550) hide show
  1. package/README.md +63 -7
  2. package/package.json +73 -249
  3. package/src/adapters/abstract/application-adapter.test.ts +172 -0
  4. package/src/adapters/abstract/application-adapter.ts +379 -0
  5. package/src/adapters/abstract/router-adapter.ts +30 -0
  6. package/src/adapters/abstract/server-adapter.ts +79 -0
  7. package/src/adapters/bun/client-bridge.ts +62 -0
  8. package/src/adapters/bun/create-app.ts +232 -0
  9. package/src/adapters/bun/hmr-manager.test.ts +265 -0
  10. package/src/adapters/bun/hmr-manager.ts +383 -0
  11. package/src/adapters/bun/index.ts +2 -0
  12. package/src/adapters/bun/server-adapter.ts +526 -0
  13. package/src/adapters/bun/server-lifecycle.ts +124 -0
  14. package/src/adapters/create-app.test.ts +10 -0
  15. package/src/adapters/create-app.ts +91 -0
  16. package/src/adapters/index.ts +2 -0
  17. package/src/adapters/node/create-app.test.ts +53 -0
  18. package/src/adapters/node/create-app.ts +183 -0
  19. package/src/adapters/node/node-client-bridge.test.ts +198 -0
  20. package/src/adapters/node/node-client-bridge.ts +79 -0
  21. package/src/adapters/node/node-hmr-manager.test.ts +320 -0
  22. package/src/adapters/node/node-hmr-manager.ts +355 -0
  23. package/src/adapters/node/server-adapter.ts +502 -0
  24. package/src/adapters/node/static-content-server.test.ts +60 -0
  25. package/src/adapters/node/static-content-server.ts +239 -0
  26. package/src/adapters/shared/api-response.test.ts +97 -0
  27. package/src/adapters/shared/api-response.ts +104 -0
  28. package/src/adapters/shared/application-adapter.ts +199 -0
  29. package/src/adapters/shared/define-api-handler.ts +66 -0
  30. package/src/adapters/shared/explicit-static-render-preparation.ts +58 -0
  31. package/src/adapters/shared/explicit-static-route-matcher.test.ts +381 -0
  32. package/src/adapters/shared/explicit-static-route-matcher.ts +131 -0
  33. package/src/adapters/shared/file-route-middleware-pipeline.test.ts +85 -0
  34. package/src/adapters/shared/file-route-middleware-pipeline.ts +118 -0
  35. package/src/adapters/shared/fs-server-response-factory.test.ts +176 -0
  36. package/src/adapters/shared/fs-server-response-factory.ts +96 -0
  37. package/src/adapters/shared/fs-server-response-matcher.test.ts +311 -0
  38. package/src/adapters/shared/fs-server-response-matcher.ts +240 -0
  39. package/src/adapters/shared/hmr-entrypoint-registrar.ts +149 -0
  40. package/src/adapters/shared/hmr-html-response.ts +52 -0
  41. package/src/adapters/shared/hmr-manager.contract.test.ts +228 -0
  42. package/src/adapters/shared/hmr-manager.dispatch.test.ts +220 -0
  43. package/src/adapters/shared/render-context.test.ts +150 -0
  44. package/src/adapters/shared/render-context.ts +123 -0
  45. package/src/adapters/shared/runtime-bootstrap.ts +79 -0
  46. package/src/adapters/shared/server-adapter.test.ts +130 -0
  47. package/src/adapters/shared/server-adapter.ts +562 -0
  48. package/src/adapters/shared/server-route-handler.test.ts +111 -0
  49. package/src/adapters/shared/server-route-handler.ts +153 -0
  50. package/src/adapters/shared/server-static-builder.test.ts +338 -0
  51. package/src/adapters/shared/server-static-builder.ts +170 -0
  52. package/src/build/build-adapter-serialization.test.ts +281 -0
  53. package/src/build/build-adapter.test.ts +1240 -0
  54. package/src/build/build-adapter.ts +1012 -0
  55. package/src/build/build-manifest.ts +54 -0
  56. package/src/build/build-types.ts +83 -0
  57. package/src/build/dev-build-coordinator.ts +220 -0
  58. package/src/build/esbuild-build-adapter.ts +660 -0
  59. package/src/build/runtime-build-executor.test.ts +81 -0
  60. package/src/build/runtime-build-executor.ts +40 -0
  61. package/src/build/runtime-specifier-alias-plugin.test.ts +67 -0
  62. package/src/build/runtime-specifier-alias-plugin.ts +62 -0
  63. package/src/build/runtime-specifier-aliases.ts +135 -0
  64. package/src/config/README.md +1 -1
  65. package/src/config/config-builder.test.ts +442 -0
  66. package/src/config/config-builder.ts +737 -0
  67. package/src/config/config-builder.typecheck.test.ts +96 -0
  68. package/src/config/{constants.d.ts → constants.ts} +22 -13
  69. package/src/dev/host-runtime.ts +34 -0
  70. package/src/dev/sc-server.ts +143 -0
  71. package/src/eco/eco.browser.test.ts +43 -0
  72. package/src/eco/eco.browser.ts +118 -0
  73. package/src/eco/eco.test.ts +654 -0
  74. package/src/eco/eco.ts +205 -0
  75. package/src/eco/eco.types.ts +221 -0
  76. package/src/eco/eco.utils.test.ts +219 -0
  77. package/src/eco/eco.utils.ts +5 -0
  78. package/src/eco/global-injector-map.test.ts +42 -0
  79. package/src/eco/global-injector-map.ts +112 -0
  80. package/src/eco/lazy-injector-map.test.ts +66 -0
  81. package/src/eco/lazy-injector-map.ts +120 -0
  82. package/src/eco/module-dependencies.test.ts +30 -0
  83. package/src/eco/module-dependencies.ts +75 -0
  84. package/src/errors/http-error.test.ts +134 -0
  85. package/src/errors/http-error.ts +72 -0
  86. package/src/errors/index.ts +3 -0
  87. package/src/errors/locals-access-error.ts +7 -0
  88. package/src/global/app-logger.ts +4 -0
  89. package/src/global/utils.test.ts +12 -0
  90. 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
  91. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
  92. package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
  93. package/src/hmr/client/hmr-runtime.ts +162 -0
  94. package/src/hmr/hmr-strategy.test.ts +124 -0
  95. package/src/hmr/hmr-strategy.ts +177 -0
  96. package/src/hmr/hmr.postcss.test.e2e.ts +41 -0
  97. package/src/hmr/hmr.test.e2e.ts +66 -0
  98. package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
  99. package/src/hmr/strategies/js-hmr-strategy.test.ts +334 -0
  100. package/src/hmr/strategies/js-hmr-strategy.ts +314 -0
  101. package/src/index.browser.ts +3 -0
  102. package/src/index.ts +15 -0
  103. package/src/integrations/ghtml/ghtml-renderer.test.ts +253 -0
  104. package/src/integrations/ghtml/ghtml-renderer.ts +87 -0
  105. package/src/integrations/ghtml/ghtml.constants.ts +1 -0
  106. package/src/integrations/ghtml/ghtml.plugin.ts +28 -0
  107. package/src/plugins/alias-resolver-plugin.test.ts +41 -0
  108. package/src/plugins/alias-resolver-plugin.ts +63 -0
  109. package/src/plugins/eco-component-meta-plugin.test.ts +406 -0
  110. package/src/plugins/eco-component-meta-plugin.ts +494 -0
  111. package/src/plugins/foreign-jsx-override-plugin.test.ts +65 -0
  112. package/src/plugins/foreign-jsx-override-plugin.ts +67 -0
  113. package/src/plugins/integration-plugin.test.ts +151 -0
  114. package/src/plugins/integration-plugin.ts +323 -0
  115. package/src/plugins/processor.test.ts +148 -0
  116. package/src/plugins/processor.ts +257 -0
  117. package/src/plugins/{runtime-capability.d.ts → runtime-capability.ts} +8 -3
  118. package/src/plugins/source-transform.test.ts +82 -0
  119. package/src/plugins/source-transform.ts +123 -0
  120. package/src/route-renderer/GRAPH.md +81 -289
  121. package/src/route-renderer/README.md +67 -105
  122. package/src/route-renderer/orchestration/component-render-context.ts +325 -0
  123. package/src/route-renderer/orchestration/declared-ownership-graph.ts +62 -0
  124. package/src/route-renderer/orchestration/foreign-subtree-execution.service.ts +383 -0
  125. package/src/route-renderer/orchestration/integration-renderer.test.ts +2085 -0
  126. package/src/route-renderer/orchestration/integration-renderer.ts +1244 -0
  127. package/src/route-renderer/orchestration/ownership-planning.service.ts +97 -0
  128. package/src/route-renderer/orchestration/ownership-validation.service.ts +76 -0
  129. package/src/route-renderer/orchestration/processed-asset-dedupe.ts +25 -0
  130. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.test.ts +324 -0
  131. package/src/route-renderer/orchestration/queued-foreign-subtree-resolution.service.ts +294 -0
  132. package/src/route-renderer/orchestration/render-output.utils.ts +310 -0
  133. package/src/route-renderer/orchestration/route-render-orchestrator.prepare-render-options.test.ts +644 -0
  134. package/src/route-renderer/orchestration/route-render-orchestrator.test.ts +265 -0
  135. package/src/route-renderer/orchestration/route-render-orchestrator.ts +592 -0
  136. package/src/route-renderer/orchestration/template-serialization.test.ts +110 -0
  137. package/src/route-renderer/orchestration/template-serialization.ts +117 -0
  138. package/src/route-renderer/page-loading/component-dependency-collection.ts +202 -0
  139. package/src/route-renderer/page-loading/declared-asset-collection.ts +153 -0
  140. package/src/route-renderer/page-loading/dependency-resolver.test.ts +761 -0
  141. package/src/route-renderer/page-loading/dependency-resolver.ts +144 -0
  142. package/src/route-renderer/page-loading/ecopages-virtual-imports.ts +75 -0
  143. package/src/route-renderer/page-loading/lazy-entry-collection.ts +167 -0
  144. package/src/route-renderer/page-loading/lazy-trigger-planning.ts +74 -0
  145. package/src/route-renderer/page-loading/module-declaration-aggregation.ts +60 -0
  146. package/src/route-renderer/page-loading/module-declaration-scripts.ts +16 -0
  147. package/src/route-renderer/page-loading/page-dependency-bundling.ts +244 -0
  148. package/src/route-renderer/page-loading/page-module-loader.test.ts +183 -0
  149. package/src/route-renderer/page-loading/page-module-loader.ts +184 -0
  150. package/src/route-renderer/route-renderer.ts +133 -0
  151. package/src/router/README.md +16 -19
  152. package/src/router/client/link-intent.test.browser.ts +51 -0
  153. package/src/router/client/link-intent.ts +92 -0
  154. package/src/router/client/navigation-coordinator.test.ts +237 -0
  155. package/src/router/client/navigation-coordinator.ts +453 -0
  156. package/src/router/server/route-registry.test.ts +176 -0
  157. package/src/router/server/route-registry.ts +382 -0
  158. package/src/services/README.md +1 -2
  159. package/src/services/assets/asset-processing-service/asset-dependency-keys.ts +66 -0
  160. package/src/services/assets/asset-processing-service/asset-processing.service.test.ts +473 -0
  161. package/src/services/assets/asset-processing-service/asset-processing.service.ts +344 -0
  162. package/src/services/assets/asset-processing-service/asset.factory.test.ts +63 -0
  163. package/src/services/assets/asset-processing-service/asset.factory.ts +105 -0
  164. package/src/services/assets/asset-processing-service/assets.types.ts +128 -0
  165. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.test.ts +74 -0
  166. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.ts +96 -0
  167. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.test.ts +67 -0
  168. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.ts +78 -0
  169. package/src/services/assets/asset-processing-service/grouped-content-bundles.ts +104 -0
  170. package/src/services/assets/asset-processing-service/index.ts +6 -0
  171. package/src/services/assets/asset-processing-service/page-package.test.ts +100 -0
  172. package/src/services/assets/asset-processing-service/page-package.ts +93 -0
  173. package/src/services/assets/asset-processing-service/{processor.interface.d.ts → processor.interface.ts} +10 -5
  174. package/src/services/assets/asset-processing-service/processor.registry.ts +18 -0
  175. package/src/services/assets/asset-processing-service/processors/base/base-processor.test.ts +59 -0
  176. package/src/services/assets/asset-processing-service/processors/base/base-processor.ts +83 -0
  177. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.ts +173 -0
  178. package/src/services/assets/asset-processing-service/processors/index.ts +5 -0
  179. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.test.ts +195 -0
  180. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.ts +137 -0
  181. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.test.ts +326 -0
  182. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.ts +116 -0
  183. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.test.ts +227 -0
  184. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.ts +89 -0
  185. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.test.ts +261 -0
  186. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +72 -0
  187. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +83 -0
  188. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.ts +65 -0
  189. package/src/services/assets/browser-bundle.service.test.ts +66 -0
  190. package/src/services/assets/browser-bundle.service.ts +109 -0
  191. package/src/services/cache/cache.types.ts +126 -0
  192. package/src/services/cache/index.ts +18 -0
  193. package/src/services/cache/memory-cache-store.test.ts +225 -0
  194. package/src/services/cache/memory-cache-store.ts +130 -0
  195. package/src/services/cache/page-cache-service.test.ts +175 -0
  196. package/src/services/cache/page-cache-service.ts +202 -0
  197. package/src/services/cache/page-request-cache-coordinator.service.test.ts +79 -0
  198. package/src/services/cache/page-request-cache-coordinator.service.ts +131 -0
  199. package/src/services/html/html-rewriter-provider.service.test.ts +183 -0
  200. package/src/services/html/html-rewriter-provider.service.ts +104 -0
  201. package/src/services/html/html-transformer.service.test.ts +476 -0
  202. package/src/services/html/html-transformer.service.ts +275 -0
  203. package/src/services/invalidation/development-invalidation.service.test.ts +87 -0
  204. package/src/services/invalidation/development-invalidation.service.ts +262 -0
  205. package/src/services/module-loading/app-module-loader.service.ts +9 -0
  206. package/src/services/module-loading/app-server-module-transpiler.service.test.ts +130 -0
  207. package/src/services/module-loading/app-server-module-transpiler.service.ts +141 -0
  208. package/src/services/module-loading/host-module-loader-registry.ts +15 -0
  209. package/src/services/module-loading/{module-loading-types.d.ts → module-loading-types.ts} +1 -0
  210. package/src/services/module-loading/node-bootstrap-plugin.test.ts +335 -0
  211. package/src/services/module-loading/node-bootstrap-plugin.ts +311 -0
  212. package/src/services/module-loading/page-module-import.service.test.ts +504 -0
  213. package/src/services/module-loading/page-module-import.service.ts +251 -0
  214. package/src/services/module-loading/server-module-transpiler.service.test.ts +243 -0
  215. package/src/services/module-loading/server-module-transpiler.service.ts +104 -0
  216. package/src/services/module-loading/source-module-support.ts +19 -0
  217. package/src/services/runtime-state/dev-graph.service.ts +217 -0
  218. package/src/services/runtime-state/entrypoint-dependency-graph.service.ts +136 -0
  219. package/src/services/runtime-state/server-invalidation-state.service.ts +68 -0
  220. package/src/services/validation/schema-validation-service.test.ts +223 -0
  221. package/src/services/validation/schema-validation-service.ts +204 -0
  222. package/src/services/validation/{standard-schema.types.d.ts → standard-schema.types.ts} +20 -17
  223. package/src/static-site-generator/static-site-generator.test.ts +408 -0
  224. package/src/static-site-generator/static-site-generator.ts +445 -0
  225. package/src/types/internal-types.ts +243 -0
  226. package/src/types/public-types.ts +1459 -0
  227. package/src/utils/deep-merge.test.ts +114 -0
  228. package/src/utils/deep-merge.ts +47 -0
  229. package/src/utils/hash.ts +5 -0
  230. package/src/utils/html-escaping.ts +9 -0
  231. package/src/utils/invariant.test.ts +22 -0
  232. package/src/utils/invariant.ts +15 -0
  233. package/src/utils/locals-utils.ts +37 -0
  234. package/src/utils/parse-cli-args.test.ts +69 -0
  235. package/src/utils/parse-cli-args.ts +105 -0
  236. package/src/utils/path-utils.module.ts +14 -0
  237. package/src/utils/path-utils.test.ts +15 -0
  238. package/src/utils/resolve-work-dir.ts +45 -0
  239. package/src/utils/runtime.ts +44 -0
  240. package/src/utils/server-utils.module.ts +67 -0
  241. package/src/utils/server-utils.test.ts +38 -0
  242. package/src/watchers/project-watcher.integration.test.ts +337 -0
  243. package/src/watchers/project-watcher.test-helpers.ts +42 -0
  244. package/src/watchers/project-watcher.test.ts +768 -0
  245. package/src/watchers/project-watcher.ts +357 -0
  246. package/CHANGELOG.md +0 -66
  247. package/src/adapters/abstract/application-adapter.d.ts +0 -194
  248. package/src/adapters/abstract/application-adapter.js +0 -121
  249. package/src/adapters/abstract/router-adapter.d.ts +0 -26
  250. package/src/adapters/abstract/router-adapter.js +0 -5
  251. package/src/adapters/abstract/server-adapter.d.ts +0 -69
  252. package/src/adapters/abstract/server-adapter.js +0 -15
  253. package/src/adapters/bun/client-bridge.d.ts +0 -34
  254. package/src/adapters/bun/client-bridge.js +0 -48
  255. package/src/adapters/bun/create-app.d.ts +0 -52
  256. package/src/adapters/bun/create-app.js +0 -116
  257. package/src/adapters/bun/hmr-manager.d.ts +0 -143
  258. package/src/adapters/bun/hmr-manager.js +0 -333
  259. package/src/adapters/bun/index.d.ts +0 -2
  260. package/src/adapters/bun/index.js +0 -8
  261. package/src/adapters/bun/server-adapter.d.ts +0 -155
  262. package/src/adapters/bun/server-adapter.js +0 -374
  263. package/src/adapters/bun/server-lifecycle.d.ts +0 -63
  264. package/src/adapters/bun/server-lifecycle.js +0 -92
  265. package/src/adapters/create-app.d.ts +0 -20
  266. package/src/adapters/create-app.js +0 -66
  267. package/src/adapters/index.d.ts +0 -2
  268. package/src/adapters/index.js +0 -8
  269. package/src/adapters/node/create-app.d.ts +0 -18
  270. package/src/adapters/node/create-app.js +0 -149
  271. package/src/adapters/node/node-client-bridge.d.ts +0 -26
  272. package/src/adapters/node/node-client-bridge.js +0 -66
  273. package/src/adapters/node/node-hmr-manager.d.ts +0 -133
  274. package/src/adapters/node/node-hmr-manager.js +0 -311
  275. package/src/adapters/node/server-adapter.d.ts +0 -162
  276. package/src/adapters/node/server-adapter.js +0 -368
  277. package/src/adapters/node/static-content-server.d.ts +0 -60
  278. package/src/adapters/node/static-content-server.js +0 -194
  279. package/src/adapters/shared/api-response.d.ts +0 -52
  280. package/src/adapters/shared/api-response.js +0 -96
  281. package/src/adapters/shared/application-adapter.d.ts +0 -18
  282. package/src/adapters/shared/application-adapter.js +0 -90
  283. package/src/adapters/shared/define-api-handler.d.ts +0 -25
  284. package/src/adapters/shared/define-api-handler.js +0 -15
  285. package/src/adapters/shared/explicit-static-route-matcher.d.ts +0 -38
  286. package/src/adapters/shared/explicit-static-route-matcher.js +0 -103
  287. package/src/adapters/shared/file-route-middleware-pipeline.d.ts +0 -65
  288. package/src/adapters/shared/file-route-middleware-pipeline.js +0 -99
  289. package/src/adapters/shared/fs-server-response-factory.d.ts +0 -19
  290. package/src/adapters/shared/fs-server-response-factory.js +0 -97
  291. package/src/adapters/shared/fs-server-response-matcher.d.ts +0 -67
  292. package/src/adapters/shared/fs-server-response-matcher.js +0 -147
  293. package/src/adapters/shared/hmr-entrypoint-registrar.d.ts +0 -55
  294. package/src/adapters/shared/hmr-entrypoint-registrar.js +0 -87
  295. package/src/adapters/shared/hmr-html-response.d.ts +0 -22
  296. package/src/adapters/shared/hmr-html-response.js +0 -32
  297. package/src/adapters/shared/render-context.d.ts +0 -15
  298. package/src/adapters/shared/render-context.js +0 -72
  299. package/src/adapters/shared/runtime-bootstrap.d.ts +0 -38
  300. package/src/adapters/shared/runtime-bootstrap.js +0 -43
  301. package/src/adapters/shared/server-adapter.d.ts +0 -97
  302. package/src/adapters/shared/server-adapter.js +0 -390
  303. package/src/adapters/shared/server-route-handler.d.ts +0 -89
  304. package/src/adapters/shared/server-route-handler.js +0 -111
  305. package/src/adapters/shared/server-static-builder.d.ts +0 -71
  306. package/src/adapters/shared/server-static-builder.js +0 -100
  307. package/src/build/build-adapter.d.ts +0 -239
  308. package/src/build/build-adapter.js +0 -642
  309. package/src/build/build-manifest.d.ts +0 -27
  310. package/src/build/build-manifest.js +0 -30
  311. package/src/build/build-types.d.ts +0 -57
  312. package/src/build/build-types.js +0 -0
  313. package/src/build/dev-build-coordinator.d.ts +0 -72
  314. package/src/build/dev-build-coordinator.js +0 -154
  315. package/src/build/esbuild-build-adapter.d.ts +0 -78
  316. package/src/build/esbuild-build-adapter.js +0 -505
  317. package/src/build/runtime-build-executor.d.ts +0 -14
  318. package/src/build/runtime-build-executor.js +0 -22
  319. package/src/build/runtime-specifier-alias-plugin.d.ts +0 -15
  320. package/src/build/runtime-specifier-alias-plugin.js +0 -35
  321. package/src/build/runtime-specifier-aliases.d.ts +0 -5
  322. package/src/build/runtime-specifier-aliases.js +0 -95
  323. package/src/config/config-builder.d.ts +0 -252
  324. package/src/config/config-builder.js +0 -603
  325. package/src/config/constants.js +0 -25
  326. package/src/dev/sc-server.d.ts +0 -30
  327. package/src/dev/sc-server.js +0 -111
  328. package/src/eco/eco.browser.d.ts +0 -2
  329. package/src/eco/eco.browser.js +0 -83
  330. package/src/eco/eco.d.ts +0 -9
  331. package/src/eco/eco.js +0 -85
  332. package/src/eco/eco.types.d.ts +0 -178
  333. package/src/eco/eco.types.js +0 -0
  334. package/src/eco/eco.utils.d.ts +0 -1
  335. package/src/eco/eco.utils.js +0 -10
  336. package/src/eco/global-injector-map.d.ts +0 -16
  337. package/src/eco/global-injector-map.js +0 -80
  338. package/src/eco/lazy-injector-map.d.ts +0 -8
  339. package/src/eco/lazy-injector-map.js +0 -70
  340. package/src/eco/module-dependencies.d.ts +0 -18
  341. package/src/eco/module-dependencies.js +0 -49
  342. package/src/errors/http-error.d.ts +0 -31
  343. package/src/errors/http-error.js +0 -50
  344. package/src/errors/index.d.ts +0 -2
  345. package/src/errors/index.js +0 -4
  346. package/src/errors/locals-access-error.d.ts +0 -4
  347. package/src/errors/locals-access-error.js +0 -9
  348. package/src/global/app-logger.d.ts +0 -2
  349. package/src/global/app-logger.js +0 -6
  350. package/src/hmr/client/hmr-runtime.d.ts +0 -5
  351. package/src/hmr/client/hmr-runtime.js +0 -117
  352. package/src/hmr/hmr-strategy.d.ts +0 -162
  353. package/src/hmr/hmr-strategy.js +0 -44
  354. package/src/hmr/hmr.postcss.test.e2e.d.ts +0 -1
  355. package/src/hmr/hmr.postcss.test.e2e.js +0 -31
  356. package/src/hmr/hmr.test.e2e.d.ts +0 -1
  357. package/src/hmr/hmr.test.e2e.js +0 -43
  358. package/src/hmr/strategies/default-hmr-strategy.d.ts +0 -43
  359. package/src/hmr/strategies/default-hmr-strategy.js +0 -34
  360. package/src/hmr/strategies/js-hmr-strategy.d.ts +0 -139
  361. package/src/hmr/strategies/js-hmr-strategy.js +0 -178
  362. package/src/index.browser.d.ts +0 -3
  363. package/src/index.browser.js +0 -4
  364. package/src/index.d.ts +0 -6
  365. package/src/index.js +0 -21
  366. package/src/integrations/ghtml/ghtml-renderer.d.ts +0 -20
  367. package/src/integrations/ghtml/ghtml-renderer.js +0 -63
  368. package/src/integrations/ghtml/ghtml.constants.d.ts +0 -1
  369. package/src/integrations/ghtml/ghtml.constants.js +0 -4
  370. package/src/integrations/ghtml/ghtml.plugin.d.ts +0 -16
  371. package/src/integrations/ghtml/ghtml.plugin.js +0 -20
  372. package/src/plugins/alias-resolver-plugin.d.ts +0 -2
  373. package/src/plugins/alias-resolver-plugin.js +0 -53
  374. package/src/plugins/eco-component-meta-plugin.d.ts +0 -108
  375. package/src/plugins/eco-component-meta-plugin.js +0 -163
  376. package/src/plugins/foreign-jsx-override-plugin.d.ts +0 -31
  377. package/src/plugins/foreign-jsx-override-plugin.js +0 -35
  378. package/src/plugins/integration-plugin.d.ts +0 -219
  379. package/src/plugins/integration-plugin.js +0 -196
  380. package/src/plugins/processor.d.ts +0 -95
  381. package/src/plugins/processor.js +0 -136
  382. package/src/plugins/runtime-capability.js +0 -0
  383. package/src/plugins/source-transform.d.ts +0 -46
  384. package/src/plugins/source-transform.js +0 -71
  385. package/src/route-renderer/orchestration/boundary-planning.service.d.ts +0 -25
  386. package/src/route-renderer/orchestration/boundary-planning.service.js +0 -97
  387. package/src/route-renderer/orchestration/component-render-context.d.ts +0 -83
  388. package/src/route-renderer/orchestration/component-render-context.js +0 -147
  389. package/src/route-renderer/orchestration/integration-renderer.d.ts +0 -556
  390. package/src/route-renderer/orchestration/integration-renderer.js +0 -932
  391. package/src/route-renderer/orchestration/page-packaging.service.d.ts +0 -16
  392. package/src/route-renderer/orchestration/page-packaging.service.js +0 -66
  393. package/src/route-renderer/orchestration/processed-asset-dedupe.d.ts +0 -2
  394. package/src/route-renderer/orchestration/processed-asset-dedupe.js +0 -23
  395. package/src/route-renderer/orchestration/queued-boundary-runtime.service.d.ts +0 -89
  396. package/src/route-renderer/orchestration/queued-boundary-runtime.service.js +0 -155
  397. package/src/route-renderer/orchestration/render-execution.service.d.ts +0 -43
  398. package/src/route-renderer/orchestration/render-execution.service.js +0 -106
  399. package/src/route-renderer/orchestration/render-output.utils.d.ts +0 -66
  400. package/src/route-renderer/orchestration/render-output.utils.js +0 -171
  401. package/src/route-renderer/orchestration/render-preparation.service.d.ts +0 -120
  402. package/src/route-renderer/orchestration/render-preparation.service.js +0 -364
  403. package/src/route-renderer/orchestration/route-shell-composer.service.d.ts +0 -50
  404. package/src/route-renderer/orchestration/route-shell-composer.service.js +0 -81
  405. package/src/route-renderer/orchestration/template-serialization.d.ts +0 -38
  406. package/src/route-renderer/orchestration/template-serialization.js +0 -45
  407. package/src/route-renderer/page-loading/component-dependency-collection.d.ts +0 -37
  408. package/src/route-renderer/page-loading/component-dependency-collection.js +0 -125
  409. package/src/route-renderer/page-loading/declared-asset-collection.d.ts +0 -24
  410. package/src/route-renderer/page-loading/declared-asset-collection.js +0 -106
  411. package/src/route-renderer/page-loading/dependency-resolver.d.ts +0 -35
  412. package/src/route-renderer/page-loading/dependency-resolver.js +0 -117
  413. package/src/route-renderer/page-loading/ecopages-virtual-imports.d.ts +0 -11
  414. package/src/route-renderer/page-loading/ecopages-virtual-imports.js +0 -57
  415. package/src/route-renderer/page-loading/lazy-entry-collection.d.ts +0 -45
  416. package/src/route-renderer/page-loading/lazy-entry-collection.js +0 -105
  417. package/src/route-renderer/page-loading/lazy-trigger-planning.d.ts +0 -19
  418. package/src/route-renderer/page-loading/lazy-trigger-planning.js +0 -40
  419. package/src/route-renderer/page-loading/module-declaration-aggregation.d.ts +0 -5
  420. package/src/route-renderer/page-loading/module-declaration-aggregation.js +0 -33
  421. package/src/route-renderer/page-loading/module-declaration-scripts.d.ts +0 -3
  422. package/src/route-renderer/page-loading/module-declaration-scripts.js +0 -18
  423. package/src/route-renderer/page-loading/page-dependency-bundling.d.ts +0 -13
  424. package/src/route-renderer/page-loading/page-dependency-bundling.js +0 -115
  425. package/src/route-renderer/page-loading/page-module-loader.d.ts +0 -90
  426. package/src/route-renderer/page-loading/page-module-loader.js +0 -127
  427. package/src/route-renderer/route-renderer.d.ts +0 -67
  428. package/src/route-renderer/route-renderer.js +0 -103
  429. package/src/router/client/link-intent.js +0 -34
  430. package/src/router/client/link-intent.test.browser.d.ts +0 -1
  431. package/src/router/client/link-intent.test.browser.js +0 -43
  432. package/src/router/client/navigation-coordinator.d.ts +0 -169
  433. package/src/router/client/navigation-coordinator.js +0 -215
  434. package/src/router/server/fs-router-scanner.d.ts +0 -41
  435. package/src/router/server/fs-router-scanner.js +0 -161
  436. package/src/router/server/fs-router.d.ts +0 -26
  437. package/src/router/server/fs-router.js +0 -100
  438. package/src/services/assets/asset-processing-service/asset-dependency-keys.d.ts +0 -3
  439. package/src/services/assets/asset-processing-service/asset-dependency-keys.js +0 -56
  440. package/src/services/assets/asset-processing-service/asset-processing.service.d.ts +0 -103
  441. package/src/services/assets/asset-processing-service/asset-processing.service.js +0 -285
  442. package/src/services/assets/asset-processing-service/asset.factory.d.ts +0 -17
  443. package/src/services/assets/asset-processing-service/asset.factory.js +0 -82
  444. package/src/services/assets/asset-processing-service/assets.types.d.ts +0 -100
  445. package/src/services/assets/asset-processing-service/assets.types.js +0 -0
  446. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.d.ts +0 -55
  447. package/src/services/assets/asset-processing-service/browser-runtime-asset.factory.js +0 -49
  448. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.d.ts +0 -20
  449. package/src/services/assets/asset-processing-service/browser-runtime-entry.factory.js +0 -41
  450. package/src/services/assets/asset-processing-service/grouped-content-bundles.d.ts +0 -30
  451. package/src/services/assets/asset-processing-service/grouped-content-bundles.js +0 -65
  452. package/src/services/assets/asset-processing-service/index.d.ts +0 -5
  453. package/src/services/assets/asset-processing-service/index.js +0 -5
  454. package/src/services/assets/asset-processing-service/processor.interface.js +0 -6
  455. package/src/services/assets/asset-processing-service/processor.registry.d.ts +0 -8
  456. package/src/services/assets/asset-processing-service/processor.registry.js +0 -15
  457. package/src/services/assets/asset-processing-service/processors/base/base-processor.d.ts +0 -24
  458. package/src/services/assets/asset-processing-service/processors/base/base-processor.js +0 -65
  459. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.d.ts +0 -22
  460. package/src/services/assets/asset-processing-service/processors/base/base-script-processor.js +0 -136
  461. package/src/services/assets/asset-processing-service/processors/index.d.ts +0 -5
  462. package/src/services/assets/asset-processing-service/processors/index.js +0 -5
  463. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.d.ts +0 -6
  464. package/src/services/assets/asset-processing-service/processors/script/content-script.processor.js +0 -116
  465. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.d.ts +0 -9
  466. package/src/services/assets/asset-processing-service/processors/script/file-script.processor.js +0 -91
  467. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.d.ts +0 -7
  468. package/src/services/assets/asset-processing-service/processors/script/node-module-script.processor.js +0 -77
  469. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +0 -8
  470. package/src/services/assets/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +0 -58
  471. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +0 -9
  472. package/src/services/assets/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +0 -67
  473. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.d.ts +0 -18
  474. package/src/services/assets/asset-processing-service/ungrouped-dependency-processing.js +0 -45
  475. package/src/services/assets/browser-bundle.service.d.ts +0 -73
  476. package/src/services/assets/browser-bundle.service.js +0 -41
  477. package/src/services/cache/cache.types.d.ts +0 -107
  478. package/src/services/cache/cache.types.js +0 -0
  479. package/src/services/cache/index.d.ts +0 -7
  480. package/src/services/cache/index.js +0 -7
  481. package/src/services/cache/memory-cache-store.d.ts +0 -42
  482. package/src/services/cache/memory-cache-store.js +0 -98
  483. package/src/services/cache/page-cache-service.d.ts +0 -70
  484. package/src/services/cache/page-cache-service.js +0 -152
  485. package/src/services/cache/page-request-cache-coordinator.service.d.ts +0 -75
  486. package/src/services/cache/page-request-cache-coordinator.service.js +0 -109
  487. package/src/services/html/html-rewriter-provider.service.d.ts +0 -37
  488. package/src/services/html/html-rewriter-provider.service.js +0 -68
  489. package/src/services/html/html-transformer.service.d.ts +0 -87
  490. package/src/services/html/html-transformer.service.js +0 -216
  491. package/src/services/invalidation/development-invalidation.service.d.ts +0 -74
  492. package/src/services/invalidation/development-invalidation.service.js +0 -190
  493. package/src/services/module-loading/app-module-loader.service.d.ts +0 -7
  494. package/src/services/module-loading/app-module-loader.service.js +0 -0
  495. package/src/services/module-loading/app-server-module-transpiler.service.d.ts +0 -24
  496. package/src/services/module-loading/app-server-module-transpiler.service.js +0 -115
  497. package/src/services/module-loading/host-module-loader-registry.d.ts +0 -4
  498. package/src/services/module-loading/host-module-loader-registry.js +0 -15
  499. package/src/services/module-loading/module-loading-types.js +0 -0
  500. package/src/services/module-loading/node-bootstrap-plugin.d.ts +0 -42
  501. package/src/services/module-loading/node-bootstrap-plugin.js +0 -204
  502. package/src/services/module-loading/page-module-import.service.d.ts +0 -76
  503. package/src/services/module-loading/page-module-import.service.js +0 -170
  504. package/src/services/module-loading/server-module-transpiler.service.d.ts +0 -63
  505. package/src/services/module-loading/server-module-transpiler.service.js +0 -64
  506. package/src/services/module-loading/source-module-support.d.ts +0 -5
  507. package/src/services/module-loading/source-module-support.js +0 -8
  508. package/src/services/runtime-state/dev-graph.service.d.ts +0 -118
  509. package/src/services/runtime-state/dev-graph.service.js +0 -162
  510. package/src/services/runtime-state/entrypoint-dependency-graph.service.d.ts +0 -41
  511. package/src/services/runtime-state/entrypoint-dependency-graph.service.js +0 -85
  512. package/src/services/runtime-state/runtime-specifier-registry.service.d.ts +0 -69
  513. package/src/services/runtime-state/runtime-specifier-registry.service.js +0 -37
  514. package/src/services/runtime-state/server-invalidation-state.service.d.ts +0 -26
  515. package/src/services/runtime-state/server-invalidation-state.service.js +0 -35
  516. package/src/services/validation/schema-validation-service.d.ts +0 -122
  517. package/src/services/validation/schema-validation-service.js +0 -101
  518. package/src/services/validation/standard-schema.types.js +0 -0
  519. package/src/static-site-generator/static-site-generator.d.ts +0 -105
  520. package/src/static-site-generator/static-site-generator.js +0 -349
  521. package/src/types/internal-types.d.ts +0 -231
  522. package/src/types/internal-types.js +0 -0
  523. package/src/types/public-types.d.ts +0 -1257
  524. package/src/types/public-types.js +0 -0
  525. package/src/utils/deep-merge.d.ts +0 -14
  526. package/src/utils/deep-merge.js +0 -32
  527. package/src/utils/hash.d.ts +0 -1
  528. package/src/utils/hash.js +0 -7
  529. package/src/utils/html-escaping.d.ts +0 -7
  530. package/src/utils/html-escaping.js +0 -6
  531. package/src/utils/html.js +0 -4
  532. package/src/utils/invariant.d.ts +0 -5
  533. package/src/utils/invariant.js +0 -11
  534. package/src/utils/locals-utils.d.ts +0 -15
  535. package/src/utils/locals-utils.js +0 -24
  536. package/src/utils/parse-cli-args.d.ts +0 -27
  537. package/src/utils/parse-cli-args.js +0 -62
  538. package/src/utils/path-utils.module.d.ts +0 -5
  539. package/src/utils/path-utils.module.js +0 -14
  540. package/src/utils/resolve-work-dir.d.ts +0 -11
  541. package/src/utils/resolve-work-dir.js +0 -31
  542. package/src/utils/runtime.d.ts +0 -11
  543. package/src/utils/runtime.js +0 -40
  544. package/src/utils/server-utils.module.d.ts +0 -19
  545. package/src/utils/server-utils.module.js +0 -56
  546. package/src/watchers/project-watcher.d.ts +0 -136
  547. package/src/watchers/project-watcher.js +0 -275
  548. package/src/watchers/project-watcher.test-helpers.d.ts +0 -4
  549. package/src/watchers/project-watcher.test-helpers.js +0 -52
  550. /package/src/utils/{html.d.ts → html.ts} +0 -0
@@ -0,0 +1,175 @@
1
+ /**
2
+ * Unit tests for PageCacheService
3
+ */
4
+
5
+ import { describe, expect, test, beforeEach, vi } from 'vitest';
6
+ import { PageCacheService, getCacheControlHeader } from './page-cache-service.js';
7
+ import { MemoryCacheStore } from './memory-cache-store.js';
8
+
9
+ describe('PageCacheService', () => {
10
+ let service: PageCacheService;
11
+ let store: MemoryCacheStore;
12
+
13
+ beforeEach(() => {
14
+ store = new MemoryCacheStore();
15
+ service = new PageCacheService({ store });
16
+ });
17
+
18
+ describe('generateCacheKey', () => {
19
+ test('should use full URL as key', () => {
20
+ expect(service.generateCacheKey('/blog/post-1')).toBe('/blog/post-1');
21
+ expect(service.generateCacheKey('/search?q=test')).toBe('/search?q=test');
22
+ });
23
+ });
24
+
25
+ describe('getOrCreate', () => {
26
+ test('should return miss and render on cache miss', async () => {
27
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy: 'static' as const });
28
+
29
+ const result = await service.getOrCreate('/page', 'static', renderFn);
30
+
31
+ expect(result.status).toBe('miss');
32
+ expect(result.html).toBe('<html>hello</html>');
33
+ expect(result.strategy).toBe('static');
34
+ expect(renderFn).toHaveBeenCalledTimes(1);
35
+ });
36
+
37
+ test('should return hit on cache hit', async () => {
38
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy: 'static' as const });
39
+
40
+ await service.getOrCreate('/page', 'static', renderFn);
41
+
42
+ renderFn.mockClear();
43
+ const result = await service.getOrCreate('/page', 'static', renderFn);
44
+
45
+ expect(result.status).toBe('hit');
46
+ expect(result.html).toBe('<html>hello</html>');
47
+ expect(result.strategy).toBe('static');
48
+ expect(renderFn).not.toHaveBeenCalled();
49
+ });
50
+
51
+ test('should bypass cache for dynamic strategy', async () => {
52
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy: 'dynamic' as const });
53
+
54
+ const result1 = await service.getOrCreate('/page', 'dynamic', renderFn);
55
+ const result2 = await service.getOrCreate('/page', 'dynamic', renderFn);
56
+
57
+ expect(result1.status).toBe('miss');
58
+ expect(result2.status).toBe('miss');
59
+ expect(renderFn).toHaveBeenCalledTimes(2);
60
+ });
61
+
62
+ test('should return stale when entry is past revalidation time', async () => {
63
+ const revalidateStrategy = { revalidate: 60 };
64
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>v1</html>', strategy: revalidateStrategy });
65
+
66
+ const staleEntry = {
67
+ html: '<html>stale</html>',
68
+ createdAt: Date.now() - 120000,
69
+ revalidateAfter: Date.now() - 60000,
70
+ tags: [],
71
+ strategy: revalidateStrategy,
72
+ };
73
+ await store.set('/stale-page', staleEntry);
74
+
75
+ const result = await service.getOrCreate('/stale-page', { revalidate: 60 }, renderFn);
76
+ expect(result.status).toBe('stale');
77
+ expect(result.html).toBe('<html>stale</html>');
78
+ });
79
+
80
+ test('should deduplicate concurrent regeneration requests', async () => {
81
+ let callCount = 0;
82
+ const renderFn = vi.fn().mockImplementation(async () => {
83
+ callCount++;
84
+ await new Promise((resolve) => setTimeout(resolve, 10));
85
+ return { html: `<html>v${callCount}</html>`, strategy: { revalidate: 60 } };
86
+ });
87
+
88
+ const staleEntry = {
89
+ html: '<html>stale</html>',
90
+ createdAt: Date.now() - 120000,
91
+ revalidateAfter: Date.now() - 60000,
92
+ tags: [],
93
+ strategy: { revalidate: 60 },
94
+ };
95
+ await store.set('/dedup-page', staleEntry);
96
+
97
+ const results = await Promise.all([
98
+ service.getOrCreate('/dedup-page', { revalidate: 60 }, renderFn),
99
+ service.getOrCreate('/dedup-page', { revalidate: 60 }, renderFn),
100
+ service.getOrCreate('/dedup-page', { revalidate: 60 }, renderFn),
101
+ ]);
102
+
103
+ for (const result of results) {
104
+ expect(result.status).toBe('stale');
105
+ expect(result.html).toBe('<html>stale</html>');
106
+ }
107
+
108
+ await new Promise((resolve) => setTimeout(resolve, 50));
109
+
110
+ expect(renderFn).toHaveBeenCalledTimes(1);
111
+ });
112
+ });
113
+
114
+ describe('invalidation', () => {
115
+ test('should invalidate by tags', async () => {
116
+ const strategy = { revalidate: 3600, tags: ['blog'] };
117
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy });
118
+
119
+ await service.getOrCreate('/blog/1', strategy, renderFn);
120
+ await service.getOrCreate('/blog/2', strategy, renderFn);
121
+
122
+ const count = await service.invalidateByTags(['blog']);
123
+ expect(count).toBe(2);
124
+
125
+ renderFn.mockClear();
126
+ const result = await service.getOrCreate('/blog/1', 'static', renderFn);
127
+ expect(result.status).toBe('miss');
128
+ });
129
+
130
+ test('should invalidate by paths', async () => {
131
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy: 'static' as const });
132
+
133
+ await service.getOrCreate('/blog/1', 'static', renderFn);
134
+
135
+ const count = await service.invalidateByPaths(['/blog/1']);
136
+ expect(count).toBe(1);
137
+
138
+ renderFn.mockClear();
139
+ const result = await service.getOrCreate('/blog/1', 'static', renderFn);
140
+ expect(result.status).toBe('miss');
141
+ });
142
+ });
143
+
144
+ describe('disabled cache', () => {
145
+ test('should bypass cache when disabled', async () => {
146
+ const disabledService = new PageCacheService({ store, enabled: false });
147
+ const renderFn = vi.fn().mockResolvedValue({ html: '<html>hello</html>', strategy: 'static' as const });
148
+
149
+ const result1 = await disabledService.getOrCreate('/page', 'static', renderFn);
150
+ const result2 = await disabledService.getOrCreate('/page', 'static', renderFn);
151
+
152
+ expect(result1.status).toBe('miss');
153
+ expect(result2.status).toBe('miss');
154
+ expect(renderFn).toHaveBeenCalledTimes(2);
155
+ });
156
+ });
157
+ });
158
+
159
+ describe('getCacheControlHeader', () => {
160
+ test('should return immutable for static strategy', () => {
161
+ expect(getCacheControlHeader('static')).toBe('public, max-age=31536000, immutable');
162
+ });
163
+
164
+ test('should return no-store for dynamic strategy', () => {
165
+ expect(getCacheControlHeader('dynamic')).toBe('no-store, must-revalidate');
166
+ });
167
+
168
+ test('should return max-age and stale-while-revalidate for object strategy', () => {
169
+ expect(getCacheControlHeader({ revalidate: 3600 })).toBe('public, max-age=3600, stale-while-revalidate=7200');
170
+ });
171
+
172
+ test('should return no-store when disabled', () => {
173
+ expect(getCacheControlHeader('disabled')).toBe('no-store, must-revalidate');
174
+ });
175
+ });
@@ -0,0 +1,202 @@
1
+ /**
2
+ * Page cache service with ISR (Incremental Static Regeneration) support.
3
+ * Handles stale-while-revalidate semantics and background regeneration.
4
+ * @module
5
+ */
6
+
7
+ import { appLogger } from '../../global/app-logger.ts';
8
+ import type { CacheEntry, CacheResult, CacheStore, CacheStrategy, RenderResult } from './cache.types.ts';
9
+ import { MemoryCacheStore } from './memory-cache-store.ts';
10
+
11
+ export interface PageCacheServiceOptions {
12
+ store?: CacheStore;
13
+ enabled?: boolean;
14
+ }
15
+
16
+ /**
17
+ * Core page caching service with ISR support.
18
+ */
19
+ export class PageCacheService {
20
+ private store: CacheStore;
21
+ private enabled: boolean;
22
+ private regenerationPromises = new Map<string, Promise<string>>();
23
+
24
+ constructor(options: PageCacheServiceOptions = {}) {
25
+ this.store = options.store ?? new MemoryCacheStore();
26
+ this.enabled = options.enabled ?? true;
27
+ }
28
+
29
+ /**
30
+ * Generate a cache key from URL and optional params.
31
+ * Uses full URL (path + query) as the key.
32
+ */
33
+ generateCacheKey(url: string): string {
34
+ return url;
35
+ }
36
+
37
+ /**
38
+ * Check if an entry is stale (past its revalidation time).
39
+ */
40
+ private isStale(entry: CacheEntry): boolean {
41
+ if (entry.revalidateAfter === null) return false;
42
+ return Date.now() > entry.revalidateAfter;
43
+ }
44
+
45
+ /**
46
+ * Create a cache entry from rendered HTML.
47
+ */
48
+ private createEntry(html: string, strategy: CacheStrategy): CacheEntry {
49
+ const now = Date.now();
50
+
51
+ let revalidateAfter: number | null = null;
52
+ let tags: string[] = [];
53
+
54
+ if (strategy === 'static') {
55
+ revalidateAfter = null;
56
+ } else if (strategy === 'dynamic') {
57
+ revalidateAfter = 0;
58
+ } else if (typeof strategy === 'object') {
59
+ revalidateAfter = now + strategy.revalidate * 1000;
60
+ tags = strategy.tags ?? [];
61
+ }
62
+
63
+ return {
64
+ html,
65
+ createdAt: now,
66
+ revalidateAfter,
67
+ tags,
68
+ strategy,
69
+ };
70
+ }
71
+
72
+ /**
73
+ * Get cached content or create new content with stale-while-revalidate semantics.
74
+ * @param key - Cache key (URL path + query)
75
+ * @param defaultStrategy - Default strategy if page doesn't specify one
76
+ * @param renderFn - Function that renders the page and returns HTML + strategy
77
+ */
78
+ async getOrCreate(
79
+ key: string,
80
+ defaultStrategy: CacheStrategy,
81
+ renderFn: () => Promise<RenderResult>,
82
+ ): Promise<CacheResult> {
83
+ if (!this.enabled) {
84
+ const { html, strategy } = await renderFn();
85
+ return { html, status: 'miss', strategy };
86
+ }
87
+
88
+ const entry = await this.store.get(key);
89
+
90
+ if (!entry) {
91
+ const { html, strategy } = await renderFn();
92
+ const effectiveStrategy = strategy ?? defaultStrategy;
93
+
94
+ if (effectiveStrategy === 'dynamic') {
95
+ return { html, status: 'miss', strategy: effectiveStrategy };
96
+ }
97
+
98
+ const newEntry = this.createEntry(html, effectiveStrategy);
99
+ await this.store.set(key, newEntry);
100
+ return { html, status: 'miss', strategy: effectiveStrategy };
101
+ }
102
+
103
+ if (!this.isStale(entry)) {
104
+ return { html: entry.html, status: 'hit', strategy: entry.strategy };
105
+ }
106
+
107
+ this.regenerateInBackground(key, entry.strategy, renderFn);
108
+ return { html: entry.html, status: 'stale', strategy: entry.strategy };
109
+ }
110
+
111
+ /**
112
+ * Regenerate content in the background without blocking the response.
113
+ * Uses promise deduplication to prevent multiple concurrent regenerations.
114
+ */
115
+ private regenerateInBackground(
116
+ key: string,
117
+ fallbackStrategy: CacheStrategy,
118
+ renderFn: () => Promise<{ html: string; strategy: CacheStrategy }>,
119
+ ): void {
120
+ if (this.regenerationPromises.has(key)) {
121
+ return;
122
+ }
123
+
124
+ const regeneratePromise = (async () => {
125
+ try {
126
+ const { html, strategy } = await renderFn();
127
+ const effectiveStrategy = strategy ?? fallbackStrategy;
128
+ const newEntry = this.createEntry(html, effectiveStrategy);
129
+ await this.store.set(key, newEntry);
130
+ return html;
131
+ } finally {
132
+ this.regenerationPromises.delete(key);
133
+ }
134
+ })();
135
+
136
+ this.regenerationPromises.set(key, regeneratePromise);
137
+
138
+ queueMicrotask(() => {
139
+ regeneratePromise.catch((error) => {
140
+ appLogger.error(`[PageCacheService] Failed to regenerate: ${key}`, error);
141
+ });
142
+ });
143
+ }
144
+
145
+ /**
146
+ * Invalidate cache entries by tags.
147
+ */
148
+ async invalidateByTags(tags: string[]): Promise<number> {
149
+ return this.store.invalidateByTags(tags);
150
+ }
151
+
152
+ /**
153
+ * Invalidate cache entries by paths.
154
+ */
155
+ async invalidateByPaths(paths: string[]): Promise<number> {
156
+ return this.store.invalidateByPaths(paths);
157
+ }
158
+
159
+ /**
160
+ * Clear all cached entries.
161
+ */
162
+ async clear(): Promise<void> {
163
+ return this.store.clear();
164
+ }
165
+
166
+ /**
167
+ * Get cache statistics.
168
+ */
169
+ async stats() {
170
+ return this.store.stats?.() ?? { entries: 0 };
171
+ }
172
+
173
+ /**
174
+ * Get the underlying cache store.
175
+ */
176
+ getStore(): CacheStore {
177
+ return this.store;
178
+ }
179
+ }
180
+
181
+ /**
182
+ * Generate Cache-Control header value from cache strategy.
183
+ */
184
+ export function getCacheControlHeader(strategy: CacheStrategy | 'disabled'): string {
185
+ if (strategy === 'disabled') {
186
+ return 'no-store, must-revalidate';
187
+ }
188
+ if (strategy === 'static') {
189
+ return 'public, max-age=31536000, immutable';
190
+ }
191
+
192
+ if (strategy === 'dynamic') {
193
+ return 'no-store, must-revalidate';
194
+ }
195
+
196
+ if (typeof strategy === 'object') {
197
+ const swr = strategy.revalidate * 2;
198
+ return `public, max-age=${strategy.revalidate}, stale-while-revalidate=${swr}`;
199
+ }
200
+
201
+ return 'no-store';
202
+ }
@@ -0,0 +1,79 @@
1
+ import { describe, expect, it, vi } from 'vitest';
2
+ import { PageRequestCacheCoordinator } from './page-request-cache-coordinator.service.ts';
3
+ import type { PageCacheService } from './page-cache-service.js';
4
+
5
+ describe('PageRequestCacheCoordinator', () => {
6
+ it('should build cache keys with query parameters', () => {
7
+ const service = new PageRequestCacheCoordinator(null, 'static');
8
+
9
+ expect(service.buildCacheKey({ pathname: '/blog' })).toBe('/blog');
10
+ expect(service.buildCacheKey({ pathname: '/blog', query: { page: '2', tag: 'eco' } })).toBe(
11
+ '/blog?page=2&tag=eco',
12
+ );
13
+ });
14
+
15
+ it('should bypass the cache service for dynamic pages', async () => {
16
+ const cacheService = {
17
+ getOrCreate: vi.fn(),
18
+ } as unknown as PageCacheService;
19
+ const renderFn = vi.fn(async () => ({
20
+ html: '<html><body>dynamic</body></html>',
21
+ strategy: 'dynamic' as const,
22
+ }));
23
+ const service = new PageRequestCacheCoordinator(cacheService, 'static');
24
+
25
+ const response = await service.render({
26
+ cacheKey: '/dynamic',
27
+ pageCacheStrategy: 'dynamic',
28
+ renderFn,
29
+ });
30
+
31
+ expect(await response.text()).toBe('<html><body>dynamic</body></html>');
32
+ expect(response.headers.get('X-Cache')).toBe('DISABLED');
33
+ expect(cacheService.getOrCreate).not.toHaveBeenCalled();
34
+ expect(renderFn).toHaveBeenCalledTimes(1);
35
+ });
36
+
37
+ it('should delegate cached rendering to the cache service', async () => {
38
+ const cacheService = {
39
+ getOrCreate: vi.fn(async () => ({
40
+ html: '<html><body>cached</body></html>',
41
+ strategy: { revalidate: 60 },
42
+ status: 'hit',
43
+ })),
44
+ } as unknown as PageCacheService;
45
+ const service = new PageRequestCacheCoordinator(cacheService, 'static');
46
+
47
+ const response = await service.render({
48
+ cacheKey: '/cached',
49
+ pageCacheStrategy: { revalidate: 60 },
50
+ renderFn: async () => ({
51
+ html: '<html><body>fresh</body></html>',
52
+ strategy: { revalidate: 60 },
53
+ }),
54
+ });
55
+
56
+ expect(await response.text()).toBe('<html><body>cached</body></html>');
57
+ expect(response.headers.get('X-Cache')).toBe('HIT');
58
+ expect(response.headers.get('Cache-Control')).toContain('max-age=60');
59
+ expect(cacheService.getOrCreate).toHaveBeenCalledWith('/cached', { revalidate: 60 }, expect.any(Function));
60
+ });
61
+
62
+ it('should normalize supported body types to strings', async () => {
63
+ const service = new PageRequestCacheCoordinator(null, 'static');
64
+
65
+ expect(await service.bodyToString('plain')).toBe('plain');
66
+ expect(await service.bodyToString(Buffer.from('buffered'))).toBe('buffered');
67
+ expect(await service.bodyToString(new Uint8Array([104, 101, 108, 108, 111]))).toBe('hello');
68
+ expect(
69
+ await service.bodyToString(
70
+ new ReadableStream({
71
+ start: (controller) => {
72
+ controller.enqueue(new TextEncoder().encode('streamed'));
73
+ controller.close();
74
+ },
75
+ }),
76
+ ),
77
+ ).toBe('streamed');
78
+ });
79
+ });
@@ -0,0 +1,131 @@
1
+ import { getCacheControlHeader, type PageCacheService } from './page-cache-service.ts';
2
+ import type { CacheStrategy, RenderResult } from './cache.types.ts';
3
+
4
+ type CacheStatus = 'hit' | 'miss' | 'stale' | 'expired' | 'disabled';
5
+
6
+ /**
7
+ * Coordinates request-time page caching concerns around one render invocation.
8
+ *
9
+ * This service keeps `FileSystemResponseMatcher` from owning low-level cache
10
+ * policy mechanics such as cache key construction, `dynamic` bypass behavior,
11
+ * body normalization for cache storage, and final cache header generation.
12
+ */
13
+ export class PageRequestCacheCoordinator {
14
+ private cacheService: PageCacheService | null;
15
+ private defaultCacheStrategy: CacheStrategy;
16
+
17
+ constructor(cacheService: PageCacheService | null, defaultCacheStrategy: CacheStrategy) {
18
+ this.cacheService = cacheService;
19
+ this.defaultCacheStrategy = defaultCacheStrategy;
20
+ }
21
+
22
+ /**
23
+ * Builds the cache key used for page lookups.
24
+ *
25
+ * Query parameters are part of the key so two requests that hit the same
26
+ * pathname but differ by search params do not share the same rendered entry.
27
+ *
28
+ * @param input Pathname plus optional query record.
29
+ * @returns Stable cache key for the request.
30
+ */
31
+ buildCacheKey(input: { pathname: string; query?: Record<string, string> }): string {
32
+ let key = input.pathname;
33
+ if (input.query && Object.keys(input.query).length > 0) {
34
+ const queryString = new URLSearchParams(input.query).toString();
35
+ key += `?${queryString}`;
36
+ }
37
+ return key;
38
+ }
39
+
40
+ /**
41
+ * Resolves a render request through the configured cache policy.
42
+ *
43
+ * Pages using `dynamic` rendering, or applications without a cache service,
44
+ * bypass cache lookup entirely and still receive the same response header
45
+ * contract as cached pages.
46
+ *
47
+ * @param options Cache coordination inputs for one page request.
48
+ * @returns HTTP response with cache headers applied.
49
+ */
50
+ async render(options: {
51
+ cacheKey: string;
52
+ pageCacheStrategy: CacheStrategy;
53
+ renderFn: () => Promise<RenderResult>;
54
+ }): Promise<Response> {
55
+ if (!this.cacheService || options.pageCacheStrategy === 'dynamic') {
56
+ const { html, strategy } = await options.renderFn();
57
+ return this.createCachedResponse(html, strategy, 'disabled');
58
+ }
59
+
60
+ const result = await this.cacheService.getOrCreate(
61
+ options.cacheKey,
62
+ options.pageCacheStrategy,
63
+ options.renderFn,
64
+ );
65
+
66
+ return this.createCachedResponse(result.html, result.strategy, result.status);
67
+ }
68
+
69
+ /**
70
+ * Exposes the underlying cache service for invalidation and adapter plumbing.
71
+ *
72
+ * @returns Configured cache service or `null` when caching is disabled.
73
+ */
74
+ getCacheService(): PageCacheService | null {
75
+ return this.cacheService;
76
+ }
77
+
78
+ /**
79
+ * Returns the default render strategy used when a page does not declare one.
80
+ *
81
+ * @returns Application-level fallback cache strategy.
82
+ */
83
+ getDefaultCacheStrategy(): CacheStrategy {
84
+ return this.defaultCacheStrategy;
85
+ }
86
+
87
+ /**
88
+ * Normalizes various route render body shapes into a cacheable string.
89
+ *
90
+ * Page rendering may produce strings, buffers, byte arrays, or streams. The
91
+ * matcher needs a single representation before passing HTML through the cache
92
+ * layer, so this method centralizes the conversion rules.
93
+ *
94
+ * @param body Render output body in any supported form.
95
+ * @returns HTML string representation.
96
+ */
97
+ async bodyToString(body: unknown): Promise<string> {
98
+ if (typeof body === 'string') {
99
+ return body;
100
+ }
101
+ if (Buffer.isBuffer(body)) {
102
+ return body.toString('utf-8');
103
+ }
104
+ if (body instanceof ReadableStream) {
105
+ return new Response(body).text();
106
+ }
107
+ if (body instanceof Uint8Array) {
108
+ return new TextDecoder().decode(body);
109
+ }
110
+ return String(body);
111
+ }
112
+
113
+ /**
114
+ * Creates the final HTML response with the current cache semantics encoded in
115
+ * response headers.
116
+ *
117
+ * @param html Rendered page HTML.
118
+ * @param strategy Effective cache strategy for the response.
119
+ * @param cacheStatus Status used for `X-Cache` and `Cache-Control` generation.
120
+ * @returns HTTP response ready to send to the client.
121
+ */
122
+ private createCachedResponse(html: string, strategy: CacheStrategy, cacheStatus: CacheStatus): Response {
123
+ const headers: HeadersInit = {
124
+ 'Content-Type': 'text/html',
125
+ 'Cache-Control': getCacheControlHeader(cacheStatus === 'disabled' ? 'disabled' : strategy),
126
+ 'X-Cache': cacheStatus.toUpperCase(),
127
+ };
128
+
129
+ return new Response(html, { headers });
130
+ }
131
+ }