UNPKG

mediabunny

Version:

Pure TypeScript media toolkit for reading, writing, and converting media files, directly in the browser.

909 lines (804 loc) 29.4 kB
/*! * Copyright (c) 2026-present, Vanilagy and contributors * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ import { assert, AsyncMutex, EventEmitter, isIso639Dash2LanguageCode, MaybePromise, Rotation, toArray } from './misc'; import { MetadataTags, TrackDisposition, validateMetadataTags, validateTrackDisposition } from './metadata'; import { Muxer } from './muxer'; import { OutputFormat } from './output-format'; import { AudioSource, MediaSource, SubtitleSource, VideoSource } from './media-source'; import { PathedTarget, Target, TargetRequest } from './target'; import { Writer } from './writer'; /** * List of all track types. * @group Miscellaneous * @public */ export const ALL_TRACK_TYPES = ['video', 'audio', 'subtitle'] as const; /** * Union type of all track types. * @group Miscellaneous * @public */ export type TrackType = typeof ALL_TRACK_TYPES[number]; /** * Represents a track added to an {@link Output}. * @group Output files * @public */ export abstract class OutputTrack { /** @internal */ readonly id: number; /** The {@link Output} this track belongs to. */ readonly output: Output; /** The type of this track. */ readonly type: TrackType; /** The media source providing data for this track. */ readonly source: MediaSource; /** The metadata associated with this track. */ readonly metadata: BaseTrackMetadata; /** @internal */ protected constructor( id: number, output: Output, type: TrackType, source: MediaSource, metadata: BaseTrackMetadata, ) { this.id = id; this.output = output; this.type = type; this.source = source; this.metadata = metadata; } /** Returns true if and only if this track is a video track. */ isVideoTrack(): this is OutputVideoTrack { return this.type === 'video'; } /** Returns true if and only if this track is an audio track. */ isAudioTrack(): this is OutputAudioTrack { return this.type === 'audio'; } /** Returns true if and only if this track is a subtitle track. */ isSubtitleTrack(): this is OutputSubtitleTrack { return this.type === 'subtitle'; } /** * Returns true if and only if this track can be paired with the given other track. Pairability can be set using * the {@link BaseTrackMetadata.group} option. */ canBePairedWith(other: OutputTrack) { if (!(other instanceof OutputTrack)) { throw new TypeError('other must be an OutputTrack.'); } if (this === other) { return false; } const thisGroups = toArray(this.metadata.group!); const otherGroups = toArray(other.metadata.group!); for (const aGroup of thisGroups) { const pairableInSameGroup = this.type !== other.type && otherGroups.some(bGroup => aGroup === bGroup); if (pairableInSameGroup) { return true; } const pairableAcrossGroups = otherGroups.some( bGroup => aGroup._pairedGroups.has(bGroup), ); if (pairableAcrossGroups) { return true; } } return false; } } /** * An {@link OutputTrack} providing video data, created using {@link Output.addVideoTrack}. * @group Output files * @public */ export class OutputVideoTrack extends OutputTrack { declare readonly type: 'video'; declare readonly source: VideoSource; declare readonly metadata: VideoTrackMetadata; /** @internal */ constructor(id: number, output: Output, source: VideoSource, metadata: VideoTrackMetadata) { super(id, output, 'video', source, metadata); } } /** * An {@link OutputTrack} providing audio data, created using {@link Output.addAudioTrack}. * @group Output files * @public */ export class OutputAudioTrack extends OutputTrack { declare readonly type: 'audio'; declare readonly source: AudioSource; declare readonly metadata: AudioTrackMetadata; /** @internal */ constructor(id: number, output: Output, source: AudioSource, metadata: AudioTrackMetadata) { super(id, output, 'audio', source, metadata); } } /** * An {@link OutputTrack} providing subtitle data, created using {@link Output.addSubtitleTrack}. * @group Output files * @public */ export class OutputSubtitleTrack extends OutputTrack { declare readonly type: 'subtitle'; declare readonly source: SubtitleSource; declare readonly metadata: SubtitleTrackMetadata; /** @internal */ constructor(id: number, output: Output, source: SubtitleSource, metadata: SubtitleTrackMetadata) { super(id, output, 'subtitle', source, metadata); } } /** * Used to define pairability between {@link OutputTrack} instances. First create the group, then assign tracks to it * via {@link BaseTrackMetadata.group}. * * Two tracks are considered _pairable_ if they are in the same group but have a different {@link TrackType}, or if they * are in different groups that are paired with each other. Groups can be paired with each other using the * {@link OutputTrackGroup.pairWith} method. * * @group Output files * @public */ export class OutputTrackGroup { /** @internal */ _pairedGroups = new Set<OutputTrackGroup>(); /** Creates a new {@link OutputTrackGroup}. */ constructor() { // The object's identity is the state } /** * Marks this group as being pairable with another group, symmetrically. Output tracks where each track is assigned * to one half of a group pairing are then considered pairable. * * You cannot pair a group with itself. */ pairWith(other: OutputTrackGroup) { if (!(other instanceof OutputTrackGroup)) { throw new TypeError('other must be an OutputTrackGroup.'); } if (this === other) { throw new TypeError('Cannot pair a group with itself.'); } this._pairedGroups.add(other); other._pairedGroups.add(this); } } /** * Base track metadata, applicable to all tracks. * @group Output files * @public */ export type BaseTrackMetadata = { /** The three-letter, ISO 639-2/T language code specifying the language of this track. */ languageCode?: string; /** A user-defined name for this track, like "English" or "Director Commentary". */ name?: string; /** The track's disposition, i.e. information about its intended usage. */ disposition?: Partial<TrackDisposition>; /** * The maximum amount of encoded packets that will be added to this track. Setting this field provides the muxer * with an additional signal that it can use to preallocate space in the file. * * When this field is set, it is an error to provide more packets than whatever this field specifies. * * Predicting the maximum packet count requires considering both the maximum duration as well as the codec. * - For video codecs, you can assume one packet per frame. * - For audio codecs, there is one packet for each "audio chunk", the duration of which depends on the codec. For * simplicity, you can assume each packet is roughly 10 ms or 512 samples long, whichever is shorter. * - For subtitles, assume each cue and each gap in the subtitles adds a packet. * * If you're not fully sure, make sure to add a buffer of around 33% to make sure you stay below the maximum. */ maximumPacketCount?: number; /** * Whether the timestamps of this track are relative to the Unix epoch (January 1, 1970, 00:00:00 UTC). When `true`, * each timestamp maps to a definitive point in time. */ isRelativeToUnixEpoch?: boolean; /** * Defines the group(s) this track is a part of. Group assignment determines track pairability, determining which * tracks can be presented together with other tracks. This is needed for configuring things like HLS master * playlists. * * Two groups are considered pairable if they are in the same group but are of different {@link TrackType}, or if * they are in two separate groups that have been paired with each other. * * If left blank, a track is automatically assigned to {@link Output.defaultTrackGroup}. */ group?: OutputTrackGroup | OutputTrackGroup[]; }; /** * Additional metadata for video tracks. * @group Output files * @public */ export type VideoTrackMetadata = BaseTrackMetadata & { /** The angle in degrees by which the track's frames should be rotated (clockwise). */ rotation?: Rotation; /** * The expected video frame rate in hertz. If set, all timestamps and durations of this track will be snapped to * this frame rate. You should avoid adding more frames than the rate allows, as this will lead to multiple frames * with the same timestamp. */ frameRate?: number; /** * When true, this track is marked as being made only out of key frames (I-frames). It is an error to add a non-key * frame to this track. */ hasOnlyKeyPackets?: boolean; }; /** * Additional metadata for audio tracks. * @group Output files * @public */ export type AudioTrackMetadata = BaseTrackMetadata & {}; /** * Additional metadata for subtitle tracks. * @group Output files * @public */ export type SubtitleTrackMetadata = BaseTrackMetadata & {}; const validateBaseTrackMetadata = (metadata: BaseTrackMetadata) => { if (!metadata || typeof metadata !== 'object') { throw new TypeError('metadata must be an object.'); } if (metadata.languageCode !== undefined && !isIso639Dash2LanguageCode(metadata.languageCode)) { throw new TypeError('metadata.languageCode, when provided, must be a three-letter, ISO 639-2/T language code.'); } if (metadata.name !== undefined && typeof metadata.name !== 'string') { throw new TypeError('metadata.name, when provided, must be a string.'); } if (metadata.disposition !== undefined) { validateTrackDisposition(metadata.disposition); } if ( metadata.maximumPacketCount !== undefined && (!Number.isInteger(metadata.maximumPacketCount) || metadata.maximumPacketCount < 0) ) { throw new TypeError('metadata.maximumPacketCount, when provided, must be a non-negative integer.'); } if ( metadata.group !== undefined && !(metadata.group instanceof OutputTrackGroup) && (!Array.isArray(metadata.group) || metadata.group.some(group => !(group instanceof OutputTrackGroup))) ) { throw new TypeError( 'metadata.group, when provided, must be an OutputTrackGroup instance or an array of' + ' OutputTrackGroup instances.', ); } }; /** * The options for creating an Output object. * @group Output files * @public */ export type OutputOptions< F extends OutputFormat = OutputFormat, T extends Target = Target, > = { /** The format of the output file. */ format: F; /** The target to which the file will be written. */ target: T | PathedTarget<T>; /** * Optional; the target to which the track initialization data will be written. Most formats do not make use of * this, but some do, such as {@link CmafOutputFormat}. * * When this is a function, it will only be called if an init target is needed. */ initTarget?: T | (() => MaybePromise<T>); /** * Optional; a callback to be called at the end of {@link Output.finalize}. Can be used to run logic once the * output has completed. If a promise is returned, it will be awaited internally by {@link Output.finalize}. */ onFinalize?: () => MaybePromise<unknown>; }; /** * Describes the events that an {@link Output} emits, with each key being an event name and its value being the * event data. * * @group Output files * @public */ export type OutputEvents = { /** Emitted whenever a {@link Target} is obtained by the output. Useful to track writes. */ target: { /** The target that was obtained. */ target: Target; /** The request that led to the target being obtained, or `null` if the output is not pathed. */ request: TargetRequest | null; /** Whether the target is the root file of the media. */ isRoot: boolean; }; }; /** * Main class orchestrating the creation of new media files. * @group Output files * @public */ export class Output< F extends OutputFormat = OutputFormat, T extends Target = Target, > extends EventEmitter<OutputEvents> { /** The format of the output file. */ readonly format: F; /** @internal */ _target: T | PathedTarget<T>; /** The current state of the output. */ state: 'pending' | 'started' | 'canceled' | 'finalizing' | 'finalized' = 'pending'; /** * The {@link OutputTrackGroup} that all tracks are assigned to by default unless otherwise specified by * {@link BaseTrackMetadata.group}. */ readonly defaultTrackGroup = new OutputTrackGroup(); /** @internal */ private _initTarget: T | (() => MaybePromise<T>) | null; /** @internal */ _onFinalize: (() => MaybePromise<unknown>) | null = null; /** @internal */ _muxer: Muxer; /** @internal */ _unfinalizedTargets = new Set<Target>(); /** @internal */ _rootWriterPromise: Promise<Writer> | null = null; /** @internal */ _tracks: OutputTrack[] = []; /** @internal */ _startPromise: Promise<void> | null = null; /** @internal */ _cancelPromise: Promise<void> | null = null; /** @internal */ _finalizePromise: Promise<void> | null = null; /** @internal */ _mutex = new AsyncMutex(); /** @internal */ _metadataTags: MetadataTags = {}; /** @internal */ _rootTarget: T | null = null; /** @internal */ _rootTargetPromise: Promise<T> | null = null; /** * This field is used to synchronize multiple MediaStreamTracks. They use the same time coordinate system across * tracks, and to ensure correct audio-video sync, we must use the same offset for all of them. The reason an offset * is needed at all is because the timestamps typically don't start at zero. * @internal */ _firstMediaStreamTimestamp: number | null = null; /** * The target to which the root file will be written. Throws when using {@link PathedTarget} with an async callback; * prefer the `'target'` event for those cases. */ get target(): T { const errorMessage = 'Output.target cannot be used when using PathedTarget with an async callback.' + ' Use the \'target\' event instead.'; // We use this field to make sure we can reliably throw in the `target` getter whenever retrieving the target // requires awaiting a promise. We do this so there is no different behavior based on order: if the target has // already been retrieved via the normal internal operations, and then somebody calls the `target` getter, even // if the target is now available, the getter should still throw to be consistent in behavior and in definition. if (this._rootTargetPromise) { throw new TypeError(errorMessage); } const rootTargetResult = this._getRootTarget(); if (rootTargetResult instanceof Promise) { throw new TypeError(errorMessage); } return rootTargetResult; } /** * Creates a new instance of {@link Output} which can then be used to create a new media file according to the * specified {@link OutputOptions}. */ constructor(options: OutputOptions<F, T>) { super(); if (!options || typeof options !== 'object') { throw new TypeError('options must be an object.'); } if (!(options.format instanceof OutputFormat)) { throw new TypeError('options.format must be an OutputFormat.'); } if (!(options.target instanceof Target || options.target instanceof PathedTarget)) { throw new TypeError('options.target must be a Target or a PathedTarget.'); } if (options.target instanceof Target) { this._rememberTarget(options.target); } if ( options.initTarget !== undefined && !(options.initTarget instanceof Target) && typeof options.initTarget !== 'function' ) { throw new Error( 'options.initTarget, when provided, must be a Target or a function that returns or resolves to' + ' a Target.', ); } if (options.onFinalize !== undefined && typeof options.onFinalize !== 'function') { throw new TypeError('options.onFinalize, when provided, must be a function.'); } this.format = options.format; this._target = options.target; this._onFinalize = options.onFinalize ?? null; this._initTarget = options.initTarget ?? null; if (this._initTarget instanceof Target) { this._rememberTarget(this._initTarget); } this._muxer = options.format._createMuxer(this); } /** @internal */ _getTargetValidated(request: TargetRequest): MaybePromise<T> { assert(this._target instanceof PathedTarget); const result = this._target.getTarget(request); const handleResult = (result: T) => { if (!(result instanceof Target)) { throw new TypeError('getTarget must return a Target.'); } return result; }; if (result instanceof Promise) { return result.then(handleResult); } else { return handleResult(result); } } /** @internal */ async _getTarget(request: TargetRequest) { assert(this._target instanceof PathedTarget); const target = await this._getTargetValidated(request); this._emit('target', { target, request, isRoot: request.isRoot }); if (this.state === 'canceled') { await target._close(); } else { this._rememberTarget(target); } return target; } /** @internal */ _rememberTarget(target: Target) { this._unfinalizedTargets.add(target); target.on('finalized', () => this._unfinalizedTargets.delete(target), { once: true }); } /** @internal */ async _getInitTarget(): Promise<T> { assert(this._initTarget !== null); if (this._initTarget instanceof Target) { return this._initTarget; } const target = await this._initTarget(); if (this.state === 'canceled') { await target._close(); } else { this._rememberTarget(target); } return target; } /** @internal */ _hasInitTarget() { return this._initTarget !== null; } /** @internal */ _getRootTarget(): MaybePromise<T> { if (this._rootTarget) { return this._rootTarget; } if (this._rootTargetPromise) { return this._rootTargetPromise; } if (this._target instanceof Target) { this._emit('target', { target: this._target, request: null, isRoot: true }); this._rootTarget = this._target; return this._target; } const request: TargetRequest = { path: this._target.rootPath, isRoot: true, mimeType: this.format.mimeType, }; const result = this._getTargetValidated(request); const handleResult = (target: T) => { if (this.state === 'canceled') { // Promise thrown away here, but no way to surface it to the user really void target._close(); } else { this._rememberTarget(target); } this._emit('target', { target, request, isRoot: true }); this._rootTarget = target; return target; }; if (result instanceof Promise) { return this._rootTargetPromise = result.then(handleResult); } else { return handleResult(result); } } /** @internal */ _getRootWriter(isMonotonic: boolean | ((target: Target) => boolean)) { return this._rootWriterPromise ??= (async () => { const target = await this._getRootTarget(); const writer = new Writer(target, typeof isMonotonic === 'boolean' ? isMonotonic : isMonotonic(target)); writer.start(); return writer; })(); } /** Adds a video track to the output with the given source. Can only be called before the output is started. */ addVideoTrack(source: VideoSource, metadata: VideoTrackMetadata = {}) { if (!(source instanceof VideoSource)) { throw new TypeError('source must be a VideoSource.'); } validateBaseTrackMetadata(metadata); if (metadata.rotation !== undefined && ![0, 90, 180, 270].includes(metadata.rotation)) { throw new TypeError(`Invalid video rotation: ${metadata.rotation}. Has to be 0, 90, 180 or 270.`); } if (!this.format.supportsVideoRotationMetadata && metadata.rotation) { throw new Error(`${this.format._name} does not support video rotation metadata.`); } if ( metadata.frameRate !== undefined && (!Number.isFinite(metadata.frameRate) || metadata.frameRate <= 0) ) { throw new TypeError( `Invalid video frame rate: ${metadata.frameRate}. Must be a positive number.`, ); } const metadataCopy = { ...metadata }; metadataCopy.group ??= this.defaultTrackGroup; return this._addTrack(new OutputVideoTrack( this._tracks.length + 1, this, source, metadataCopy, )); } /** Adds an audio track to the output with the given source. Can only be called before the output is started. */ addAudioTrack(source: AudioSource, metadata: AudioTrackMetadata = {}) { if (!(source instanceof AudioSource)) { throw new TypeError('source must be an AudioSource.'); } validateBaseTrackMetadata(metadata); const metadataCopy = { ...metadata }; metadataCopy.group ??= this.defaultTrackGroup; return this._addTrack(new OutputAudioTrack( this._tracks.length + 1, this, source, metadataCopy, )); } /** Adds a subtitle track to the output with the given source. Can only be called before the output is started. */ addSubtitleTrack(source: SubtitleSource, metadata: SubtitleTrackMetadata = {}) { if (!(source instanceof SubtitleSource)) { throw new TypeError('source must be a SubtitleSource.'); } validateBaseTrackMetadata(metadata); const metadataCopy = { ...metadata }; metadataCopy.group ??= this.defaultTrackGroup; return this._addTrack(new OutputSubtitleTrack( this._tracks.length + 1, this, source, metadataCopy, )); } /** * Sets descriptive metadata tags about the media file, such as title, author, date, or cover art. When called * multiple times, only the metadata from the last call will be used. * * Can only be called before the output is started. */ setMetadataTags(tags: MetadataTags) { validateMetadataTags(tags); if (this.state !== 'pending') { throw new Error('Cannot set metadata tags after output has been started or canceled.'); } this._metadataTags = tags; } /** @internal */ private _addTrack<T extends OutputTrack>(track: T) { if (this.state !== 'pending') { throw new Error('Cannot add track after output has been started or canceled.'); } if (track.source._connectedTrack) { throw new Error('Source is already used for a track.'); } // Verify maximum track count constraints const supportedTrackCounts = this.format.getSupportedTrackCounts(); const presentTracksOfThisType = this._tracks.reduce( (count, t) => count + (t.type === track.type ? 1 : 0), 0, ); const maxCount = supportedTrackCounts[track.type].max; if (presentTracksOfThisType === maxCount) { throw new Error( maxCount === 0 ? `${this.format._name} does not support ${track.type} tracks.` : (`${this.format._name} does not support more than ${maxCount} ${track.type} track` + `${maxCount === 1 ? '' : 's'}.`), ); } const maxTotalCount = supportedTrackCounts.total.max; if (this._tracks.length === maxTotalCount) { throw new Error( `${this.format._name} does not support more than ${maxTotalCount} tracks` + `${maxTotalCount === 1 ? '' : 's'} in total.`, ); } if (track.isVideoTrack()) { const supportedVideoCodecs = this.format.getSupportedVideoCodecs(); if (supportedVideoCodecs.length === 0) { throw new Error( `${this.format._name} does not support video tracks.` + this.format._codecUnsupportedHint(track.source._codec), ); } else if (!supportedVideoCodecs.includes(track.source._codec)) { throw new Error( `Codec '${track.source._codec}' cannot be contained within ${this.format._name}. Supported` + ` video codecs are: ${supportedVideoCodecs.map(codec => `'${codec}'`).join(', ')}.` + this.format._codecUnsupportedHint(track.source._codec), ); } } else if (track.isAudioTrack()) { const supportedAudioCodecs = this.format.getSupportedAudioCodecs(); if (supportedAudioCodecs.length === 0) { throw new Error( `${this.format._name} does not support audio tracks.` + this.format._codecUnsupportedHint(track.source._codec), ); } else if (!supportedAudioCodecs.includes(track.source._codec)) { throw new Error( `Codec '${track.source._codec}' cannot be contained within ${this.format._name}. Supported` + ` audio codecs are: ${supportedAudioCodecs.map(codec => `'${codec}'`).join(', ')}.` + this.format._codecUnsupportedHint(track.source._codec), ); } } else if (track.isSubtitleTrack()) { const supportedSubtitleCodecs = this.format.getSupportedSubtitleCodecs(); if (supportedSubtitleCodecs.length === 0) { throw new Error( `${this.format._name} does not support subtitle tracks.` + this.format._codecUnsupportedHint(track.source._codec), ); } else if (!supportedSubtitleCodecs.includes(track.source._codec)) { throw new Error( `Codec '${track.source._codec}' cannot be contained within ${this.format._name}. Supported` + ` subtitle codecs are: ${supportedSubtitleCodecs.map(codec => `'${codec}'`).join(', ')}.` + this.format._codecUnsupportedHint(track.source._codec), ); } } this._tracks.push(track); track.source._connectedTrack = track; return track; } /** * Starts the creation of the output file. This method should be called after all tracks have been added. Only after * the output has started can media samples be added to the tracks. * * @returns A promise that resolves when the output has successfully started and is ready to receive media samples. */ async start() { // Verify minimum track count constraints const supportedTrackCounts = this.format.getSupportedTrackCounts(); for (const trackType of ALL_TRACK_TYPES) { const presentTracksOfThisType = this._tracks.reduce( (count, track) => count + (track.type === trackType ? 1 : 0), 0, ); const minCount = supportedTrackCounts[trackType].min; if (presentTracksOfThisType < minCount) { throw new Error( minCount === supportedTrackCounts[trackType].max ? (`${this.format._name} requires exactly ${minCount} ${trackType}` + ` track${minCount === 1 ? '' : 's'}.`) : (`${this.format._name} requires at least ${minCount} ${trackType}` + ` track${minCount === 1 ? '' : 's'}.`), ); } } const totalMinCount = supportedTrackCounts.total.min; if (this._tracks.length < totalMinCount) { throw new Error( totalMinCount === supportedTrackCounts.total.max ? (`${this.format._name} requires exactly ${totalMinCount} track` + `${totalMinCount === 1 ? '' : 's'}.`) : (`${this.format._name} requires at least ${totalMinCount} track` + `${totalMinCount === 1 ? '' : 's'}.`), ); } if (this.state === 'canceled') { throw new Error('Output has been canceled.'); } if (this._startPromise) { console.warn('Output has already been started.'); return this._startPromise; } return this._startPromise = (async () => { this.state = 'started'; const release = await this._mutex.acquire(); try { await this._muxer.start(); const promises = this._tracks.map(track => track.source._start()); await Promise.all(promises); } finally { release(); } })(); } /** * Resolves with the full MIME type of the output file, including track codecs. * * The returned promise will resolve only once the precise codec strings of all tracks are known. */ getMimeType() { return this._muxer.getMimeType(); } /** * Cancels the creation of the output file, releasing internal resources like encoders and preventing further * samples from being added. * * @returns A promise that resolves once all internal resources have been released. */ async cancel() { if (this._cancelPromise) { console.warn('Output has already been canceled.'); return this._cancelPromise; } else if (this.state === 'finalizing' || this.state === 'finalized') { // Don't wanna warn when finalizing since that shows a warning when finalization fails and then cancel // is called if (this.state === 'finalized') { console.warn('Output has already been finalized.'); } return; } return this._cancelPromise = (async () => { this.state = 'canceled'; const release = await this._mutex.acquire(); try { const promises = this._tracks.map(x => x.source._flushOrWaitForOngoingClose(true)); // Force close await Promise.all(promises); await Promise.all([...this._unfinalizedTargets].map(target => target._close())); this._unfinalizedTargets.clear(); } finally { release(); } })(); } /** * Finalizes the output file. This method must be called after all media samples across all tracks have been added. * Once the Promise returned by this method completes, the output file is ready. */ async finalize() { if (this.state === 'pending') { throw new Error('Cannot finalize before starting.'); } if (this.state === 'canceled') { throw new Error('Cannot finalize after canceling.'); } if (this._finalizePromise) { console.warn('Output has already been finalized.'); return this._finalizePromise; } return this._finalizePromise = (async () => { this.state = 'finalizing'; const release = await this._mutex.acquire(); try { const promises = this._tracks.map(x => x.source._flushOrWaitForOngoingClose(false)); await Promise.all(promises); await this._muxer.finalize(); if (this._rootWriterPromise) { const rootWriter = await this._rootWriterPromise; if (!rootWriter.finalized) { await rootWriter.flush(); await rootWriter.finalize(); } } if (this._onFinalize) { await this._onFinalize(); } this.state = 'finalized'; } finally { release(); } })(); } }