@remotion/media-parser 4.0.201 → 4.0.202
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/boxes/iso-base-media/mvhd.js +2 -2
- package/dist/boxes/iso-base-media/stsd/keys.js +1 -1
- package/dist/boxes/webm/av1-codec-private.js +1 -1
- package/dist/boxes/webm/description.d.ts +2 -2
- package/dist/boxes/webm/description.js +2 -2
- package/dist/boxes/webm/ebml.d.ts +2 -2
- package/dist/boxes/webm/ebml.js +23 -1
- package/dist/boxes/webm/get-ready-tracks.d.ts +1 -1
- package/dist/boxes/webm/get-ready-tracks.js +3 -3
- package/dist/boxes/webm/get-sample-from-block.d.ts +17 -0
- package/dist/boxes/webm/get-sample-from-block.js +78 -0
- package/dist/boxes/webm/get-track.d.ts +2 -2
- package/dist/boxes/webm/get-track.js +26 -25
- package/dist/boxes/webm/make-header.d.ts +3 -8
- package/dist/boxes/webm/make-header.js +43 -20
- package/dist/boxes/webm/parse-ebml.d.ts +9 -4
- package/dist/boxes/webm/parse-ebml.js +122 -13
- package/dist/boxes/webm/segments/all-segments.d.ts +421 -107
- package/dist/boxes/webm/segments/all-segments.js +260 -33
- package/dist/boxes/webm/segments/seek-position.js +1 -1
- package/dist/boxes/webm/segments/seek.d.ts +1 -1
- package/dist/boxes/webm/segments/seek.js +2 -8
- package/dist/boxes/webm/segments/timestamp-scale.js +1 -1
- package/dist/boxes/webm/segments/track-entry.d.ts +3 -191
- package/dist/boxes/webm/segments/track-entry.js +2 -456
- package/dist/boxes/webm/segments.d.ts +3 -16
- package/dist/boxes/webm/segments.js +12 -196
- package/dist/boxes/webm/traversal.d.ts +5 -6
- package/dist/boxes/webm/traversal.js +6 -6
- package/dist/buffer-iterator.d.ts +1 -1
- package/dist/buffer-iterator.js +3 -3
- package/dist/create/create-media.d.ts +2 -0
- package/dist/create/create-media.js +36 -0
- package/dist/create/matroska-header.d.ts +1 -0
- package/dist/create/matroska-header.js +66 -0
- package/dist/create/matroska-info.d.ts +4 -0
- package/dist/create/matroska-info.js +39 -0
- package/dist/create/matroska-segment.d.ts +1 -0
- package/dist/create/matroska-segment.js +12 -0
- package/dist/create/matroska-trackentry.d.ts +21 -0
- package/dist/create/matroska-trackentry.js +191 -0
- package/dist/create-media.d.ts +1 -0
- package/dist/create-media.js +78 -0
- package/dist/get-audio-codec.d.ts +1 -1
- package/dist/get-audio-codec.js +13 -13
- package/dist/get-duration.js +12 -14
- package/dist/get-tracks.js +2 -2
- package/dist/get-video-codec.js +13 -13
- package/dist/parse-media.js +4 -1
- package/dist/parser-context.d.ts +1 -0
- package/dist/parser-state.js +3 -2
- package/dist/readers/from-fetch.d.ts +2 -0
- package/dist/readers/from-fetch.js +64 -0
- package/dist/readers/from-node.d.ts +2 -0
- package/dist/readers/from-node.js +40 -0
- package/dist/readers/from-web-file.d.ts +2 -0
- package/dist/readers/from-web-file.js +39 -0
- package/dist/readers/reader.d.ts +11 -0
- package/dist/readers/reader.js +2 -0
- package/dist/traversal.d.ts +19 -17
- package/dist/traversal.js +38 -39
- package/dist/writers/web-fs.d.ts +2 -0
- package/dist/writers/web-fs.js +28 -0
- package/dist/writers/writer.d.ts +9 -0
- package/dist/writers/writer.js +2 -0
- package/input.webm +0 -0
- package/package.json +2 -2
- package/src/boxes/iso-base-media/mvhd.ts +2 -2
- package/src/boxes/iso-base-media/stsd/keys.ts +1 -1
- package/src/boxes/webm/av1-codec-private.ts +1 -1
- package/src/boxes/webm/description.ts +7 -4
- package/src/boxes/webm/ebml.ts +24 -4
- package/src/boxes/webm/get-ready-tracks.ts +4 -4
- package/src/boxes/webm/get-sample-from-block.ts +125 -0
- package/src/boxes/webm/get-track.ts +38 -31
- package/src/boxes/webm/make-header.ts +58 -51
- package/src/boxes/webm/parse-ebml.ts +170 -16
- package/src/boxes/webm/segments/all-segments.ts +379 -62
- package/src/boxes/webm/segments/track-entry.ts +3 -846
- package/src/boxes/webm/segments.ts +18 -410
- package/src/boxes/webm/traversal.ts +17 -17
- package/src/buffer-iterator.ts +5 -4
- package/src/get-audio-codec.ts +14 -16
- package/src/get-duration.ts +15 -16
- package/src/get-tracks.ts +2 -2
- package/src/get-video-codec.ts +13 -15
- package/src/parse-media.ts +6 -1
- package/src/parser-context.ts +1 -0
- package/src/parser-state.ts +2 -2
- package/src/test/create-matroska.test.ts +237 -23
- package/src/test/matroska.test.ts +283 -348
- package/src/test/mvhd.test.ts +1 -1
- package/src/test/parse-esds.test.ts +2 -2
- package/src/test/parse-stco.test.ts +2 -2
- package/src/test/parse-stsc.test.ts +2 -2
- package/src/test/parse-stsz.test.ts +2 -2
- package/src/test/parse-stts.test.ts +1 -1
- package/src/test/stsd.test.ts +4 -2
- package/src/test/tkhd.test.ts +1 -1
- package/src/traversal.ts +62 -85
- package/tsconfig.tsbuildinfo +1 -1
- package/src/boxes/webm/segments/duration.ts +0 -29
- package/src/boxes/webm/segments/info.ts +0 -34
- package/src/boxes/webm/segments/main.ts +0 -6
- package/src/boxes/webm/segments/muxing.ts +0 -18
- package/src/boxes/webm/segments/seek-head.ts +0 -34
- package/src/boxes/webm/segments/seek-position.ts +0 -18
- package/src/boxes/webm/segments/seek.ts +0 -55
- package/src/boxes/webm/segments/timestamp-scale.ts +0 -17
- package/src/boxes/webm/segments/tracks.ts +0 -32
- package/src/boxes/webm/segments/void.ts +0 -18
- package/src/boxes/webm/segments/writing.ts +0 -18
- package/src/combine-uint8array.ts +0 -13
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.makeMatroskaTracks = exports.makeMatroskaVideoTrackEntryBytes = exports.makeMatroskaVideoBytes = exports.makeMatroskaColorBytes = void 0;
|
|
4
|
+
const make_header_1 = require("../boxes/webm/make-header");
|
|
5
|
+
const makeMatroskaColorBytes = ({ transferChracteristics, matrixCoefficients, primaries, fullRange, }) => {
|
|
6
|
+
const rangeValue = transferChracteristics && matrixCoefficients
|
|
7
|
+
? 3
|
|
8
|
+
: fullRange === true
|
|
9
|
+
? 2
|
|
10
|
+
: fullRange === false
|
|
11
|
+
? 1
|
|
12
|
+
: 0;
|
|
13
|
+
// https://datatracker.ietf.org/doc/draft-ietf-cellar-matroska/
|
|
14
|
+
// 5.1.4.1.28.27
|
|
15
|
+
const primariesValue = primaries === 'bt709'
|
|
16
|
+
? 1
|
|
17
|
+
: primaries === 'smpte170m'
|
|
18
|
+
? 6
|
|
19
|
+
: primaries === 'bt470bg'
|
|
20
|
+
? 5
|
|
21
|
+
: 2;
|
|
22
|
+
const transferChracteristicsValue = transferChracteristics === 'bt709'
|
|
23
|
+
? 1
|
|
24
|
+
: transferChracteristics === 'smpte170m'
|
|
25
|
+
? 6
|
|
26
|
+
: transferChracteristics === 'iec61966-2-1'
|
|
27
|
+
? 13
|
|
28
|
+
: 2;
|
|
29
|
+
if (matrixCoefficients === 'rgb') {
|
|
30
|
+
throw new Error('Cannot encode Matroska in RGB');
|
|
31
|
+
}
|
|
32
|
+
const matrixCoefficientsValue = matrixCoefficients === 'bt709'
|
|
33
|
+
? 1
|
|
34
|
+
: matrixCoefficients === 'bt470bg'
|
|
35
|
+
? 5
|
|
36
|
+
: matrixCoefficients === 'smpte170m'
|
|
37
|
+
? 6
|
|
38
|
+
: 2;
|
|
39
|
+
return (0, make_header_1.makeMatroskaBytes)({
|
|
40
|
+
type: 'Colour',
|
|
41
|
+
minVintWidth: null,
|
|
42
|
+
value: [
|
|
43
|
+
{
|
|
44
|
+
type: 'TransferCharacteristics',
|
|
45
|
+
value: {
|
|
46
|
+
value: transferChracteristicsValue,
|
|
47
|
+
byteLength: null,
|
|
48
|
+
},
|
|
49
|
+
minVintWidth: null,
|
|
50
|
+
},
|
|
51
|
+
{
|
|
52
|
+
type: 'MatrixCoefficients',
|
|
53
|
+
value: {
|
|
54
|
+
value: matrixCoefficientsValue,
|
|
55
|
+
byteLength: null,
|
|
56
|
+
},
|
|
57
|
+
minVintWidth: null,
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
type: 'Primaries',
|
|
61
|
+
value: {
|
|
62
|
+
value: primariesValue,
|
|
63
|
+
byteLength: null,
|
|
64
|
+
},
|
|
65
|
+
minVintWidth: null,
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
type: 'Range',
|
|
69
|
+
value: {
|
|
70
|
+
value: rangeValue,
|
|
71
|
+
byteLength: null,
|
|
72
|
+
},
|
|
73
|
+
minVintWidth: null,
|
|
74
|
+
},
|
|
75
|
+
],
|
|
76
|
+
});
|
|
77
|
+
};
|
|
78
|
+
exports.makeMatroskaColorBytes = makeMatroskaColorBytes;
|
|
79
|
+
const makeMatroskaVideoBytes = ({ color, width, height, }) => {
|
|
80
|
+
return (0, make_header_1.makeMatroskaBytes)({
|
|
81
|
+
type: 'Video',
|
|
82
|
+
value: [
|
|
83
|
+
{
|
|
84
|
+
type: 'PixelWidth',
|
|
85
|
+
value: {
|
|
86
|
+
value: width,
|
|
87
|
+
byteLength: null,
|
|
88
|
+
},
|
|
89
|
+
minVintWidth: null,
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
type: 'PixelHeight',
|
|
93
|
+
value: {
|
|
94
|
+
value: height,
|
|
95
|
+
byteLength: null,
|
|
96
|
+
},
|
|
97
|
+
minVintWidth: null,
|
|
98
|
+
},
|
|
99
|
+
{
|
|
100
|
+
type: 'FlagInterlaced',
|
|
101
|
+
value: {
|
|
102
|
+
// https://datatracker.ietf.org/doc/draft-ietf-cellar-matroska/
|
|
103
|
+
// 5.1.4.1.28.1.
|
|
104
|
+
value: 2, // 2 - progressive, no interlaced
|
|
105
|
+
byteLength: null,
|
|
106
|
+
},
|
|
107
|
+
minVintWidth: null,
|
|
108
|
+
},
|
|
109
|
+
(0, exports.makeMatroskaColorBytes)(color),
|
|
110
|
+
],
|
|
111
|
+
minVintWidth: null,
|
|
112
|
+
});
|
|
113
|
+
};
|
|
114
|
+
exports.makeMatroskaVideoBytes = makeMatroskaVideoBytes;
|
|
115
|
+
const makeMatroskaVideoTrackEntryBytes = ({ color, width, height, defaultDuration, trackNumber, codecId, }) => {
|
|
116
|
+
return (0, make_header_1.makeMatroskaBytes)({
|
|
117
|
+
type: 'TrackEntry',
|
|
118
|
+
minVintWidth: null,
|
|
119
|
+
value: [
|
|
120
|
+
{
|
|
121
|
+
type: 'TrackNumber',
|
|
122
|
+
value: {
|
|
123
|
+
value: trackNumber,
|
|
124
|
+
byteLength: null,
|
|
125
|
+
},
|
|
126
|
+
minVintWidth: null,
|
|
127
|
+
},
|
|
128
|
+
{
|
|
129
|
+
type: 'TrackUID',
|
|
130
|
+
value: '0xab2171012bb9020a',
|
|
131
|
+
minVintWidth: null,
|
|
132
|
+
},
|
|
133
|
+
{
|
|
134
|
+
type: 'FlagLacing',
|
|
135
|
+
value: {
|
|
136
|
+
value: 0,
|
|
137
|
+
byteLength: null,
|
|
138
|
+
},
|
|
139
|
+
minVintWidth: null,
|
|
140
|
+
},
|
|
141
|
+
{
|
|
142
|
+
type: 'Language',
|
|
143
|
+
value: 'und',
|
|
144
|
+
minVintWidth: null,
|
|
145
|
+
},
|
|
146
|
+
{
|
|
147
|
+
type: 'FlagDefault',
|
|
148
|
+
value: {
|
|
149
|
+
value: 0,
|
|
150
|
+
byteLength: null,
|
|
151
|
+
},
|
|
152
|
+
minVintWidth: null,
|
|
153
|
+
},
|
|
154
|
+
{
|
|
155
|
+
type: 'CodecID',
|
|
156
|
+
value: codecId,
|
|
157
|
+
minVintWidth: null,
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
type: 'TrackType',
|
|
161
|
+
value: {
|
|
162
|
+
value: 1, // 'video'
|
|
163
|
+
byteLength: null,
|
|
164
|
+
},
|
|
165
|
+
minVintWidth: null,
|
|
166
|
+
},
|
|
167
|
+
{
|
|
168
|
+
type: 'DefaultDuration',
|
|
169
|
+
value: {
|
|
170
|
+
value: defaultDuration,
|
|
171
|
+
byteLength: null,
|
|
172
|
+
},
|
|
173
|
+
minVintWidth: null,
|
|
174
|
+
},
|
|
175
|
+
(0, exports.makeMatroskaVideoBytes)({
|
|
176
|
+
color,
|
|
177
|
+
width,
|
|
178
|
+
height,
|
|
179
|
+
}),
|
|
180
|
+
],
|
|
181
|
+
});
|
|
182
|
+
};
|
|
183
|
+
exports.makeMatroskaVideoTrackEntryBytes = makeMatroskaVideoTrackEntryBytes;
|
|
184
|
+
const makeMatroskaTracks = (tracks) => {
|
|
185
|
+
return (0, make_header_1.makeMatroskaBytes)({
|
|
186
|
+
type: 'Tracks',
|
|
187
|
+
value: tracks,
|
|
188
|
+
minVintWidth: null,
|
|
189
|
+
});
|
|
190
|
+
};
|
|
191
|
+
exports.makeMatroskaTracks = makeMatroskaTracks;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare const createMedia: () => Promise<void>;
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.createMedia = void 0;
|
|
4
|
+
const make_header_1 = require("./boxes/webm/make-header");
|
|
5
|
+
const createMedia = async () => {
|
|
6
|
+
const header = (0, make_header_1.makeMatroskaBytes)({
|
|
7
|
+
type: 'Header',
|
|
8
|
+
value: [
|
|
9
|
+
{
|
|
10
|
+
minVintWidth: null,
|
|
11
|
+
type: 'EBMLVersion',
|
|
12
|
+
value: {
|
|
13
|
+
value: 1,
|
|
14
|
+
byteLength: null,
|
|
15
|
+
},
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
minVintWidth: null,
|
|
19
|
+
type: 'EBMLReadVersion',
|
|
20
|
+
value: {
|
|
21
|
+
value: 1,
|
|
22
|
+
byteLength: null,
|
|
23
|
+
},
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
type: 'EBMLMaxIDLength',
|
|
27
|
+
value: {
|
|
28
|
+
byteLength: null,
|
|
29
|
+
value: 4,
|
|
30
|
+
},
|
|
31
|
+
minVintWidth: null,
|
|
32
|
+
},
|
|
33
|
+
{
|
|
34
|
+
type: 'EBMLMaxSizeLength',
|
|
35
|
+
value: {
|
|
36
|
+
byteLength: null,
|
|
37
|
+
value: 8,
|
|
38
|
+
},
|
|
39
|
+
minVintWidth: null,
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
type: 'DocType',
|
|
43
|
+
value: 'webm',
|
|
44
|
+
minVintWidth: null,
|
|
45
|
+
},
|
|
46
|
+
{
|
|
47
|
+
type: 'DocTypeVersion',
|
|
48
|
+
value: {
|
|
49
|
+
byteLength: null,
|
|
50
|
+
value: 4,
|
|
51
|
+
},
|
|
52
|
+
minVintWidth: null,
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
type: 'DocTypeReadVersion',
|
|
56
|
+
value: {
|
|
57
|
+
byteLength: null,
|
|
58
|
+
value: 2,
|
|
59
|
+
},
|
|
60
|
+
minVintWidth: null,
|
|
61
|
+
},
|
|
62
|
+
],
|
|
63
|
+
minVintWidth: null,
|
|
64
|
+
});
|
|
65
|
+
const handle = await window.showSaveFilePicker({
|
|
66
|
+
suggestedName: 'out.webm',
|
|
67
|
+
types: [
|
|
68
|
+
{
|
|
69
|
+
description: 'WebM video',
|
|
70
|
+
accept: { 'video/webm': ['.webm'] },
|
|
71
|
+
},
|
|
72
|
+
],
|
|
73
|
+
});
|
|
74
|
+
const writable = await handle.createWritable();
|
|
75
|
+
await writable.write(header);
|
|
76
|
+
writable.close();
|
|
77
|
+
};
|
|
78
|
+
exports.createMedia = createMedia;
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import type { MoovBox } from './boxes/iso-base-media/moov/moov';
|
|
2
2
|
import type { TrakBox } from './boxes/iso-base-media/trak/trak';
|
|
3
|
-
import type { MainSegment } from './boxes/webm/segments/
|
|
3
|
+
import type { MainSegment } from './boxes/webm/segments/all-segments';
|
|
4
4
|
import type { KnownAudioCodecs } from './options';
|
|
5
5
|
import type { AnySegment } from './parse-result';
|
|
6
6
|
export declare const hasAudioCodec: (boxes: AnySegment[]) => boolean;
|
package/dist/get-audio-codec.js
CHANGED
|
@@ -115,27 +115,27 @@ const getAudioCodecFromIso = (moov) => {
|
|
|
115
115
|
};
|
|
116
116
|
exports.getAudioCodecFromIso = getAudioCodecFromIso;
|
|
117
117
|
const getAudioCodecFromMatroska = (mainSegment) => {
|
|
118
|
-
const tracksSegment = mainSegment.
|
|
119
|
-
if (!tracksSegment || tracksSegment.type !== '
|
|
118
|
+
const tracksSegment = mainSegment.value.find((b) => b.type === 'Tracks');
|
|
119
|
+
if (!tracksSegment || tracksSegment.type !== 'Tracks') {
|
|
120
120
|
return null;
|
|
121
121
|
}
|
|
122
|
-
for (const track of tracksSegment.
|
|
123
|
-
if (track.type === '
|
|
124
|
-
const trackType = track.
|
|
125
|
-
if (trackType && trackType.type === '
|
|
126
|
-
if (trackType.
|
|
122
|
+
for (const track of tracksSegment.value) {
|
|
123
|
+
if (track.type === 'TrackEntry') {
|
|
124
|
+
const trackType = track.value.find((b) => b.type === 'CodecID');
|
|
125
|
+
if (trackType && trackType.type === 'CodecID') {
|
|
126
|
+
if (trackType.value === 'A_OPUS') {
|
|
127
127
|
return 'opus';
|
|
128
128
|
}
|
|
129
|
-
if (trackType.
|
|
129
|
+
if (trackType.value === 'A_VORBIS') {
|
|
130
130
|
return 'vorbis';
|
|
131
131
|
}
|
|
132
|
-
if (trackType.
|
|
132
|
+
if (trackType.value === 'A_PCM/INT/LIT') {
|
|
133
133
|
return 'pcm';
|
|
134
134
|
}
|
|
135
|
-
if (trackType.
|
|
135
|
+
if (trackType.value === 'A_AAC') {
|
|
136
136
|
return 'aac';
|
|
137
137
|
}
|
|
138
|
-
if (trackType.
|
|
138
|
+
if (trackType.value === 'A_MPEG/L3') {
|
|
139
139
|
return 'mp3';
|
|
140
140
|
}
|
|
141
141
|
}
|
|
@@ -190,8 +190,8 @@ const getAudioCodec = (boxes) => {
|
|
|
190
190
|
}
|
|
191
191
|
throw new Error('Unknown audio format: ' + codec.format);
|
|
192
192
|
}
|
|
193
|
-
const mainSegment = boxes.find((b) => b.type === '
|
|
194
|
-
if (!mainSegment || mainSegment.type !== '
|
|
193
|
+
const mainSegment = boxes.find((b) => b.type === 'Segment');
|
|
194
|
+
if (!mainSegment || mainSegment.type !== 'Segment') {
|
|
195
195
|
return null;
|
|
196
196
|
}
|
|
197
197
|
return (0, exports.getAudioCodecFromMatroska)(mainSegment);
|
package/dist/get-duration.js
CHANGED
|
@@ -3,33 +3,31 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.hasDuration = exports.getDuration = void 0;
|
|
4
4
|
const traversal_1 = require("./traversal");
|
|
5
5
|
const getDurationFromMatroska = (segments) => {
|
|
6
|
-
const mainSegment = segments.find((s) => s.type === '
|
|
7
|
-
if (!mainSegment || mainSegment.type !== '
|
|
6
|
+
const mainSegment = segments.find((s) => s.type === 'Segment');
|
|
7
|
+
if (!mainSegment || mainSegment.type !== 'Segment') {
|
|
8
8
|
return null;
|
|
9
9
|
}
|
|
10
|
-
const { children } = mainSegment;
|
|
10
|
+
const { value: children } = mainSegment;
|
|
11
11
|
if (!children) {
|
|
12
12
|
return null;
|
|
13
13
|
}
|
|
14
|
-
const infoSegment = children.find((s) => s.type === '
|
|
14
|
+
const infoSegment = children.find((s) => s.type === 'Info');
|
|
15
15
|
const relevantBoxes = [
|
|
16
|
-
...mainSegment.
|
|
17
|
-
...(infoSegment && infoSegment.type === '
|
|
18
|
-
? infoSegment.children
|
|
19
|
-
: []),
|
|
16
|
+
...mainSegment.value,
|
|
17
|
+
...(infoSegment && infoSegment.type === 'Info' ? infoSegment.value : []),
|
|
20
18
|
];
|
|
21
|
-
const timestampScale = relevantBoxes.find((s) => s.type === '
|
|
22
|
-
if (!timestampScale || timestampScale.type !== '
|
|
19
|
+
const timestampScale = relevantBoxes.find((s) => s.type === 'TimestampScale');
|
|
20
|
+
if (!timestampScale || timestampScale.type !== 'TimestampScale') {
|
|
23
21
|
return null;
|
|
24
22
|
}
|
|
25
|
-
const duration = relevantBoxes.find((s) => s.type === '
|
|
26
|
-
if (!duration || duration.type !== '
|
|
23
|
+
const duration = relevantBoxes.find((s) => s.type === 'Duration');
|
|
24
|
+
if (!duration || duration.type !== 'Duration') {
|
|
27
25
|
return null;
|
|
28
26
|
}
|
|
29
|
-
return (duration.
|
|
27
|
+
return (duration.value.value / timestampScale.value.value) * 1000;
|
|
30
28
|
};
|
|
31
29
|
const getDuration = (boxes) => {
|
|
32
|
-
const matroskaBox = boxes.find((b) => b.type === '
|
|
30
|
+
const matroskaBox = boxes.find((b) => b.type === 'Segment');
|
|
33
31
|
if (matroskaBox) {
|
|
34
32
|
return getDurationFromMatroska(boxes);
|
|
35
33
|
}
|
package/dist/get-tracks.js
CHANGED
|
@@ -31,8 +31,8 @@ const getTracks = (segments, state) => {
|
|
|
31
31
|
const videoTracks = [];
|
|
32
32
|
const audioTracks = [];
|
|
33
33
|
const otherTracks = [];
|
|
34
|
-
const mainSegment = segments.find((s) => s.type === '
|
|
35
|
-
if (mainSegment && mainSegment.type === '
|
|
34
|
+
const mainSegment = segments.find((s) => s.type === 'Segment');
|
|
35
|
+
if (mainSegment && mainSegment.type === 'Segment') {
|
|
36
36
|
const matroskaTracks = (0, get_ready_tracks_1.getTracksFromMatroska)(mainSegment, state.getTimescale());
|
|
37
37
|
for (const track of matroskaTracks) {
|
|
38
38
|
if (track.type === 'video') {
|
package/dist/get-video-codec.js
CHANGED
|
@@ -60,31 +60,31 @@ const getVideoCodec = (boxes) => {
|
|
|
60
60
|
}
|
|
61
61
|
}
|
|
62
62
|
}
|
|
63
|
-
const mainSegment = boxes.find((b) => b.type === '
|
|
64
|
-
if (!mainSegment || mainSegment.type !== '
|
|
63
|
+
const mainSegment = boxes.find((b) => b.type === 'Segment');
|
|
64
|
+
if (!mainSegment || mainSegment.type !== 'Segment') {
|
|
65
65
|
return null;
|
|
66
66
|
}
|
|
67
|
-
const tracksSegment = mainSegment.
|
|
68
|
-
if (!tracksSegment || tracksSegment.type !== '
|
|
67
|
+
const tracksSegment = mainSegment.value.find((b) => b.type === 'Tracks');
|
|
68
|
+
if (!tracksSegment || tracksSegment.type !== 'Tracks') {
|
|
69
69
|
return null;
|
|
70
70
|
}
|
|
71
|
-
for (const track of tracksSegment.
|
|
72
|
-
if (track.type === '
|
|
73
|
-
const trackType = track.
|
|
74
|
-
if (trackType && trackType.type === '
|
|
75
|
-
if (trackType.
|
|
71
|
+
for (const track of tracksSegment.value) {
|
|
72
|
+
if (track.type === 'TrackEntry') {
|
|
73
|
+
const trackType = track.value.find((b) => b.type === 'CodecID');
|
|
74
|
+
if (trackType && trackType.type === 'CodecID') {
|
|
75
|
+
if (trackType.value === 'V_VP8') {
|
|
76
76
|
return 'vp8';
|
|
77
77
|
}
|
|
78
|
-
if (trackType.
|
|
78
|
+
if (trackType.value === 'V_VP9') {
|
|
79
79
|
return 'vp9';
|
|
80
80
|
}
|
|
81
|
-
if (trackType.
|
|
81
|
+
if (trackType.value === 'V_AV1') {
|
|
82
82
|
return 'av1';
|
|
83
83
|
}
|
|
84
|
-
if (trackType.
|
|
84
|
+
if (trackType.value === 'V_MPEG4/ISO/AVC') {
|
|
85
85
|
return 'h264';
|
|
86
86
|
}
|
|
87
|
-
if (trackType.
|
|
87
|
+
if (trackType.value === 'V_MPEGH/ISO/HEVC') {
|
|
88
88
|
return 'h265';
|
|
89
89
|
}
|
|
90
90
|
}
|
package/dist/parse-media.js
CHANGED
|
@@ -28,6 +28,9 @@ const parseMedia = async ({ src, fields, reader: readerInterface = from_fetch_1.
|
|
|
28
28
|
onAudioTrack: onAudioTrack !== null && onAudioTrack !== void 0 ? onAudioTrack : null,
|
|
29
29
|
onVideoTrack: onVideoTrack !== null && onVideoTrack !== void 0 ? onVideoTrack : null,
|
|
30
30
|
parserState: state,
|
|
31
|
+
nullifySamples: !(typeof process !== 'undefined' &&
|
|
32
|
+
typeof process.env !== 'undefined' &&
|
|
33
|
+
process.env.KEEP_SAMPLES === 'true'),
|
|
31
34
|
};
|
|
32
35
|
while (parseResult === null || parseResult.status === 'incomplete') {
|
|
33
36
|
if (signal === null || signal === void 0 ? void 0 : signal.aborted) {
|
|
@@ -43,7 +46,7 @@ const parseMedia = async ({ src, fields, reader: readerInterface = from_fetch_1.
|
|
|
43
46
|
if (result.done) {
|
|
44
47
|
throw new Error('Unexpectedly reached EOF');
|
|
45
48
|
}
|
|
46
|
-
iterator = (0, buffer_iterator_1.getArrayBufferIterator)(result.value, contentLength !== null && contentLength !== void 0 ? contentLength :
|
|
49
|
+
iterator = (0, buffer_iterator_1.getArrayBufferIterator)(result.value, contentLength !== null && contentLength !== void 0 ? contentLength : 1000000000);
|
|
47
50
|
}
|
|
48
51
|
if (parseResult && parseResult.status === 'incomplete') {
|
|
49
52
|
parseResult = await parseResult.continueParsing();
|
package/dist/parser-context.d.ts
CHANGED
package/dist/parser-state.js
CHANGED
|
@@ -6,6 +6,7 @@ const traversal_2 = require("./traversal");
|
|
|
6
6
|
const makeParserState = ({ hasAudioCallbacks, hasVideoCallbacks, signal, }) => {
|
|
7
7
|
const trackEntries = {};
|
|
8
8
|
const onTrackEntrySegment = (trackEntry) => {
|
|
9
|
+
var _a;
|
|
9
10
|
const trackId = (0, traversal_2.getTrackId)(trackEntry);
|
|
10
11
|
if (!trackId) {
|
|
11
12
|
throw new Error('Expected track id');
|
|
@@ -19,8 +20,8 @@ const makeParserState = ({ hasAudioCallbacks, hasVideoCallbacks, signal, }) => {
|
|
|
19
20
|
}
|
|
20
21
|
const trackTimescale = (0, traversal_1.getTrackTimestampScale)(trackEntry);
|
|
21
22
|
trackEntries[trackId] = {
|
|
22
|
-
codec: codec.
|
|
23
|
-
trackTimescale,
|
|
23
|
+
codec: codec.value,
|
|
24
|
+
trackTimescale: (_a = trackTimescale === null || trackTimescale === void 0 ? void 0 : trackTimescale.value) !== null && _a !== void 0 ? _a : null,
|
|
24
25
|
};
|
|
25
26
|
};
|
|
26
27
|
const videoSampleCallbacks = {};
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.fetchReader = void 0;
|
|
4
|
+
exports.fetchReader = {
|
|
5
|
+
read: async (src, range, signal) => {
|
|
6
|
+
if (typeof src !== 'string') {
|
|
7
|
+
throw new Error('src must be a string when using `fetchReader`');
|
|
8
|
+
}
|
|
9
|
+
const resolvedUrl = typeof window !== 'undefined' && typeof window.location !== 'undefined'
|
|
10
|
+
? new URL(src, window.location.origin).toString()
|
|
11
|
+
: src;
|
|
12
|
+
if (!resolvedUrl.startsWith('https://') &&
|
|
13
|
+
!resolvedUrl.startsWith('http://')) {
|
|
14
|
+
return Promise.reject(new Error(resolvedUrl +
|
|
15
|
+
' is not a URL - needs to start with http:// or https://. If you want to read a local file, pass `nodeReader` to parseMedia().'));
|
|
16
|
+
}
|
|
17
|
+
const res = await fetch(resolvedUrl, {
|
|
18
|
+
headers: range === null
|
|
19
|
+
? {}
|
|
20
|
+
: typeof range === 'number'
|
|
21
|
+
? {
|
|
22
|
+
Range: `bytes=${range}`,
|
|
23
|
+
}
|
|
24
|
+
: {
|
|
25
|
+
Range: `bytes=${`${range[0]}-${range[1]}`}`,
|
|
26
|
+
},
|
|
27
|
+
signal,
|
|
28
|
+
// Disable Next.js caching
|
|
29
|
+
cache: 'no-store',
|
|
30
|
+
});
|
|
31
|
+
if (res.status.toString().startsWith('4') ||
|
|
32
|
+
res.status.toString().startsWith('5')) {
|
|
33
|
+
throw new Error(`Server returned status code ${res.status} for ${src}`);
|
|
34
|
+
}
|
|
35
|
+
if (!res.body) {
|
|
36
|
+
throw new Error('No body');
|
|
37
|
+
}
|
|
38
|
+
const length = res.headers.get('content-length');
|
|
39
|
+
const contentLength = length === null ? null : parseInt(length, 10);
|
|
40
|
+
const reader = res.body.getReader();
|
|
41
|
+
if (signal) {
|
|
42
|
+
signal.addEventListener('abort', () => {
|
|
43
|
+
reader.cancel();
|
|
44
|
+
}, { once: true });
|
|
45
|
+
}
|
|
46
|
+
return { reader, contentLength };
|
|
47
|
+
},
|
|
48
|
+
getLength: async (src) => {
|
|
49
|
+
if (typeof src !== 'string') {
|
|
50
|
+
throw new Error('src must be a string when using `fetchReader`');
|
|
51
|
+
}
|
|
52
|
+
const res = await fetch(src, {
|
|
53
|
+
method: 'HEAD',
|
|
54
|
+
});
|
|
55
|
+
if (!res.body) {
|
|
56
|
+
throw new Error('No body');
|
|
57
|
+
}
|
|
58
|
+
const length = res.headers.get('content-length');
|
|
59
|
+
if (!length) {
|
|
60
|
+
throw new Error('No content-length');
|
|
61
|
+
}
|
|
62
|
+
return parseInt(length, 10);
|
|
63
|
+
},
|
|
64
|
+
};
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.nodeReader = void 0;
|
|
4
|
+
const fs_1 = require("fs");
|
|
5
|
+
const promises_1 = require("node:fs/promises");
|
|
6
|
+
const stream_1 = require("stream");
|
|
7
|
+
exports.nodeReader = {
|
|
8
|
+
read: async (src, range, signal) => {
|
|
9
|
+
if (typeof src !== 'string') {
|
|
10
|
+
throw new Error('src must be a string when using `nodeReader`');
|
|
11
|
+
}
|
|
12
|
+
const stream = (0, fs_1.createReadStream)(src, {
|
|
13
|
+
start: range === null ? 0 : typeof range === 'number' ? range : range[0],
|
|
14
|
+
end: range === null
|
|
15
|
+
? Infinity
|
|
16
|
+
: typeof range === 'number'
|
|
17
|
+
? Infinity
|
|
18
|
+
: range[1],
|
|
19
|
+
signal,
|
|
20
|
+
});
|
|
21
|
+
const stats = await (0, promises_1.stat)(src);
|
|
22
|
+
const reader = stream_1.Readable.toWeb(stream).getReader();
|
|
23
|
+
if (signal) {
|
|
24
|
+
signal.addEventListener('abort', () => {
|
|
25
|
+
reader.cancel();
|
|
26
|
+
}, { once: true });
|
|
27
|
+
}
|
|
28
|
+
return {
|
|
29
|
+
reader,
|
|
30
|
+
contentLength: stats.size,
|
|
31
|
+
};
|
|
32
|
+
},
|
|
33
|
+
getLength: async (src) => {
|
|
34
|
+
if (typeof src !== 'string') {
|
|
35
|
+
throw new Error('src must be a string when using `nodeReader`');
|
|
36
|
+
}
|
|
37
|
+
const stats = await (0, promises_1.stat)(src);
|
|
38
|
+
return stats.size;
|
|
39
|
+
},
|
|
40
|
+
};
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.webFileReader = void 0;
|
|
4
|
+
exports.webFileReader = {
|
|
5
|
+
read: (file, range, signal) => {
|
|
6
|
+
if (typeof file === 'string') {
|
|
7
|
+
throw new Error('`inputTypeFileReader` only supports `File` objects');
|
|
8
|
+
}
|
|
9
|
+
const part = range === null
|
|
10
|
+
? file
|
|
11
|
+
: typeof range === 'number'
|
|
12
|
+
? file.slice(range)
|
|
13
|
+
: file.slice(range[0], range[1]);
|
|
14
|
+
const reader = new FileReader();
|
|
15
|
+
reader.readAsArrayBuffer(file);
|
|
16
|
+
if (signal) {
|
|
17
|
+
signal.addEventListener('abort', () => {
|
|
18
|
+
reader.abort();
|
|
19
|
+
}, { once: true });
|
|
20
|
+
}
|
|
21
|
+
return new Promise((resolve, reject) => {
|
|
22
|
+
reader.onload = () => {
|
|
23
|
+
resolve({
|
|
24
|
+
reader: part.stream().getReader(),
|
|
25
|
+
contentLength: file.size,
|
|
26
|
+
});
|
|
27
|
+
};
|
|
28
|
+
reader.onerror = (error) => {
|
|
29
|
+
reject(error);
|
|
30
|
+
};
|
|
31
|
+
});
|
|
32
|
+
},
|
|
33
|
+
getLength: (src) => {
|
|
34
|
+
if (typeof src === 'string') {
|
|
35
|
+
throw new Error('`inputTypeFileReader` only supports `File` objects');
|
|
36
|
+
}
|
|
37
|
+
return Promise.resolve(src.size);
|
|
38
|
+
},
|
|
39
|
+
};
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
type ReadResult = {
|
|
2
|
+
reader: ReadableStreamDefaultReader<Uint8Array>;
|
|
3
|
+
contentLength: number | null;
|
|
4
|
+
};
|
|
5
|
+
type ReadContent = (src: string | File, range: [number, number] | number | null, signal: AbortSignal | undefined) => Promise<ReadResult>;
|
|
6
|
+
type GetLength = (src: string | File) => Promise<number>;
|
|
7
|
+
export type ReaderInterface = {
|
|
8
|
+
read: ReadContent;
|
|
9
|
+
getLength: GetLength;
|
|
10
|
+
};
|
|
11
|
+
export {};
|