@kenzuya/mediabunny 1.26.0 → 1.28.6

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 (238) hide show
  1. package/README.md +1 -1
  2. package/dist/bundles/{mediabunny.mjs → mediabunny.js} +21963 -21390
  3. package/dist/bundles/mediabunny.min.js +490 -0
  4. package/dist/modules/shared/mp3-misc.d.ts.map +1 -1
  5. package/dist/modules/src/adts/adts-demuxer.d.ts +6 -6
  6. package/dist/modules/src/adts/adts-demuxer.d.ts.map +1 -1
  7. package/dist/modules/src/adts/adts-muxer.d.ts +4 -4
  8. package/dist/modules/src/adts/adts-muxer.d.ts.map +1 -1
  9. package/dist/modules/src/adts/adts-reader.d.ts +1 -1
  10. package/dist/modules/src/adts/adts-reader.d.ts.map +1 -1
  11. package/dist/modules/src/avi/avi-demuxer.d.ts +44 -0
  12. package/dist/modules/src/avi/avi-demuxer.d.ts.map +1 -0
  13. package/dist/modules/src/avi/avi-misc.d.ts +88 -0
  14. package/dist/modules/src/avi/avi-misc.d.ts.map +1 -0
  15. package/dist/modules/src/avi/avi-muxer.d.ts +45 -0
  16. package/dist/modules/src/avi/avi-muxer.d.ts.map +1 -0
  17. package/dist/modules/src/avi/riff-writer.d.ts +26 -0
  18. package/dist/modules/src/avi/riff-writer.d.ts.map +1 -0
  19. package/dist/modules/src/codec-data.d.ts +8 -3
  20. package/dist/modules/src/codec-data.d.ts.map +1 -1
  21. package/dist/modules/src/codec.d.ts +10 -10
  22. package/dist/modules/src/codec.d.ts.map +1 -1
  23. package/dist/modules/src/conversion.d.ts +33 -16
  24. package/dist/modules/src/conversion.d.ts.map +1 -1
  25. package/dist/modules/src/custom-coder.d.ts +8 -8
  26. package/dist/modules/src/custom-coder.d.ts.map +1 -1
  27. package/dist/modules/src/demuxer.d.ts +3 -3
  28. package/dist/modules/src/demuxer.d.ts.map +1 -1
  29. package/dist/modules/src/encode.d.ts +8 -8
  30. package/dist/modules/src/encode.d.ts.map +1 -1
  31. package/dist/modules/src/flac/flac-demuxer.d.ts +7 -7
  32. package/dist/modules/src/flac/flac-demuxer.d.ts.map +1 -1
  33. package/dist/modules/src/flac/flac-misc.d.ts +3 -3
  34. package/dist/modules/src/flac/flac-misc.d.ts.map +1 -1
  35. package/dist/modules/src/flac/flac-muxer.d.ts +5 -5
  36. package/dist/modules/src/flac/flac-muxer.d.ts.map +1 -1
  37. package/dist/modules/src/id3.d.ts +3 -3
  38. package/dist/modules/src/id3.d.ts.map +1 -1
  39. package/dist/modules/src/index.d.ts +20 -20
  40. package/dist/modules/src/index.d.ts.map +1 -1
  41. package/dist/modules/src/input-format.d.ts +22 -0
  42. package/dist/modules/src/input-format.d.ts.map +1 -1
  43. package/dist/modules/src/input-track.d.ts +8 -8
  44. package/dist/modules/src/input-track.d.ts.map +1 -1
  45. package/dist/modules/src/input.d.ts +12 -12
  46. package/dist/modules/src/isobmff/isobmff-boxes.d.ts +2 -2
  47. package/dist/modules/src/isobmff/isobmff-boxes.d.ts.map +1 -1
  48. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts +12 -12
  49. package/dist/modules/src/isobmff/isobmff-demuxer.d.ts.map +1 -1
  50. package/dist/modules/src/isobmff/isobmff-misc.d.ts.map +1 -1
  51. package/dist/modules/src/isobmff/isobmff-muxer.d.ts +11 -11
  52. package/dist/modules/src/isobmff/isobmff-muxer.d.ts.map +1 -1
  53. package/dist/modules/src/isobmff/isobmff-reader.d.ts +2 -2
  54. package/dist/modules/src/isobmff/isobmff-reader.d.ts.map +1 -1
  55. package/dist/modules/src/matroska/ebml.d.ts +3 -3
  56. package/dist/modules/src/matroska/ebml.d.ts.map +1 -1
  57. package/dist/modules/src/matroska/matroska-demuxer.d.ts +13 -13
  58. package/dist/modules/src/matroska/matroska-demuxer.d.ts.map +1 -1
  59. package/dist/modules/src/matroska/matroska-input.d.ts +33 -0
  60. package/dist/modules/src/matroska/matroska-input.d.ts.map +1 -0
  61. package/dist/modules/src/matroska/matroska-misc.d.ts.map +1 -1
  62. package/dist/modules/src/matroska/matroska-muxer.d.ts +5 -5
  63. package/dist/modules/src/matroska/matroska-muxer.d.ts.map +1 -1
  64. package/dist/modules/src/media-sink.d.ts +5 -5
  65. package/dist/modules/src/media-sink.d.ts.map +1 -1
  66. package/dist/modules/src/media-source.d.ts +22 -4
  67. package/dist/modules/src/media-source.d.ts.map +1 -1
  68. package/dist/modules/src/metadata.d.ts +2 -2
  69. package/dist/modules/src/metadata.d.ts.map +1 -1
  70. package/dist/modules/src/misc.d.ts +5 -4
  71. package/dist/modules/src/misc.d.ts.map +1 -1
  72. package/dist/modules/src/mp3/mp3-demuxer.d.ts +7 -7
  73. package/dist/modules/src/mp3/mp3-demuxer.d.ts.map +1 -1
  74. package/dist/modules/src/mp3/mp3-muxer.d.ts +4 -4
  75. package/dist/modules/src/mp3/mp3-muxer.d.ts.map +1 -1
  76. package/dist/modules/src/mp3/mp3-reader.d.ts +2 -2
  77. package/dist/modules/src/mp3/mp3-reader.d.ts.map +1 -1
  78. package/dist/modules/src/mp3/mp3-writer.d.ts +1 -1
  79. package/dist/modules/src/mp3/mp3-writer.d.ts.map +1 -1
  80. package/dist/modules/src/muxer.d.ts +4 -4
  81. package/dist/modules/src/muxer.d.ts.map +1 -1
  82. package/dist/modules/src/ogg/ogg-demuxer.d.ts +7 -7
  83. package/dist/modules/src/ogg/ogg-demuxer.d.ts.map +1 -1
  84. package/dist/modules/src/ogg/ogg-misc.d.ts +1 -1
  85. package/dist/modules/src/ogg/ogg-misc.d.ts.map +1 -1
  86. package/dist/modules/src/ogg/ogg-muxer.d.ts +5 -5
  87. package/dist/modules/src/ogg/ogg-muxer.d.ts.map +1 -1
  88. package/dist/modules/src/ogg/ogg-reader.d.ts +1 -1
  89. package/dist/modules/src/ogg/ogg-reader.d.ts.map +1 -1
  90. package/dist/modules/src/output-format.d.ts +51 -6
  91. package/dist/modules/src/output-format.d.ts.map +1 -1
  92. package/dist/modules/src/output.d.ts +13 -13
  93. package/dist/modules/src/output.d.ts.map +1 -1
  94. package/dist/modules/src/packet.d.ts +1 -1
  95. package/dist/modules/src/packet.d.ts.map +1 -1
  96. package/dist/modules/src/pcm.d.ts.map +1 -1
  97. package/dist/modules/src/reader.d.ts +2 -2
  98. package/dist/modules/src/reader.d.ts.map +1 -1
  99. package/dist/modules/src/sample.d.ts +57 -15
  100. package/dist/modules/src/sample.d.ts.map +1 -1
  101. package/dist/modules/src/source.d.ts +3 -3
  102. package/dist/modules/src/source.d.ts.map +1 -1
  103. package/dist/modules/src/subtitles.d.ts +1 -1
  104. package/dist/modules/src/subtitles.d.ts.map +1 -1
  105. package/dist/modules/src/target.d.ts +2 -2
  106. package/dist/modules/src/target.d.ts.map +1 -1
  107. package/dist/modules/src/tsconfig.tsbuildinfo +1 -1
  108. package/dist/modules/src/wave/riff-writer.d.ts +1 -1
  109. package/dist/modules/src/wave/riff-writer.d.ts.map +1 -1
  110. package/dist/modules/src/wave/wave-demuxer.d.ts +6 -6
  111. package/dist/modules/src/wave/wave-demuxer.d.ts.map +1 -1
  112. package/dist/modules/src/wave/wave-muxer.d.ts +4 -4
  113. package/dist/modules/src/wave/wave-muxer.d.ts.map +1 -1
  114. package/dist/modules/src/writer.d.ts +1 -1
  115. package/dist/modules/src/writer.d.ts.map +1 -1
  116. package/dist/packages/eac3/eac3.wasm +0 -0
  117. package/dist/packages/eac3/mediabunny-eac3.js +1058 -0
  118. package/dist/packages/eac3/mediabunny-eac3.min.js +44 -0
  119. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.js +694 -0
  120. package/dist/packages/mp3-encoder/mediabunny-mp3-encoder.min.js +58 -0
  121. package/dist/packages/mpeg4/mediabunny-mpeg4.js +1198 -0
  122. package/dist/packages/mpeg4/mediabunny-mpeg4.min.js +44 -0
  123. package/dist/packages/mpeg4/xvid.wasm +0 -0
  124. package/package.json +18 -57
  125. package/dist/bundles/mediabunny.cjs +0 -26140
  126. package/dist/bundles/mediabunny.min.cjs +0 -147
  127. package/dist/bundles/mediabunny.min.mjs +0 -146
  128. package/dist/mediabunny.d.ts +0 -3319
  129. package/dist/modules/shared/mp3-misc.js +0 -147
  130. package/dist/modules/src/adts/adts-demuxer.js +0 -239
  131. package/dist/modules/src/adts/adts-muxer.js +0 -80
  132. package/dist/modules/src/adts/adts-reader.js +0 -63
  133. package/dist/modules/src/codec-data.js +0 -1730
  134. package/dist/modules/src/codec.js +0 -869
  135. package/dist/modules/src/conversion.js +0 -1459
  136. package/dist/modules/src/custom-coder.js +0 -117
  137. package/dist/modules/src/demuxer.js +0 -12
  138. package/dist/modules/src/encode.js +0 -442
  139. package/dist/modules/src/flac/flac-demuxer.js +0 -504
  140. package/dist/modules/src/flac/flac-misc.js +0 -135
  141. package/dist/modules/src/flac/flac-muxer.js +0 -222
  142. package/dist/modules/src/id3.js +0 -848
  143. package/dist/modules/src/index.js +0 -28
  144. package/dist/modules/src/input-format.js +0 -480
  145. package/dist/modules/src/input-track.js +0 -372
  146. package/dist/modules/src/input.js +0 -188
  147. package/dist/modules/src/isobmff/isobmff-boxes.js +0 -1480
  148. package/dist/modules/src/isobmff/isobmff-demuxer.js +0 -2618
  149. package/dist/modules/src/isobmff/isobmff-misc.js +0 -20
  150. package/dist/modules/src/isobmff/isobmff-muxer.js +0 -966
  151. package/dist/modules/src/isobmff/isobmff-reader.js +0 -72
  152. package/dist/modules/src/matroska/ebml.js +0 -653
  153. package/dist/modules/src/matroska/matroska-demuxer.js +0 -2133
  154. package/dist/modules/src/matroska/matroska-misc.js +0 -20
  155. package/dist/modules/src/matroska/matroska-muxer.js +0 -1017
  156. package/dist/modules/src/media-sink.js +0 -1736
  157. package/dist/modules/src/media-source.js +0 -1825
  158. package/dist/modules/src/metadata.js +0 -193
  159. package/dist/modules/src/misc.js +0 -623
  160. package/dist/modules/src/mp3/mp3-demuxer.js +0 -285
  161. package/dist/modules/src/mp3/mp3-muxer.js +0 -123
  162. package/dist/modules/src/mp3/mp3-reader.js +0 -26
  163. package/dist/modules/src/mp3/mp3-writer.js +0 -78
  164. package/dist/modules/src/muxer.js +0 -50
  165. package/dist/modules/src/node.d.ts +0 -9
  166. package/dist/modules/src/node.d.ts.map +0 -1
  167. package/dist/modules/src/node.js +0 -9
  168. package/dist/modules/src/ogg/ogg-demuxer.js +0 -763
  169. package/dist/modules/src/ogg/ogg-misc.js +0 -78
  170. package/dist/modules/src/ogg/ogg-muxer.js +0 -353
  171. package/dist/modules/src/ogg/ogg-reader.js +0 -65
  172. package/dist/modules/src/output-format.js +0 -527
  173. package/dist/modules/src/output.js +0 -300
  174. package/dist/modules/src/packet.js +0 -182
  175. package/dist/modules/src/pcm.js +0 -85
  176. package/dist/modules/src/reader.js +0 -236
  177. package/dist/modules/src/sample.js +0 -1056
  178. package/dist/modules/src/source.js +0 -1182
  179. package/dist/modules/src/subtitles.js +0 -575
  180. package/dist/modules/src/target.js +0 -140
  181. package/dist/modules/src/wave/riff-writer.js +0 -30
  182. package/dist/modules/src/wave/wave-demuxer.js +0 -447
  183. package/dist/modules/src/wave/wave-muxer.js +0 -318
  184. package/dist/modules/src/writer.js +0 -370
  185. package/src/adts/adts-demuxer.ts +0 -331
  186. package/src/adts/adts-muxer.ts +0 -111
  187. package/src/adts/adts-reader.ts +0 -85
  188. package/src/codec-data.ts +0 -2078
  189. package/src/codec.ts +0 -1092
  190. package/src/conversion.ts +0 -2112
  191. package/src/custom-coder.ts +0 -197
  192. package/src/demuxer.ts +0 -24
  193. package/src/encode.ts +0 -739
  194. package/src/flac/flac-demuxer.ts +0 -730
  195. package/src/flac/flac-misc.ts +0 -164
  196. package/src/flac/flac-muxer.ts +0 -320
  197. package/src/id3.ts +0 -925
  198. package/src/index.ts +0 -221
  199. package/src/input-format.ts +0 -541
  200. package/src/input-track.ts +0 -529
  201. package/src/input.ts +0 -235
  202. package/src/isobmff/isobmff-boxes.ts +0 -1719
  203. package/src/isobmff/isobmff-demuxer.ts +0 -3190
  204. package/src/isobmff/isobmff-misc.ts +0 -29
  205. package/src/isobmff/isobmff-muxer.ts +0 -1348
  206. package/src/isobmff/isobmff-reader.ts +0 -91
  207. package/src/matroska/ebml.ts +0 -730
  208. package/src/matroska/matroska-demuxer.ts +0 -2481
  209. package/src/matroska/matroska-misc.ts +0 -29
  210. package/src/matroska/matroska-muxer.ts +0 -1276
  211. package/src/media-sink.ts +0 -2179
  212. package/src/media-source.ts +0 -2243
  213. package/src/metadata.ts +0 -320
  214. package/src/misc.ts +0 -798
  215. package/src/mp3/mp3-demuxer.ts +0 -383
  216. package/src/mp3/mp3-muxer.ts +0 -166
  217. package/src/mp3/mp3-reader.ts +0 -34
  218. package/src/mp3/mp3-writer.ts +0 -120
  219. package/src/muxer.ts +0 -88
  220. package/src/node.ts +0 -11
  221. package/src/ogg/ogg-demuxer.ts +0 -1053
  222. package/src/ogg/ogg-misc.ts +0 -116
  223. package/src/ogg/ogg-muxer.ts +0 -497
  224. package/src/ogg/ogg-reader.ts +0 -93
  225. package/src/output-format.ts +0 -945
  226. package/src/output.ts +0 -488
  227. package/src/packet.ts +0 -263
  228. package/src/pcm.ts +0 -112
  229. package/src/reader.ts +0 -323
  230. package/src/sample.ts +0 -1461
  231. package/src/source.ts +0 -1688
  232. package/src/subtitles.ts +0 -711
  233. package/src/target.ts +0 -204
  234. package/src/tsconfig.json +0 -16
  235. package/src/wave/riff-writer.ts +0 -36
  236. package/src/wave/wave-demuxer.ts +0 -529
  237. package/src/wave/wave-muxer.ts +0 -371
  238. package/src/writer.ts +0 -490
@@ -1,383 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
-
9
- import { AudioCodec } from '../codec';
10
- import { Demuxer } from '../demuxer';
11
- import { Input } from '../input';
12
- import { InputAudioTrack, InputAudioTrackBacking } from '../input-track';
13
- import { DEFAULT_TRACK_DISPOSITION, MetadataTags } from '../metadata';
14
- import { PacketRetrievalOptions } from '../media-sink';
15
- import { assert, AsyncMutex, binarySearchExact, binarySearchLessOrEqual, UNDETERMINED_LANGUAGE } from '../misc';
16
- import { EncodedPacket, PLACEHOLDER_DATA } from '../packet';
17
- import { FrameHeader, getXingOffset, INFO, XING } from '../../shared/mp3-misc';
18
- import {
19
- ID3_V1_TAG_SIZE,
20
- ID3_V2_HEADER_SIZE,
21
- parseId3V1Tag,
22
- parseId3V2Tag,
23
- readId3V2Header,
24
- } from '../id3';
25
- import { readNextFrameHeader } from './mp3-reader';
26
- import { readAscii, readBytes, Reader, readU32Be } from '../reader';
27
-
28
- type Sample = {
29
- timestamp: number;
30
- duration: number;
31
- dataStart: number;
32
- dataSize: number;
33
- };
34
-
35
- export class Mp3Demuxer extends Demuxer {
36
- reader: Reader;
37
-
38
- metadataPromise: Promise<void> | null = null;
39
- firstFrameHeader: FrameHeader | null = null;
40
- loadedSamples: Sample[] = []; // All samples from the start of the file to lastLoadedPos
41
- metadataTags: MetadataTags | null = null;
42
-
43
- tracks: InputAudioTrack[] = [];
44
-
45
- readingMutex = new AsyncMutex();
46
- lastSampleLoaded = false;
47
- lastLoadedPos = 0;
48
- nextTimestampInSamples = 0;
49
-
50
- constructor(input: Input) {
51
- super(input);
52
-
53
- this.reader = input._reader;
54
- }
55
-
56
- async readMetadata() {
57
- return this.metadataPromise ??= (async () => {
58
- // Keep loading until we find the first frame header
59
- while (!this.firstFrameHeader && !this.lastSampleLoaded) {
60
- await this.advanceReader();
61
- }
62
-
63
- if (!this.firstFrameHeader) {
64
- throw new Error('No valid MP3 frame found.');
65
- }
66
-
67
- this.tracks = [new InputAudioTrack(this.input, new Mp3AudioTrackBacking(this))];
68
- })();
69
- }
70
-
71
- async advanceReader() {
72
- if (this.lastLoadedPos === 0) {
73
- // Let's skip all ID3v2 tags at the start of the file
74
- while (true) {
75
- let slice = this.reader.requestSlice(this.lastLoadedPos, ID3_V2_HEADER_SIZE);
76
- if (slice instanceof Promise) slice = await slice;
77
-
78
- if (!slice) {
79
- this.lastSampleLoaded = true;
80
- return;
81
- }
82
-
83
- const id3V2Header = readId3V2Header(slice);
84
- if (!id3V2Header) {
85
- break;
86
- }
87
-
88
- this.lastLoadedPos = slice.filePos + id3V2Header.size;
89
- }
90
- }
91
-
92
- const result = await readNextFrameHeader(this.reader, this.lastLoadedPos, this.reader.fileSize);
93
- if (!result) {
94
- this.lastSampleLoaded = true;
95
- return;
96
- }
97
-
98
- const header = result.header;
99
-
100
- this.lastLoadedPos = result.startPos + header.totalSize - 1; // -1 in case the frame is 1 byte too short
101
-
102
- const xingOffset = getXingOffset(header.mpegVersionId, header.channel);
103
-
104
- let slice = this.reader.requestSlice(result.startPos + xingOffset, 4);
105
- if (slice instanceof Promise) slice = await slice;
106
- if (slice) {
107
- const word = readU32Be(slice);
108
- const isXing = word === XING || word === INFO;
109
-
110
- if (isXing) {
111
- // There's no actual audio data in this frame, so let's skip it
112
- return;
113
- }
114
- }
115
-
116
- if (!this.firstFrameHeader) {
117
- this.firstFrameHeader = header;
118
- }
119
-
120
- if (header.sampleRate !== this.firstFrameHeader.sampleRate) {
121
- console.warn(
122
- `MP3 changed sample rate mid-file: ${this.firstFrameHeader.sampleRate} Hz to ${header.sampleRate} Hz.`
123
- + ` Might be a bug, so please report this file.`,
124
- );
125
- }
126
-
127
- const sampleDuration = header.audioSamplesInFrame / this.firstFrameHeader.sampleRate;
128
- const sample: Sample = {
129
- timestamp: this.nextTimestampInSamples / this.firstFrameHeader.sampleRate,
130
- duration: sampleDuration,
131
- dataStart: result.startPos,
132
- dataSize: header.totalSize,
133
- };
134
-
135
- this.loadedSamples.push(sample);
136
- this.nextTimestampInSamples += header.audioSamplesInFrame;
137
-
138
- return;
139
- }
140
-
141
- async getMimeType() {
142
- return 'audio/mpeg';
143
- }
144
-
145
- async getTracks() {
146
- await this.readMetadata();
147
- return this.tracks;
148
- }
149
-
150
- async computeDuration() {
151
- await this.readMetadata();
152
-
153
- const track = this.tracks[0];
154
- assert(track);
155
-
156
- return track.computeDuration();
157
- }
158
-
159
- async getMetadataTags() {
160
- const release = await this.readingMutex.acquire();
161
-
162
- try {
163
- await this.readMetadata();
164
-
165
- if (this.metadataTags) {
166
- return this.metadataTags;
167
- }
168
-
169
- this.metadataTags = {};
170
- let currentPos = 0;
171
- let id3V2HeaderFound = false;
172
-
173
- while (true) {
174
- let headerSlice = this.reader.requestSlice(currentPos, ID3_V2_HEADER_SIZE);
175
- if (headerSlice instanceof Promise) headerSlice = await headerSlice;
176
- if (!headerSlice) break;
177
-
178
- const id3V2Header = readId3V2Header(headerSlice);
179
- if (!id3V2Header) {
180
- break;
181
- }
182
-
183
- id3V2HeaderFound = true;
184
-
185
- let contentSlice = this.reader.requestSlice(headerSlice.filePos, id3V2Header.size);
186
- if (contentSlice instanceof Promise) contentSlice = await contentSlice;
187
- if (!contentSlice) break;
188
-
189
- parseId3V2Tag(contentSlice, id3V2Header, this.metadataTags);
190
-
191
- currentPos = headerSlice.filePos + id3V2Header.size;
192
- }
193
-
194
- if (!id3V2HeaderFound && this.reader.fileSize !== null && this.reader.fileSize >= ID3_V1_TAG_SIZE) {
195
- // Try reading an ID3v1 tag at the end of the file
196
- let slice = this.reader.requestSlice(this.reader.fileSize - ID3_V1_TAG_SIZE, ID3_V1_TAG_SIZE);
197
- if (slice instanceof Promise) slice = await slice;
198
- assert(slice);
199
-
200
- const tag = readAscii(slice, 3);
201
- if (tag === 'TAG') {
202
- parseId3V1Tag(slice, this.metadataTags);
203
- }
204
- }
205
-
206
- return this.metadataTags;
207
- } finally {
208
- release();
209
- }
210
- }
211
- }
212
-
213
- class Mp3AudioTrackBacking implements InputAudioTrackBacking {
214
- constructor(public demuxer: Mp3Demuxer) {}
215
-
216
- getId() {
217
- return 1;
218
- }
219
-
220
- async getFirstTimestamp() {
221
- return 0;
222
- }
223
-
224
- getTimeResolution() {
225
- assert(this.demuxer.firstFrameHeader);
226
- return this.demuxer.firstFrameHeader.sampleRate / this.demuxer.firstFrameHeader.audioSamplesInFrame;
227
- }
228
-
229
- async computeDuration() {
230
- const lastPacket = await this.getPacket(Infinity, { metadataOnly: true });
231
- return (lastPacket?.timestamp ?? 0) + (lastPacket?.duration ?? 0);
232
- }
233
-
234
- getName() {
235
- return null;
236
- }
237
-
238
- getLanguageCode() {
239
- return UNDETERMINED_LANGUAGE;
240
- }
241
-
242
- getCodec(): AudioCodec {
243
- return 'mp3';
244
- }
245
-
246
- getInternalCodecId() {
247
- return null;
248
- }
249
-
250
- getNumberOfChannels() {
251
- assert(this.demuxer.firstFrameHeader);
252
- return this.demuxer.firstFrameHeader.channel === 3 ? 1 : 2;
253
- }
254
-
255
- getSampleRate() {
256
- assert(this.demuxer.firstFrameHeader);
257
- return this.demuxer.firstFrameHeader.sampleRate;
258
- }
259
-
260
- getDisposition() {
261
- return {
262
- ...DEFAULT_TRACK_DISPOSITION,
263
- };
264
- }
265
-
266
- async getDecoderConfig(): Promise<AudioDecoderConfig> {
267
- assert(this.demuxer.firstFrameHeader);
268
-
269
- return {
270
- codec: 'mp3',
271
- numberOfChannels: this.demuxer.firstFrameHeader.channel === 3 ? 1 : 2,
272
- sampleRate: this.demuxer.firstFrameHeader.sampleRate,
273
- };
274
- }
275
-
276
- async getPacketAtIndex(sampleIndex: number, options: PacketRetrievalOptions) {
277
- if (sampleIndex === -1) {
278
- return null;
279
- }
280
-
281
- const rawSample = this.demuxer.loadedSamples[sampleIndex];
282
- if (!rawSample) {
283
- return null;
284
- }
285
-
286
- let data: Uint8Array;
287
- if (options.metadataOnly) {
288
- data = PLACEHOLDER_DATA;
289
- } else {
290
- let slice = this.demuxer.reader.requestSlice(rawSample.dataStart, rawSample.dataSize);
291
- if (slice instanceof Promise) slice = await slice;
292
-
293
- if (!slice) {
294
- return null; // Data didn't fit into the rest of the file
295
- }
296
-
297
- data = readBytes(slice, rawSample.dataSize);
298
- }
299
-
300
- return new EncodedPacket(
301
- data,
302
- 'key',
303
- rawSample.timestamp,
304
- rawSample.duration,
305
- sampleIndex,
306
- rawSample.dataSize,
307
- );
308
- }
309
-
310
- getFirstPacket(options: PacketRetrievalOptions) {
311
- return this.getPacketAtIndex(0, options);
312
- }
313
-
314
- async getNextPacket(packet: EncodedPacket, options: PacketRetrievalOptions) {
315
- const release = await this.demuxer.readingMutex.acquire();
316
-
317
- try {
318
- const sampleIndex = binarySearchExact(
319
- this.demuxer.loadedSamples,
320
- packet.timestamp,
321
- x => x.timestamp,
322
- );
323
- if (sampleIndex === -1) {
324
- throw new Error('Packet was not created from this track.');
325
- }
326
-
327
- const nextIndex = sampleIndex + 1;
328
- // Ensure the next sample exists
329
- while (
330
- nextIndex >= this.demuxer.loadedSamples.length
331
- && !this.demuxer.lastSampleLoaded
332
- ) {
333
- await this.demuxer.advanceReader();
334
- }
335
-
336
- return this.getPacketAtIndex(nextIndex, options);
337
- } finally {
338
- release();
339
- }
340
- }
341
-
342
- async getPacket(timestamp: number, options: PacketRetrievalOptions) {
343
- const release = await this.demuxer.readingMutex.acquire();
344
-
345
- try {
346
- while (true) {
347
- const index = binarySearchLessOrEqual(
348
- this.demuxer.loadedSamples,
349
- timestamp,
350
- x => x.timestamp,
351
- );
352
-
353
- if (index === -1 && this.demuxer.loadedSamples.length > 0) {
354
- // We're before the first sample
355
- return null;
356
- }
357
-
358
- if (this.demuxer.lastSampleLoaded) {
359
- // All data is loaded, return what we found
360
- return this.getPacketAtIndex(index, options);
361
- }
362
-
363
- if (index >= 0 && index + 1 < this.demuxer.loadedSamples.length) {
364
- // The next packet also exists, we're done
365
- return this.getPacketAtIndex(index, options);
366
- }
367
-
368
- // Otherwise, keep loading data
369
- await this.demuxer.advanceReader();
370
- }
371
- } finally {
372
- release();
373
- }
374
- }
375
-
376
- getKeyPacket(timestamp: number, options: PacketRetrievalOptions) {
377
- return this.getPacket(timestamp, options);
378
- }
379
-
380
- getNextKeyPacket(packet: EncodedPacket, options: PacketRetrievalOptions) {
381
- return this.getNextPacket(packet, options);
382
- }
383
- }
@@ -1,166 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
-
9
- import { assert, toDataView } from '../misc';
10
- import { metadataTagsAreEmpty } from '../metadata';
11
- import { Muxer } from '../muxer';
12
- import { Output, OutputAudioTrack } from '../output';
13
- import { Mp3OutputFormat } from '../output-format';
14
- import { EncodedPacket } from '../packet';
15
- import { Writer } from '../writer';
16
- import { getXingOffset, INFO, readFrameHeader, XING } from '../../shared/mp3-misc';
17
- import { Mp3Writer, XingFrameData } from './mp3-writer';
18
- import { Id3V2Writer } from '../id3';
19
-
20
- export class Mp3Muxer extends Muxer {
21
- private format: Mp3OutputFormat;
22
- private writer: Writer;
23
- private mp3Writer: Mp3Writer;
24
- private xingFrameData: XingFrameData | null = null;
25
- private frameCount = 0;
26
- private framePositions: number[] = [];
27
- private xingFramePos: number | null = null;
28
-
29
- constructor(output: Output, format: Mp3OutputFormat) {
30
- super(output);
31
-
32
- this.format = format;
33
- this.writer = output._writer;
34
- this.mp3Writer = new Mp3Writer(output._writer);
35
- }
36
-
37
- async start() {
38
- if (!metadataTagsAreEmpty(this.output._metadataTags)) {
39
- const id3Writer = new Id3V2Writer(this.writer);
40
- id3Writer.writeId3V2Tag(this.output._metadataTags);
41
- }
42
- }
43
-
44
- async getMimeType() {
45
- return 'audio/mpeg';
46
- }
47
-
48
- async addEncodedVideoPacket() {
49
- throw new Error('MP3 does not support video.');
50
- }
51
-
52
- async addEncodedAudioPacket(
53
- track: OutputAudioTrack,
54
- packet: EncodedPacket,
55
- ) {
56
- const release = await this.mutex.acquire();
57
-
58
- try {
59
- const writeXingHeader = this.format._options.xingHeader !== false;
60
-
61
- if (!this.xingFrameData && writeXingHeader) {
62
- const view = toDataView(packet.data);
63
- if (view.byteLength < 4) {
64
- throw new Error('Invalid MP3 header in sample.');
65
- }
66
-
67
- const word = view.getUint32(0, false);
68
- const header = readFrameHeader(word, null).header;
69
- if (!header) {
70
- throw new Error('Invalid MP3 header in sample.');
71
- }
72
-
73
- const xingOffset = getXingOffset(header.mpegVersionId, header.channel);
74
- if (view.byteLength >= xingOffset + 4) {
75
- const word = view.getUint32(xingOffset, false);
76
- const isXing = word === XING || word === INFO;
77
-
78
- if (isXing) {
79
- // This is not a data frame, so let's completely ignore this sample
80
- return;
81
- }
82
- }
83
-
84
- this.xingFrameData = {
85
- mpegVersionId: header.mpegVersionId,
86
- layer: header.layer,
87
- frequencyIndex: header.frequencyIndex,
88
- sampleRate: header.sampleRate,
89
- channel: header.channel,
90
- modeExtension: header.modeExtension,
91
- copyright: header.copyright,
92
- original: header.original,
93
- emphasis: header.emphasis,
94
-
95
- frameCount: null,
96
- fileSize: null,
97
- toc: null,
98
- };
99
-
100
- // Write a Xing frame because this muxer doesn't make any bitrate constraints, meaning we don't know if
101
- // this will be a constant or variable bitrate file. Therefore, always write the Xing frame.
102
- this.xingFramePos = this.writer.getPos();
103
- this.mp3Writer.writeXingFrame(this.xingFrameData);
104
-
105
- this.frameCount++;
106
- }
107
-
108
- this.validateAndNormalizeTimestamp(track, packet.timestamp, packet.type === 'key');
109
-
110
- this.writer.write(packet.data);
111
- this.frameCount++;
112
-
113
- await this.writer.flush();
114
-
115
- if (writeXingHeader) {
116
- this.framePositions.push(this.writer.getPos());
117
- }
118
- } finally {
119
- release();
120
- }
121
- }
122
-
123
- async addSubtitleCue() {
124
- throw new Error('MP3 does not support subtitles.');
125
- }
126
-
127
- async finalize() {
128
- if (!this.xingFrameData || this.xingFramePos === null) {
129
- return;
130
- }
131
-
132
- const release = await this.mutex.acquire();
133
-
134
- const endPos = this.writer.getPos();
135
-
136
- this.writer.seek(this.xingFramePos);
137
-
138
- const toc = new Uint8Array(100);
139
- for (let i = 0; i < 100; i++) {
140
- const index = Math.floor(this.framePositions.length * (i / 100));
141
- assert(index !== -1 && index < this.framePositions.length);
142
-
143
- const byteOffset = this.framePositions[index]!;
144
- toc[i] = 256 * (byteOffset / endPos);
145
- }
146
-
147
- this.xingFrameData.frameCount = this.frameCount;
148
- this.xingFrameData.fileSize = endPos;
149
- this.xingFrameData.toc = toc;
150
-
151
- if (this.format._options.onXingFrame) {
152
- this.writer.startTrackingWrites();
153
- }
154
-
155
- this.mp3Writer.writeXingFrame(this.xingFrameData);
156
-
157
- if (this.format._options.onXingFrame) {
158
- const { data, start } = this.writer.stopTrackingWrites();
159
- this.format._options.onXingFrame(data, start);
160
- }
161
-
162
- this.writer.seek(endPos);
163
-
164
- release();
165
- }
166
- }
@@ -1,34 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
-
9
- import { FRAME_HEADER_SIZE, FrameHeader, readFrameHeader } from '../../shared/mp3-misc';
10
- import { Reader, readU32Be } from '../reader';
11
-
12
- export const readNextFrameHeader = async (reader: Reader, startPos: number, until: number | null): Promise<{
13
- header: FrameHeader;
14
- startPos: number;
15
- } | null> => {
16
- let currentPos = startPos;
17
-
18
- while (until === null || currentPos < until) {
19
- let slice = reader.requestSlice(currentPos, FRAME_HEADER_SIZE);
20
- if (slice instanceof Promise) slice = await slice;
21
- if (!slice) break;
22
-
23
- const word = readU32Be(slice);
24
-
25
- const result = readFrameHeader(word, reader.fileSize !== null ? reader.fileSize - currentPos : null);
26
- if (result.header) {
27
- return { header: result.header, startPos: currentPos };
28
- }
29
-
30
- currentPos += result.bytesAdvanced;
31
- }
32
-
33
- return null;
34
- };
@@ -1,120 +0,0 @@
1
- /*!
2
- * Copyright (c) 2025-present, Vanilagy and contributors
3
- *
4
- * This Source Code Form is subject to the terms of the Mozilla Public
5
- * License, v. 2.0. If a copy of the MPL was not distributed with this
6
- * file, You can obtain one at https://mozilla.org/MPL/2.0/.
7
- */
8
-
9
- import { Writer } from '../writer';
10
- import {
11
- computeMp3FrameSize,
12
- getXingOffset,
13
- KILOBIT_RATES,
14
- XING,
15
- } from '../../shared/mp3-misc';
16
-
17
- export type XingFrameData = {
18
- mpegVersionId: number;
19
- layer: number;
20
- frequencyIndex: number;
21
- sampleRate: number;
22
- channel: number;
23
- modeExtension: number;
24
- copyright: number;
25
- original: number;
26
- emphasis: number;
27
-
28
- frameCount: number | null;
29
- fileSize: number | null;
30
- toc: Uint8Array | null;
31
- };
32
-
33
- export class Mp3Writer {
34
- private helper = new Uint8Array(8);
35
- private helperView = new DataView(this.helper.buffer);
36
-
37
- constructor(private writer: Writer) {}
38
-
39
- writeU32(value: number) {
40
- this.helperView.setUint32(0, value, false);
41
- this.writer.write(this.helper.subarray(0, 4));
42
- }
43
-
44
- writeXingFrame(data: XingFrameData) {
45
- const startPos = this.writer.getPos();
46
-
47
- const firstByte = 0xff;
48
- const secondByte = 0xe0 | (data.mpegVersionId << 3) | (data.layer << 1);
49
-
50
- let lowSamplingFrequency: number;
51
- if (data.mpegVersionId & 2) {
52
- lowSamplingFrequency = (data.mpegVersionId & 1) ? 0 : 1;
53
- } else {
54
- lowSamplingFrequency = 1;
55
- }
56
-
57
- const padding = 0;
58
- const neededBytes = 155;
59
-
60
- let bitrateIndex = -1;
61
- const bitrateOffset = lowSamplingFrequency * 16 * 4 + data.layer * 16;
62
-
63
- // Let's find the lowest bitrate for which the frame size is sufficiently large to fit all the data
64
- for (let i = 0; i < 16; i++) {
65
- const kbr = KILOBIT_RATES[bitrateOffset + i]!;
66
- const size = computeMp3FrameSize(lowSamplingFrequency, data.layer, 1000 * kbr, data.sampleRate, padding);
67
-
68
- if (size >= neededBytes) {
69
- bitrateIndex = i;
70
- break;
71
- }
72
- }
73
-
74
- if (bitrateIndex === -1) {
75
- throw new Error('No suitable bitrate found.');
76
- }
77
-
78
- const thirdByte = (bitrateIndex << 4) | (data.frequencyIndex << 2) | padding << 1;
79
- const fourthByte = (data.channel << 6)
80
- | (data.modeExtension << 4)
81
- | (data.copyright << 3)
82
- | (data.original << 2)
83
- | data.emphasis;
84
-
85
- this.helper[0] = firstByte;
86
- this.helper[1] = secondByte;
87
- this.helper[2] = thirdByte;
88
- this.helper[3] = fourthByte;
89
-
90
- this.writer.write(this.helper.subarray(0, 4));
91
-
92
- const xingOffset = getXingOffset(data.mpegVersionId, data.channel);
93
-
94
- this.writer.seek(startPos + xingOffset);
95
- this.writeU32(XING);
96
-
97
- let flags = 0;
98
- if (data.frameCount !== null) {
99
- flags |= 1;
100
- }
101
- if (data.fileSize !== null) {
102
- flags |= 2;
103
- }
104
- if (data.toc !== null) {
105
- flags |= 4;
106
- }
107
-
108
- this.writeU32(flags);
109
-
110
- this.writeU32(data.frameCount ?? 0);
111
- this.writeU32(data.fileSize ?? 0);
112
- this.writer.write(data.toc ?? new Uint8Array(100));
113
-
114
- const kilobitRate = KILOBIT_RATES[bitrateOffset + bitrateIndex]!;
115
- const frameSize = computeMp3FrameSize(
116
- lowSamplingFrequency, data.layer, 1000 * kilobitRate, data.sampleRate, padding,
117
- );
118
- this.writer.seek(startPos + frameSize);
119
- }
120
- }