@editframe/elements 0.17.6-beta.0 → 0.18.7-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 (211) hide show
  1. package/dist/EF_FRAMEGEN.js +1 -1
  2. package/dist/elements/EFAudio.d.ts +21 -2
  3. package/dist/elements/EFAudio.js +41 -11
  4. package/dist/elements/EFImage.d.ts +1 -0
  5. package/dist/elements/EFImage.js +11 -3
  6. package/dist/elements/EFMedia/AssetIdMediaEngine.d.ts +18 -0
  7. package/dist/elements/EFMedia/AssetIdMediaEngine.js +41 -0
  8. package/dist/elements/EFMedia/AssetMediaEngine.browsertest.d.ts +0 -0
  9. package/dist/elements/EFMedia/AssetMediaEngine.d.ts +45 -0
  10. package/dist/elements/EFMedia/AssetMediaEngine.js +135 -0
  11. package/dist/elements/EFMedia/BaseMediaEngine.d.ts +55 -0
  12. package/dist/elements/EFMedia/BaseMediaEngine.js +115 -0
  13. package/dist/elements/EFMedia/BufferedSeekingInput.d.ts +43 -0
  14. package/dist/elements/EFMedia/BufferedSeekingInput.js +179 -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 +81 -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 +141 -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 +30 -0
  29. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.chunkboundary.regression.browsertest.d.ts +0 -0
  30. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.d.ts +7 -0
  31. package/dist/elements/EFMedia/audioTasks/makeAudioSeekTask.js +32 -0
  32. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.d.ts +4 -0
  33. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.js +28 -0
  34. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.d.ts +4 -0
  35. package/dist/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.js +17 -0
  36. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.d.ts +3 -0
  37. package/dist/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.js +107 -0
  38. package/dist/elements/EFMedia/shared/AudioSpanUtils.d.ts +7 -0
  39. package/dist/elements/EFMedia/shared/AudioSpanUtils.js +54 -0
  40. package/dist/elements/EFMedia/shared/BufferUtils.d.ts +70 -0
  41. package/dist/elements/EFMedia/shared/BufferUtils.js +89 -0
  42. package/dist/elements/EFMedia/shared/MediaTaskUtils.d.ts +23 -0
  43. package/dist/elements/EFMedia/shared/PrecisionUtils.d.ts +28 -0
  44. package/dist/elements/EFMedia/shared/PrecisionUtils.js +29 -0
  45. package/dist/elements/EFMedia/shared/RenditionHelpers.d.ts +19 -0
  46. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.d.ts +18 -0
  47. package/dist/elements/EFMedia/tasks/makeMediaEngineTask.js +60 -0
  48. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.d.ts +9 -0
  49. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.d.ts +16 -0
  50. package/dist/elements/EFMedia/videoTasks/makeVideoBufferTask.js +46 -0
  51. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.d.ts +9 -0
  52. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.d.ts +4 -0
  53. package/dist/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.js +16 -0
  54. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.d.ts +9 -0
  55. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.d.ts +3 -0
  56. package/dist/elements/EFMedia/videoTasks/makeVideoInputTask.js +27 -0
  57. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.d.ts +7 -0
  58. package/dist/elements/EFMedia/videoTasks/makeVideoSeekTask.js +34 -0
  59. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.d.ts +9 -0
  60. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.d.ts +4 -0
  61. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.js +28 -0
  62. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.d.ts +9 -0
  63. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.d.ts +4 -0
  64. package/dist/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.js +17 -0
  65. package/dist/elements/EFMedia.browsertest.d.ts +1 -0
  66. package/dist/elements/EFMedia.d.ts +63 -111
  67. package/dist/elements/EFMedia.js +117 -1113
  68. package/dist/elements/EFTemporal.d.ts +1 -1
  69. package/dist/elements/EFTemporal.js +1 -1
  70. package/dist/elements/EFTimegroup.d.ts +11 -0
  71. package/dist/elements/EFTimegroup.js +83 -13
  72. package/dist/elements/EFVideo.d.ts +54 -32
  73. package/dist/elements/EFVideo.js +100 -207
  74. package/dist/elements/EFWaveform.js +2 -2
  75. package/dist/elements/SampleBuffer.d.ts +14 -0
  76. package/dist/elements/SampleBuffer.js +52 -0
  77. package/dist/getRenderInfo.js +2 -1
  78. package/dist/gui/ContextMixin.js +3 -2
  79. package/dist/gui/EFFilmstrip.d.ts +3 -3
  80. package/dist/gui/EFFilmstrip.js +1 -1
  81. package/dist/gui/EFFitScale.d.ts +2 -2
  82. package/dist/gui/TWMixin.js +1 -1
  83. package/dist/style.css +1 -1
  84. package/dist/transcoding/cache/CacheManager.d.ts +73 -0
  85. package/dist/transcoding/cache/RequestDeduplicator.d.ts +29 -0
  86. package/dist/transcoding/cache/RequestDeduplicator.js +53 -0
  87. package/dist/transcoding/cache/RequestDeduplicator.test.d.ts +1 -0
  88. package/dist/transcoding/types/index.d.ts +242 -0
  89. package/dist/transcoding/utils/MediaUtils.d.ts +9 -0
  90. package/dist/transcoding/utils/UrlGenerator.d.ts +26 -0
  91. package/dist/transcoding/utils/UrlGenerator.js +45 -0
  92. package/dist/transcoding/utils/constants.d.ts +27 -0
  93. package/dist/utils/LRUCache.d.ts +34 -0
  94. package/dist/utils/LRUCache.js +115 -0
  95. package/package.json +3 -3
  96. package/src/elements/EFAudio.browsertest.ts +189 -49
  97. package/src/elements/EFAudio.ts +59 -13
  98. package/src/elements/EFImage.browsertest.ts +42 -0
  99. package/src/elements/EFImage.ts +23 -3
  100. package/src/elements/EFMedia/AssetIdMediaEngine.test.ts +222 -0
  101. package/src/elements/EFMedia/AssetIdMediaEngine.ts +70 -0
  102. package/src/elements/EFMedia/AssetMediaEngine.browsertest.ts +100 -0
  103. package/src/elements/EFMedia/AssetMediaEngine.ts +255 -0
  104. package/src/elements/EFMedia/BaseMediaEngine.test.ts +164 -0
  105. package/src/elements/EFMedia/BaseMediaEngine.ts +219 -0
  106. package/src/elements/EFMedia/BufferedSeekingInput.browsertest.ts +481 -0
  107. package/src/elements/EFMedia/BufferedSeekingInput.ts +324 -0
  108. package/src/elements/EFMedia/JitMediaEngine.browsertest.ts +165 -0
  109. package/src/elements/EFMedia/JitMediaEngine.ts +166 -0
  110. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.browsertest.ts +554 -0
  111. package/src/elements/EFMedia/audioTasks/makeAudioBufferTask.ts +81 -0
  112. package/src/elements/EFMedia/audioTasks/makeAudioFrequencyAnalysisTask.ts +250 -0
  113. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.browsertest.ts +59 -0
  114. package/src/elements/EFMedia/audioTasks/makeAudioInitSegmentFetchTask.ts +23 -0
  115. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.browsertest.ts +55 -0
  116. package/src/elements/EFMedia/audioTasks/makeAudioInputTask.ts +43 -0
  117. package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.chunkboundary.regression.browsertest.ts +199 -0
  118. package/src/elements/EFMedia/audioTasks/makeAudioSeekTask.ts +64 -0
  119. package/src/elements/EFMedia/audioTasks/makeAudioSegmentFetchTask.ts +45 -0
  120. package/src/elements/EFMedia/audioTasks/makeAudioSegmentIdTask.ts +24 -0
  121. package/src/elements/EFMedia/audioTasks/makeAudioTimeDomainAnalysisTask.ts +183 -0
  122. package/src/elements/EFMedia/shared/AudioSpanUtils.ts +128 -0
  123. package/src/elements/EFMedia/shared/BufferUtils.ts +310 -0
  124. package/src/elements/EFMedia/shared/MediaTaskUtils.ts +44 -0
  125. package/src/elements/EFMedia/shared/PrecisionUtils.ts +46 -0
  126. package/src/elements/EFMedia/shared/RenditionHelpers.browsertest.ts +247 -0
  127. package/src/elements/EFMedia/shared/RenditionHelpers.ts +79 -0
  128. package/src/elements/EFMedia/tasks/makeMediaEngineTask.browsertest.ts +128 -0
  129. package/src/elements/EFMedia/tasks/makeMediaEngineTask.test.ts +233 -0
  130. package/src/elements/EFMedia/tasks/makeMediaEngineTask.ts +89 -0
  131. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.browsertest.ts +555 -0
  132. package/src/elements/EFMedia/videoTasks/makeVideoBufferTask.ts +79 -0
  133. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.browsertest.ts +59 -0
  134. package/src/elements/EFMedia/videoTasks/makeVideoInitSegmentFetchTask.ts +23 -0
  135. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.browsertest.ts +55 -0
  136. package/src/elements/EFMedia/videoTasks/makeVideoInputTask.ts +45 -0
  137. package/src/elements/EFMedia/videoTasks/makeVideoSeekTask.ts +68 -0
  138. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.browsertest.ts +57 -0
  139. package/src/elements/EFMedia/videoTasks/makeVideoSegmentFetchTask.ts +43 -0
  140. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.browsertest.ts +56 -0
  141. package/src/elements/EFMedia/videoTasks/makeVideoSegmentIdTask.ts +24 -0
  142. package/src/elements/EFMedia.browsertest.ts +706 -273
  143. package/src/elements/EFMedia.ts +136 -1769
  144. package/src/elements/EFTemporal.ts +3 -4
  145. package/src/elements/EFTimegroup.browsertest.ts +6 -3
  146. package/src/elements/EFTimegroup.ts +147 -21
  147. package/src/elements/EFVideo.browsertest.ts +980 -169
  148. package/src/elements/EFVideo.ts +113 -458
  149. package/src/elements/EFWaveform.ts +1 -1
  150. package/src/elements/MediaController.ts +2 -12
  151. package/src/elements/SampleBuffer.ts +95 -0
  152. package/src/gui/ContextMixin.ts +3 -6
  153. package/src/transcoding/cache/CacheManager.ts +208 -0
  154. package/src/transcoding/cache/RequestDeduplicator.test.ts +170 -0
  155. package/src/transcoding/cache/RequestDeduplicator.ts +65 -0
  156. package/src/transcoding/types/index.ts +269 -0
  157. package/src/transcoding/utils/MediaUtils.ts +63 -0
  158. package/src/transcoding/utils/UrlGenerator.ts +68 -0
  159. package/src/transcoding/utils/constants.ts +36 -0
  160. package/src/utils/LRUCache.ts +153 -0
  161. package/test/EFVideo.framegen.browsertest.ts +39 -30
  162. 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
  163. 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
  164. package/test/__cache__/GET__api_v1_transcode_audio_1_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__9ed2d25c675aa6bb6ff5b3ae23887c71/data.bin +0 -0
  165. package/test/__cache__/GET__api_v1_transcode_audio_1_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__9ed2d25c675aa6bb6ff5b3ae23887c71/metadata.json +22 -0
  166. 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
  167. 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
  168. package/test/__cache__/GET__api_v1_transcode_audio_2_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__d5a3309a2bf756dd6e304807eb402f56/data.bin +0 -0
  169. package/test/__cache__/GET__api_v1_transcode_audio_2_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__d5a3309a2bf756dd6e304807eb402f56/metadata.json +22 -0
  170. 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
  171. 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
  172. package/test/__cache__/GET__api_v1_transcode_audio_3_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__773254bb671e3466fca8677139fb239e/data.bin +0 -0
  173. package/test/__cache__/GET__api_v1_transcode_audio_3_mp4_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4_bytes_0__773254bb671e3466fca8677139fb239e/metadata.json +22 -0
  174. 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
  175. 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
  176. 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
  177. 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
  178. 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
  179. 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
  180. 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
  181. 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
  182. 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
  183. 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
  184. 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
  185. 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
  186. package/test/__cache__/GET__api_v1_transcode_high_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a6fb05a22b18d850f7f2950bbcdbdeed/data.bin +0 -0
  187. package/test/__cache__/GET__api_v1_transcode_high_4_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a6fb05a22b18d850f7f2950bbcdbdeed/metadata.json +21 -0
  188. package/test/__cache__/GET__api_v1_transcode_high_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a50058c7c3602e90879fe3428ed891f4/data.bin +0 -0
  189. package/test/__cache__/GET__api_v1_transcode_high_5_m4s_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__a50058c7c3602e90879fe3428ed891f4/metadata.json +21 -0
  190. 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
  191. 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
  192. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/data.bin +1 -0
  193. package/test/__cache__/GET__api_v1_transcode_manifest_json_url_http_3A_2F_2Fweb_3A3000_2Fhead_moov_480p_mp4__3be92a0437de726b431ed5af2369158a/metadata.json +19 -0
  194. package/test/createJitTestClips.ts +320 -188
  195. package/test/recordReplayProxyPlugin.js +352 -0
  196. package/test/useAssetMSW.ts +1 -1
  197. package/test/useMSW.ts +35 -22
  198. package/types.json +1 -1
  199. package/dist/JitTranscodingClient.d.ts +0 -167
  200. package/dist/JitTranscodingClient.js +0 -373
  201. package/dist/ScrubTrackManager.d.ts +0 -96
  202. package/dist/ScrubTrackManager.js +0 -216
  203. package/dist/elements/printTaskStatus.js +0 -11
  204. package/src/elements/__screenshots__/EFMedia.browsertest.ts/EFMedia-JIT-audio-playback-audioBufferTask-should-work-in-JIT-mode-without-URL-errors-1.png +0 -0
  205. package/test/EFVideo.frame-tasks.browsertest.ts +0 -524
  206. /package/dist/{DecoderResetFrequency.test.d.ts → elements/EFMedia/AssetIdMediaEngine.test.d.ts} +0 -0
  207. /package/dist/{DecoderResetRecovery.test.d.ts → elements/EFMedia/BaseMediaEngine.test.d.ts} +0 -0
  208. /package/dist/{JitTranscodingClient.browsertest.d.ts → elements/EFMedia/BufferedSeekingInput.browsertest.d.ts} +0 -0
  209. /package/dist/{JitTranscodingClient.test.d.ts → elements/EFMedia/shared/RenditionHelpers.browsertest.d.ts} +0 -0
  210. /package/dist/{ScrubTrackIntegration.test.d.ts → elements/EFMedia/tasks/makeMediaEngineTask.browsertest.d.ts} +0 -0
  211. /package/dist/{SegmentSwitchLoading.test.d.ts → elements/EFMedia/tasks/makeMediaEngineTask.test.d.ts} +0 -0
@@ -0,0 +1,269 @@
1
+ /**
2
+ * Core types and interfaces for JIT Transcoding System
3
+ */
4
+
5
+ import type { MediaRendition } from "../../elements/EFMedia/shared/MediaTaskUtils";
6
+
7
+ export interface QualityPreset {
8
+ name: string;
9
+ width: number;
10
+ height: number;
11
+ videoBitrate: number;
12
+ audioBitrate: number;
13
+ audioChannels: number;
14
+ audioSampleRate: number;
15
+ audioCodec: "aac";
16
+ }
17
+
18
+ export type RenditionId = "high" | "medium" | "low" | "audio" | "scrub";
19
+
20
+ export interface VideoMetadata {
21
+ url: string;
22
+ durationMs: number;
23
+ streams: Array<{
24
+ index: number;
25
+ type: "video" | "audio" | "subtitle" | "other";
26
+ codecName: string;
27
+ duration: number;
28
+ durationMs: number;
29
+ width?: number;
30
+ height?: number;
31
+ frameRate?: { num: number; den: number };
32
+ channels?: number;
33
+ sampleRate?: number;
34
+ }>;
35
+ presets: string[];
36
+ segmentDuration: number;
37
+ supportedFormats: string[];
38
+ extractedAt: string;
39
+ }
40
+
41
+ export interface JitTranscodingConfig {
42
+ baseUrl: string;
43
+ defaultQuality: keyof QualityPresets;
44
+ segmentCacheSize: number;
45
+ enableNetworkAdaptation: boolean;
46
+ enablePrefetch: boolean;
47
+ prefetchSegments: number;
48
+ }
49
+
50
+ export interface QualityPresets {
51
+ low: QualityPreset;
52
+ medium: QualityPreset;
53
+ high: QualityPreset;
54
+ scrub: QualityPreset;
55
+ }
56
+
57
+ export interface NetworkCondition {
58
+ bandwidth: number; // bits per second
59
+ rtt: number; // round trip time in ms
60
+ connectionType: string;
61
+ }
62
+
63
+ export interface SegmentInfo {
64
+ url: string;
65
+ startTimeMs: number;
66
+ durationMs: number;
67
+ quality: string;
68
+ cached: boolean;
69
+ }
70
+
71
+ export interface ManifestVideoRendition {
72
+ /** Unique identifier for the video rendition */
73
+ id: string; // "high", "medium", "low", "scrub"
74
+ /** Duration of each segment in milliseconds */
75
+ segmentDuration: number;
76
+ /** Duration of each segment in milliseconds */
77
+ segmentDurationMs: number;
78
+ /** Actual segment durations array (overrides fixed segmentDurationMs if provided) */
79
+ segmentDurationsMs?: number[];
80
+ /** Video width in pixels */
81
+ width: number;
82
+ /** Video height in pixels */
83
+ height: number;
84
+ /** Target bitrate in bits per second */
85
+ bitrate: number;
86
+ /** Video codec string (e.g., "avc1.640029") */
87
+ codec: string;
88
+ /** Container format (e.g., "video/mp4") */
89
+ container: string;
90
+ /** Complete MIME type with codecs */
91
+ mimeType: string; // 'video/mp4; codecs="avc1.640029,mp4a.40.2"'
92
+ /** Optional frame rate */
93
+ frameRate?: number;
94
+ /** Optional profile indication */
95
+ profile?: string;
96
+ /** Optional level indication */
97
+ level?: string;
98
+ }
99
+
100
+ export interface ManifestAudioRendition {
101
+ /** Unique identifier for the audio rendition */
102
+ id: string; // "audio"
103
+ /** Duration of each segment in milliseconds */
104
+ segmentDuration: number;
105
+ /** Duration of each segment in milliseconds */
106
+ segmentDurationMs: number;
107
+ /** Actual segment durations array (overrides fixed segmentDurationMs if provided) */
108
+ segmentDurationsMs?: number[];
109
+ /** Number of audio channels */
110
+ channels: number;
111
+ /** Sample rate in Hz */
112
+ sampleRate: number;
113
+ /** Target bitrate in bits per second */
114
+ bitrate: number;
115
+ /** Audio codec string (e.g., "mp4a.40.2") */
116
+ codec: string;
117
+ /** Container format (e.g., "audio/mp4") */
118
+ container: string;
119
+ /** Complete MIME type with codecs */
120
+ mimeType: string; // 'audio/mp4; codecs="mp4a.40.2"'
121
+ /** Optional language code */
122
+ language?: string;
123
+ }
124
+
125
+ /**
126
+ * CMAF Manifest Response Structure
127
+ * Matches the server response from /api/v1/transcode/manifest.json
128
+ */
129
+ export interface ManifestResponse {
130
+ /** Manifest version for compatibility tracking */
131
+ version: string;
132
+ /** Type of manifest (e.g., "cmaf", "dash", "hls") */
133
+ type: string;
134
+ /** Original source URL for the media */
135
+ sourceUrl: string;
136
+ /** Total duration of the content in milliseconds */
137
+ duration: number;
138
+ /** Total duration of the content in milliseconds */
139
+ durationMs: number;
140
+ /** Duration of each segment in milliseconds */
141
+ segmentDuration: number;
142
+ /** Base URL for all relative URLs */
143
+ baseUrl: string;
144
+ /** Video renditions available for adaptive streaming */
145
+ videoRenditions: ManifestVideoRendition[];
146
+ /** Audio renditions available for adaptive streaming */
147
+ audioRenditions: ManifestAudioRendition[];
148
+ /** URL templates for segment access */
149
+ endpoints: {
150
+ /** Initialization segment URL template with {rendition} placeholder */
151
+ initSegment: string;
152
+ /** Media segment URL template with {rendition} and {segmentId} placeholders */
153
+ mediaSegment: string;
154
+ };
155
+ /** JIT transcoding specific information */
156
+ jitInfo: {
157
+ /** Whether parallel transcoding is supported */
158
+ parallelTranscodingSupported: boolean;
159
+ /** Expected transcoding latency in milliseconds */
160
+ expectedTranscodeLatency: number;
161
+ /** Total number of segments */
162
+ segmentCount: number;
163
+ };
164
+ /** Optional timescale for the media timeline */
165
+ timescale?: number;
166
+ /** Optional start number for segments */
167
+ startNumber?: number;
168
+ /** Optional minimum buffer time in milliseconds */
169
+ minBufferTime?: number;
170
+ /** Optional suggested presentation delay in milliseconds */
171
+ suggestedPresentationDelay?: number;
172
+ }
173
+
174
+ /**
175
+ * Cache statistics interface
176
+ */
177
+ export interface CacheStats {
178
+ size: number;
179
+ maxSize: number;
180
+ hitRate: number;
181
+ efficiency: number;
182
+ totalRequests: number;
183
+ recentKeys: string[];
184
+ }
185
+
186
+ // TODO: JitTranscodingClient interface removed - will return in a later release
187
+ export interface AudioRendition {
188
+ id?: RenditionId;
189
+ trackId: number | undefined;
190
+ src: string;
191
+ segmentDurationMs?: number;
192
+ segmentDurationsMs?: number[];
193
+ startTimeOffsetMs?: number;
194
+ }
195
+ export interface VideoRendition {
196
+ id?: RenditionId;
197
+ trackId: number | undefined;
198
+ src: string;
199
+ segmentDurationMs?: number;
200
+ segmentDurationsMs?: number[];
201
+ startTimeOffsetMs?: number;
202
+ }
203
+ export interface MediaEngine {
204
+ durationMs: number;
205
+ src: string;
206
+ audioRendition?: AudioRendition;
207
+ videoRendition?: VideoRendition;
208
+ templates: {
209
+ initSegment: string;
210
+ mediaSegment: string;
211
+ };
212
+ fetchInitSegment: (
213
+ rendition: {
214
+ id?: RenditionId;
215
+ trackId: number | undefined;
216
+ src: string;
217
+ },
218
+ signal: AbortSignal,
219
+ ) => Promise<ArrayBuffer>;
220
+ fetchMediaSegment: (
221
+ segmentId: number,
222
+ rendition: { id?: RenditionId; trackId: number | undefined; src: string },
223
+ signal?: AbortSignal,
224
+ ) => Promise<ArrayBuffer>;
225
+ computeSegmentId: (
226
+ desiredSeekTimeMs: number,
227
+ rendition: MediaRendition,
228
+ ) => number | undefined;
229
+
230
+ /**
231
+ * Get the video rendition, or throws if no video rendition is available
232
+ */
233
+ getVideoRendition: () => VideoRendition;
234
+
235
+ /**
236
+ * Get the audio rendition, or throws if no audio rendition is available
237
+ */
238
+ getAudioRendition: () => AudioRendition;
239
+
240
+ /**
241
+ * Calculate audio segments needed for a time range
242
+ * Each media engine implements this based on their segment structure
243
+ */
244
+ calculateAudioSegmentRange: (
245
+ fromMs: number,
246
+ toMs: number,
247
+ rendition: AudioRendition,
248
+ durationMs: number,
249
+ ) => SegmentTimeRange[];
250
+ }
251
+ interface InitSegmentPath {
252
+ path: string;
253
+ pos: number;
254
+ size: number;
255
+ }
256
+ export interface InitSegmentPaths {
257
+ audio?: InitSegmentPath;
258
+ video?: InitSegmentPath;
259
+ }
260
+ export interface AudioSpan {
261
+ startMs: number;
262
+ endMs: number;
263
+ blob: Blob;
264
+ }
265
+ export interface SegmentTimeRange {
266
+ segmentId: number;
267
+ startMs: number;
268
+ endMs: number;
269
+ }
@@ -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
+ }