@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.
Files changed (218) hide show
  1. package/dist/EF_FRAMEGEN.js +1 -1
  2. package/dist/ScrubTrackManager.d.ts +2 -2
  3. package/dist/elements/EFAudio.d.ts +21 -2
  4. package/dist/elements/EFAudio.js +41 -11
  5. package/dist/elements/EFImage.d.ts +1 -0
  6. package/dist/elements/EFImage.js +11 -3
  7. package/dist/elements/EFMedia/AssetIdMediaEngine.d.ts +18 -0
  8. package/dist/elements/EFMedia/AssetIdMediaEngine.js +41 -0
  9. package/dist/elements/EFMedia/AssetMediaEngine.d.ts +47 -0
  10. package/dist/elements/EFMedia/AssetMediaEngine.js +116 -0
  11. package/dist/elements/EFMedia/BaseMediaEngine.d.ts +55 -0
  12. package/dist/elements/EFMedia/BaseMediaEngine.js +96 -0
  13. package/dist/elements/EFMedia/BufferedSeekingInput.d.ts +43 -0
  14. package/dist/elements/EFMedia/BufferedSeekingInput.js +159 -0
  15. package/dist/elements/EFMedia/JitMediaEngine.browsertest.d.ts +0 -0
  16. package/dist/elements/EFMedia/JitMediaEngine.d.ts +31 -0
  17. package/dist/elements/EFMedia/JitMediaEngine.js +62 -0
  18. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.d.ts +9 -0
  19. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.d.ts +16 -0
  20. package/dist/elements/EFMedia/audioTasks/makeAudioBufferTask.js +48 -0
  21. package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.d.ts +3 -0
  22. package/dist/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.js +138 -0
  23. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.d.ts +9 -0
  24. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.d.ts +4 -0
  25. package/dist/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.js +16 -0
  26. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.d.ts +9 -0
  27. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.d.ts +3 -0
  28. package/dist/elements/EFMedia/audioTasks/makeAudioInputTask.js +22 -0
  29. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.d.ts +7 -0
  30. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.js +24 -0
  31. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.d.ts +4 -0
  32. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.js +18 -0
  33. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.d.ts +4 -0
  34. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.js +16 -0
  35. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.d.ts +3 -0
  36. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.js +104 -0
  37. package/dist/elements/EFMedia/services/AudioElementFactory.d.ts +22 -0
  38. package/dist/elements/EFMedia/services/AudioElementFactory.js +72 -0
  39. package/dist/elements/EFMedia/services/MediaSourceService.browsertest.d.ts +1 -0
  40. package/dist/elements/EFMedia/services/MediaSourceService.d.ts +47 -0
  41. package/dist/elements/EFMedia/services/MediaSourceService.js +73 -0
  42. package/dist/elements/EFMedia/shared/AudioSpanUtils.d.ts +7 -0
  43. package/dist/elements/EFMedia/shared/AudioSpanUtils.js +54 -0
  44. package/dist/elements/EFMedia/shared/BufferUtils.d.ts +70 -0
  45. package/dist/elements/EFMedia/shared/BufferUtils.js +89 -0
  46. package/dist/elements/EFMedia/shared/MediaTaskUtils.d.ts +23 -0
  47. package/dist/elements/EFMedia/shared/RenditionHelpers.browsertest.d.ts +1 -0
  48. package/dist/elements/EFMedia/shared/RenditionHelpers.d.ts +19 -0
  49. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.d.ts +1 -0
  50. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.d.ts +18 -0
  51. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.js +60 -0
  52. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.test.d.ts +1 -0
  53. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.d.ts +9 -0
  54. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.d.ts +16 -0
  55. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.js +46 -0
  56. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.d.ts +9 -0
  57. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.d.ts +4 -0
  58. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.js +16 -0
  59. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.d.ts +9 -0
  60. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.d.ts +3 -0
  61. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.js +27 -0
  62. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.d.ts +7 -0
  63. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.js +25 -0
  64. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.d.ts +9 -0
  65. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.d.ts +4 -0
  66. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.js +18 -0
  67. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.d.ts +9 -0
  68. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.d.ts +4 -0
  69. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.js +16 -0
  70. package/dist/elements/EFMedia.browsertest.d.ts +1 -0
  71. package/dist/elements/EFMedia.d.ts +75 -111
  72. package/dist/elements/EFMedia.js +141 -1111
  73. package/dist/elements/EFTemporal.d.ts +1 -1
  74. package/dist/elements/EFTemporal.js +1 -1
  75. package/dist/elements/EFTimegroup.d.ts +11 -0
  76. package/dist/elements/EFTimegroup.js +88 -13
  77. package/dist/elements/EFVideo.d.ts +60 -29
  78. package/dist/elements/EFVideo.js +103 -203
  79. package/dist/elements/EFWaveform.js +2 -2
  80. package/dist/elements/SampleBuffer.d.ts +14 -0
  81. package/dist/elements/SampleBuffer.js +52 -0
  82. package/dist/getRenderInfo.d.ts +2 -2
  83. package/dist/getRenderInfo.js +2 -1
  84. package/dist/gui/ContextMixin.js +17 -70
  85. package/dist/gui/EFFilmstrip.d.ts +3 -3
  86. package/dist/gui/EFFilmstrip.js +1 -1
  87. package/dist/gui/EFFitScale.d.ts +2 -2
  88. package/dist/gui/TWMixin.js +1 -1
  89. package/dist/gui/services/ElementConnectionManager.browsertest.d.ts +1 -0
  90. package/dist/gui/services/ElementConnectionManager.d.ts +59 -0
  91. package/dist/gui/services/ElementConnectionManager.js +128 -0
  92. package/dist/gui/services/PlaybackController.browsertest.d.ts +1 -0
  93. package/dist/gui/services/PlaybackController.d.ts +103 -0
  94. package/dist/gui/services/PlaybackController.js +290 -0
  95. package/dist/services/MediaSourceManager.d.ts +62 -0
  96. package/dist/services/MediaSourceManager.js +211 -0
  97. package/dist/style.css +1 -1
  98. package/dist/transcoding/cache/CacheManager.d.ts +73 -0
  99. package/dist/transcoding/cache/RequestDeduplicator.d.ts +29 -0
  100. package/dist/transcoding/cache/RequestDeduplicator.js +53 -0
  101. package/dist/transcoding/cache/RequestDeduplicator.test.d.ts +1 -0
  102. package/dist/transcoding/types/index.d.ts +242 -0
  103. package/dist/transcoding/utils/MediaUtils.d.ts +9 -0
  104. package/dist/transcoding/utils/UrlGenerator.d.ts +26 -0
  105. package/dist/transcoding/utils/UrlGenerator.js +45 -0
  106. package/dist/transcoding/utils/constants.d.ts +27 -0
  107. package/dist/utils/LRUCache.d.ts +34 -0
  108. package/dist/utils/LRUCache.js +115 -0
  109. package/package.json +3 -2
  110. package/src/elements/EFAudio.browsertest.ts +183 -43
  111. package/src/elements/EFAudio.ts +59 -13
  112. package/src/elements/EFImage.browsertest.ts +42 -0
  113. package/src/elements/EFImage.ts +23 -3
  114. package/src/elements/EFMedia/AssetIdMediaEngine.test.ts +222 -0
  115. package/src/elements/EFMedia/AssetIdMediaEngine.ts +70 -0
  116. package/src/elements/EFMedia/AssetMediaEngine.ts +210 -0
  117. package/src/elements/EFMedia/BaseMediaEngine.test.ts +164 -0
  118. package/src/elements/EFMedia/BaseMediaEngine.ts +170 -0
  119. package/src/elements/EFMedia/BufferedSeekingInput.browsertest.ts +400 -0
  120. package/src/elements/EFMedia/BufferedSeekingInput.ts +267 -0
  121. package/src/elements/EFMedia/JitMediaEngine.browsertest.ts +165 -0
  122. package/src/elements/EFMedia/JitMediaEngine.ts +110 -0
  123. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.ts +554 -0
  124. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.ts +81 -0
  125. package/src/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.ts +241 -0
  126. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.ts +59 -0
  127. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.ts +23 -0
  128. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.ts +55 -0
  129. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.ts +35 -0
  130. package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.ts +42 -0
  131. package/src/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.ts +34 -0
  132. package/src/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.ts +23 -0
  133. package/src/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.ts +174 -0
  134. package/src/elements/EFMedia/services/AudioElementFactory.browsertest.ts +325 -0
  135. package/src/elements/EFMedia/services/AudioElementFactory.ts +119 -0
  136. package/src/elements/EFMedia/services/MediaSourceService.browsertest.ts +257 -0
  137. package/src/elements/EFMedia/services/MediaSourceService.ts +102 -0
  138. package/src/elements/EFMedia/shared/AudioSpanUtils.ts +128 -0
  139. package/src/elements/EFMedia/shared/BufferUtils.ts +310 -0
  140. package/src/elements/EFMedia/shared/MediaTaskUtils.ts +44 -0
  141. package/src/elements/EFMedia/shared/RenditionHelpers.browsertest.ts +247 -0
  142. package/src/elements/EFMedia/shared/RenditionHelpers.ts +79 -0
  143. package/src/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.ts +128 -0
  144. package/src/elements/EFMedia/tasks/makeMediaEngineTask.test.ts +233 -0
  145. package/src/elements/EFMedia/tasks/makeMediaEngineTask.ts +89 -0
  146. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.ts +555 -0
  147. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.ts +79 -0
  148. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.ts +59 -0
  149. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.ts +23 -0
  150. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.ts +55 -0
  151. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.ts +45 -0
  152. package/src/elements/EFMedia/videoTasks/makeVideoSeekTask.ts +44 -0
  153. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.ts +57 -0
  154. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.ts +32 -0
  155. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.ts +56 -0
  156. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.ts +23 -0
  157. package/src/elements/EFMedia.browsertest.ts +658 -265
  158. package/src/elements/EFMedia.ts +173 -1763
  159. package/src/elements/EFTemporal.ts +3 -4
  160. package/src/elements/EFTimegroup.browsertest.ts +6 -3
  161. package/src/elements/EFTimegroup.ts +152 -21
  162. package/src/elements/EFVideo.browsertest.ts +115 -37
  163. package/src/elements/EFVideo.ts +123 -452
  164. package/src/elements/EFWaveform.ts +1 -1
  165. package/src/elements/MediaController.ts +2 -12
  166. package/src/elements/SampleBuffer.ts +97 -0
  167. package/src/gui/ContextMixin.ts +23 -104
  168. package/src/gui/services/ElementConnectionManager.browsertest.ts +263 -0
  169. package/src/gui/services/ElementConnectionManager.ts +224 -0
  170. package/src/gui/services/PlaybackController.browsertest.ts +437 -0
  171. package/src/gui/services/PlaybackController.ts +521 -0
  172. package/src/services/MediaSourceManager.ts +333 -0
  173. package/src/transcoding/cache/CacheManager.ts +208 -0
  174. package/src/transcoding/cache/RequestDeduplicator.test.ts +170 -0
  175. package/src/transcoding/cache/RequestDeduplicator.ts +65 -0
  176. package/src/transcoding/types/index.ts +265 -0
  177. package/src/transcoding/utils/MediaUtils.ts +63 -0
  178. package/src/transcoding/utils/UrlGenerator.ts +68 -0
  179. package/src/transcoding/utils/constants.ts +36 -0
  180. package/src/utils/LRUCache.ts +153 -0
  181. package/test/EFVideo.framegen.browsertest.ts +38 -29
  182. 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
  183. 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
  184. 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
  185. 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
  186. 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
  187. 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
  188. 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
  189. 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
  190. 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
  191. 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
  192. 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
  193. 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
  194. 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
  195. 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
  196. 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
  197. 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
  198. 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
  199. 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
  200. 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
  201. 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
  202. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/data.bin +1 -0
  203. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/metadata.json +19 -0
  204. package/test/createJitTestClips.ts +320 -188
  205. package/test/recordReplayProxyPlugin.js +302 -0
  206. package/test/useAssetMSW.ts +1 -1
  207. package/test/useMSW.ts +35 -22
  208. package/types.json +1 -1
  209. package/dist/JitTranscodingClient.d.ts +0 -167
  210. package/dist/JitTranscodingClient.js +0 -373
  211. package/dist/ScrubTrackManager.js +0 -216
  212. package/dist/elements/printTaskStatus.js +0 -11
  213. package/src/elements/__screenshots__/EFMedia.browsertest.ts/EFMedia-JIT-audio-playback-audioBufferTask-should-work-in-JIT-mode-without-URL-errors-1.png +0 -0
  214. package/test/EFVideo.frame-tasks.browsertest.ts +0 -524
  215. /package/dist/{JitTranscodingClient.browsertest.d.ts → elements/EFMedia/AssetIdMediaEngine.test.d.ts} +0 -0
  216. /package/dist/{JitTranscodingClient.test.d.ts → elements/EFMedia/BaseMediaEngine.test.d.ts} +0 -0
  217. /package/dist/{ScrubTrackIntegration.test.d.ts → elements/EFMedia/BufferedSeekingInput.browsertest.d.ts} +0 -0
  218. /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
+ }