| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015 | #pragma warning disable CS1591using System;using System.Collections.Generic;using System.Globalization;using System.Linq;using Jellyfin.Data.Enums;using MediaBrowser.Model.Drawing;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.MediaInfo;using MediaBrowser.Model.Session;namespace MediaBrowser.Model.Dlna{    /// <summary>    /// Class StreamInfo.    /// </summary>    public class StreamInfo    {        public StreamInfo()        {            AudioCodecs = Array.Empty<string>();            VideoCodecs = Array.Empty<string>();            SubtitleCodecs = Array.Empty<string>();            StreamOptions = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);        }        public Guid ItemId { get; set; }        public PlayMethod PlayMethod { get; set; }        public EncodingContext Context { get; set; }        public DlnaProfileType MediaType { get; set; }        public string? Container { get; set; }        public string? SubProtocol { get; set; }        public long StartPositionTicks { get; set; }        public int? SegmentLength { get; set; }        public int? MinSegments { get; set; }        public bool BreakOnNonKeyFrames { get; set; }        public bool RequireAvc { get; set; }        public bool RequireNonAnamorphic { get; set; }        public bool CopyTimestamps { get; set; }        public bool EnableMpegtsM2TsMode { get; set; }        public bool EnableSubtitlesInManifest { get; set; }        public string[] AudioCodecs { get; set; }        public string[] VideoCodecs { get; set; }        public int? AudioStreamIndex { get; set; }        public int? SubtitleStreamIndex { get; set; }        public int? TranscodingMaxAudioChannels { get; set; }        public int? GlobalMaxAudioChannels { get; set; }        public int? AudioBitrate { get; set; }        public int? AudioSampleRate { get; set; }        public int? VideoBitrate { get; set; }        public int? MaxWidth { get; set; }        public int? MaxHeight { get; set; }        public float? MaxFramerate { get; set; }        public required DeviceProfile DeviceProfile { get; set; }        public string? DeviceProfileId { get; set; }        public string? DeviceId { get; set; }        public long? RunTimeTicks { get; set; }        public TranscodeSeekInfo TranscodeSeekInfo { get; set; }        public bool EstimateContentLength { get; set; }        public MediaSourceInfo? MediaSource { get; set; }        public string[] SubtitleCodecs { get; set; }        public SubtitleDeliveryMethod SubtitleDeliveryMethod { get; set; }        public string? SubtitleFormat { get; set; }        public string? PlaySessionId { get; set; }        public TranscodeReason TranscodeReasons { get; set; }        public Dictionary<string, string> StreamOptions { get; private set; }        public string? MediaSourceId => MediaSource?.Id;        public bool IsDirectStream => MediaSource?.VideoType is not (VideoType.Dvd or VideoType.BluRay)            && PlayMethod is PlayMethod.DirectStream or PlayMethod.DirectPlay;        /// <summary>        /// Gets the audio stream that will be used.        /// </summary>        public MediaStream? TargetAudioStream => MediaSource?.GetDefaultAudioStream(AudioStreamIndex);        /// <summary>        /// Gets the video stream that will be used.        /// </summary>        public MediaStream? TargetVideoStream => MediaSource?.VideoStream;        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public int? TargetAudioSampleRate        {            get            {                var stream = TargetAudioStream;                return AudioSampleRate.HasValue && !IsDirectStream                    ? AudioSampleRate                    : stream?.SampleRate;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public int? TargetAudioBitDepth        {            get            {                if (IsDirectStream)                {                    return TargetAudioStream?.BitDepth;                }                var targetAudioCodecs = TargetAudioCodec;                var audioCodec = targetAudioCodecs.Length == 0 ? null : targetAudioCodecs[0];                if (!string.IsNullOrEmpty(audioCodec))                {                    return GetTargetAudioBitDepth(audioCodec);                }                return TargetAudioStream?.BitDepth;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public int? TargetVideoBitDepth        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.BitDepth;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec))                {                    return GetTargetVideoBitDepth(videoCodec);                }                return TargetVideoStream?.BitDepth;            }        }        /// <summary>        /// Gets the target reference frames.        /// </summary>        /// <value>The target reference frames.</value>        public int? TargetRefFrames        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.RefFrames;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec))                {                    return GetTargetRefFrames(videoCodec);                }                return TargetVideoStream?.RefFrames;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public float? TargetFramerate        {            get            {                var stream = TargetVideoStream;                return MaxFramerate.HasValue && !IsDirectStream                    ? MaxFramerate                    : stream is null ? null : stream.AverageFrameRate ?? stream.RealFrameRate;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public double? TargetVideoLevel        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.Level;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec))                {                    return GetTargetVideoLevel(videoCodec);                }                return TargetVideoStream?.Level;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public int? TargetPacketLength        {            get            {                var stream = TargetVideoStream;                return !IsDirectStream                    ? null                    : stream?.PacketLength;            }        }        /// <summary>        /// Gets the audio sample rate that will be in the output stream.        /// </summary>        public string? TargetVideoProfile        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.Profile;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec))                {                    return GetOption(videoCodec, "profile");                }                return TargetVideoStream?.Profile;            }        }        /// <summary>        /// Gets the target video range type that will be in the output stream.        /// </summary>        public VideoRangeType TargetVideoRangeType        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.VideoRangeType ?? VideoRangeType.Unknown;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec)                    && Enum.TryParse(GetOption(videoCodec, "rangetype"), true, out VideoRangeType videoRangeType))                {                    return videoRangeType;                }                return TargetVideoStream?.VideoRangeType ?? VideoRangeType.Unknown;            }        }        /// <summary>        /// Gets the target video codec tag.        /// </summary>        /// <value>The target video codec tag.</value>        public string? TargetVideoCodecTag        {            get            {                var stream = TargetVideoStream;                return !IsDirectStream                    ? null                    : stream?.CodecTag;            }        }        /// <summary>        /// Gets the audio bitrate that will be in the output stream.        /// </summary>        public int? TargetAudioBitrate        {            get            {                var stream = TargetAudioStream;                return AudioBitrate.HasValue && !IsDirectStream                    ? AudioBitrate                    : stream?.BitRate;            }        }        /// <summary>        /// Gets the audio channels that will be in the output stream.        /// </summary>        public int? TargetAudioChannels        {            get            {                if (IsDirectStream)                {                    return TargetAudioStream?.Channels;                }                var targetAudioCodecs = TargetAudioCodec;                var codec = targetAudioCodecs.Length == 0 ? null : targetAudioCodecs[0];                if (!string.IsNullOrEmpty(codec))                {                    return GetTargetRefFrames(codec);                }                return TargetAudioStream?.Channels;            }        }        /// <summary>        /// Gets the audio codec that will be in the output stream.        /// </summary>        public string[] TargetAudioCodec        {            get            {                var stream = TargetAudioStream;                string? inputCodec = stream?.Codec;                if (IsDirectStream)                {                    return string.IsNullOrEmpty(inputCodec) ? Array.Empty<string>() : new[] { inputCodec };                }                foreach (string codec in AudioCodecs)                {                    if (string.Equals(codec, inputCodec, StringComparison.OrdinalIgnoreCase))                    {                        return string.IsNullOrEmpty(codec) ? Array.Empty<string>() : new[] { codec };                    }                }                return AudioCodecs;            }        }        public string[] TargetVideoCodec        {            get            {                var stream = TargetVideoStream;                string? inputCodec = stream?.Codec;                if (IsDirectStream)                {                    return string.IsNullOrEmpty(inputCodec) ? Array.Empty<string>() : new[] { inputCodec };                }                foreach (string codec in VideoCodecs)                {                    if (string.Equals(codec, inputCodec, StringComparison.OrdinalIgnoreCase))                    {                        return string.IsNullOrEmpty(codec) ? Array.Empty<string>() : new[] { codec };                    }                }                return VideoCodecs;            }        }        /// <summary>        /// Gets the audio channels that will be in the output stream.        /// </summary>        public long? TargetSize        {            get            {                if (IsDirectStream)                {                    return MediaSource?.Size;                }                if (RunTimeTicks.HasValue)                {                    int? totalBitrate = TargetTotalBitrate;                    double totalSeconds = RunTimeTicks.Value;                    // Convert to ms                    totalSeconds /= 10000;                    // Convert to seconds                    totalSeconds /= 1000;                    return totalBitrate.HasValue ?                        Convert.ToInt64(totalBitrate.Value * totalSeconds) :                        null;                }                return null;            }        }        public int? TargetVideoBitrate        {            get            {                var stream = TargetVideoStream;                return VideoBitrate.HasValue && !IsDirectStream                    ? VideoBitrate                    : stream?.BitRate;            }        }        public TransportStreamTimestamp TargetTimestamp        {            get            {                var defaultValue = string.Equals(Container, "m2ts", StringComparison.OrdinalIgnoreCase)                    ? TransportStreamTimestamp.Valid                    : TransportStreamTimestamp.None;                return !IsDirectStream                    ? defaultValue                    : MediaSource is null ? defaultValue : MediaSource.Timestamp ?? TransportStreamTimestamp.None;            }        }        public int? TargetTotalBitrate => (TargetAudioBitrate ?? 0) + (TargetVideoBitrate ?? 0);        public bool? IsTargetAnamorphic        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.IsAnamorphic;                }                return false;            }        }        public bool? IsTargetInterlaced        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.IsInterlaced;                }                var targetVideoCodecs = TargetVideoCodec;                var videoCodec = targetVideoCodecs.Length == 0 ? null : targetVideoCodecs[0];                if (!string.IsNullOrEmpty(videoCodec))                {                    if (string.Equals(GetOption(videoCodec, "deinterlace"), "true", StringComparison.OrdinalIgnoreCase))                    {                        return false;                    }                }                return TargetVideoStream?.IsInterlaced;            }        }        public bool? IsTargetAVC        {            get            {                if (IsDirectStream)                {                    return TargetVideoStream?.IsAVC;                }                return true;            }        }        public int? TargetWidth        {            get            {                var videoStream = TargetVideoStream;                if (videoStream is not null && videoStream.Width.HasValue && videoStream.Height.HasValue)                {                    ImageDimensions size = new ImageDimensions(videoStream.Width.Value, videoStream.Height.Value);                    size = DrawingUtils.Resize(size, 0, 0, MaxWidth ?? 0, MaxHeight ?? 0);                    return size.Width;                }                return MaxWidth;            }        }        public int? TargetHeight        {            get            {                var videoStream = TargetVideoStream;                if (videoStream is not null && videoStream.Width.HasValue && videoStream.Height.HasValue)                {                    ImageDimensions size = new ImageDimensions(videoStream.Width.Value, videoStream.Height.Value);                    size = DrawingUtils.Resize(size, 0, 0, MaxWidth ?? 0, MaxHeight ?? 0);                    return size.Height;                }                return MaxHeight;            }        }        public int? TargetVideoStreamCount        {            get            {                if (IsDirectStream)                {                    return GetMediaStreamCount(MediaStreamType.Video, int.MaxValue);                }                return GetMediaStreamCount(MediaStreamType.Video, 1);            }        }        public int? TargetAudioStreamCount        {            get            {                if (IsDirectStream)                {                    return GetMediaStreamCount(MediaStreamType.Audio, int.MaxValue);                }                return GetMediaStreamCount(MediaStreamType.Audio, 1);            }        }        public void SetOption(string? qualifier, string name, string value)        {            if (string.IsNullOrEmpty(qualifier))            {                SetOption(name, value);            }            else            {                SetOption(qualifier + "-" + name, value);            }        }        public void SetOption(string name, string value)        {            StreamOptions[name] = value;        }        public string? GetOption(string? qualifier, string name)        {            var value = GetOption(qualifier + "-" + name);            if (string.IsNullOrEmpty(value))            {                value = GetOption(name);            }            return value;        }        public string? GetOption(string name)        {            if (StreamOptions.TryGetValue(name, out var value))            {                return value;            }            return null;        }        public string ToUrl(string baseUrl, string? accessToken)        {            ArgumentException.ThrowIfNullOrEmpty(baseUrl);            var list = new List<string>();            foreach (NameValuePair pair in BuildParams(this, accessToken))            {                if (string.IsNullOrEmpty(pair.Value))                {                    continue;                }                // Try to keep the url clean by omitting defaults                if (string.Equals(pair.Name, "StartTimeTicks", StringComparison.OrdinalIgnoreCase)                    && string.Equals(pair.Value, "0", StringComparison.OrdinalIgnoreCase))                {                    continue;                }                if (string.Equals(pair.Name, "SubtitleStreamIndex", StringComparison.OrdinalIgnoreCase)                    && string.Equals(pair.Value, "-1", StringComparison.OrdinalIgnoreCase))                {                    continue;                }                if (string.Equals(pair.Name, "Static", StringComparison.OrdinalIgnoreCase)                    && string.Equals(pair.Value, "false", StringComparison.OrdinalIgnoreCase))                {                    continue;                }                var encodedValue = pair.Value.Replace(" ", "%20", StringComparison.Ordinal);                list.Add(string.Format(CultureInfo.InvariantCulture, "{0}={1}", pair.Name, encodedValue));            }            string queryString = string.Join('&', list);            return GetUrl(baseUrl, queryString);        }        private string GetUrl(string baseUrl, string queryString)        {            ArgumentException.ThrowIfNullOrEmpty(baseUrl);            string extension = string.IsNullOrEmpty(Container) ? string.Empty : "." + Container;            baseUrl = baseUrl.TrimEnd('/');            if (MediaType == DlnaProfileType.Audio)            {                if (string.Equals(SubProtocol, "hls", StringComparison.OrdinalIgnoreCase))                {                    return string.Format(CultureInfo.InvariantCulture, "{0}/audio/{1}/master.m3u8?{2}", baseUrl, ItemId, queryString);                }                return string.Format(CultureInfo.InvariantCulture, "{0}/audio/{1}/stream{2}?{3}", baseUrl, ItemId, extension, queryString);            }            if (string.Equals(SubProtocol, "hls", StringComparison.OrdinalIgnoreCase))            {                return string.Format(CultureInfo.InvariantCulture, "{0}/videos/{1}/master.m3u8?{2}", baseUrl, ItemId, queryString);            }            return string.Format(CultureInfo.InvariantCulture, "{0}/videos/{1}/stream{2}?{3}", baseUrl, ItemId, extension, queryString);        }        private static IEnumerable<NameValuePair> BuildParams(StreamInfo item, string? accessToken)        {            var list = new List<NameValuePair>();            string audioCodecs = item.AudioCodecs.Length == 0 ?                string.Empty :                string.Join(',', item.AudioCodecs);            string videoCodecs = item.VideoCodecs.Length == 0 ?                string.Empty :                string.Join(',', item.VideoCodecs);            list.Add(new NameValuePair("DeviceProfileId", item.DeviceProfileId ?? string.Empty));            list.Add(new NameValuePair("DeviceId", item.DeviceId ?? string.Empty));            list.Add(new NameValuePair("MediaSourceId", item.MediaSourceId ?? string.Empty));            list.Add(new NameValuePair("Static", item.IsDirectStream.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));            list.Add(new NameValuePair("VideoCodec", videoCodecs));            list.Add(new NameValuePair("AudioCodec", audioCodecs));            list.Add(new NameValuePair("AudioStreamIndex", item.AudioStreamIndex.HasValue ? item.AudioStreamIndex.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("SubtitleStreamIndex", item.SubtitleStreamIndex.HasValue && item.SubtitleDeliveryMethod != SubtitleDeliveryMethod.External ? item.SubtitleStreamIndex.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("VideoBitrate", item.VideoBitrate.HasValue ? item.VideoBitrate.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("AudioBitrate", item.AudioBitrate.HasValue ? item.AudioBitrate.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("AudioSampleRate", item.AudioSampleRate.HasValue ? item.AudioSampleRate.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("MaxFramerate", item.MaxFramerate.HasValue ? item.MaxFramerate.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("MaxWidth", item.MaxWidth.HasValue ? item.MaxWidth.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            list.Add(new NameValuePair("MaxHeight", item.MaxHeight.HasValue ? item.MaxHeight.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));            long startPositionTicks = item.StartPositionTicks;            var isHls = string.Equals(item.SubProtocol, "hls", StringComparison.OrdinalIgnoreCase);            if (isHls)            {                list.Add(new NameValuePair("StartTimeTicks", string.Empty));            }            else            {                list.Add(new NameValuePair("StartTimeTicks", startPositionTicks.ToString(CultureInfo.InvariantCulture)));            }            list.Add(new NameValuePair("PlaySessionId", item.PlaySessionId ?? string.Empty));            list.Add(new NameValuePair("api_key", accessToken ?? string.Empty));            string? liveStreamId = item.MediaSource?.LiveStreamId;            list.Add(new NameValuePair("LiveStreamId", liveStreamId ?? string.Empty));            list.Add(new NameValuePair("SubtitleMethod", item.SubtitleStreamIndex.HasValue && item.SubtitleDeliveryMethod != SubtitleDeliveryMethod.External ? item.SubtitleDeliveryMethod.ToString() : string.Empty));            if (!item.IsDirectStream)            {                if (item.RequireNonAnamorphic)                {                    list.Add(new NameValuePair("RequireNonAnamorphic", item.RequireNonAnamorphic.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));                }                list.Add(new NameValuePair("TranscodingMaxAudioChannels", item.TranscodingMaxAudioChannels.HasValue ? item.TranscodingMaxAudioChannels.Value.ToString(CultureInfo.InvariantCulture) : string.Empty));                if (item.EnableSubtitlesInManifest)                {                    list.Add(new NameValuePair("EnableSubtitlesInManifest", item.EnableSubtitlesInManifest.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));                }                if (item.EnableMpegtsM2TsMode)                {                    list.Add(new NameValuePair("EnableMpegtsM2TsMode", item.EnableMpegtsM2TsMode.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));                }                if (item.EstimateContentLength)                {                    list.Add(new NameValuePair("EstimateContentLength", item.EstimateContentLength.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));                }                if (item.TranscodeSeekInfo != TranscodeSeekInfo.Auto)                {                    list.Add(new NameValuePair("TranscodeSeekInfo", item.TranscodeSeekInfo.ToString().ToLowerInvariant()));                }                if (item.CopyTimestamps)                {                    list.Add(new NameValuePair("CopyTimestamps", item.CopyTimestamps.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));                }                list.Add(new NameValuePair("RequireAvc", item.RequireAvc.ToString(CultureInfo.InvariantCulture).ToLowerInvariant()));            }            list.Add(new NameValuePair("Tag", item.MediaSource?.ETag ?? string.Empty));            string subtitleCodecs = item.SubtitleCodecs.Length == 0 ?               string.Empty :               string.Join(",", item.SubtitleCodecs);            list.Add(new NameValuePair("SubtitleCodec", item.SubtitleStreamIndex.HasValue && item.SubtitleDeliveryMethod == SubtitleDeliveryMethod.Embed ? subtitleCodecs : string.Empty));            if (isHls)            {                list.Add(new NameValuePair("SegmentContainer", item.Container ?? string.Empty));                if (item.SegmentLength.HasValue)                {                    list.Add(new NameValuePair("SegmentLength", item.SegmentLength.Value.ToString(CultureInfo.InvariantCulture)));                }                if (item.MinSegments.HasValue)                {                    list.Add(new NameValuePair("MinSegments", item.MinSegments.Value.ToString(CultureInfo.InvariantCulture)));                }                list.Add(new NameValuePair("BreakOnNonKeyFrames", item.BreakOnNonKeyFrames.ToString(CultureInfo.InvariantCulture)));            }            foreach (var pair in item.StreamOptions)            {                if (string.IsNullOrEmpty(pair.Value))                {                    continue;                }                // strip spaces to avoid having to encode h264 profile names                list.Add(new NameValuePair(pair.Key, pair.Value.Replace(" ", string.Empty, StringComparison.Ordinal)));            }            if (!item.IsDirectStream)            {                list.Add(new NameValuePair("TranscodeReasons", item.TranscodeReasons.ToString()));            }            return list;        }        public IEnumerable<SubtitleStreamInfo> GetSubtitleProfiles(ITranscoderSupport transcoderSupport, bool includeSelectedTrackOnly, string baseUrl, string? accessToken)        {            return GetSubtitleProfiles(transcoderSupport, includeSelectedTrackOnly, false, baseUrl, accessToken);        }        public IEnumerable<SubtitleStreamInfo> GetSubtitleProfiles(ITranscoderSupport transcoderSupport, bool includeSelectedTrackOnly, bool enableAllProfiles, string baseUrl, string? accessToken)        {            if (MediaSource is null)            {                return Enumerable.Empty<SubtitleStreamInfo>();            }            var list = new List<SubtitleStreamInfo>();            // HLS will preserve timestamps so we can just grab the full subtitle stream            long startPositionTicks = string.Equals(SubProtocol, "hls", StringComparison.OrdinalIgnoreCase)                ? 0                : (PlayMethod == PlayMethod.Transcode && !CopyTimestamps ? StartPositionTicks : 0);            // First add the selected track            if (SubtitleStreamIndex.HasValue)            {                foreach (var stream in MediaSource.MediaStreams)                {                    if (stream.Type == MediaStreamType.Subtitle && stream.Index == SubtitleStreamIndex.Value)                    {                        AddSubtitleProfiles(list, stream, transcoderSupport, enableAllProfiles, baseUrl, accessToken, startPositionTicks);                    }                }            }            if (!includeSelectedTrackOnly)            {                foreach (var stream in MediaSource.MediaStreams)                {                    if (stream.Type == MediaStreamType.Subtitle && (!SubtitleStreamIndex.HasValue || stream.Index != SubtitleStreamIndex.Value))                    {                        AddSubtitleProfiles(list, stream, transcoderSupport, enableAllProfiles, baseUrl, accessToken, startPositionTicks);                    }                }            }            return list;        }        private void AddSubtitleProfiles(List<SubtitleStreamInfo> list, MediaStream stream, ITranscoderSupport transcoderSupport, bool enableAllProfiles, string baseUrl, string? accessToken, long startPositionTicks)        {            if (enableAllProfiles)            {                foreach (var profile in DeviceProfile.SubtitleProfiles)                {                    var info = GetSubtitleStreamInfo(stream, baseUrl, accessToken, startPositionTicks, new[] { profile }, transcoderSupport);                    if (info is not null)                    {                        list.Add(info);                    }                }            }            else            {                var info = GetSubtitleStreamInfo(stream, baseUrl, accessToken, startPositionTicks, DeviceProfile.SubtitleProfiles, transcoderSupport);                if (info is not null)                {                    list.Add(info);                }            }        }        private SubtitleStreamInfo? GetSubtitleStreamInfo(MediaStream stream, string baseUrl, string? accessToken, long startPositionTicks, SubtitleProfile[] subtitleProfiles, ITranscoderSupport transcoderSupport)        {            if (MediaSource is null)            {                return null;            }            var subtitleProfile = StreamBuilder.GetSubtitleProfile(MediaSource, stream, subtitleProfiles, PlayMethod, transcoderSupport, Container, SubProtocol);            var info = new SubtitleStreamInfo            {                IsForced = stream.IsForced,                Language = stream.Language,                Name = stream.Language ?? "Unknown",                Format = subtitleProfile.Format,                Index = stream.Index,                DeliveryMethod = subtitleProfile.Method,                DisplayTitle = stream.DisplayTitle            };            if (info.DeliveryMethod == SubtitleDeliveryMethod.External)            {                if (MediaSource.Protocol == MediaProtocol.File || !string.Equals(stream.Codec, subtitleProfile.Format, StringComparison.OrdinalIgnoreCase) || !stream.IsExternal)                {                    info.Url = string.Format(                        CultureInfo.InvariantCulture,                        "{0}/Videos/{1}/{2}/Subtitles/{3}/{4}/Stream.{5}",                        baseUrl,                        ItemId,                        MediaSourceId,                        stream.Index.ToString(CultureInfo.InvariantCulture),                        startPositionTicks.ToString(CultureInfo.InvariantCulture),                        subtitleProfile.Format);                    if (!string.IsNullOrEmpty(accessToken))                    {                        info.Url += "?api_key=" + accessToken;                    }                    info.IsExternalUrl = false;                }                else                {                    info.Url = stream.Path;                    info.IsExternalUrl = true;                }            }            return info;        }        public int? GetTargetVideoBitDepth(string? codec)        {            var value = GetOption(codec, "videobitdepth");            if (int.TryParse(value, CultureInfo.InvariantCulture, out var result))            {                return result;            }            return null;        }        public int? GetTargetAudioBitDepth(string? codec)        {            var value = GetOption(codec, "audiobitdepth");            if (int.TryParse(value, CultureInfo.InvariantCulture, out var result))            {                return result;            }            return null;        }        public double? GetTargetVideoLevel(string? codec)        {            var value = GetOption(codec, "level");            if (double.TryParse(value, CultureInfo.InvariantCulture, out var result))            {                return result;            }            return null;        }        public int? GetTargetRefFrames(string? codec)        {            var value = GetOption(codec, "maxrefframes");            if (int.TryParse(value, CultureInfo.InvariantCulture, out var result))            {                return result;            }            return null;        }        public int? GetTargetAudioChannels(string? codec)        {            var defaultValue = GlobalMaxAudioChannels ?? TranscodingMaxAudioChannels;            var value = GetOption(codec, "audiochannels");            if (string.IsNullOrEmpty(value))            {                return defaultValue;            }            if (int.TryParse(value, NumberStyles.Integer, CultureInfo.InvariantCulture, out var result))            {                return Math.Min(result, defaultValue ?? result);            }            return defaultValue;        }        private int? GetMediaStreamCount(MediaStreamType type, int limit)        {            var count = MediaSource?.GetStreamCount(type);            if (count.HasValue)            {                count = Math.Min(count.Value, limit);            }            return count;        }    }}
 |