VideosController.cs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. using System;
  2. using System.Globalization;
  3. using System.Linq;
  4. using System.Threading;
  5. using Jellyfin.Api.Constants;
  6. using Jellyfin.Api.Extensions;
  7. using Jellyfin.Api.Helpers;
  8. using MediaBrowser.Controller.Dto;
  9. using MediaBrowser.Controller.Entities;
  10. using MediaBrowser.Controller.Library;
  11. using MediaBrowser.Model.Dto;
  12. using MediaBrowser.Model.Entities;
  13. using MediaBrowser.Model.Querying;
  14. using Microsoft.AspNetCore.Authorization;
  15. using Microsoft.AspNetCore.Http;
  16. using Microsoft.AspNetCore.Mvc;
  17. namespace Jellyfin.Api.Controllers
  18. {
  19. /// <summary>
  20. /// The videos controller.
  21. /// </summary>
  22. [Route("Videos")]
  23. public class VideosController : BaseJellyfinApiController
  24. {
  25. private readonly ILibraryManager _libraryManager;
  26. private readonly IUserManager _userManager;
  27. private readonly IDtoService _dtoService;
  28. /// <summary>
  29. /// Initializes a new instance of the <see cref="VideosController"/> class.
  30. /// </summary>
  31. /// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
  32. /// <param name="userManager">Instance of the <see cref="IUserManager"/> interface.</param>
  33. /// <param name="dtoService">Instance of the <see cref="IDtoService"/> interface.</param>
  34. public VideosController(
  35. ILibraryManager libraryManager,
  36. IUserManager userManager,
  37. IDtoService dtoService)
  38. {
  39. _libraryManager = libraryManager;
  40. _userManager = userManager;
  41. _dtoService = dtoService;
  42. }
  43. /// <summary>
  44. /// Gets additional parts for a video.
  45. /// </summary>
  46. /// <param name="itemId">The item id.</param>
  47. /// <param name="userId">Optional. Filter by user id, and attach user data.</param>
  48. /// <response code="200">Additional parts returned.</response>
  49. /// <returns>A <see cref="QueryResult{BaseItemDto}"/> with the parts.</returns>
  50. [HttpGet("{itemId}/AdditionalParts")]
  51. [Authorize(Policy = Policies.DefaultAuthorization)]
  52. [ProducesResponseType(StatusCodes.Status200OK)]
  53. public ActionResult<QueryResult<BaseItemDto>> GetAdditionalPart([FromRoute] Guid itemId, [FromQuery] Guid? userId)
  54. {
  55. var user = userId.HasValue && !userId.Equals(Guid.Empty)
  56. ? _userManager.GetUserById(userId.Value)
  57. : null;
  58. var item = itemId.Equals(Guid.Empty)
  59. ? (!userId.Equals(Guid.Empty)
  60. ? _libraryManager.GetUserRootFolder()
  61. : _libraryManager.RootFolder)
  62. : _libraryManager.GetItemById(itemId);
  63. var dtoOptions = new DtoOptions();
  64. dtoOptions = dtoOptions.AddClientFields(Request);
  65. BaseItemDto[] items;
  66. if (item is Video video)
  67. {
  68. items = video.GetAdditionalParts()
  69. .Select(i => _dtoService.GetBaseItemDto(i, dtoOptions, user, video))
  70. .ToArray();
  71. }
  72. else
  73. {
  74. items = Array.Empty<BaseItemDto>();
  75. }
  76. var result = new QueryResult<BaseItemDto>
  77. {
  78. Items = items,
  79. TotalRecordCount = items.Length
  80. };
  81. return result;
  82. }
  83. /// <summary>
  84. /// Removes alternate video sources.
  85. /// </summary>
  86. /// <param name="itemId">The item id.</param>
  87. /// <response code="204">Alternate sources deleted.</response>
  88. /// <response code="404">Video not found.</response>
  89. /// <returns>A <see cref="NoContentResult"/> indicating success, or a <see cref="NotFoundResult"/> if the video doesn't exist.</returns>
  90. [HttpDelete("{itemId}/AlternateSources")]
  91. [Authorize(Policy = Policies.RequiresElevation)]
  92. [ProducesResponseType(StatusCodes.Status200OK)]
  93. [ProducesResponseType(StatusCodes.Status404NotFound)]
  94. public ActionResult DeleteAlternateSources([FromRoute] Guid itemId)
  95. {
  96. var video = (Video)_libraryManager.GetItemById(itemId);
  97. if (video == null)
  98. {
  99. return NotFound("The video either does not exist or the id does not belong to a video.");
  100. }
  101. foreach (var link in video.GetLinkedAlternateVersions())
  102. {
  103. link.SetPrimaryVersionId(null);
  104. link.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  105. link.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  106. }
  107. video.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  108. video.SetPrimaryVersionId(null);
  109. video.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  110. return NoContent();
  111. }
  112. /// <summary>
  113. /// Merges videos into a single record.
  114. /// </summary>
  115. /// <param name="itemIds">Item id list. This allows multiple, comma delimited.</param>
  116. /// <response code="204">Videos merged.</response>
  117. /// <response code="400">Supply at least 2 video ids.</response>
  118. /// <returns>A <see cref="NoContentResult"/> indicating success, or a <see cref="BadRequestResult"/> if less than two ids were supplied.</returns>
  119. [HttpPost("MergeVersions")]
  120. [Authorize(Policy = Policies.RequiresElevation)]
  121. [ProducesResponseType(StatusCodes.Status204NoContent)]
  122. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  123. public ActionResult MergeVersions([FromQuery] string? itemIds)
  124. {
  125. var items = RequestHelpers.Split(itemIds, ',', true)
  126. .Select(i => _libraryManager.GetItemById(i))
  127. .OfType<Video>()
  128. .OrderBy(i => i.Id)
  129. .ToList();
  130. if (items.Count < 2)
  131. {
  132. return BadRequest("Please supply at least two videos to merge.");
  133. }
  134. var videosWithVersions = items.Where(i => i.MediaSourceCount > 1).ToList();
  135. var primaryVersion = videosWithVersions.FirstOrDefault();
  136. if (primaryVersion == null)
  137. {
  138. primaryVersion = items
  139. .OrderBy(i =>
  140. {
  141. if (i.Video3DFormat.HasValue || i.VideoType != VideoType.VideoFile)
  142. {
  143. return 1;
  144. }
  145. return 0;
  146. })
  147. .ThenByDescending(i => i.GetDefaultVideoStream()?.Width ?? 0)
  148. .First();
  149. }
  150. var list = primaryVersion.LinkedAlternateVersions.ToList();
  151. foreach (var item in items.Where(i => i.Id != primaryVersion.Id))
  152. {
  153. item.SetPrimaryVersionId(primaryVersion.Id.ToString("N", CultureInfo.InvariantCulture));
  154. item.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  155. list.Add(new LinkedChild
  156. {
  157. Path = item.Path,
  158. ItemId = item.Id
  159. });
  160. foreach (var linkedItem in item.LinkedAlternateVersions)
  161. {
  162. if (!list.Any(i => string.Equals(i.Path, linkedItem.Path, StringComparison.OrdinalIgnoreCase)))
  163. {
  164. list.Add(linkedItem);
  165. }
  166. }
  167. if (item.LinkedAlternateVersions.Length > 0)
  168. {
  169. item.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  170. item.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  171. }
  172. }
  173. primaryVersion.LinkedAlternateVersions = list.ToArray();
  174. primaryVersion.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  175. return NoContent();
  176. }
  177. }
  178. }