@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.
- package/CHANGELOG.md +89 -0
- package/LICENSE +21 -0
- package/README.md +32 -0
- package/package.json +279 -0
- package/src/adapters/abstract/application-adapter.d.ts +168 -0
- package/src/adapters/abstract/application-adapter.js +109 -0
- package/src/adapters/abstract/application-adapter.ts +337 -0
- package/src/adapters/abstract/router-adapter.d.ts +26 -0
- package/src/adapters/abstract/router-adapter.js +5 -0
- package/src/adapters/abstract/router-adapter.ts +30 -0
- package/src/adapters/abstract/server-adapter.d.ts +69 -0
- package/src/adapters/abstract/server-adapter.js +15 -0
- package/src/adapters/abstract/server-adapter.ts +79 -0
- package/src/adapters/bun/client-bridge.d.ts +34 -0
- package/src/adapters/bun/client-bridge.js +48 -0
- package/src/adapters/bun/client-bridge.ts +62 -0
- package/src/adapters/bun/create-app.d.ts +60 -0
- package/src/adapters/bun/create-app.js +117 -0
- package/src/adapters/bun/create-app.ts +189 -0
- package/src/adapters/bun/define-api-handler.d.ts +61 -0
- package/src/adapters/bun/define-api-handler.js +15 -0
- package/src/adapters/bun/define-api-handler.ts +114 -0
- package/src/adapters/bun/hmr-manager.d.ts +84 -0
- package/src/adapters/bun/hmr-manager.js +227 -0
- package/src/adapters/bun/hmr-manager.ts +281 -0
- package/src/adapters/bun/index.d.ts +3 -0
- package/src/adapters/bun/index.js +8 -0
- package/src/adapters/bun/index.ts +3 -0
- package/src/adapters/bun/server-adapter.d.ts +155 -0
- package/src/adapters/bun/server-adapter.js +368 -0
- package/src/adapters/bun/server-adapter.ts +492 -0
- package/src/adapters/bun/server-lifecycle.d.ts +52 -0
- package/src/adapters/bun/server-lifecycle.js +120 -0
- package/src/adapters/bun/server-lifecycle.ts +154 -0
- package/src/adapters/index.d.ts +6 -0
- package/src/adapters/index.js +14 -0
- package/src/adapters/index.ts +6 -0
- package/src/adapters/node/create-app.d.ts +21 -0
- package/src/adapters/node/create-app.js +143 -0
- package/src/adapters/node/create-app.ts +179 -0
- package/src/adapters/node/index.d.ts +4 -0
- package/src/adapters/node/index.js +8 -0
- package/src/adapters/node/index.ts +9 -0
- package/src/adapters/node/node-client-bridge.d.ts +26 -0
- package/src/adapters/node/node-client-bridge.js +66 -0
- package/src/adapters/node/node-client-bridge.ts +79 -0
- package/src/adapters/node/node-hmr-manager.d.ts +62 -0
- package/src/adapters/node/node-hmr-manager.js +221 -0
- package/src/adapters/node/node-hmr-manager.ts +271 -0
- package/src/adapters/node/server-adapter.d.ts +190 -0
- package/src/adapters/node/server-adapter.js +420 -0
- package/src/adapters/node/server-adapter.ts +561 -0
- package/src/adapters/node/static-content-server.d.ts +24 -0
- package/src/adapters/node/static-content-server.js +166 -0
- package/src/adapters/node/static-content-server.ts +203 -0
- package/src/adapters/shared/api-response.d.ts +52 -0
- package/src/adapters/shared/api-response.js +96 -0
- package/src/adapters/shared/api-response.ts +104 -0
- package/src/adapters/shared/application-adapter.d.ts +18 -0
- package/src/adapters/shared/application-adapter.js +90 -0
- package/src/adapters/shared/application-adapter.ts +199 -0
- package/src/adapters/shared/explicit-static-route-matcher.d.ts +38 -0
- package/src/adapters/shared/explicit-static-route-matcher.js +100 -0
- package/src/adapters/shared/explicit-static-route-matcher.ts +134 -0
- package/src/adapters/shared/file-route-middleware-pipeline.d.ts +65 -0
- package/src/adapters/shared/file-route-middleware-pipeline.js +98 -0
- package/src/adapters/shared/file-route-middleware-pipeline.ts +123 -0
- package/src/adapters/shared/fs-server-response-factory.d.ts +19 -0
- package/src/adapters/shared/fs-server-response-factory.js +97 -0
- package/src/adapters/shared/fs-server-response-factory.ts +118 -0
- package/src/adapters/shared/fs-server-response-matcher.d.ts +71 -0
- package/src/adapters/shared/fs-server-response-matcher.js +155 -0
- package/src/adapters/shared/fs-server-response-matcher.ts +198 -0
- package/src/adapters/shared/render-context.d.ts +14 -0
- package/src/adapters/shared/render-context.js +69 -0
- package/src/adapters/shared/render-context.ts +105 -0
- package/src/adapters/shared/server-adapter.d.ts +87 -0
- package/src/adapters/shared/server-adapter.js +353 -0
- package/src/adapters/shared/server-adapter.ts +442 -0
- package/src/adapters/shared/server-route-handler.d.ts +89 -0
- package/src/adapters/shared/server-route-handler.js +120 -0
- package/src/adapters/shared/server-route-handler.ts +166 -0
- package/src/adapters/shared/server-static-builder.d.ts +38 -0
- package/src/adapters/shared/server-static-builder.js +46 -0
- package/src/adapters/shared/server-static-builder.ts +82 -0
- package/src/build/build-adapter.d.ts +74 -0
- package/src/build/build-adapter.js +54 -0
- package/src/build/build-adapter.ts +132 -0
- package/src/build/build-types.d.ts +57 -0
- package/src/build/build-types.js +0 -0
- package/src/build/build-types.ts +83 -0
- package/src/build/esbuild-build-adapter.d.ts +69 -0
- package/src/build/esbuild-build-adapter.js +390 -0
- package/src/build/esbuild-build-adapter.ts +510 -0
- package/src/config/config-builder.d.ts +227 -0
- package/src/config/config-builder.js +392 -0
- package/src/config/config-builder.ts +474 -0
- package/src/constants.d.ts +32 -0
- package/src/constants.js +21 -0
- package/src/constants.ts +39 -0
- package/src/create-app.d.ts +17 -0
- package/src/create-app.js +66 -0
- package/src/create-app.ts +87 -0
- package/src/declarations.d.ts +26 -0
- package/src/define-api-handler.d.ts +25 -0
- package/src/define-api-handler.js +15 -0
- package/src/define-api-handler.ts +66 -0
- package/src/dev/sc-server.d.ts +30 -0
- package/src/dev/sc-server.js +111 -0
- package/src/dev/sc-server.ts +143 -0
- package/src/eco/README.md +636 -0
- package/src/eco/component-render-context.d.ts +105 -0
- package/src/eco/component-render-context.js +77 -0
- package/src/eco/component-render-context.ts +202 -0
- package/src/eco/eco.d.ts +9 -0
- package/src/eco/eco.js +110 -0
- package/src/eco/eco.ts +221 -0
- package/src/eco/eco.types.d.ts +170 -0
- package/src/eco/eco.types.js +0 -0
- package/src/eco/eco.types.ts +202 -0
- package/src/eco/eco.utils.d.ts +40 -0
- package/src/eco/eco.utils.js +40 -0
- package/src/eco/eco.utils.ts +89 -0
- package/src/eco/global-injector-map.d.ts +16 -0
- package/src/eco/global-injector-map.js +80 -0
- package/src/eco/global-injector-map.ts +112 -0
- package/src/eco/lazy-injector-map.d.ts +8 -0
- package/src/eco/lazy-injector-map.js +70 -0
- package/src/eco/lazy-injector-map.ts +120 -0
- package/src/eco/module-dependencies.d.ts +18 -0
- package/src/eco/module-dependencies.js +49 -0
- package/src/eco/module-dependencies.ts +75 -0
- package/src/env.d.ts +20 -0
- package/src/errors/http-error.d.ts +31 -0
- package/src/errors/http-error.js +50 -0
- package/src/errors/http-error.ts +72 -0
- package/src/errors/index.d.ts +2 -0
- package/src/errors/index.js +4 -0
- package/src/errors/index.ts +2 -0
- package/src/errors/locals-access-error.d.ts +4 -0
- package/src/errors/locals-access-error.js +9 -0
- package/src/errors/locals-access-error.ts +7 -0
- package/src/global/app-logger.d.ts +2 -0
- package/src/global/app-logger.js +6 -0
- package/src/global/app-logger.ts +4 -0
- 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
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-HMR-Server-Integration-should-load-fixture-app-page-1.png +0 -0
- package/src/hmr/client/__screenshots__/hmr-runtime.test.browser.ts/HMR-Runtime-WebSocket-Connection-should-connect-to-correct-HMR-endpoint-1.png +0 -0
- package/src/hmr/client/hmr-runtime.d.ts +10 -0
- package/src/hmr/client/hmr-runtime.js +86 -0
- package/src/hmr/client/hmr-runtime.ts +121 -0
- package/src/hmr/hmr-strategy.d.ts +159 -0
- package/src/hmr/hmr-strategy.js +29 -0
- package/src/hmr/hmr-strategy.ts +172 -0
- package/src/hmr/hmr.test.e2e.d.ts +1 -0
- package/src/hmr/hmr.test.e2e.js +50 -0
- package/src/hmr/hmr.test.e2e.ts +75 -0
- package/src/hmr/strategies/default-hmr-strategy.d.ts +43 -0
- package/src/hmr/strategies/default-hmr-strategy.js +34 -0
- package/src/hmr/strategies/default-hmr-strategy.ts +60 -0
- package/src/hmr/strategies/js-hmr-strategy.d.ts +136 -0
- package/src/hmr/strategies/js-hmr-strategy.js +179 -0
- package/src/hmr/strategies/js-hmr-strategy.ts +308 -0
- package/src/index.browser.d.ts +3 -0
- package/src/index.browser.js +4 -0
- package/src/index.browser.ts +3 -0
- package/src/index.d.ts +5 -0
- package/src/index.js +10 -0
- package/src/index.ts +5 -0
- package/src/integrations/ghtml/ghtml-renderer.d.ts +15 -0
- package/src/integrations/ghtml/ghtml-renderer.js +60 -0
- package/src/integrations/ghtml/ghtml-renderer.ts +93 -0
- package/src/integrations/ghtml/ghtml.plugin.d.ts +20 -0
- package/src/integrations/ghtml/ghtml.plugin.js +21 -0
- package/src/integrations/ghtml/ghtml.plugin.ts +32 -0
- package/src/internal-types.d.ts +200 -0
- package/src/internal-types.js +0 -0
- package/src/internal-types.ts +212 -0
- package/src/plugins/alias-resolver-plugin.d.ts +2 -0
- package/src/plugins/alias-resolver-plugin.js +39 -0
- package/src/plugins/alias-resolver-plugin.ts +45 -0
- package/src/plugins/eco-component-meta-plugin.d.ts +95 -0
- package/src/plugins/eco-component-meta-plugin.js +157 -0
- package/src/plugins/eco-component-meta-plugin.ts +474 -0
- package/src/plugins/integration-plugin.d.ts +102 -0
- package/src/plugins/integration-plugin.js +100 -0
- package/src/plugins/integration-plugin.ts +184 -0
- package/src/plugins/processor.d.ts +82 -0
- package/src/plugins/processor.js +122 -0
- package/src/plugins/processor.ts +220 -0
- package/src/public-types.d.ts +1094 -0
- package/src/public-types.js +0 -0
- package/src/public-types.ts +1255 -0
- package/src/route-renderer/GRAPH.md +387 -0
- package/src/route-renderer/README.md +135 -0
- package/src/route-renderer/component-graph-executor.d.ts +32 -0
- package/src/route-renderer/component-graph-executor.js +31 -0
- package/src/route-renderer/component-graph-executor.ts +84 -0
- package/src/route-renderer/component-graph.d.ts +42 -0
- package/src/route-renderer/component-graph.js +72 -0
- package/src/route-renderer/component-graph.ts +159 -0
- package/src/route-renderer/component-marker.d.ts +52 -0
- package/src/route-renderer/component-marker.js +46 -0
- package/src/route-renderer/component-marker.ts +117 -0
- package/src/route-renderer/dependency-resolver.d.ts +24 -0
- package/src/route-renderer/dependency-resolver.js +428 -0
- package/src/route-renderer/dependency-resolver.ts +596 -0
- package/src/route-renderer/html-post-processing.service.d.ts +40 -0
- package/src/route-renderer/html-post-processing.service.js +86 -0
- package/src/route-renderer/html-post-processing.service.ts +103 -0
- package/src/route-renderer/integration-renderer.d.ts +339 -0
- package/src/route-renderer/integration-renderer.js +526 -0
- package/src/route-renderer/integration-renderer.ts +696 -0
- package/src/route-renderer/marker-graph-resolver.d.ts +76 -0
- package/src/route-renderer/marker-graph-resolver.js +93 -0
- package/src/route-renderer/marker-graph-resolver.ts +153 -0
- package/src/route-renderer/page-module-loader.d.ts +61 -0
- package/src/route-renderer/page-module-loader.js +102 -0
- package/src/route-renderer/page-module-loader.ts +153 -0
- package/src/route-renderer/render-execution.service.d.ts +69 -0
- package/src/route-renderer/render-execution.service.js +91 -0
- package/src/route-renderer/render-execution.service.ts +158 -0
- package/src/route-renderer/render-preparation.service.d.ts +112 -0
- package/src/route-renderer/render-preparation.service.js +243 -0
- package/src/route-renderer/render-preparation.service.ts +358 -0
- package/src/route-renderer/route-renderer.d.ts +26 -0
- package/src/route-renderer/route-renderer.js +68 -0
- package/src/route-renderer/route-renderer.ts +80 -0
- package/src/router/fs-router-scanner.d.ts +41 -0
- package/src/router/fs-router-scanner.js +155 -0
- package/src/router/fs-router-scanner.ts +217 -0
- package/src/router/fs-router.d.ts +26 -0
- package/src/router/fs-router.js +100 -0
- package/src/router/fs-router.ts +122 -0
- package/src/services/asset-processing-service/asset-processing.service.d.ts +41 -0
- package/src/services/asset-processing-service/asset-processing.service.js +250 -0
- package/src/services/asset-processing-service/asset-processing.service.ts +306 -0
- package/src/services/asset-processing-service/asset.factory.d.ts +17 -0
- package/src/services/asset-processing-service/asset.factory.js +82 -0
- package/src/services/asset-processing-service/asset.factory.ts +105 -0
- package/src/services/asset-processing-service/assets.types.d.ts +88 -0
- package/src/services/asset-processing-service/assets.types.js +0 -0
- package/src/services/asset-processing-service/assets.types.ts +112 -0
- package/src/services/asset-processing-service/index.d.ts +3 -0
- package/src/services/asset-processing-service/index.js +3 -0
- package/src/services/asset-processing-service/index.ts +3 -0
- package/src/services/asset-processing-service/processor.interface.d.ts +22 -0
- package/src/services/asset-processing-service/processor.interface.js +6 -0
- package/src/services/asset-processing-service/processor.interface.ts +27 -0
- package/src/services/asset-processing-service/processor.registry.d.ts +8 -0
- package/src/services/asset-processing-service/processor.registry.js +15 -0
- package/src/services/asset-processing-service/processor.registry.ts +18 -0
- package/src/services/asset-processing-service/processors/base/base-processor.d.ts +24 -0
- package/src/services/asset-processing-service/processors/base/base-processor.js +59 -0
- package/src/services/asset-processing-service/processors/base/base-processor.ts +76 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.d.ts +16 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.js +80 -0
- package/src/services/asset-processing-service/processors/base/base-script-processor.ts +105 -0
- package/src/services/asset-processing-service/processors/index.d.ts +5 -0
- package/src/services/asset-processing-service/processors/index.js +5 -0
- package/src/services/asset-processing-service/processors/index.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.js +57 -0
- package/src/services/asset-processing-service/processors/script/content-script.processor.ts +66 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.d.ts +8 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.js +76 -0
- package/src/services/asset-processing-service/processors/script/file-script.processor.ts +88 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.d.ts +7 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.js +74 -0
- package/src/services/asset-processing-service/processors/script/node-module-script.processor.ts +84 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.d.ts +5 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.js +25 -0
- package/src/services/asset-processing-service/processors/stylesheet/content-stylesheet.processor.ts +27 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.d.ts +9 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.js +63 -0
- package/src/services/asset-processing-service/processors/stylesheet/file-stylesheet.processor.ts +77 -0
- package/src/services/cache/cache.types.d.ts +107 -0
- package/src/services/cache/cache.types.js +0 -0
- package/src/services/cache/cache.types.ts +126 -0
- package/src/services/cache/index.d.ts +7 -0
- package/src/services/cache/index.js +7 -0
- package/src/services/cache/index.ts +18 -0
- package/src/services/cache/memory-cache-store.d.ts +42 -0
- package/src/services/cache/memory-cache-store.js +98 -0
- package/src/services/cache/memory-cache-store.ts +130 -0
- package/src/services/cache/page-cache-service.d.ts +70 -0
- package/src/services/cache/page-cache-service.js +152 -0
- package/src/services/cache/page-cache-service.ts +202 -0
- package/src/services/html-transformer.service.d.ts +50 -0
- package/src/services/html-transformer.service.js +163 -0
- package/src/services/html-transformer.service.ts +217 -0
- package/src/services/page-module-import.service.d.ts +37 -0
- package/src/services/page-module-import.service.js +88 -0
- package/src/services/page-module-import.service.ts +129 -0
- package/src/services/page-request-cache-coordinator.service.d.ts +75 -0
- package/src/services/page-request-cache-coordinator.service.js +107 -0
- package/src/services/page-request-cache-coordinator.service.ts +128 -0
- package/src/services/schema-validation-service.d.ts +122 -0
- package/src/services/schema-validation-service.js +101 -0
- package/src/services/schema-validation-service.ts +204 -0
- package/src/services/validation/standard-schema.types.d.ts +65 -0
- package/src/services/validation/standard-schema.types.js +0 -0
- package/src/services/validation/standard-schema.types.ts +68 -0
- package/src/static-site-generator/static-site-generator.d.ts +57 -0
- package/src/static-site-generator/static-site-generator.js +272 -0
- package/src/static-site-generator/static-site-generator.ts +359 -0
- package/src/utils/css.d.ts +1 -0
- package/src/utils/css.js +7 -0
- package/src/utils/css.ts +5 -0
- package/src/utils/deep-merge.d.ts +14 -0
- package/src/utils/deep-merge.js +32 -0
- package/src/utils/deep-merge.ts +47 -0
- package/src/utils/hash.d.ts +1 -0
- package/src/utils/hash.js +7 -0
- package/src/utils/hash.ts +5 -0
- package/src/utils/html.d.ts +1 -0
- package/src/utils/html.js +4 -0
- package/src/utils/html.ts +1 -0
- package/src/utils/invariant.d.ts +5 -0
- package/src/utils/invariant.js +11 -0
- package/src/utils/invariant.ts +15 -0
- package/src/utils/locals-utils.d.ts +15 -0
- package/src/utils/locals-utils.js +24 -0
- package/src/utils/locals-utils.ts +37 -0
- package/src/utils/parse-cli-args.d.ts +24 -0
- package/src/utils/parse-cli-args.js +47 -0
- package/src/utils/parse-cli-args.ts +83 -0
- package/src/utils/path-utils.module.d.ts +5 -0
- package/src/utils/path-utils.module.js +14 -0
- package/src/utils/path-utils.module.ts +14 -0
- package/src/utils/runtime.d.ts +11 -0
- package/src/utils/runtime.js +40 -0
- package/src/utils/runtime.ts +44 -0
- package/src/utils/server-utils.module.d.ts +19 -0
- package/src/utils/server-utils.module.js +56 -0
- package/src/utils/server-utils.module.ts +67 -0
- package/src/watchers/project-watcher.d.ts +120 -0
- package/src/watchers/project-watcher.js +238 -0
- package/src/watchers/project-watcher.test-helpers.d.ts +4 -0
- package/src/watchers/project-watcher.test-helpers.js +51 -0
- package/src/watchers/project-watcher.test-helpers.ts +40 -0
- package/src/watchers/project-watcher.ts +306 -0
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Page cache service with ISR (Incremental Static Regeneration) support.
|
|
3
|
+
* Handles stale-while-revalidate semantics and background regeneration.
|
|
4
|
+
* @module
|
|
5
|
+
*/
|
|
6
|
+
import type { CacheResult, CacheStore, CacheStrategy, RenderResult } from './cache.types.js';
|
|
7
|
+
export interface PageCacheServiceOptions {
|
|
8
|
+
store?: CacheStore;
|
|
9
|
+
enabled?: boolean;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Core page caching service with ISR support.
|
|
13
|
+
*/
|
|
14
|
+
export declare class PageCacheService {
|
|
15
|
+
private store;
|
|
16
|
+
private enabled;
|
|
17
|
+
private regenerationPromises;
|
|
18
|
+
constructor(options?: PageCacheServiceOptions);
|
|
19
|
+
/**
|
|
20
|
+
* Generate a cache key from URL and optional params.
|
|
21
|
+
* Uses full URL (path + query) as the key.
|
|
22
|
+
*/
|
|
23
|
+
generateCacheKey(url: string): string;
|
|
24
|
+
/**
|
|
25
|
+
* Check if an entry is stale (past its revalidation time).
|
|
26
|
+
*/
|
|
27
|
+
private isStale;
|
|
28
|
+
/**
|
|
29
|
+
* Create a cache entry from rendered HTML.
|
|
30
|
+
*/
|
|
31
|
+
private createEntry;
|
|
32
|
+
/**
|
|
33
|
+
* Get cached content or create new content with stale-while-revalidate semantics.
|
|
34
|
+
* @param key - Cache key (URL path + query)
|
|
35
|
+
* @param defaultStrategy - Default strategy if page doesn't specify one
|
|
36
|
+
* @param renderFn - Function that renders the page and returns HTML + strategy
|
|
37
|
+
*/
|
|
38
|
+
getOrCreate(key: string, defaultStrategy: CacheStrategy, renderFn: () => Promise<RenderResult>): Promise<CacheResult>;
|
|
39
|
+
/**
|
|
40
|
+
* Regenerate content in the background without blocking the response.
|
|
41
|
+
* Uses promise deduplication to prevent multiple concurrent regenerations.
|
|
42
|
+
*/
|
|
43
|
+
private regenerateInBackground;
|
|
44
|
+
/**
|
|
45
|
+
* Invalidate cache entries by tags.
|
|
46
|
+
*/
|
|
47
|
+
invalidateByTags(tags: string[]): Promise<number>;
|
|
48
|
+
/**
|
|
49
|
+
* Invalidate cache entries by paths.
|
|
50
|
+
*/
|
|
51
|
+
invalidateByPaths(paths: string[]): Promise<number>;
|
|
52
|
+
/**
|
|
53
|
+
* Clear all cached entries.
|
|
54
|
+
*/
|
|
55
|
+
clear(): Promise<void>;
|
|
56
|
+
/**
|
|
57
|
+
* Get cache statistics.
|
|
58
|
+
*/
|
|
59
|
+
stats(): Promise<import("./cache.types.js").CacheStats | {
|
|
60
|
+
entries: number;
|
|
61
|
+
}>;
|
|
62
|
+
/**
|
|
63
|
+
* Get the underlying cache store.
|
|
64
|
+
*/
|
|
65
|
+
getStore(): CacheStore;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Generate Cache-Control header value from cache strategy.
|
|
69
|
+
*/
|
|
70
|
+
export declare function getCacheControlHeader(strategy: CacheStrategy | 'disabled'): string;
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import { appLogger } from "../../global/app-logger.js";
|
|
2
|
+
import { MemoryCacheStore } from "./memory-cache-store.js";
|
|
3
|
+
class PageCacheService {
|
|
4
|
+
store;
|
|
5
|
+
enabled;
|
|
6
|
+
regenerationPromises = /* @__PURE__ */ new Map();
|
|
7
|
+
constructor(options = {}) {
|
|
8
|
+
this.store = options.store ?? new MemoryCacheStore();
|
|
9
|
+
this.enabled = options.enabled ?? true;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Generate a cache key from URL and optional params.
|
|
13
|
+
* Uses full URL (path + query) as the key.
|
|
14
|
+
*/
|
|
15
|
+
generateCacheKey(url) {
|
|
16
|
+
return url;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Check if an entry is stale (past its revalidation time).
|
|
20
|
+
*/
|
|
21
|
+
isStale(entry) {
|
|
22
|
+
if (entry.revalidateAfter === null) return false;
|
|
23
|
+
return Date.now() > entry.revalidateAfter;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Create a cache entry from rendered HTML.
|
|
27
|
+
*/
|
|
28
|
+
createEntry(html, strategy) {
|
|
29
|
+
const now = Date.now();
|
|
30
|
+
let revalidateAfter = null;
|
|
31
|
+
let tags = [];
|
|
32
|
+
if (strategy === "static") {
|
|
33
|
+
revalidateAfter = null;
|
|
34
|
+
} else if (strategy === "dynamic") {
|
|
35
|
+
revalidateAfter = 0;
|
|
36
|
+
} else if (typeof strategy === "object") {
|
|
37
|
+
revalidateAfter = now + strategy.revalidate * 1e3;
|
|
38
|
+
tags = strategy.tags ?? [];
|
|
39
|
+
}
|
|
40
|
+
return {
|
|
41
|
+
html,
|
|
42
|
+
createdAt: now,
|
|
43
|
+
revalidateAfter,
|
|
44
|
+
tags,
|
|
45
|
+
strategy
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Get cached content or create new content with stale-while-revalidate semantics.
|
|
50
|
+
* @param key - Cache key (URL path + query)
|
|
51
|
+
* @param defaultStrategy - Default strategy if page doesn't specify one
|
|
52
|
+
* @param renderFn - Function that renders the page and returns HTML + strategy
|
|
53
|
+
*/
|
|
54
|
+
async getOrCreate(key, defaultStrategy, renderFn) {
|
|
55
|
+
if (!this.enabled) {
|
|
56
|
+
const { html, strategy } = await renderFn();
|
|
57
|
+
return { html, status: "miss", strategy };
|
|
58
|
+
}
|
|
59
|
+
const entry = await this.store.get(key);
|
|
60
|
+
if (!entry) {
|
|
61
|
+
const { html, strategy } = await renderFn();
|
|
62
|
+
const effectiveStrategy = strategy ?? defaultStrategy;
|
|
63
|
+
if (effectiveStrategy === "dynamic") {
|
|
64
|
+
return { html, status: "miss", strategy: effectiveStrategy };
|
|
65
|
+
}
|
|
66
|
+
const newEntry = this.createEntry(html, effectiveStrategy);
|
|
67
|
+
await this.store.set(key, newEntry);
|
|
68
|
+
return { html, status: "miss", strategy: effectiveStrategy };
|
|
69
|
+
}
|
|
70
|
+
if (!this.isStale(entry)) {
|
|
71
|
+
return { html: entry.html, status: "hit", strategy: entry.strategy };
|
|
72
|
+
}
|
|
73
|
+
this.regenerateInBackground(key, entry.strategy, renderFn);
|
|
74
|
+
return { html: entry.html, status: "stale", strategy: entry.strategy };
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Regenerate content in the background without blocking the response.
|
|
78
|
+
* Uses promise deduplication to prevent multiple concurrent regenerations.
|
|
79
|
+
*/
|
|
80
|
+
regenerateInBackground(key, fallbackStrategy, renderFn) {
|
|
81
|
+
if (this.regenerationPromises.has(key)) {
|
|
82
|
+
return;
|
|
83
|
+
}
|
|
84
|
+
const regeneratePromise = (async () => {
|
|
85
|
+
try {
|
|
86
|
+
const { html, strategy } = await renderFn();
|
|
87
|
+
const effectiveStrategy = strategy ?? fallbackStrategy;
|
|
88
|
+
const newEntry = this.createEntry(html, effectiveStrategy);
|
|
89
|
+
await this.store.set(key, newEntry);
|
|
90
|
+
return html;
|
|
91
|
+
} finally {
|
|
92
|
+
this.regenerationPromises.delete(key);
|
|
93
|
+
}
|
|
94
|
+
})();
|
|
95
|
+
this.regenerationPromises.set(key, regeneratePromise);
|
|
96
|
+
queueMicrotask(() => {
|
|
97
|
+
regeneratePromise.catch((error) => {
|
|
98
|
+
appLogger.error(`[PageCacheService] Failed to regenerate: ${key}`, error);
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Invalidate cache entries by tags.
|
|
104
|
+
*/
|
|
105
|
+
async invalidateByTags(tags) {
|
|
106
|
+
return this.store.invalidateByTags(tags);
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* Invalidate cache entries by paths.
|
|
110
|
+
*/
|
|
111
|
+
async invalidateByPaths(paths) {
|
|
112
|
+
return this.store.invalidateByPaths(paths);
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Clear all cached entries.
|
|
116
|
+
*/
|
|
117
|
+
async clear() {
|
|
118
|
+
return this.store.clear();
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Get cache statistics.
|
|
122
|
+
*/
|
|
123
|
+
async stats() {
|
|
124
|
+
return this.store.stats?.() ?? { entries: 0 };
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Get the underlying cache store.
|
|
128
|
+
*/
|
|
129
|
+
getStore() {
|
|
130
|
+
return this.store;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
function getCacheControlHeader(strategy) {
|
|
134
|
+
if (strategy === "disabled") {
|
|
135
|
+
return "no-store, must-revalidate";
|
|
136
|
+
}
|
|
137
|
+
if (strategy === "static") {
|
|
138
|
+
return "public, max-age=31536000, immutable";
|
|
139
|
+
}
|
|
140
|
+
if (strategy === "dynamic") {
|
|
141
|
+
return "no-store, must-revalidate";
|
|
142
|
+
}
|
|
143
|
+
if (typeof strategy === "object") {
|
|
144
|
+
const swr = strategy.revalidate * 2;
|
|
145
|
+
return `public, max-age=${strategy.revalidate}, stale-while-revalidate=${swr}`;
|
|
146
|
+
}
|
|
147
|
+
return "no-store";
|
|
148
|
+
}
|
|
149
|
+
export {
|
|
150
|
+
PageCacheService,
|
|
151
|
+
getCacheControlHeader
|
|
152
|
+
};
|
|
@@ -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,50 @@
|
|
|
1
|
+
import type { ProcessedAsset } from './asset-processing-service/assets.types';
|
|
2
|
+
export type HtmlRewriterMode = 'auto' | 'native' | 'worker-tools' | 'fallback';
|
|
3
|
+
export interface HtmlTransformerServiceOptions {
|
|
4
|
+
htmlRewriterMode?: HtmlRewriterMode;
|
|
5
|
+
}
|
|
6
|
+
export declare class HtmlTransformerService {
|
|
7
|
+
private processedDependencies;
|
|
8
|
+
private htmlRewriterConstructorPromise?;
|
|
9
|
+
private htmlRewriterMode;
|
|
10
|
+
constructor(options?: HtmlTransformerServiceOptions);
|
|
11
|
+
/**
|
|
12
|
+
* Overrides the HTML rewriter runtime selection.
|
|
13
|
+
*
|
|
14
|
+
* This is intended for internal/runtime tests that need deterministic
|
|
15
|
+
* selection between native, worker-tools, and string fallback behavior.
|
|
16
|
+
*
|
|
17
|
+
* @param mode Requested runtime selection strategy.
|
|
18
|
+
*/
|
|
19
|
+
setHtmlRewriterMode(mode: HtmlRewriterMode): void;
|
|
20
|
+
/**
|
|
21
|
+
* Creates an HTML rewriter instance from the best available runtime.
|
|
22
|
+
*
|
|
23
|
+
* Resolution order is:
|
|
24
|
+
* 1. native `globalThis.HTMLRewriter`
|
|
25
|
+
* 2. `@worker-tools/html-rewriter/base64`
|
|
26
|
+
* 3. `null`, which triggers the string-based fallback path
|
|
27
|
+
*
|
|
28
|
+
* @returns HTML rewriter instance when available; otherwise `null`.
|
|
29
|
+
*/
|
|
30
|
+
private createHtmlRewriter;
|
|
31
|
+
/**
|
|
32
|
+
* Resolves the constructor used for HTML rewriting.
|
|
33
|
+
*
|
|
34
|
+
* The worker-tools fallback is loaded lazily so native runtimes avoid the WASM
|
|
35
|
+
* dependency cost unless it is actually needed.
|
|
36
|
+
*
|
|
37
|
+
* @returns Rewriter constructor when available; otherwise `null`.
|
|
38
|
+
*/
|
|
39
|
+
private resolveHtmlRewriterConstructor;
|
|
40
|
+
private formatAttributes;
|
|
41
|
+
private generateScriptTag;
|
|
42
|
+
private generateStylesheetTag;
|
|
43
|
+
private appendDependencies;
|
|
44
|
+
private buildDependencyTags;
|
|
45
|
+
private injectBeforeClosingTag;
|
|
46
|
+
setProcessedDependencies(processedDependencies: ProcessedAsset[]): void;
|
|
47
|
+
getProcessedDependencies(): ProcessedAsset[];
|
|
48
|
+
transform(res: Response): Promise<Response>;
|
|
49
|
+
private groupDependenciesByPosition;
|
|
50
|
+
}
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
import { appLogger } from "../global/app-logger.js";
|
|
2
|
+
class HtmlTransformerService {
|
|
3
|
+
processedDependencies = [];
|
|
4
|
+
htmlRewriterConstructorPromise;
|
|
5
|
+
htmlRewriterMode = "auto";
|
|
6
|
+
constructor(options = {}) {
|
|
7
|
+
this.setHtmlRewriterMode(options.htmlRewriterMode ?? "auto");
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Overrides the HTML rewriter runtime selection.
|
|
11
|
+
*
|
|
12
|
+
* This is intended for internal/runtime tests that need deterministic
|
|
13
|
+
* selection between native, worker-tools, and string fallback behavior.
|
|
14
|
+
*
|
|
15
|
+
* @param mode Requested runtime selection strategy.
|
|
16
|
+
*/
|
|
17
|
+
setHtmlRewriterMode(mode) {
|
|
18
|
+
this.htmlRewriterMode = mode;
|
|
19
|
+
this.htmlRewriterConstructorPromise = void 0;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Creates an HTML rewriter instance from the best available runtime.
|
|
23
|
+
*
|
|
24
|
+
* Resolution order is:
|
|
25
|
+
* 1. native `globalThis.HTMLRewriter`
|
|
26
|
+
* 2. `@worker-tools/html-rewriter/base64`
|
|
27
|
+
* 3. `null`, which triggers the string-based fallback path
|
|
28
|
+
*
|
|
29
|
+
* @returns HTML rewriter instance when available; otherwise `null`.
|
|
30
|
+
*/
|
|
31
|
+
async createHtmlRewriter() {
|
|
32
|
+
const RuntimeHtmlRewriter = await this.resolveHtmlRewriterConstructor();
|
|
33
|
+
return RuntimeHtmlRewriter ? new RuntimeHtmlRewriter() : null;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Resolves the constructor used for HTML rewriting.
|
|
37
|
+
*
|
|
38
|
+
* The worker-tools fallback is loaded lazily so native runtimes avoid the WASM
|
|
39
|
+
* dependency cost unless it is actually needed.
|
|
40
|
+
*
|
|
41
|
+
* @returns Rewriter constructor when available; otherwise `null`.
|
|
42
|
+
*/
|
|
43
|
+
async resolveHtmlRewriterConstructor() {
|
|
44
|
+
if (!this.htmlRewriterConstructorPromise) {
|
|
45
|
+
const mode = this.htmlRewriterMode;
|
|
46
|
+
const RuntimeHtmlRewriter = globalThis.HTMLRewriter;
|
|
47
|
+
if (mode === "fallback") {
|
|
48
|
+
this.htmlRewriterConstructorPromise = Promise.resolve(null);
|
|
49
|
+
} else if (mode === "native") {
|
|
50
|
+
if (RuntimeHtmlRewriter) {
|
|
51
|
+
this.htmlRewriterConstructorPromise = Promise.resolve(RuntimeHtmlRewriter);
|
|
52
|
+
} else {
|
|
53
|
+
appLogger.warn(
|
|
54
|
+
"[HtmlTransformerService] Native HTMLRewriter was forced but is unavailable, falling back to string injection."
|
|
55
|
+
);
|
|
56
|
+
this.htmlRewriterConstructorPromise = Promise.resolve(null);
|
|
57
|
+
}
|
|
58
|
+
} else if (mode === "auto" && RuntimeHtmlRewriter) {
|
|
59
|
+
this.htmlRewriterConstructorPromise = Promise.resolve(RuntimeHtmlRewriter);
|
|
60
|
+
} else {
|
|
61
|
+
this.htmlRewriterConstructorPromise = import("@worker-tools/html-rewriter/base64").then((module) => module.HTMLRewriter).catch((error) => {
|
|
62
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
63
|
+
appLogger.warn(
|
|
64
|
+
`[HtmlTransformerService] Failed to load @worker-tools/html-rewriter/base64, falling back to string injection: ${message}`
|
|
65
|
+
);
|
|
66
|
+
return null;
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
return this.htmlRewriterConstructorPromise;
|
|
71
|
+
}
|
|
72
|
+
formatAttributes(attrs) {
|
|
73
|
+
if (!attrs) return "";
|
|
74
|
+
return ` ${Object.entries(attrs).map(([key, value]) => `${key}="${value}"`).join(" ")}`;
|
|
75
|
+
}
|
|
76
|
+
generateScriptTag(dep) {
|
|
77
|
+
return dep.inline ? `<script${this.formatAttributes(dep.attributes)}>${dep.content}<\/script>` : `<script src="${dep.srcUrl}"${this.formatAttributes(dep.attributes)}><\/script>`;
|
|
78
|
+
}
|
|
79
|
+
generateStylesheetTag(dep) {
|
|
80
|
+
return dep.inline ? `<style${this.formatAttributes(dep.attributes)}>${dep.content}</style>` : `<link rel="stylesheet" href="${dep.srcUrl}"${this.formatAttributes(dep.attributes)}>`;
|
|
81
|
+
}
|
|
82
|
+
appendDependencies(element, dependencies) {
|
|
83
|
+
for (const dep of dependencies) {
|
|
84
|
+
const tag = dep.kind === "script" ? this.generateScriptTag(dep) : this.generateStylesheetTag(dep);
|
|
85
|
+
element.append(tag, { html: true });
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
buildDependencyTags(dependencies) {
|
|
89
|
+
return dependencies.map(
|
|
90
|
+
(dep) => dep.kind === "script" ? this.generateScriptTag(dep) : this.generateStylesheetTag(dep)
|
|
91
|
+
).join("");
|
|
92
|
+
}
|
|
93
|
+
injectBeforeClosingTag(html, tag, content) {
|
|
94
|
+
if (!content) {
|
|
95
|
+
return html;
|
|
96
|
+
}
|
|
97
|
+
const closingTag = `</${tag}>`;
|
|
98
|
+
const lowerHtml = html.toLowerCase();
|
|
99
|
+
const closingTagIndex = lowerHtml.lastIndexOf(closingTag);
|
|
100
|
+
if (closingTagIndex !== -1) {
|
|
101
|
+
return `${html.slice(0, closingTagIndex)}${content}${html.slice(closingTagIndex)}`;
|
|
102
|
+
}
|
|
103
|
+
if (tag === "head") {
|
|
104
|
+
return `${content}${html}`;
|
|
105
|
+
}
|
|
106
|
+
return `${html}${content}`;
|
|
107
|
+
}
|
|
108
|
+
setProcessedDependencies(processedDependencies) {
|
|
109
|
+
this.processedDependencies = processedDependencies;
|
|
110
|
+
}
|
|
111
|
+
getProcessedDependencies() {
|
|
112
|
+
return this.processedDependencies;
|
|
113
|
+
}
|
|
114
|
+
async transform(res) {
|
|
115
|
+
const { head, body } = this.groupDependenciesByPosition();
|
|
116
|
+
const htmlRewriter = await this.createHtmlRewriter();
|
|
117
|
+
const html = await res.text();
|
|
118
|
+
const headers = new Headers(res.headers);
|
|
119
|
+
if (htmlRewriter) {
|
|
120
|
+
htmlRewriter.on("head", {
|
|
121
|
+
element: (element) => this.appendDependencies(element, head)
|
|
122
|
+
}).on("body", {
|
|
123
|
+
element: (element) => this.appendDependencies(element, body)
|
|
124
|
+
});
|
|
125
|
+
return htmlRewriter.transform(
|
|
126
|
+
new Response(html, {
|
|
127
|
+
headers,
|
|
128
|
+
status: res.status,
|
|
129
|
+
statusText: res.statusText
|
|
130
|
+
})
|
|
131
|
+
);
|
|
132
|
+
}
|
|
133
|
+
const withHeadDependencies = this.injectBeforeClosingTag(html, "head", this.buildDependencyTags(head));
|
|
134
|
+
const transformedHtml = this.injectBeforeClosingTag(
|
|
135
|
+
withHeadDependencies,
|
|
136
|
+
"body",
|
|
137
|
+
this.buildDependencyTags(body)
|
|
138
|
+
);
|
|
139
|
+
return new Response(transformedHtml, {
|
|
140
|
+
headers,
|
|
141
|
+
status: res.status,
|
|
142
|
+
statusText: res.statusText
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
groupDependenciesByPosition() {
|
|
146
|
+
return this.processedDependencies.reduce(
|
|
147
|
+
(acc, dep) => {
|
|
148
|
+
if (dep.kind === "script") {
|
|
149
|
+
if (dep.excludeFromHtml) return acc;
|
|
150
|
+
const position = dep.position || "body";
|
|
151
|
+
acc[position].push(dep);
|
|
152
|
+
} else if (dep.kind === "stylesheet") {
|
|
153
|
+
acc.head.push(dep);
|
|
154
|
+
}
|
|
155
|
+
return acc;
|
|
156
|
+
},
|
|
157
|
+
{ head: [], body: [] }
|
|
158
|
+
);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
export {
|
|
162
|
+
HtmlTransformerService
|
|
163
|
+
};
|