@editframe/elements 0.18.26-beta.0 → 0.19.2-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/elements/EFMedia/AssetMediaEngine.d.ts +10 -0
- package/dist/elements/EFMedia/AssetMediaEngine.js +13 -1
- package/dist/elements/EFMedia/BaseMediaEngine.js +1 -5
- package/dist/elements/EFMedia/JitMediaEngine.d.ts +10 -0
- package/dist/elements/EFMedia/JitMediaEngine.js +12 -0
- package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.js +16 -12
- package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.d.ts +1 -1
- package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.js +0 -4
- package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.d.ts +1 -1
- package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.js +0 -4
- package/dist/elements/EFMedia/videoTasks/makeScrubVideoBufferTask.js +1 -1
- package/dist/elements/EFMedia/videoTasks/makeUnifiedVideoSeekTask.js +3 -2
- package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.js +16 -12
- package/dist/elements/EFMedia.d.ts +2 -3
- package/dist/elements/EFMedia.js +0 -4
- package/dist/elements/EFTemporal.d.ts +9 -6
- package/dist/elements/EFTemporal.js +15 -12
- package/dist/elements/EFTimegroup.browsertest.d.ts +26 -0
- package/dist/elements/EFTimegroup.d.ts +12 -9
- package/dist/elements/EFTimegroup.js +114 -65
- package/dist/elements/EFVideo.d.ts +5 -1
- package/dist/elements/EFVideo.js +16 -8
- package/dist/elements/EFWaveform.js +2 -3
- package/dist/elements/FetchContext.browsertest.d.ts +0 -0
- package/dist/elements/FetchMixin.js +14 -9
- package/dist/elements/TimegroupController.js +2 -1
- package/dist/elements/updateAnimations.browsertest.d.ts +0 -0
- package/dist/elements/updateAnimations.d.ts +19 -9
- package/dist/elements/updateAnimations.js +64 -25
- package/dist/gui/ContextMixin.js +34 -27
- package/dist/gui/EFConfiguration.d.ts +1 -1
- package/dist/gui/EFConfiguration.js +1 -0
- package/dist/gui/EFFilmstrip.d.ts +1 -0
- package/dist/gui/EFFilmstrip.js +12 -14
- package/dist/gui/TWMixin.js +1 -1
- package/dist/style.css +1 -1
- package/dist/transcoding/cache/URLTokenDeduplicator.d.ts +38 -0
- package/dist/transcoding/cache/URLTokenDeduplicator.js +66 -0
- package/dist/transcoding/cache/URLTokenDeduplicator.test.d.ts +1 -0
- package/dist/transcoding/types/index.d.ts +10 -0
- package/package.json +2 -2
- package/src/elements/EFMedia/AssetMediaEngine.ts +16 -2
- package/src/elements/EFMedia/BaseMediaEngine.ts +0 -6
- package/src/elements/EFMedia/JitMediaEngine.ts +14 -0
- package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.ts +0 -1
- package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.ts +11 -4
- package/src/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.ts +0 -4
- package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.chunkboundary.regression.browsertest.ts +4 -1
- package/src/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.ts +0 -5
- package/src/elements/EFMedia/videoTasks/makeScrubVideoBufferTask.ts +2 -2
- package/src/elements/EFMedia/videoTasks/makeUnifiedVideoSeekTask.ts +7 -3
- package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.ts +11 -4
- package/src/elements/EFMedia.browsertest.ts +13 -4
- package/src/elements/EFMedia.ts +6 -10
- package/src/elements/EFTemporal.ts +21 -26
- package/src/elements/EFTimegroup.browsertest.ts +186 -2
- package/src/elements/EFTimegroup.ts +190 -94
- package/src/elements/EFVideo.browsertest.ts +53 -132
- package/src/elements/EFVideo.ts +26 -13
- package/src/elements/EFWaveform.ts +2 -3
- package/src/elements/FetchContext.browsertest.ts +396 -0
- package/src/elements/FetchMixin.ts +25 -8
- package/src/elements/TimegroupController.ts +2 -1
- package/src/elements/updateAnimations.browsertest.ts +559 -0
- package/src/elements/updateAnimations.ts +113 -50
- package/src/gui/ContextMixin.browsertest.ts +4 -9
- package/src/gui/ContextMixin.ts +52 -33
- package/src/gui/EFConfiguration.ts +1 -1
- package/src/gui/EFFilmstrip.ts +15 -18
- package/src/transcoding/cache/URLTokenDeduplicator.test.ts +182 -0
- package/src/transcoding/cache/URLTokenDeduplicator.ts +101 -0
- package/src/transcoding/types/index.ts +11 -0
- package/test/EFVideo.framegen.browsertest.ts +1 -1
- package/test/setup.ts +2 -0
- package/types.json +1 -1
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Global URL token deduplication utility
|
|
3
|
+
* Ensures that multiple EFMedia elements requesting tokens for the same resource
|
|
4
|
+
* share a single token request, preventing unnecessary duplicate token generation
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
export interface TokenCacheEntry {
|
|
8
|
+
tokenPromise: Promise<string>;
|
|
9
|
+
expiration: number;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export class URLTokenDeduplicator {
|
|
13
|
+
private tokenCache = new Map<string, TokenCacheEntry>();
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Get or create a URL token with global deduplication
|
|
17
|
+
* Multiple requests for the same cache key will share the same token promise
|
|
18
|
+
*/
|
|
19
|
+
async getToken(
|
|
20
|
+
cacheKey: string,
|
|
21
|
+
tokenFactory: () => Promise<string>,
|
|
22
|
+
parseExpiration: (token: string) => number,
|
|
23
|
+
): Promise<string> {
|
|
24
|
+
const now = Date.now();
|
|
25
|
+
const cached = this.tokenCache.get(cacheKey);
|
|
26
|
+
|
|
27
|
+
// Check if we have a valid cached token
|
|
28
|
+
if (cached && now < cached.expiration) {
|
|
29
|
+
return cached.tokenPromise;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// Create new token request
|
|
33
|
+
const tokenPromise = tokenFactory()
|
|
34
|
+
.then(async (token) => {
|
|
35
|
+
// Update expiration after we get the token
|
|
36
|
+
const expiration = parseExpiration(token);
|
|
37
|
+
this.tokenCache.set(cacheKey, { tokenPromise, expiration });
|
|
38
|
+
return token;
|
|
39
|
+
})
|
|
40
|
+
.catch((error) => {
|
|
41
|
+
// Remove failed request from cache
|
|
42
|
+
this.tokenCache.delete(cacheKey);
|
|
43
|
+
throw error;
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
// Cache the promise immediately to deduplicate concurrent requests
|
|
47
|
+
this.tokenCache.set(cacheKey, {
|
|
48
|
+
tokenPromise,
|
|
49
|
+
expiration: now + 60000, // Temporary 1-minute expiration, will be updated when token is received
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
return tokenPromise;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Clear all cached tokens (used in testing)
|
|
57
|
+
*/
|
|
58
|
+
clear(): void {
|
|
59
|
+
this.tokenCache.clear();
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Get number of cached tokens
|
|
64
|
+
*/
|
|
65
|
+
getCachedCount(): number {
|
|
66
|
+
return this.tokenCache.size;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Check if a token is cached and valid
|
|
71
|
+
*/
|
|
72
|
+
hasValidToken(cacheKey: string): boolean {
|
|
73
|
+
const cached = this.tokenCache.get(cacheKey);
|
|
74
|
+
if (!cached) return false;
|
|
75
|
+
|
|
76
|
+
const now = Date.now();
|
|
77
|
+
return now < cached.expiration;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* Get all cached token keys
|
|
82
|
+
*/
|
|
83
|
+
getCachedKeys(): string[] {
|
|
84
|
+
return Array.from(this.tokenCache.keys());
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Remove expired tokens from cache
|
|
89
|
+
*/
|
|
90
|
+
cleanup(): void {
|
|
91
|
+
const now = Date.now();
|
|
92
|
+
for (const [key, entry] of this.tokenCache.entries()) {
|
|
93
|
+
if (now >= entry.expiration) {
|
|
94
|
+
this.tokenCache.delete(key);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// Global instance shared across all context providers
|
|
101
|
+
export const globalURLTokenDeduplicator = new URLTokenDeduplicator();
|
|
@@ -263,6 +263,17 @@ export interface MediaEngine {
|
|
|
263
263
|
rendition: AudioRendition,
|
|
264
264
|
durationMs: number,
|
|
265
265
|
) => SegmentTimeRange[];
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* Get buffer configuration for this media engine
|
|
269
|
+
* Returns preferred buffer settings that may override host defaults
|
|
270
|
+
*/
|
|
271
|
+
getBufferConfig: () => {
|
|
272
|
+
videoBufferDurationMs: number;
|
|
273
|
+
audioBufferDurationMs: number;
|
|
274
|
+
maxVideoBufferFetches: number;
|
|
275
|
+
maxAudioBufferFetches: number;
|
|
276
|
+
};
|
|
266
277
|
}
|
|
267
278
|
interface InitSegmentPath {
|
|
268
279
|
path: string;
|
|
@@ -51,7 +51,7 @@ describe("EFVideo Frame Generation", () => {
|
|
|
51
51
|
});
|
|
52
52
|
|
|
53
53
|
test("fragmentIndexTask is pending", ({ expect }) => {
|
|
54
|
-
expect(video.fragmentIndexTask.status).toEqual(TaskStatus.
|
|
54
|
+
expect(video.fragmentIndexTask.status).toEqual(TaskStatus.INITIAL);
|
|
55
55
|
});
|
|
56
56
|
|
|
57
57
|
// Note: Timing-dependent tests disabled due to seek range issues
|
package/test/setup.ts
CHANGED
|
@@ -8,9 +8,11 @@ import {
|
|
|
8
8
|
globalRequestDeduplicator,
|
|
9
9
|
mediaCache,
|
|
10
10
|
} from "../src/elements/EFMedia/BaseMediaEngine.js";
|
|
11
|
+
import { globalURLTokenDeduplicator } from "../src/transcoding/cache/URLTokenDeduplicator.js";
|
|
11
12
|
|
|
12
13
|
// Clear global caches before each test to ensure isolation
|
|
13
14
|
beforeEach(() => {
|
|
14
15
|
globalRequestDeduplicator.clear();
|
|
15
16
|
mediaCache.clear();
|
|
17
|
+
globalURLTokenDeduplicator.clear();
|
|
16
18
|
});
|