2
0

VideosController.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  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.Equals(Guid.Empty) ? _userManager.GetUserById(userId) : null;
  56. var item = itemId.Equals(Guid.Empty)
  57. ? (!userId.Equals(Guid.Empty)
  58. ? _libraryManager.GetUserRootFolder()
  59. : _libraryManager.RootFolder)
  60. : _libraryManager.GetItemById(itemId);
  61. var dtoOptions = new DtoOptions();
  62. dtoOptions = dtoOptions.AddClientFields(Request);
  63. BaseItemDto[] items;
  64. if (item is Video video)
  65. {
  66. items = video.GetAdditionalParts()
  67. .Select(i => _dtoService.GetBaseItemDto(i, dtoOptions, user, video))
  68. .ToArray();
  69. }
  70. else
  71. {
  72. items = Array.Empty<BaseItemDto>();
  73. }
  74. var result = new QueryResult<BaseItemDto>
  75. {
  76. Items = items,
  77. TotalRecordCount = items.Length
  78. };
  79. return result;
  80. }
  81. /// <summary>
  82. /// Removes alternate video sources.
  83. /// </summary>
  84. /// <param name="itemId">The item id.</param>
  85. /// <response code="204">Alternate sources deleted.</response>
  86. /// <response code="404">Video not found.</response>
  87. /// <returns>A <see cref="NoContentResult"/> indicating success, or a <see cref="NotFoundResult"/> if the video doesn't exist.</returns>
  88. [HttpDelete("{itemId}/AlternateSources")]
  89. [Authorize(Policy = Policies.RequiresElevation)]
  90. [ProducesResponseType(StatusCodes.Status200OK)]
  91. [ProducesResponseType(StatusCodes.Status404NotFound)]
  92. public ActionResult DeleteAlternateSources([FromRoute] Guid itemId)
  93. {
  94. var video = (Video)_libraryManager.GetItemById(itemId);
  95. if (video == null)
  96. {
  97. return NotFound("The video either does not exist or the id does not belong to a video.");
  98. }
  99. foreach (var link in video.GetLinkedAlternateVersions())
  100. {
  101. link.SetPrimaryVersionId(null);
  102. link.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  103. link.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  104. }
  105. video.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  106. video.SetPrimaryVersionId(null);
  107. video.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  108. return NoContent();
  109. }
  110. /// <summary>
  111. /// Merges videos into a single record.
  112. /// </summary>
  113. /// <param name="itemIds">Item id list. This allows multiple, comma delimited.</param>
  114. /// <response code="204">Videos merged.</response>
  115. /// <response code="400">Supply at least 2 video ids.</response>
  116. /// <returns>A <see cref="NoContentResult"/> indicating success, or a <see cref="BadRequestResult"/> if less than two ids were supplied.</returns>
  117. [HttpPost("MergeVersions")]
  118. [Authorize(Policy = Policies.RequiresElevation)]
  119. [ProducesResponseType(StatusCodes.Status204NoContent)]
  120. [ProducesResponseType(StatusCodes.Status400BadRequest)]
  121. public ActionResult MergeVersions([FromQuery] string itemIds)
  122. {
  123. var items = RequestHelpers.Split(itemIds, ',', true)
  124. .Select(i => _libraryManager.GetItemById(i))
  125. .OfType<Video>()
  126. .OrderBy(i => i.Id)
  127. .ToList();
  128. if (items.Count < 2)
  129. {
  130. return BadRequest("Please supply at least two videos to merge.");
  131. }
  132. var videosWithVersions = items.Where(i => i.MediaSourceCount > 1).ToList();
  133. var primaryVersion = videosWithVersions.FirstOrDefault();
  134. if (primaryVersion == null)
  135. {
  136. primaryVersion = items
  137. .OrderBy(i =>
  138. {
  139. if (i.Video3DFormat.HasValue || i.VideoType != VideoType.VideoFile)
  140. {
  141. return 1;
  142. }
  143. return 0;
  144. })
  145. .ThenByDescending(i => i.GetDefaultVideoStream()?.Width ?? 0)
  146. .First();
  147. }
  148. var list = primaryVersion.LinkedAlternateVersions.ToList();
  149. foreach (var item in items.Where(i => i.Id != primaryVersion.Id))
  150. {
  151. item.SetPrimaryVersionId(primaryVersion.Id.ToString("N", CultureInfo.InvariantCulture));
  152. item.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  153. list.Add(new LinkedChild
  154. {
  155. Path = item.Path,
  156. ItemId = item.Id
  157. });
  158. foreach (var linkedItem in item.LinkedAlternateVersions)
  159. {
  160. if (!list.Any(i => string.Equals(i.Path, linkedItem.Path, StringComparison.OrdinalIgnoreCase)))
  161. {
  162. list.Add(linkedItem);
  163. }
  164. }
  165. if (item.LinkedAlternateVersions.Length > 0)
  166. {
  167. item.LinkedAlternateVersions = Array.Empty<LinkedChild>();
  168. item.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  169. }
  170. }
  171. primaryVersion.LinkedAlternateVersions = list.ToArray();
  172. primaryVersion.UpdateToRepository(ItemUpdateType.MetadataEdit, CancellationToken.None);
  173. return NoContent();
  174. }
  175. }
  176. }