@editframe/elements 0.17.6-beta.0 → 0.18.3-beta.0
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/dist/EF_FRAMEGEN.js +1 -1
- package/dist/ScrubTrackManager.d.ts +2 -2
- package/dist/elements/EFAudio.d.ts +21 -2
- package/dist/elements/EFAudio.js +41 -11
- package/dist/elements/EFImage.d.ts +1 -0
- package/dist/elements/EFImage.js +11 -3
- package/dist/elements/EFMedia/AssetIdMediaEngine.d.ts +18 -0
- package/dist/elements/EFMedia/AssetIdMediaEngine.js +41 -0
- package/dist/elements/EFMedia/AssetMediaEngine.d.ts +47 -0
- package/dist/elements/EFMedia/AssetMediaEngine.js +116 -0
- package/dist/elements/EFMedia/BaseMediaEngine.d.ts +55 -0
- package/dist/elements/EFMedia/BaseMediaEngine.js +96 -0
- package/dist/elements/EFMedia/BufferedSeekingInput.d.ts +43 -0
- package/dist/elements/EFMedia/BufferedSeekingInput.js +159 -0
- package/dist/elements/EFMedia/JitMediaEngine.browsertest.d.ts +0 -0
- package/dist/elements/EFMedia/JitMediaEngine.d.ts +31 -0
- package/dist/elements/EFMedia/JitMediaEngine.js +62 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.d.ts +16 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.js +48 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.d.ts +3 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.js +138 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.d.ts +4 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.js +16 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.d.ts +3 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.js +22 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.d.ts +7 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.js +24 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.d.ts +4 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.js +18 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.d.ts +4 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.js +16 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.d.ts +3 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.js +104 -0
- package/dist/elements/EFMedia/services/AudioElementFactory.d.ts +22 -0
- package/dist/elements/EFMedia/services/AudioElementFactory.js +72 -0
- package/dist/elements/EFMedia/services/MediaSourceService.browsertest.d.ts +1 -0
- package/dist/elements/EFMedia/services/MediaSourceService.d.ts +47 -0
- package/dist/elements/EFMedia/services/MediaSourceService.js +73 -0
- package/dist/elements/EFMedia/shared/AudioSpanUtils.d.ts +7 -0
- package/dist/elements/EFMedia/shared/AudioSpanUtils.js +54 -0
- package/dist/elements/EFMedia/shared/BufferUtils.d.ts +70 -0
- package/dist/elements/EFMedia/shared/BufferUtils.js +89 -0
- package/dist/elements/EFMedia/shared/MediaTaskUtils.d.ts +23 -0
- package/dist/elements/EFMedia/shared/RenditionHelpers.browsertest.d.ts +1 -0
- package/dist/elements/EFMedia/shared/RenditionHelpers.d.ts +19 -0
- package/dist/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.d.ts +1 -0
- package/dist/elements/EFMedia/tasks/makeMediaEngineTask.d.ts +18 -0
- package/dist/elements/EFMedia/tasks/makeMediaEngineTask.js +60 -0
- package/dist/elements/EFMedia/tasks/makeMediaEngineTask.test.d.ts +1 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.d.ts +16 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.js +46 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.d.ts +4 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.js +16 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.d.ts +3 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.js +27 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.d.ts +7 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.js +25 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.d.ts +4 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.js +18 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.d.ts +9 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.d.ts +4 -0
- package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.js +16 -0
- package/dist/elements/EFMedia.browsertest.d.ts +1 -0
- package/dist/elements/EFMedia.d.ts +75 -111
- package/dist/elements/EFMedia.js +141 -1111
- package/dist/elements/EFTemporal.d.ts +1 -1
- package/dist/elements/EFTemporal.js +1 -1
- package/dist/elements/EFTimegroup.d.ts +11 -0
- package/dist/elements/EFTimegroup.js +88 -13
- package/dist/elements/EFVideo.d.ts +60 -29
- package/dist/elements/EFVideo.js +103 -203
- package/dist/elements/EFWaveform.js +2 -2
- package/dist/elements/SampleBuffer.d.ts +14 -0
- package/dist/elements/SampleBuffer.js +52 -0
- package/dist/getRenderInfo.d.ts +2 -2
- package/dist/getRenderInfo.js +2 -1
- package/dist/gui/ContextMixin.js +17 -70
- package/dist/gui/EFFilmstrip.d.ts +3 -3
- package/dist/gui/EFFilmstrip.js +1 -1
- package/dist/gui/EFFitScale.d.ts +2 -2
- package/dist/gui/TWMixin.js +1 -1
- package/dist/gui/services/ElementConnectionManager.browsertest.d.ts +1 -0
- package/dist/gui/services/ElementConnectionManager.d.ts +59 -0
- package/dist/gui/services/ElementConnectionManager.js +128 -0
- package/dist/gui/services/PlaybackController.browsertest.d.ts +1 -0
- package/dist/gui/services/PlaybackController.d.ts +103 -0
- package/dist/gui/services/PlaybackController.js +290 -0
- package/dist/services/MediaSourceManager.d.ts +62 -0
- package/dist/services/MediaSourceManager.js +211 -0
- package/dist/style.css +1 -1
- package/dist/transcoding/cache/CacheManager.d.ts +73 -0
- package/dist/transcoding/cache/RequestDeduplicator.d.ts +29 -0
- package/dist/transcoding/cache/RequestDeduplicator.js +53 -0
- package/dist/transcoding/cache/RequestDeduplicator.test.d.ts +1 -0
- package/dist/transcoding/types/index.d.ts +242 -0
- package/dist/transcoding/utils/MediaUtils.d.ts +9 -0
- package/dist/transcoding/utils/UrlGenerator.d.ts +26 -0
- package/dist/transcoding/utils/UrlGenerator.js +45 -0
- package/dist/transcoding/utils/constants.d.ts +27 -0
- package/dist/utils/LRUCache.d.ts +34 -0
- package/dist/utils/LRUCache.js +115 -0
- package/package.json +3 -2
- package/src/elements/EFAudio.browsertest.ts +183 -43
- package/src/elements/EFAudio.ts +59 -13
- package/src/elements/EFImage.browsertest.ts +42 -0
- package/src/elements/EFImage.ts +23 -3
- package/src/elements/EFMedia/AssetIdMediaEngine.test.ts +222 -0
- package/src/elements/EFMedia/AssetIdMediaEngine.ts +70 -0
- package/src/elements/EFMedia/AssetMediaEngine.ts +210 -0
- package/src/elements/EFMedia/BaseMediaEngine.test.ts +164 -0
- package/src/elements/EFMedia/BaseMediaEngine.ts +170 -0
- package/src/elements/EFMedia/BufferedSeekingInput.browsertest.ts +400 -0
- package/src/elements/EFMedia/BufferedSeekingInput.ts +267 -0
- package/src/elements/EFMedia/JitMediaEngine.browsertest.ts +165 -0
- package/src/elements/EFMedia/JitMediaEngine.ts +110 -0
- package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.ts +554 -0
- package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.ts +81 -0
- package/src/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.ts +241 -0
- package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.ts +59 -0
- package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.ts +23 -0
- package/src/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.ts +55 -0
- package/src/elements/EFMedia/audioTasks/makeAudioInputTask.ts +35 -0
- package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.ts +42 -0
- package/src/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.ts +34 -0
- package/src/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.ts +23 -0
- package/src/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.ts +174 -0
- package/src/elements/EFMedia/services/AudioElementFactory.browsertest.ts +325 -0
- package/src/elements/EFMedia/services/AudioElementFactory.ts +119 -0
- package/src/elements/EFMedia/services/MediaSourceService.browsertest.ts +257 -0
- package/src/elements/EFMedia/services/MediaSourceService.ts +102 -0
- package/src/elements/EFMedia/shared/AudioSpanUtils.ts +128 -0
- package/src/elements/EFMedia/shared/BufferUtils.ts +310 -0
- package/src/elements/EFMedia/shared/MediaTaskUtils.ts +44 -0
- package/src/elements/EFMedia/shared/RenditionHelpers.browsertest.ts +247 -0
- package/src/elements/EFMedia/shared/RenditionHelpers.ts +79 -0
- package/src/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.ts +128 -0
- package/src/elements/EFMedia/tasks/makeMediaEngineTask.test.ts +233 -0
- package/src/elements/EFMedia/tasks/makeMediaEngineTask.ts +89 -0
- package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.ts +555 -0
- package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.ts +79 -0
- package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.ts +59 -0
- package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.ts +23 -0
- package/src/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.ts +55 -0
- package/src/elements/EFMedia/videoTasks/makeVideoInputTask.ts +45 -0
- package/src/elements/EFMedia/videoTasks/makeVideoSeekTask.ts +44 -0
- package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.ts +57 -0
- package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.ts +32 -0
- package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.ts +56 -0
- package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.ts +23 -0
- package/src/elements/EFMedia.browsertest.ts +658 -265
- package/src/elements/EFMedia.ts +173 -1763
- package/src/elements/EFTemporal.ts +3 -4
- package/src/elements/EFTimegroup.browsertest.ts +6 -3
- package/src/elements/EFTimegroup.ts +152 -21
- package/src/elements/EFVideo.browsertest.ts +115 -37
- package/src/elements/EFVideo.ts +123 -452
- package/src/elements/EFWaveform.ts +1 -1
- package/src/elements/MediaController.ts +2 -12
- package/src/elements/SampleBuffer.ts +97 -0
- package/src/gui/ContextMixin.ts +23 -104
- package/src/gui/services/ElementConnectionManager.browsertest.ts +263 -0
- package/src/gui/services/ElementConnectionManager.ts +224 -0
- package/src/gui/services/PlaybackController.browsertest.ts +437 -0
- package/src/gui/services/PlaybackController.ts +521 -0
- package/src/services/MediaSourceManager.ts +333 -0
- package/src/transcoding/cache/CacheManager.ts +208 -0
- package/src/transcoding/cache/RequestDeduplicator.test.ts +170 -0
- package/src/transcoding/cache/RequestDeduplicator.ts +65 -0
- package/src/transcoding/types/index.ts +265 -0
- package/src/transcoding/utils/MediaUtils.ts +63 -0
- package/src/transcoding/utils/UrlGenerator.ts +68 -0
- package/src/transcoding/utils/constants.ts +36 -0
- package/src/utils/LRUCache.ts +153 -0
- package/test/EFVideo.framegen.browsertest.ts +38 -29
- package/test/__cache__/GET__api_v1_transcode_audio_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__32da3954ba60c96ad732020c65a08ebc/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__32da3954ba60c96ad732020c65a08ebc/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_audio_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__b0b2b07efcf607de8ee0f650328c32f7/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__b0b2b07efcf607de8ee0f650328c32f7/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_audio_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a75c2252b542e0c152c780e9a8d7b154/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a75c2252b542e0c152c780e9a8d7b154/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_audio_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a64ff1cfb1b52cae14df4b5dfa1e222b/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a64ff1cfb1b52cae14df4b5dfa1e222b/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_audio_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__91e8a522f950809b9f09f4173113b4b0/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__91e8a522f950809b9f09f4173113b4b0/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_audio_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__e66d2c831d951e74ad0aeaa6489795d0/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_audio_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__e66d2c831d951e74ad0aeaa6489795d0/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_high_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__26197f6f7c46cacb0a71134131c3f775/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_high_1_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__26197f6f7c46cacb0a71134131c3f775/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_high_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__4cb6774cd3650ccf59c8f8dc6678c0b9/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_high_2_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__4cb6774cd3650ccf59c8f8dc6678c0b9/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_high_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0b3b2b1c8933f7fcf8a9ecaa88d58b41/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_high_3_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0b3b2b1c8933f7fcf8a9ecaa88d58b41/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_high_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0798c479b44aaeef850609a430f6e613/data.bin +0 -0
- package/test/__cache__/GET__api_v1_transcode_high_init_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__0798c479b44aaeef850609a430f6e613/metadata.json +21 -0
- package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/data.bin +1 -0
- package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/metadata.json +19 -0
- package/test/createJitTestClips.ts +320 -188
- package/test/recordReplayProxyPlugin.js +302 -0
- package/test/useAssetMSW.ts +1 -1
- package/test/useMSW.ts +35 -22
- package/types.json +1 -1
- package/dist/JitTranscodingClient.d.ts +0 -167
- package/dist/JitTranscodingClient.js +0 -373
- package/dist/ScrubTrackManager.js +0 -216
- package/dist/elements/printTaskStatus.js +0 -11
- package/src/elements/__screenshots__/EFMedia.browsertest.ts/EFMedia-JIT-audio-playback-audioBufferTask-should-work-in-JIT-mode-without-URL-errors-1.png +0 -0
- package/test/EFVideo.frame-tasks.browsertest.ts +0 -524
- /package/dist/{JitTranscodingClient.browsertest.d.ts → elements/EFMedia/AssetIdMediaEngine.test.d.ts} +0 -0
- /package/dist/{JitTranscodingClient.test.d.ts → elements/EFMedia/BaseMediaEngine.test.d.ts} +0 -0
- /package/dist/{ScrubTrackIntegration.test.d.ts → elements/EFMedia/BufferedSeekingInput.browsertest.d.ts} +0 -0
- /package/dist/{SegmentSwitchLoading.test.d.ts → elements/EFMedia/services/AudioElementFactory.browsertest.d.ts} +0 -0
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Request deduplication utility
|
|
3
|
+
* Manages pending requests to prevent concurrent duplicate requests
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export class RequestDeduplicator {
|
|
7
|
+
private pendingRequests = new Map<string, Promise<any>>();
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Execute a request with deduplication
|
|
11
|
+
* If a request with the same key is already pending, return the existing promise
|
|
12
|
+
* Otherwise, execute the request factory and track the promise
|
|
13
|
+
*/
|
|
14
|
+
async executeRequest<T>(
|
|
15
|
+
key: string,
|
|
16
|
+
requestFactory: () => Promise<T>,
|
|
17
|
+
): Promise<T> {
|
|
18
|
+
// Check if there's already a pending request for this key
|
|
19
|
+
const existingRequest = this.pendingRequests.get(key);
|
|
20
|
+
if (existingRequest) {
|
|
21
|
+
return existingRequest;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// Create and track the new request
|
|
25
|
+
const requestPromise = requestFactory();
|
|
26
|
+
this.pendingRequests.set(key, requestPromise);
|
|
27
|
+
|
|
28
|
+
try {
|
|
29
|
+
const result = await requestPromise;
|
|
30
|
+
this.pendingRequests.delete(key);
|
|
31
|
+
return result;
|
|
32
|
+
} catch (error) {
|
|
33
|
+
this.pendingRequests.delete(key);
|
|
34
|
+
throw error;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Clear all pending requests (used in cache clearing)
|
|
40
|
+
*/
|
|
41
|
+
clear(): void {
|
|
42
|
+
this.pendingRequests.clear();
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Get number of pending requests
|
|
47
|
+
*/
|
|
48
|
+
getPendingCount(): number {
|
|
49
|
+
return this.pendingRequests.size;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Check if a request is pending
|
|
54
|
+
*/
|
|
55
|
+
isPending(key: string): boolean {
|
|
56
|
+
return this.pendingRequests.has(key);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Get all pending request keys
|
|
61
|
+
*/
|
|
62
|
+
getPendingKeys(): string[] {
|
|
63
|
+
return Array.from(this.pendingRequests.keys());
|
|
64
|
+
}
|
|
65
|
+
}
|
|
@@ -0,0 +1,265 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Core types and interfaces for JIT Transcoding System
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
export interface QualityPreset {
|
|
6
|
+
name: string;
|
|
7
|
+
width: number;
|
|
8
|
+
height: number;
|
|
9
|
+
videoBitrate: number;
|
|
10
|
+
audioBitrate: number;
|
|
11
|
+
audioChannels: number;
|
|
12
|
+
audioSampleRate: number;
|
|
13
|
+
audioCodec: "aac";
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export type RenditionId = "high" | "medium" | "low" | "audio" | "scrub";
|
|
17
|
+
|
|
18
|
+
export interface VideoMetadata {
|
|
19
|
+
url: string;
|
|
20
|
+
durationMs: number;
|
|
21
|
+
streams: Array<{
|
|
22
|
+
index: number;
|
|
23
|
+
type: "video" | "audio" | "subtitle" | "other";
|
|
24
|
+
codecName: string;
|
|
25
|
+
duration: number;
|
|
26
|
+
durationMs: number;
|
|
27
|
+
width?: number;
|
|
28
|
+
height?: number;
|
|
29
|
+
frameRate?: { num: number; den: number };
|
|
30
|
+
channels?: number;
|
|
31
|
+
sampleRate?: number;
|
|
32
|
+
}>;
|
|
33
|
+
presets: string[];
|
|
34
|
+
segmentDuration: number;
|
|
35
|
+
supportedFormats: string[];
|
|
36
|
+
extractedAt: string;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export interface JitTranscodingConfig {
|
|
40
|
+
baseUrl: string;
|
|
41
|
+
defaultQuality: keyof QualityPresets;
|
|
42
|
+
segmentCacheSize: number;
|
|
43
|
+
enableNetworkAdaptation: boolean;
|
|
44
|
+
enablePrefetch: boolean;
|
|
45
|
+
prefetchSegments: number;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export interface QualityPresets {
|
|
49
|
+
low: QualityPreset;
|
|
50
|
+
medium: QualityPreset;
|
|
51
|
+
high: QualityPreset;
|
|
52
|
+
scrub: QualityPreset;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export interface NetworkCondition {
|
|
56
|
+
bandwidth: number; // bits per second
|
|
57
|
+
rtt: number; // round trip time in ms
|
|
58
|
+
connectionType: string;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export interface SegmentInfo {
|
|
62
|
+
url: string;
|
|
63
|
+
startTimeMs: number;
|
|
64
|
+
durationMs: number;
|
|
65
|
+
quality: string;
|
|
66
|
+
cached: boolean;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export interface ManifestVideoRendition {
|
|
70
|
+
/** Unique identifier for the video rendition */
|
|
71
|
+
id: string; // "high", "medium", "low", "scrub"
|
|
72
|
+
/** Duration of each segment in milliseconds */
|
|
73
|
+
segmentDuration: number;
|
|
74
|
+
/** Duration of each segment in milliseconds */
|
|
75
|
+
segmentDurationMs: number;
|
|
76
|
+
/** Video width in pixels */
|
|
77
|
+
width: number;
|
|
78
|
+
/** Video height in pixels */
|
|
79
|
+
height: number;
|
|
80
|
+
/** Target bitrate in bits per second */
|
|
81
|
+
bitrate: number;
|
|
82
|
+
/** Video codec string (e.g., "avc1.640029") */
|
|
83
|
+
codec: string;
|
|
84
|
+
/** Container format (e.g., "video/mp4") */
|
|
85
|
+
container: string;
|
|
86
|
+
/** Complete MIME type with codecs */
|
|
87
|
+
mimeType: string; // 'video/mp4; codecs="avc1.640029,mp4a.40.2"'
|
|
88
|
+
/** Optional frame rate */
|
|
89
|
+
frameRate?: number;
|
|
90
|
+
/** Optional profile indication */
|
|
91
|
+
profile?: string;
|
|
92
|
+
/** Optional level indication */
|
|
93
|
+
level?: string;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
export interface ManifestAudioRendition {
|
|
97
|
+
/** Unique identifier for the audio rendition */
|
|
98
|
+
id: string; // "audio"
|
|
99
|
+
/** Duration of each segment in milliseconds */
|
|
100
|
+
segmentDuration: number;
|
|
101
|
+
/** Duration of each segment in milliseconds */
|
|
102
|
+
segmentDurationMs: number;
|
|
103
|
+
/** Number of audio channels */
|
|
104
|
+
channels: number;
|
|
105
|
+
/** Sample rate in Hz */
|
|
106
|
+
sampleRate: number;
|
|
107
|
+
/** Target bitrate in bits per second */
|
|
108
|
+
bitrate: number;
|
|
109
|
+
/** Audio codec string (e.g., "mp4a.40.2") */
|
|
110
|
+
codec: string;
|
|
111
|
+
/** Container format (e.g., "audio/mp4") */
|
|
112
|
+
container: string;
|
|
113
|
+
/** Complete MIME type with codecs */
|
|
114
|
+
mimeType: string; // 'audio/mp4; codecs="mp4a.40.2"'
|
|
115
|
+
/** Optional language code */
|
|
116
|
+
language?: string;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* CMAF Manifest Response Structure
|
|
121
|
+
* Matches the server response from /api/v1/transcode/manifest.json
|
|
122
|
+
*/
|
|
123
|
+
export interface ManifestResponse {
|
|
124
|
+
/** Manifest version for compatibility tracking */
|
|
125
|
+
version: string;
|
|
126
|
+
/** Type of manifest (e.g., "cmaf", "dash", "hls") */
|
|
127
|
+
type: string;
|
|
128
|
+
/** Original source URL for the media */
|
|
129
|
+
sourceUrl: string;
|
|
130
|
+
/** Total duration of the content in milliseconds */
|
|
131
|
+
duration: number;
|
|
132
|
+
/** Total duration of the content in milliseconds */
|
|
133
|
+
durationMs: number;
|
|
134
|
+
/** Duration of each segment in milliseconds */
|
|
135
|
+
segmentDuration: number;
|
|
136
|
+
/** Base URL for all relative URLs */
|
|
137
|
+
baseUrl: string;
|
|
138
|
+
/** Video renditions available for adaptive streaming */
|
|
139
|
+
videoRenditions: ManifestVideoRendition[];
|
|
140
|
+
/** Audio renditions available for adaptive streaming */
|
|
141
|
+
audioRenditions: ManifestAudioRendition[];
|
|
142
|
+
/** URL templates for segment access */
|
|
143
|
+
endpoints: {
|
|
144
|
+
/** Initialization segment URL template with {rendition} placeholder */
|
|
145
|
+
initSegment: string;
|
|
146
|
+
/** Media segment URL template with {rendition} and {segmentId} placeholders */
|
|
147
|
+
mediaSegment: string;
|
|
148
|
+
};
|
|
149
|
+
/** JIT transcoding specific information */
|
|
150
|
+
jitInfo: {
|
|
151
|
+
/** Whether parallel transcoding is supported */
|
|
152
|
+
parallelTranscodingSupported: boolean;
|
|
153
|
+
/** Expected transcoding latency in milliseconds */
|
|
154
|
+
expectedTranscodeLatency: number;
|
|
155
|
+
/** Total number of segments */
|
|
156
|
+
segmentCount: number;
|
|
157
|
+
};
|
|
158
|
+
/** Optional timescale for the media timeline */
|
|
159
|
+
timescale?: number;
|
|
160
|
+
/** Optional start number for segments */
|
|
161
|
+
startNumber?: number;
|
|
162
|
+
/** Optional minimum buffer time in milliseconds */
|
|
163
|
+
minBufferTime?: number;
|
|
164
|
+
/** Optional suggested presentation delay in milliseconds */
|
|
165
|
+
suggestedPresentationDelay?: number;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Cache statistics interface
|
|
170
|
+
*/
|
|
171
|
+
export interface CacheStats {
|
|
172
|
+
size: number;
|
|
173
|
+
maxSize: number;
|
|
174
|
+
hitRate: number;
|
|
175
|
+
efficiency: number;
|
|
176
|
+
totalRequests: number;
|
|
177
|
+
recentKeys: string[];
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// TODO: JitTranscodingClient interface removed - will return in a later release
|
|
181
|
+
export interface AudioRendition {
|
|
182
|
+
id?: RenditionId;
|
|
183
|
+
trackId: number | undefined;
|
|
184
|
+
src: string;
|
|
185
|
+
segmentDurationMs?: number;
|
|
186
|
+
}
|
|
187
|
+
export interface VideoRendition {
|
|
188
|
+
id?: RenditionId;
|
|
189
|
+
trackId: number | undefined;
|
|
190
|
+
src: string;
|
|
191
|
+
segmentDurationMs?: number;
|
|
192
|
+
startTimeOffsetMs?: number;
|
|
193
|
+
}
|
|
194
|
+
export interface MediaEngine {
|
|
195
|
+
durationMs: number;
|
|
196
|
+
src: string;
|
|
197
|
+
audioRendition?: AudioRendition;
|
|
198
|
+
videoRendition?: VideoRendition;
|
|
199
|
+
templates: {
|
|
200
|
+
initSegment: string;
|
|
201
|
+
mediaSegment: string;
|
|
202
|
+
};
|
|
203
|
+
fetchInitSegment: (
|
|
204
|
+
rendition: {
|
|
205
|
+
id?: RenditionId;
|
|
206
|
+
trackId: number | undefined;
|
|
207
|
+
src: string;
|
|
208
|
+
},
|
|
209
|
+
signal: AbortSignal,
|
|
210
|
+
) => Promise<ArrayBuffer>;
|
|
211
|
+
fetchMediaSegment: (
|
|
212
|
+
segmentId: number,
|
|
213
|
+
rendition: { id?: RenditionId; trackId: number | undefined; src: string },
|
|
214
|
+
signal?: AbortSignal,
|
|
215
|
+
) => Promise<ArrayBuffer>;
|
|
216
|
+
computeSegmentId: (
|
|
217
|
+
desiredSeekTimeMs: number,
|
|
218
|
+
rendition: {
|
|
219
|
+
id?: RenditionId;
|
|
220
|
+
trackId: number | undefined;
|
|
221
|
+
src: string;
|
|
222
|
+
segmentDurationMs?: number;
|
|
223
|
+
},
|
|
224
|
+
) => number | undefined;
|
|
225
|
+
|
|
226
|
+
/**
|
|
227
|
+
* Get the video rendition, or throws if no video rendition is available
|
|
228
|
+
*/
|
|
229
|
+
getVideoRendition: () => VideoRendition;
|
|
230
|
+
|
|
231
|
+
/**
|
|
232
|
+
* Get the audio rendition, or throws if no audio rendition is available
|
|
233
|
+
*/
|
|
234
|
+
getAudioRendition: () => AudioRendition;
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* Calculate audio segments needed for a time range
|
|
238
|
+
* Each media engine implements this based on their segment structure
|
|
239
|
+
*/
|
|
240
|
+
calculateAudioSegmentRange: (
|
|
241
|
+
fromMs: number,
|
|
242
|
+
toMs: number,
|
|
243
|
+
rendition: AudioRendition,
|
|
244
|
+
durationMs: number,
|
|
245
|
+
) => SegmentTimeRange[];
|
|
246
|
+
}
|
|
247
|
+
interface InitSegmentPath {
|
|
248
|
+
path: string;
|
|
249
|
+
pos: number;
|
|
250
|
+
size: number;
|
|
251
|
+
}
|
|
252
|
+
export interface InitSegmentPaths {
|
|
253
|
+
audio?: InitSegmentPath;
|
|
254
|
+
video?: InitSegmentPath;
|
|
255
|
+
}
|
|
256
|
+
export interface AudioSpan {
|
|
257
|
+
startMs: number;
|
|
258
|
+
endMs: number;
|
|
259
|
+
blob: Blob;
|
|
260
|
+
}
|
|
261
|
+
export interface SegmentTimeRange {
|
|
262
|
+
segmentId: number;
|
|
263
|
+
startMs: number;
|
|
264
|
+
endMs: number;
|
|
265
|
+
}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Media type detection utilities
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
export function isAudioUrl(url: string): boolean {
|
|
6
|
+
if (!url) return false;
|
|
7
|
+
|
|
8
|
+
// Extract the pathname from URL, handling both full URLs and simple paths
|
|
9
|
+
let pathname: string;
|
|
10
|
+
try {
|
|
11
|
+
const urlObj = new URL(url);
|
|
12
|
+
pathname = urlObj.pathname;
|
|
13
|
+
} catch {
|
|
14
|
+
// If URL parsing fails, treat as simple path
|
|
15
|
+
pathname = url;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// Remove query parameters and get file extension
|
|
19
|
+
const pathWithoutQuery = pathname.split("?")[0] || "";
|
|
20
|
+
const extension = pathWithoutQuery.split(".").pop()?.toLowerCase();
|
|
21
|
+
|
|
22
|
+
// Check for audio file extensions
|
|
23
|
+
const audioExtensions = ["mp3", "wav", "flac", "aac", "m4a", "ogg", "wma"];
|
|
24
|
+
return extension ? audioExtensions.includes(extension) : false;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export function isVideoUrl(url: string): boolean {
|
|
28
|
+
if (!url) return false;
|
|
29
|
+
|
|
30
|
+
// Extract the pathname from URL, handling both full URLs and simple paths
|
|
31
|
+
let pathname: string;
|
|
32
|
+
try {
|
|
33
|
+
const urlObj = new URL(url);
|
|
34
|
+
pathname = urlObj.pathname;
|
|
35
|
+
} catch {
|
|
36
|
+
// If URL parsing fails, treat as simple path
|
|
37
|
+
pathname = url;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// Remove query parameters and get file extension
|
|
41
|
+
const pathWithoutQuery = pathname.split("?")[0] || "";
|
|
42
|
+
const extension = pathWithoutQuery.split(".").pop()?.toLowerCase();
|
|
43
|
+
|
|
44
|
+
// Check for video file extensions
|
|
45
|
+
const videoExtensions = [
|
|
46
|
+
"mp4",
|
|
47
|
+
"avi",
|
|
48
|
+
"mov",
|
|
49
|
+
"wmv",
|
|
50
|
+
"flv",
|
|
51
|
+
"webm",
|
|
52
|
+
"mkv",
|
|
53
|
+
"m4v",
|
|
54
|
+
];
|
|
55
|
+
return extension ? videoExtensions.includes(extension) : false;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Check if URL is eligible for JIT transcoding
|
|
60
|
+
*/
|
|
61
|
+
export function isJitTranscodeEligible(url: string): boolean {
|
|
62
|
+
return url.startsWith("http://") || url.startsWith("https://");
|
|
63
|
+
}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* URL generation utilities for transcoding endpoints
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import type { RenditionId } from "../types/index.js";
|
|
6
|
+
import type { MediaEngine } from "../types/index.ts";
|
|
7
|
+
|
|
8
|
+
export class UrlGenerator {
|
|
9
|
+
constructor(private baseUrl: () => string) {}
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Generate video segment URL
|
|
13
|
+
*/
|
|
14
|
+
generateSegmentUrl(
|
|
15
|
+
segmentId: "init" | number,
|
|
16
|
+
renditionId: RenditionId,
|
|
17
|
+
metadata: MediaEngine,
|
|
18
|
+
): string {
|
|
19
|
+
const audioRendition = metadata.audioRendition;
|
|
20
|
+
const videoRendition = metadata.videoRendition;
|
|
21
|
+
const rendition = audioRendition ?? videoRendition;
|
|
22
|
+
if (!rendition) {
|
|
23
|
+
console.error("Rendition not found", metadata);
|
|
24
|
+
throw new Error(`Rendition ${renditionId} not found`);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
const template =
|
|
28
|
+
segmentId === "init"
|
|
29
|
+
? metadata.templates.initSegment
|
|
30
|
+
: metadata.templates.mediaSegment;
|
|
31
|
+
return template
|
|
32
|
+
.replace("{rendition}", renditionId)
|
|
33
|
+
.replace("{segmentId}", segmentId.toString())
|
|
34
|
+
.replace("{src}", metadata.src)
|
|
35
|
+
.replace("{trackId}", rendition.trackId?.toString() ?? "");
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Generate init segment URL
|
|
40
|
+
*/
|
|
41
|
+
generateInitSegmentUrl(mediaUrl: string, rendition: string): string {
|
|
42
|
+
return `${this.baseUrl()}/api/v1/transcode/${rendition}/init.m4s?url=${encodeURIComponent(mediaUrl)}`;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Generate manifest URL
|
|
47
|
+
*/
|
|
48
|
+
generateManifestUrl(mediaUrl: string): string {
|
|
49
|
+
return `${this.baseUrl()}/api/v1/transcode/manifest.json?url=${encodeURIComponent(mediaUrl)}`;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Generate track fragment index URL
|
|
54
|
+
*/
|
|
55
|
+
generateTrackFragmentIndexUrl(mediaUrl: string): string {
|
|
56
|
+
const normalizedSrc = mediaUrl.startsWith("/")
|
|
57
|
+
? mediaUrl.slice(1)
|
|
58
|
+
: mediaUrl;
|
|
59
|
+
return `/@ef-track-fragment-index/${normalizedSrc}`;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Generate quality presets URL
|
|
64
|
+
*/
|
|
65
|
+
generatePresetsUrl(): string {
|
|
66
|
+
return `${this.baseUrl()}/api/v1/transcode/presets`;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Constants for JIT transcoding system
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import type { QualityPresets } from "../types/index.js";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Segment duration constants (in milliseconds)
|
|
9
|
+
*/
|
|
10
|
+
export const SEGMENT_DURATION_VIDEO_MS = 2000; // 2 seconds
|
|
11
|
+
export const SEGMENT_DURATION_AUDIO_MS = 15000; // 15 seconds
|
|
12
|
+
export const SEGMENT_DURATION_SCRUB_MS = 30000; // 30 seconds
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Default cache sizes for different media types
|
|
16
|
+
*/
|
|
17
|
+
export const DEFAULT_CACHE_SIZE_VIDEO = 50;
|
|
18
|
+
export const DEFAULT_CACHE_SIZE_AUDIO = 20;
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Default prefetch settings
|
|
22
|
+
*/
|
|
23
|
+
export const DEFAULT_PREFETCH_SEGMENTS_VIDEO = 3;
|
|
24
|
+
export const DEFAULT_PREFETCH_SEGMENTS_AUDIO = 2;
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Retry configuration
|
|
28
|
+
*/
|
|
29
|
+
export const RETRY_MAX_ATTEMPTS = 3;
|
|
30
|
+
export const RETRY_BASE_DELAY_MS = 500;
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Default service configuration
|
|
34
|
+
*/
|
|
35
|
+
export const DEFAULT_BASE_URL = "http://localhost:3000";
|
|
36
|
+
export const DEFAULT_QUALITY: keyof QualityPresets = "medium";
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* A simple LRU (Least Recently Used) cache implementation
|
|
3
|
+
*/
|
|
4
|
+
export class LRUCache<K, V> {
|
|
5
|
+
private cache = new Map<K, V>();
|
|
6
|
+
private readonly maxSize: number;
|
|
7
|
+
|
|
8
|
+
constructor(maxSize: number) {
|
|
9
|
+
this.maxSize = maxSize;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
get(key: K): V | undefined {
|
|
13
|
+
const value = this.cache.get(key);
|
|
14
|
+
if (value) {
|
|
15
|
+
// Refresh position by removing and re-adding
|
|
16
|
+
this.cache.delete(key);
|
|
17
|
+
this.cache.set(key, value);
|
|
18
|
+
}
|
|
19
|
+
return value;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
set(key: K, value: V): void {
|
|
23
|
+
if (this.cache.has(key)) {
|
|
24
|
+
this.cache.delete(key);
|
|
25
|
+
} else if (this.cache.size >= this.maxSize) {
|
|
26
|
+
// Remove oldest entry (first item in map)
|
|
27
|
+
const firstKey = this.cache.keys().next().value;
|
|
28
|
+
if (firstKey) {
|
|
29
|
+
this.cache.delete(firstKey);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
this.cache.set(key, value);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
has(key: K): boolean {
|
|
36
|
+
return this.cache.has(key);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
delete(key: K): boolean {
|
|
40
|
+
return this.cache.delete(key);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
clear(): void {
|
|
44
|
+
this.cache.clear();
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
get size(): number {
|
|
48
|
+
return this.cache.size;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Size-aware LRU cache that tracks memory usage in bytes
|
|
54
|
+
* Evicts entries when total size exceeds the maximum
|
|
55
|
+
*/
|
|
56
|
+
export class SizeAwareLRUCache<K> {
|
|
57
|
+
private cache = new Map<K, Promise<ArrayBuffer>>();
|
|
58
|
+
private sizes = new Map<K, number>();
|
|
59
|
+
private currentSize = 0;
|
|
60
|
+
private readonly maxSizeBytes: number;
|
|
61
|
+
|
|
62
|
+
constructor(maxSizeBytes: number) {
|
|
63
|
+
this.maxSizeBytes = maxSizeBytes;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
get(key: K): Promise<ArrayBuffer> | undefined {
|
|
67
|
+
const value = this.cache.get(key);
|
|
68
|
+
if (value) {
|
|
69
|
+
// Refresh position by removing and re-adding
|
|
70
|
+
const size = this.sizes.get(key) || 0;
|
|
71
|
+
this.cache.delete(key);
|
|
72
|
+
this.cache.set(key, value);
|
|
73
|
+
this.sizes.delete(key);
|
|
74
|
+
this.sizes.set(key, size);
|
|
75
|
+
}
|
|
76
|
+
return value;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
set(key: K, value: Promise<ArrayBuffer>): void {
|
|
80
|
+
// If key already exists, remove it first
|
|
81
|
+
if (this.cache.has(key)) {
|
|
82
|
+
const oldSize = this.sizes.get(key) || 0;
|
|
83
|
+
this.currentSize -= oldSize;
|
|
84
|
+
this.cache.delete(key);
|
|
85
|
+
this.sizes.delete(key);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
// Track the size when the promise resolves
|
|
89
|
+
const sizeTrackingPromise = value
|
|
90
|
+
.then((buffer) => {
|
|
91
|
+
const bufferSize = buffer.byteLength;
|
|
92
|
+
this.sizes.set(key, bufferSize);
|
|
93
|
+
this.currentSize += bufferSize;
|
|
94
|
+
|
|
95
|
+
// Evict oldest entries if we exceed the size limit
|
|
96
|
+
this.evictIfNecessary();
|
|
97
|
+
|
|
98
|
+
return buffer;
|
|
99
|
+
})
|
|
100
|
+
.catch((error) => {
|
|
101
|
+
// If the promise fails, clean up the entry
|
|
102
|
+
this.cache.delete(key);
|
|
103
|
+
this.sizes.delete(key);
|
|
104
|
+
throw error;
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
this.cache.set(key, sizeTrackingPromise);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
private evictIfNecessary(): void {
|
|
111
|
+
while (this.currentSize > this.maxSizeBytes && this.cache.size > 0) {
|
|
112
|
+
// Remove oldest entry (first item in map)
|
|
113
|
+
const firstKey = this.cache.keys().next().value;
|
|
114
|
+
if (firstKey) {
|
|
115
|
+
const size = this.sizes.get(firstKey) || 0;
|
|
116
|
+
this.currentSize -= size;
|
|
117
|
+
this.cache.delete(firstKey);
|
|
118
|
+
this.sizes.delete(firstKey);
|
|
119
|
+
} else {
|
|
120
|
+
break;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
has(key: K): boolean {
|
|
126
|
+
return this.cache.has(key);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
delete(key: K): boolean {
|
|
130
|
+
const size = this.sizes.get(key) || 0;
|
|
131
|
+
this.currentSize -= size;
|
|
132
|
+
this.sizes.delete(key);
|
|
133
|
+
return this.cache.delete(key);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
clear(): void {
|
|
137
|
+
this.cache.clear();
|
|
138
|
+
this.sizes.clear();
|
|
139
|
+
this.currentSize = 0;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
get size(): number {
|
|
143
|
+
return this.cache.size;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
get currentSizeBytes(): number {
|
|
147
|
+
return this.currentSize;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
get maxSize(): number {
|
|
151
|
+
return this.maxSizeBytes;
|
|
152
|
+
}
|
|
153
|
+
}
|