| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334 | using System;using System.Collections.Generic;using System.ComponentModel.DataAnnotations;using System.Globalization;using System.Linq;using System.Threading.Tasks;using Jellyfin.Api.Attributes;using Jellyfin.Api.Constants;using Jellyfin.Api.Helpers;using Jellyfin.Api.Models.StreamingDtos;using MediaBrowser.Common.Extensions;using MediaBrowser.Controller.Devices;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.MediaEncoding;using MediaBrowser.Controller.Net;using MediaBrowser.Model.Dlna;using MediaBrowser.Model.MediaInfo;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;using Microsoft.Extensions.Logging;namespace Jellyfin.Api.Controllers{    /// <summary>    /// The universal audio controller.    /// </summary>    [Route("")]    public class UniversalAudioController : BaseJellyfinApiController    {        private readonly IAuthorizationContext _authorizationContext;        private readonly IDeviceManager _deviceManager;        private readonly ILibraryManager _libraryManager;        private readonly ILogger<UniversalAudioController> _logger;        private readonly MediaInfoHelper _mediaInfoHelper;        private readonly AudioHelper _audioHelper;        private readonly DynamicHlsHelper _dynamicHlsHelper;        /// <summary>        /// Initializes a new instance of the <see cref="UniversalAudioController"/> class.        /// </summary>        /// <param name="authorizationContext">Instance of the <see cref="IAuthorizationContext"/> interface.</param>        /// <param name="deviceManager">Instance of the <see cref="IDeviceManager"/> interface.</param>        /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>        /// <param name="logger">Instance of the <see cref="ILogger{UniversalAudioController}"/> interface.</param>        /// <param name="mediaInfoHelper">Instance of <see cref="MediaInfoHelper"/>.</param>        /// <param name="audioHelper">Instance of <see cref="AudioHelper"/>.</param>        /// <param name="dynamicHlsHelper">Instance of <see cref="DynamicHlsHelper"/>.</param>        public UniversalAudioController(            IAuthorizationContext authorizationContext,            IDeviceManager deviceManager,            ILibraryManager libraryManager,            ILogger<UniversalAudioController> logger,            MediaInfoHelper mediaInfoHelper,            AudioHelper audioHelper,            DynamicHlsHelper dynamicHlsHelper)        {            _authorizationContext = authorizationContext;            _deviceManager = deviceManager;            _libraryManager = libraryManager;            _logger = logger;            _mediaInfoHelper = mediaInfoHelper;            _audioHelper = audioHelper;            _dynamicHlsHelper = dynamicHlsHelper;        }        /// <summary>        /// Gets an audio stream.        /// </summary>        /// <param name="itemId">The item id.</param>        /// <param name="container">Optional. The audio container.</param>        /// <param name="mediaSourceId">The media version id, if playing an alternate version.</param>        /// <param name="deviceId">The device id of the client requesting. Used to stop encoding processes when needed.</param>        /// <param name="userId">Optional. The user id.</param>        /// <param name="audioCodec">Optional. The audio codec to transcode to.</param>        /// <param name="maxAudioChannels">Optional. The maximum number of audio channels.</param>        /// <param name="transcodingAudioChannels">Optional. The number of how many audio channels to transcode to.</param>        /// <param name="maxStreamingBitrate">Optional. The maximum streaming bitrate.</param>        /// <param name="startTimeTicks">Optional. Specify a starting offset, in ticks. 1 tick = 10000 ms.</param>        /// <param name="transcodingContainer">Optional. The container to transcode to.</param>        /// <param name="transcodingProtocol">Optional. The transcoding protocol.</param>        /// <param name="maxAudioSampleRate">Optional. The maximum audio sample rate.</param>        /// <param name="maxAudioBitDepth">Optional. The maximum audio bit depth.</param>        /// <param name="enableRemoteMedia">Optional. Whether to enable remote media.</param>        /// <param name="breakOnNonKeyFrames">Optional. Whether to break on non key frames.</param>        /// <param name="enableRedirection">Whether to enable redirection. Defaults to true.</param>        /// <response code="200">Audio stream returned.</response>        /// <response code="302">Redirected to remote audio stream.</response>        /// <returns>A <see cref="Task"/> containing the audio file.</returns>        [HttpGet("Audio/{itemId}/universal")]        [HttpGet("Audio/{itemId}/universal.{container}", Name = "GetUniversalAudioStream_2")]        [HttpHead("Audio/{itemId}/universal", Name = "HeadUniversalAudioStream")]        [HttpHead("Audio/{itemId}/universal.{container}", Name = "HeadUniversalAudioStream_2")]        [Authorize(Policy = Policies.DefaultAuthorization)]        [ProducesResponseType(StatusCodes.Status200OK)]        [ProducesResponseType(StatusCodes.Status302Found)]        [ProducesAudioFile]        public async Task<ActionResult> GetUniversalAudioStream(            [FromRoute, Required] Guid itemId,            [FromRoute] string? container,            [FromQuery] string? mediaSourceId,            [FromQuery] string? deviceId,            [FromQuery] Guid? userId,            [FromQuery] string? audioCodec,            [FromQuery] int? maxAudioChannels,            [FromQuery] int? transcodingAudioChannels,            [FromQuery] long? maxStreamingBitrate,            [FromQuery] long? startTimeTicks,            [FromQuery] string? transcodingContainer,            [FromQuery] string? transcodingProtocol,            [FromQuery] int? maxAudioSampleRate,            [FromQuery] int? maxAudioBitDepth,            [FromQuery] bool? enableRemoteMedia,            [FromQuery] bool breakOnNonKeyFrames,            [FromQuery] bool enableRedirection = true)        {            var deviceProfile = GetDeviceProfile(container, transcodingContainer, audioCodec, transcodingProtocol, breakOnNonKeyFrames, transcodingAudioChannels, maxAudioSampleRate, maxAudioBitDepth, maxAudioChannels);            _authorizationContext.GetAuthorizationInfo(Request).DeviceId = deviceId;            var authInfo = _authorizationContext.GetAuthorizationInfo(Request);            _logger.LogInformation("GetPostedPlaybackInfo profile: {@Profile}", deviceProfile);            if (deviceProfile == null)            {                var clientCapabilities = _deviceManager.GetCapabilities(authInfo.DeviceId);                if (clientCapabilities != null)                {                    deviceProfile = clientCapabilities.DeviceProfile;                }            }            var info = await _mediaInfoHelper.GetPlaybackInfo(                    itemId,                    userId,                    mediaSourceId)                .ConfigureAwait(false);            if (deviceProfile != null)            {                // set device specific data                var item = _libraryManager.GetItemById(itemId);                foreach (var sourceInfo in info.MediaSources)                {                    _mediaInfoHelper.SetDeviceSpecificData(                        item,                        sourceInfo,                        deviceProfile,                        authInfo,                        maxStreamingBitrate ?? deviceProfile.MaxStreamingBitrate,                        startTimeTicks ?? 0,                        mediaSourceId ?? string.Empty,                        null,                        null,                        maxAudioChannels,                        info!.PlaySessionId!,                        userId ?? Guid.Empty,                        true,                        true,                        true,                        true,                        true,                        Request.HttpContext.GetNormalizedRemoteIp());                }                _mediaInfoHelper.SortMediaSources(info, maxStreamingBitrate);            }            if (info.MediaSources != null)            {                foreach (var source in info.MediaSources)                {                    _mediaInfoHelper.NormalizeMediaSourceContainer(source, deviceProfile!, DlnaProfileType.Video);                }            }            var mediaSource = info.MediaSources![0];            if (mediaSource.SupportsDirectPlay && mediaSource.Protocol == MediaProtocol.Http)            {                if (enableRedirection)                {                    if (mediaSource.IsRemote && enableRemoteMedia.HasValue && enableRemoteMedia.Value)                    {                        return Redirect(mediaSource.Path);                    }                }            }            var isStatic = mediaSource.SupportsDirectStream;            if (!isStatic && string.Equals(mediaSource.TranscodingSubProtocol, "hls", StringComparison.OrdinalIgnoreCase))            {                // hls segment container can only be mpegts or fmp4 per ffmpeg documentation                // TODO: remove this when we switch back to the segment muxer                var supportedHlsContainers = new[] { "mpegts", "fmp4" };                var dynamicHlsRequestDto = new HlsAudioRequestDto                {                    Id = itemId,                    Container = ".m3u8",                    Static = isStatic,                    PlaySessionId = info.PlaySessionId,                    // fallback to mpegts if device reports some weird value unsupported by hls                    SegmentContainer = Array.Exists(supportedHlsContainers, element => element == transcodingContainer) ? transcodingContainer : "mpegts",                    MediaSourceId = mediaSourceId,                    DeviceId = deviceId,                    AudioCodec = audioCodec,                    EnableAutoStreamCopy = true,                    AllowAudioStreamCopy = true,                    AllowVideoStreamCopy = true,                    BreakOnNonKeyFrames = breakOnNonKeyFrames,                    AudioSampleRate = maxAudioSampleRate,                    MaxAudioChannels = maxAudioChannels,                    MaxAudioBitDepth = maxAudioBitDepth,                    AudioChannels = isStatic ? (int?)null : Convert.ToInt32(Math.Min(maxStreamingBitrate ?? 192000, int.MaxValue)),                    StartTimeTicks = startTimeTicks,                    SubtitleMethod = SubtitleDeliveryMethod.Hls,                    RequireAvc = true,                    DeInterlace = true,                    RequireNonAnamorphic = true,                    EnableMpegtsM2TsMode = true,                    TranscodeReasons = mediaSource.TranscodeReasons == null ? null : string.Join(",", mediaSource.TranscodeReasons.Select(i => i.ToString()).ToArray()),                    Context = EncodingContext.Static,                    StreamOptions = new Dictionary<string, string>(),                    EnableAdaptiveBitrateStreaming = true                };                return await _dynamicHlsHelper.GetMasterHlsPlaylist(TranscodingJobType.Hls, dynamicHlsRequestDto, true)                    .ConfigureAwait(false);            }            var audioStreamingDto = new StreamingRequestDto            {                Id = itemId,                Container = isStatic ? null : ("." + mediaSource.TranscodingContainer),                Static = isStatic,                PlaySessionId = info.PlaySessionId,                MediaSourceId = mediaSourceId,                DeviceId = deviceId,                AudioCodec = audioCodec,                EnableAutoStreamCopy = true,                AllowAudioStreamCopy = true,                AllowVideoStreamCopy = true,                BreakOnNonKeyFrames = breakOnNonKeyFrames,                AudioSampleRate = maxAudioSampleRate,                MaxAudioChannels = maxAudioChannels,                AudioBitRate = isStatic ? (int?)null : Convert.ToInt32(Math.Min(maxStreamingBitrate ?? 192000, int.MaxValue)),                MaxAudioBitDepth = maxAudioBitDepth,                AudioChannels = maxAudioChannels,                CopyTimestamps = true,                StartTimeTicks = startTimeTicks,                SubtitleMethod = SubtitleDeliveryMethod.Embed,                TranscodeReasons = mediaSource.TranscodeReasons == null ? null : string.Join(",", mediaSource.TranscodeReasons.Select(i => i.ToString()).ToArray()),                Context = EncodingContext.Static            };            return await _audioHelper.GetAudioStream(TranscodingJobType.Progressive, audioStreamingDto).ConfigureAwait(false);        }        private DeviceProfile GetDeviceProfile(            string? container,            string? transcodingContainer,            string? audioCodec,            string? transcodingProtocol,            bool? breakOnNonKeyFrames,            int? transcodingAudioChannels,            int? maxAudioSampleRate,            int? maxAudioBitDepth,            int? maxAudioChannels)        {            var deviceProfile = new DeviceProfile();            var directPlayProfiles = new List<DirectPlayProfile>();            var containers = RequestHelpers.Split(container, ',', true);            foreach (var cont in containers)            {                var parts = RequestHelpers.Split(cont, ',', true);                var audioCodecs = parts.Length == 1 ? null : string.Join(",", parts.Skip(1).ToArray());                directPlayProfiles.Add(new DirectPlayProfile { Type = DlnaProfileType.Audio, Container = parts[0], AudioCodec = audioCodecs });            }            deviceProfile.DirectPlayProfiles = directPlayProfiles.ToArray();            deviceProfile.TranscodingProfiles = new[]            {                new TranscodingProfile                {                    Type = DlnaProfileType.Audio,                    Context = EncodingContext.Streaming,                    Container = transcodingContainer,                    AudioCodec = audioCodec,                    Protocol = transcodingProtocol,                    BreakOnNonKeyFrames = breakOnNonKeyFrames ?? false,                    MaxAudioChannels = transcodingAudioChannels?.ToString(CultureInfo.InvariantCulture)                }            };            var codecProfiles = new List<CodecProfile>();            var conditions = new List<ProfileCondition>();            if (maxAudioSampleRate.HasValue)            {                // codec profile                conditions.Add(new ProfileCondition { Condition = ProfileConditionType.LessThanEqual, IsRequired = false, Property = ProfileConditionValue.AudioSampleRate, Value = maxAudioSampleRate.Value.ToString(CultureInfo.InvariantCulture) });            }            if (maxAudioBitDepth.HasValue)            {                // codec profile                conditions.Add(new ProfileCondition { Condition = ProfileConditionType.LessThanEqual, IsRequired = false, Property = ProfileConditionValue.AudioBitDepth, Value = maxAudioBitDepth.Value.ToString(CultureInfo.InvariantCulture) });            }            if (maxAudioChannels.HasValue)            {                // codec profile                conditions.Add(new ProfileCondition { Condition = ProfileConditionType.LessThanEqual, IsRequired = false, Property = ProfileConditionValue.AudioChannels, Value = maxAudioChannels.Value.ToString(CultureInfo.InvariantCulture) });            }            if (conditions.Count > 0)            {                // codec profile                codecProfiles.Add(new CodecProfile { Type = CodecType.Audio, Container = container, Conditions = conditions.ToArray() });            }            deviceProfile.CodecProfiles = codecProfiles.ToArray();            return deviceProfile;        }    }}
 |