123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312 |
- using System;
- using System.Buffers;
- using System.ComponentModel.DataAnnotations;
- using System.Linq;
- using System.Net.Mime;
- using System.Threading.Tasks;
- using Jellyfin.Api.Attributes;
- using Jellyfin.Api.Constants;
- using Jellyfin.Api.Extensions;
- using Jellyfin.Api.Helpers;
- using Jellyfin.Api.Models.MediaInfoDtos;
- using MediaBrowser.Common.Extensions;
- using MediaBrowser.Controller.Devices;
- using MediaBrowser.Controller.Library;
- using MediaBrowser.Model.MediaInfo;
- using Microsoft.AspNetCore.Authorization;
- using Microsoft.AspNetCore.Http;
- using Microsoft.AspNetCore.Mvc;
- using Microsoft.AspNetCore.Mvc.ModelBinding;
- using Microsoft.Extensions.Logging;
- namespace Jellyfin.Api.Controllers;
- /// <summary>
- /// The media info controller.
- /// </summary>
- [Route("")]
- [Authorize(Policy = Policies.DefaultAuthorization)]
- public class MediaInfoController : BaseJellyfinApiController
- {
- private readonly IMediaSourceManager _mediaSourceManager;
- private readonly IDeviceManager _deviceManager;
- private readonly ILibraryManager _libraryManager;
- private readonly ILogger<MediaInfoController> _logger;
- private readonly MediaInfoHelper _mediaInfoHelper;
- /// <summary>
- /// Initializes a new instance of the <see cref="MediaInfoController"/> class.
- /// </summary>
- /// <param name="mediaSourceManager">Instance of the <see cref="IMediaSourceManager"/> 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{MediaInfoController}"/> interface.</param>
- /// <param name="mediaInfoHelper">Instance of the <see cref="MediaInfoHelper"/>.</param>
- public MediaInfoController(
- IMediaSourceManager mediaSourceManager,
- IDeviceManager deviceManager,
- ILibraryManager libraryManager,
- ILogger<MediaInfoController> logger,
- MediaInfoHelper mediaInfoHelper)
- {
- _mediaSourceManager = mediaSourceManager;
- _deviceManager = deviceManager;
- _libraryManager = libraryManager;
- _logger = logger;
- _mediaInfoHelper = mediaInfoHelper;
- }
- /// <summary>
- /// Gets live playback media info for an item.
- /// </summary>
- /// <param name="itemId">The item id.</param>
- /// <param name="userId">The user id.</param>
- /// <response code="200">Playback info returned.</response>
- /// <returns>A <see cref="Task"/> containing a <see cref="PlaybackInfoResponse"/> with the playback information.</returns>
- [HttpGet("Items/{itemId}/PlaybackInfo")]
- [ProducesResponseType(StatusCodes.Status200OK)]
- public async Task<ActionResult<PlaybackInfoResponse>> GetPlaybackInfo([FromRoute, Required] Guid itemId, [FromQuery, Required] Guid userId)
- {
- return await _mediaInfoHelper.GetPlaybackInfo(
- itemId,
- userId)
- .ConfigureAwait(false);
- }
- /// <summary>
- /// Gets live playback media info for an item.
- /// </summary>
- /// <remarks>
- /// For backwards compatibility parameters can be sent via Query or Body, with Query having higher precedence.
- /// Query parameters are obsolete.
- /// </remarks>
- /// <param name="itemId">The item id.</param>
- /// <param name="userId">The user id.</param>
- /// <param name="maxStreamingBitrate">The maximum streaming bitrate.</param>
- /// <param name="startTimeTicks">The start time in ticks.</param>
- /// <param name="audioStreamIndex">The audio stream index.</param>
- /// <param name="subtitleStreamIndex">The subtitle stream index.</param>
- /// <param name="maxAudioChannels">The maximum number of audio channels.</param>
- /// <param name="mediaSourceId">The media source id.</param>
- /// <param name="liveStreamId">The livestream id.</param>
- /// <param name="autoOpenLiveStream">Whether to auto open the livestream.</param>
- /// <param name="enableDirectPlay">Whether to enable direct play. Default: true.</param>
- /// <param name="enableDirectStream">Whether to enable direct stream. Default: true.</param>
- /// <param name="enableTranscoding">Whether to enable transcoding. Default: true.</param>
- /// <param name="allowVideoStreamCopy">Whether to allow to copy the video stream. Default: true.</param>
- /// <param name="allowAudioStreamCopy">Whether to allow to copy the audio stream. Default: true.</param>
- /// <param name="playbackInfoDto">The playback info.</param>
- /// <response code="200">Playback info returned.</response>
- /// <returns>A <see cref="Task"/> containing a <see cref="PlaybackInfoResponse"/> with the playback info.</returns>
- [HttpPost("Items/{itemId}/PlaybackInfo")]
- [ProducesResponseType(StatusCodes.Status200OK)]
- public async Task<ActionResult<PlaybackInfoResponse>> GetPostedPlaybackInfo(
- [FromRoute, Required] Guid itemId,
- [FromQuery, ParameterObsolete] Guid? userId,
- [FromQuery, ParameterObsolete] int? maxStreamingBitrate,
- [FromQuery, ParameterObsolete] long? startTimeTicks,
- [FromQuery, ParameterObsolete] int? audioStreamIndex,
- [FromQuery, ParameterObsolete] int? subtitleStreamIndex,
- [FromQuery, ParameterObsolete] int? maxAudioChannels,
- [FromQuery, ParameterObsolete] string? mediaSourceId,
- [FromQuery, ParameterObsolete] string? liveStreamId,
- [FromQuery, ParameterObsolete] bool? autoOpenLiveStream,
- [FromQuery, ParameterObsolete] bool? enableDirectPlay,
- [FromQuery, ParameterObsolete] bool? enableDirectStream,
- [FromQuery, ParameterObsolete] bool? enableTranscoding,
- [FromQuery, ParameterObsolete] bool? allowVideoStreamCopy,
- [FromQuery, ParameterObsolete] bool? allowAudioStreamCopy,
- [FromBody(EmptyBodyBehavior = EmptyBodyBehavior.Allow)] PlaybackInfoDto? playbackInfoDto)
- {
- var profile = playbackInfoDto?.DeviceProfile;
- _logger.LogDebug("GetPostedPlaybackInfo profile: {@Profile}", profile);
- if (profile is null)
- {
- var caps = _deviceManager.GetCapabilities(User.GetDeviceId());
- if (caps is not null)
- {
- profile = caps.DeviceProfile;
- }
- }
- // Copy params from posted body
- // TODO clean up when breaking API compatibility.
- userId ??= playbackInfoDto?.UserId;
- maxStreamingBitrate ??= playbackInfoDto?.MaxStreamingBitrate;
- startTimeTicks ??= playbackInfoDto?.StartTimeTicks;
- audioStreamIndex ??= playbackInfoDto?.AudioStreamIndex;
- subtitleStreamIndex ??= playbackInfoDto?.SubtitleStreamIndex;
- maxAudioChannels ??= playbackInfoDto?.MaxAudioChannels;
- mediaSourceId ??= playbackInfoDto?.MediaSourceId;
- liveStreamId ??= playbackInfoDto?.LiveStreamId;
- autoOpenLiveStream ??= playbackInfoDto?.AutoOpenLiveStream ?? false;
- enableDirectPlay ??= playbackInfoDto?.EnableDirectPlay ?? true;
- enableDirectStream ??= playbackInfoDto?.EnableDirectStream ?? true;
- enableTranscoding ??= playbackInfoDto?.EnableTranscoding ?? true;
- allowVideoStreamCopy ??= playbackInfoDto?.AllowVideoStreamCopy ?? true;
- allowAudioStreamCopy ??= playbackInfoDto?.AllowAudioStreamCopy ?? true;
- var info = await _mediaInfoHelper.GetPlaybackInfo(
- itemId,
- userId,
- mediaSourceId,
- liveStreamId)
- .ConfigureAwait(false);
- if (info.ErrorCode is not null)
- {
- return info;
- }
- if (profile is not null)
- {
- // set device specific data
- var item = _libraryManager.GetItemById(itemId);
- foreach (var mediaSource in info.MediaSources)
- {
- _mediaInfoHelper.SetDeviceSpecificData(
- item,
- mediaSource,
- profile,
- User,
- maxStreamingBitrate ?? profile.MaxStreamingBitrate,
- startTimeTicks ?? 0,
- mediaSourceId ?? string.Empty,
- audioStreamIndex,
- subtitleStreamIndex,
- maxAudioChannels,
- info.PlaySessionId!,
- userId ?? Guid.Empty,
- enableDirectPlay.Value,
- enableDirectStream.Value,
- enableTranscoding.Value,
- allowVideoStreamCopy.Value,
- allowAudioStreamCopy.Value,
- Request.HttpContext.GetNormalizedRemoteIp());
- }
- _mediaInfoHelper.SortMediaSources(info, maxStreamingBitrate);
- }
- if (autoOpenLiveStream.Value)
- {
- var mediaSource = string.IsNullOrWhiteSpace(mediaSourceId) ? info.MediaSources[0] : info.MediaSources.FirstOrDefault(i => string.Equals(i.Id, mediaSourceId, StringComparison.Ordinal));
- if (mediaSource is not null && mediaSource.RequiresOpening && string.IsNullOrWhiteSpace(mediaSource.LiveStreamId))
- {
- var openStreamResult = await _mediaInfoHelper.OpenMediaSource(
- HttpContext,
- new LiveStreamRequest
- {
- AudioStreamIndex = audioStreamIndex,
- DeviceProfile = playbackInfoDto?.DeviceProfile,
- EnableDirectPlay = enableDirectPlay.Value,
- EnableDirectStream = enableDirectStream.Value,
- ItemId = itemId,
- MaxAudioChannels = maxAudioChannels,
- MaxStreamingBitrate = maxStreamingBitrate,
- PlaySessionId = info.PlaySessionId,
- StartTimeTicks = startTimeTicks,
- SubtitleStreamIndex = subtitleStreamIndex,
- UserId = userId ?? Guid.Empty,
- OpenToken = mediaSource.OpenToken
- }).ConfigureAwait(false);
- info.MediaSources = new[] { openStreamResult.MediaSource };
- }
- }
- return info;
- }
- /// <summary>
- /// Opens a media source.
- /// </summary>
- /// <param name="openToken">The open token.</param>
- /// <param name="userId">The user id.</param>
- /// <param name="playSessionId">The play session id.</param>
- /// <param name="maxStreamingBitrate">The maximum streaming bitrate.</param>
- /// <param name="startTimeTicks">The start time in ticks.</param>
- /// <param name="audioStreamIndex">The audio stream index.</param>
- /// <param name="subtitleStreamIndex">The subtitle stream index.</param>
- /// <param name="maxAudioChannels">The maximum number of audio channels.</param>
- /// <param name="itemId">The item id.</param>
- /// <param name="openLiveStreamDto">The open live stream dto.</param>
- /// <param name="enableDirectPlay">Whether to enable direct play. Default: true.</param>
- /// <param name="enableDirectStream">Whether to enable direct stream. Default: true.</param>
- /// <response code="200">Media source opened.</response>
- /// <returns>A <see cref="Task"/> containing a <see cref="LiveStreamResponse"/>.</returns>
- [HttpPost("LiveStreams/Open")]
- [ProducesResponseType(StatusCodes.Status200OK)]
- public async Task<ActionResult<LiveStreamResponse>> OpenLiveStream(
- [FromQuery] string? openToken,
- [FromQuery] Guid? userId,
- [FromQuery] string? playSessionId,
- [FromQuery] int? maxStreamingBitrate,
- [FromQuery] long? startTimeTicks,
- [FromQuery] int? audioStreamIndex,
- [FromQuery] int? subtitleStreamIndex,
- [FromQuery] int? maxAudioChannels,
- [FromQuery] Guid? itemId,
- [FromBody] OpenLiveStreamDto? openLiveStreamDto,
- [FromQuery] bool? enableDirectPlay,
- [FromQuery] bool? enableDirectStream)
- {
- var request = new LiveStreamRequest
- {
- OpenToken = openToken ?? openLiveStreamDto?.OpenToken,
- UserId = userId ?? openLiveStreamDto?.UserId ?? Guid.Empty,
- PlaySessionId = playSessionId ?? openLiveStreamDto?.PlaySessionId,
- MaxStreamingBitrate = maxStreamingBitrate ?? openLiveStreamDto?.MaxStreamingBitrate,
- StartTimeTicks = startTimeTicks ?? openLiveStreamDto?.StartTimeTicks,
- AudioStreamIndex = audioStreamIndex ?? openLiveStreamDto?.AudioStreamIndex,
- SubtitleStreamIndex = subtitleStreamIndex ?? openLiveStreamDto?.SubtitleStreamIndex,
- MaxAudioChannels = maxAudioChannels ?? openLiveStreamDto?.MaxAudioChannels,
- ItemId = itemId ?? openLiveStreamDto?.ItemId ?? Guid.Empty,
- DeviceProfile = openLiveStreamDto?.DeviceProfile,
- EnableDirectPlay = enableDirectPlay ?? openLiveStreamDto?.EnableDirectPlay ?? true,
- EnableDirectStream = enableDirectStream ?? openLiveStreamDto?.EnableDirectStream ?? true,
- DirectPlayProtocols = openLiveStreamDto?.DirectPlayProtocols ?? new[] { MediaProtocol.Http }
- };
- return await _mediaInfoHelper.OpenMediaSource(HttpContext, request).ConfigureAwait(false);
- }
- /// <summary>
- /// Closes a media source.
- /// </summary>
- /// <param name="liveStreamId">The livestream id.</param>
- /// <response code="204">Livestream closed.</response>
- /// <returns>A <see cref="NoContentResult"/> indicating success.</returns>
- [HttpPost("LiveStreams/Close")]
- [ProducesResponseType(StatusCodes.Status204NoContent)]
- public async Task<ActionResult> CloseLiveStream([FromQuery, Required] string liveStreamId)
- {
- await _mediaSourceManager.CloseLiveStream(liveStreamId).ConfigureAwait(false);
- return NoContent();
- }
- /// <summary>
- /// Tests the network with a request with the size of the bitrate.
- /// </summary>
- /// <param name="size">The bitrate. Defaults to 102400.</param>
- /// <response code="200">Test buffer returned.</response>
- /// <returns>A <see cref="FileResult"/> with specified bitrate.</returns>
- [HttpGet("Playback/BitrateTest")]
- [ProducesResponseType(StatusCodes.Status200OK)]
- [ProducesFile(MediaTypeNames.Application.Octet)]
- public ActionResult GetBitrateTestBytes([FromQuery][Range(1, 100_000_000, ErrorMessage = "The requested size must be greater than or equal to {1} and less than or equal to {2}")] int size = 102400)
- {
- byte[] buffer = ArrayPool<byte>.Shared.Rent(size);
- try
- {
- Random.Shared.NextBytes(buffer);
- return File(buffer, MediaTypeNames.Application.Octet);
- }
- finally
- {
- ArrayPool<byte>.Shared.Return(buffer);
- }
- }
- }
|