2020-06-19 15:00:50 +00:00
using System ;
2019-01-13 19:24:58 +00:00
using System.Collections.Generic ;
2020-08-06 14:17:45 +00:00
using System.ComponentModel.DataAnnotations ;
2019-01-13 19:24:58 +00:00
using System.Linq ;
using System.Threading ;
2020-08-21 20:01:19 +00:00
using System.Threading.Tasks ;
2020-06-08 18:20:33 +00:00
using Jellyfin.Api.Constants ;
2019-01-13 19:24:58 +00:00
using MediaBrowser.Controller.Configuration ;
2014-12-22 06:50:29 +00:00
using MediaBrowser.Controller.Entities ;
2013-06-27 19:06:37 +00:00
using MediaBrowser.Controller.Entities.Audio ;
using MediaBrowser.Controller.Entities.TV ;
using MediaBrowser.Controller.Library ;
2014-12-23 03:58:14 +00:00
using MediaBrowser.Controller.LiveTv ;
2014-12-21 05:57:06 +00:00
using MediaBrowser.Controller.Providers ;
2013-06-27 19:06:37 +00:00
using MediaBrowser.Model.Dto ;
2014-12-21 18:58:17 +00:00
using MediaBrowser.Model.Entities ;
2016-10-24 02:45:23 +00:00
using MediaBrowser.Model.Globalization ;
2018-09-12 17:26:21 +00:00
using MediaBrowser.Model.IO ;
2020-06-08 18:20:33 +00:00
using Microsoft.AspNetCore.Authorization ;
using Microsoft.AspNetCore.Http ;
using Microsoft.AspNetCore.Mvc ;
2013-06-27 19:06:37 +00:00
2020-06-08 18:20:33 +00:00
namespace Jellyfin.Api.Controllers
2013-06-27 19:06:37 +00:00
{
2020-06-08 18:20:33 +00:00
/// <summary>
/// Item update controller.
/// </summary>
2020-08-04 18:48:53 +00:00
[Route("")]
2020-06-08 18:20:33 +00:00
[Authorize(Policy = Policies.RequiresElevation)]
public class ItemUpdateController : BaseJellyfinApiController
2013-06-27 19:06:37 +00:00
{
private readonly ILibraryManager _libraryManager ;
2014-12-21 05:57:06 +00:00
private readonly IProviderManager _providerManager ;
private readonly ILocalizationManager _localizationManager ;
2018-09-12 17:26:21 +00:00
private readonly IFileSystem _fileSystem ;
2020-06-08 18:20:33 +00:00
private readonly IServerConfigurationManager _serverConfigurationManager ;
/// <summary>
/// Initializes a new instance of the <see cref="ItemUpdateController"/> class.
/// </summary>
/// <param name="fileSystem">Instance of the <see cref="IFileSystem"/> interface.</param>
/// <param name="libraryManager">Instance of the <see cref="ILibraryManager"/> interface.</param>
/// <param name="providerManager">Instance of the <see cref="IProviderManager"/> interface.</param>
/// <param name="localizationManager">Instance of the <see cref="ILocalizationManager"/> interface.</param>
/// <param name="serverConfigurationManager">Instance of the <see cref="IServerConfigurationManager"/> interface.</param>
public ItemUpdateController (
2019-11-17 22:05:39 +00:00
IFileSystem fileSystem ,
ILibraryManager libraryManager ,
IProviderManager providerManager ,
2020-06-08 18:20:33 +00:00
ILocalizationManager localizationManager ,
IServerConfigurationManager serverConfigurationManager )
2013-06-27 19:06:37 +00:00
{
_libraryManager = libraryManager ;
2014-12-21 05:57:06 +00:00
_providerManager = providerManager ;
_localizationManager = localizationManager ;
2018-09-12 17:26:21 +00:00
_fileSystem = fileSystem ;
2020-06-08 18:20:33 +00:00
_serverConfigurationManager = serverConfigurationManager ;
2014-12-21 05:57:06 +00:00
}
2020-06-08 18:20:33 +00:00
/// <summary>
/// Updates an item.
/// </summary>
/// <param name="itemId">The item id.</param>
/// <param name="request">The new item properties.</param>
2020-06-20 21:14:43 +00:00
/// <response code="204">Item updated.</response>
2020-06-08 18:20:33 +00:00
/// <response code="404">Item not found.</response>
2020-06-20 21:14:43 +00:00
/// <returns>An <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if the item could not be found.</returns>
2020-08-04 18:48:53 +00:00
[HttpPost("Items/{itemId}")]
2020-06-20 21:15:59 +00:00
[ProducesResponseType(StatusCodes.Status204NoContent)]
2020-06-08 18:20:33 +00:00
[ProducesResponseType(StatusCodes.Status404NotFound)]
2020-09-06 15:07:27 +00:00
public async Task < ActionResult > UpdateItem ( [ FromRoute , Required ] Guid itemId , [ FromBody , Required ] BaseItemDto request )
2014-12-21 05:57:06 +00:00
{
2020-06-08 18:20:33 +00:00
var item = _libraryManager . GetItemById ( itemId ) ;
if ( item = = null )
2014-12-21 18:58:17 +00:00
{
2020-06-08 18:20:33 +00:00
return NotFound ( ) ;
2014-12-21 18:58:17 +00:00
}
2014-02-19 05:21:03 +00:00
var newLockData = request . LockData ? ? false ;
2014-04-27 03:42:05 +00:00
var isLockedChanged = item . IsLocked ! = newLockData ;
2013-12-01 06:25:05 +00:00
2018-09-12 17:26:21 +00:00
var series = item as Series ;
2020-06-08 18:20:33 +00:00
var displayOrderChanged = series ! = null & & ! string . Equals (
series . DisplayOrder ? ? string . Empty ,
request . DisplayOrder ? ? string . Empty ,
StringComparison . OrdinalIgnoreCase ) ;
2018-09-12 17:26:21 +00:00
2017-06-06 06:13:49 +00:00
// Do this first so that metadata savers can pull the updates from the database.
2015-06-21 03:35:22 +00:00
if ( request . People ! = null )
{
2020-06-08 18:20:33 +00:00
_libraryManager . UpdatePeople (
item ,
request . People . Select ( x = > new PersonInfo
{
Name = x . Name ,
Role = x . Role ,
Type = x . Type
} ) . ToList ( ) ) ;
2015-06-21 03:35:22 +00:00
}
2017-06-06 06:13:49 +00:00
UpdateItem ( request , item ) ;
2017-09-13 18:41:48 +00:00
item . OnMetadataChanged ( ) ;
2020-08-21 20:01:19 +00:00
await item . UpdateToRepositoryAsync ( ItemUpdateType . MetadataEdit , CancellationToken . None ) . ConfigureAwait ( false ) ;
2017-06-06 06:13:49 +00:00
2014-04-27 03:42:05 +00:00
if ( isLockedChanged & & item . IsFolder )
2013-12-01 06:25:05 +00:00
{
var folder = ( Folder ) item ;
2015-01-25 06:34:50 +00:00
foreach ( var child in folder . GetRecursiveChildren ( ) )
2013-12-01 06:25:05 +00:00
{
2014-04-27 03:42:05 +00:00
child . IsLocked = newLockData ;
2020-08-21 20:01:19 +00:00
await child . UpdateToRepositoryAsync ( ItemUpdateType . MetadataEdit , CancellationToken . None ) . ConfigureAwait ( false ) ;
2013-12-01 06:25:05 +00:00
}
}
2018-09-12 17:26:21 +00:00
if ( displayOrderChanged )
{
2019-09-10 20:37:53 +00:00
_providerManager . QueueRefresh (
2020-06-08 18:20:33 +00:00
series ! . Id ,
2019-09-10 20:37:53 +00:00
new MetadataRefreshOptions ( new DirectoryService ( _fileSystem ) )
{
MetadataRefreshMode = MetadataRefreshMode . FullRefresh ,
ImageRefreshMode = MetadataRefreshMode . FullRefresh ,
ReplaceAllMetadata = true
} ,
RefreshPriority . High ) ;
2018-09-12 17:26:21 +00:00
}
2020-06-08 18:20:33 +00:00
2020-06-20 21:14:43 +00:00
return NoContent ( ) ;
2013-06-27 19:06:37 +00:00
}
2020-06-08 18:20:33 +00:00
/// <summary>
/// Gets metadata editor info for an item.
/// </summary>
/// <param name="itemId">The item id.</param>
/// <response code="200">Item metadata editor returned.</response>
/// <response code="404">Item not found.</response>
/// <returns>An <see cref="OkResult"/> on success containing the metadata editor, or a <see cref="NotFoundResult"/> if the item could not be found.</returns>
2020-08-04 18:48:53 +00:00
[HttpGet("Items/{itemId}/MetadataEditor")]
2020-06-08 18:20:33 +00:00
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
2020-09-06 15:07:27 +00:00
public ActionResult < MetadataEditorInfo > GetMetadataEditorInfo ( [ FromRoute , Required ] Guid itemId )
2014-07-13 04:55:56 +00:00
{
2020-06-08 18:20:33 +00:00
var item = _libraryManager . GetItemById ( itemId ) ;
var info = new MetadataEditorInfo
{
ParentalRatingOptions = _localizationManager . GetParentalRatings ( ) . ToArray ( ) ,
ExternalIdInfos = _providerManager . GetExternalIdInfos ( item ) . ToArray ( ) ,
Countries = _localizationManager . GetCountries ( ) . ToArray ( ) ,
Cultures = _localizationManager . GetCultures ( ) . ToArray ( )
} ;
if ( ! item . IsVirtualItem
& & ! ( item is ICollectionFolder )
& & ! ( item is UserView )
& & ! ( item is AggregateFolder )
& & ! ( item is LiveTvChannel )
& & ! ( item is IItemByName )
& & item . SourceType = = SourceType . Library )
{
var inheritedContentType = _libraryManager . GetInheritedContentType ( item ) ;
var configuredContentType = _libraryManager . GetConfiguredContentType ( item ) ;
if ( string . IsNullOrWhiteSpace ( inheritedContentType ) | |
! string . IsNullOrWhiteSpace ( configuredContentType ) )
{
info . ContentTypeOptions = GetContentTypeOptions ( true ) . ToArray ( ) ;
info . ContentType = configuredContentType ;
if ( string . IsNullOrWhiteSpace ( inheritedContentType )
| | string . Equals ( inheritedContentType , CollectionType . TvShows , StringComparison . OrdinalIgnoreCase ) )
{
info . ContentTypeOptions = info . ContentTypeOptions
. Where ( i = > string . IsNullOrWhiteSpace ( i . Value )
| | string . Equals ( i . Value , CollectionType . TvShows , StringComparison . OrdinalIgnoreCase ) )
. ToArray ( ) ;
}
}
}
return info ;
}
/// <summary>
/// Updates an item's content type.
/// </summary>
/// <param name="itemId">The item id.</param>
/// <param name="contentType">The content type of the item.</param>
2020-06-20 21:14:43 +00:00
/// <response code="204">Item content type updated.</response>
2020-06-08 18:20:33 +00:00
/// <response code="404">Item not found.</response>
2020-06-20 21:14:43 +00:00
/// <returns>An <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if the item could not be found.</returns>
2020-08-04 18:48:53 +00:00
[HttpPost("Items/{itemId}/ContentType")]
2020-06-20 21:15:59 +00:00
[ProducesResponseType(StatusCodes.Status204NoContent)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
2020-09-06 15:07:27 +00:00
public ActionResult UpdateItemContentType ( [ FromRoute , Required ] Guid itemId , [ FromQuery , Required ] string? contentType )
2020-06-08 18:20:33 +00:00
{
var item = _libraryManager . GetItemById ( itemId ) ;
if ( item = = null )
{
return NotFound ( ) ;
}
var path = item . ContainingFolderPath ;
var types = _serverConfigurationManager . Configuration . ContentTypes
. Where ( i = > ! string . IsNullOrWhiteSpace ( i . Name ) )
. Where ( i = > ! string . Equals ( i . Name , path , StringComparison . OrdinalIgnoreCase ) )
. ToList ( ) ;
if ( ! string . IsNullOrWhiteSpace ( contentType ) )
{
types . Add ( new NameValuePair
{
Name = path ,
Value = contentType
} ) ;
}
_serverConfigurationManager . Configuration . ContentTypes = types . ToArray ( ) ;
_serverConfigurationManager . SaveConfiguration ( ) ;
2020-06-20 21:14:43 +00:00
return NoContent ( ) ;
2014-07-13 04:55:56 +00:00
}
2013-06-27 19:06:37 +00:00
private void UpdateItem ( BaseItemDto request , BaseItem item )
{
item . Name = request . Name ;
2014-03-24 17:54:45 +00:00
item . ForcedSortName = request . ForcedSortName ;
2013-06-27 19:06:37 +00:00
2016-10-01 07:06:00 +00:00
item . OriginalTitle = string . IsNullOrWhiteSpace ( request . OriginalTitle ) ? null : request . OriginalTitle ;
2016-04-20 05:21:40 +00:00
2016-10-17 16:35:29 +00:00
item . CriticRating = request . CriticRating ;
2013-09-17 02:08:18 +00:00
2013-09-11 17:54:59 +00:00
item . CommunityRating = request . CommunityRating ;
2013-06-27 19:06:37 +00:00
item . IndexNumber = request . IndexNumber ;
item . ParentIndexNumber = request . ParentIndexNumber ;
item . Overview = request . Overview ;
item . Genres = request . Genres ;
2013-12-05 16:50:21 +00:00
2020-04-05 16:07:43 +00:00
if ( item is Episode episode )
2013-12-08 20:33:24 +00:00
{
episode . AirsAfterSeasonNumber = request . AirsAfterSeasonNumber ;
episode . AirsBeforeEpisodeNumber = request . AirsBeforeEpisodeNumber ;
episode . AirsBeforeSeasonNumber = request . AirsBeforeSeasonNumber ;
}
2014-04-27 03:42:05 +00:00
2016-06-02 17:43:29 +00:00
item . Tags = request . Tags ;
2013-11-24 23:37:38 +00:00
2016-10-08 05:57:38 +00:00
if ( request . Taglines ! = null )
2014-06-29 02:30:20 +00:00
{
2016-10-08 05:57:38 +00:00
item . Tagline = request . Taglines . FirstOrDefault ( ) ;
2014-06-29 02:30:20 +00:00
}
2013-11-24 23:37:38 +00:00
if ( request . Studios ! = null )
{
2017-08-09 19:56:38 +00:00
item . Studios = request . Studios . Select ( x = > x . Name ) . ToArray ( ) ;
2013-11-24 23:37:38 +00:00
}
2013-09-03 14:26:17 +00:00
if ( request . DateCreated . HasValue )
{
2014-07-13 04:55:56 +00:00
item . DateCreated = NormalizeDateTime ( request . DateCreated . Value ) ;
2013-09-03 14:26:17 +00:00
}
2014-07-13 04:55:56 +00:00
item . EndDate = request . EndDate . HasValue ? NormalizeDateTime ( request . EndDate . Value ) : ( DateTime ? ) null ;
item . PremiereDate = request . PremiereDate . HasValue ? NormalizeDateTime ( request . PremiereDate . Value ) : ( DateTime ? ) null ;
2013-06-27 19:06:37 +00:00
item . ProductionYear = request . ProductionYear ;
2016-08-26 17:24:04 +00:00
item . OfficialRating = string . IsNullOrWhiteSpace ( request . OfficialRating ) ? null : request . OfficialRating ;
2013-06-27 19:06:37 +00:00
item . CustomRating = request . CustomRating ;
2013-09-03 14:26:17 +00:00
2016-10-21 18:41:49 +00:00
if ( request . ProductionLocations ! = null )
{
2017-08-09 19:56:38 +00:00
item . ProductionLocations = request . ProductionLocations ;
2016-10-21 18:41:49 +00:00
}
2015-09-29 17:35:23 +00:00
item . PreferredMetadataCountryCode = request . PreferredMetadataCountryCode ;
item . PreferredMetadataLanguage = request . PreferredMetadataLanguage ;
2014-01-03 20:32:27 +00:00
2020-04-05 16:07:43 +00:00
if ( item is IHasDisplayOrder hasDisplayOrder )
2014-01-03 20:32:27 +00:00
{
hasDisplayOrder . DisplayOrder = request . DisplayOrder ;
}
2014-04-27 03:42:05 +00:00
2020-04-05 16:07:43 +00:00
if ( item is IHasAspectRatio hasAspectRatio )
2013-11-20 03:15:48 +00:00
{
hasAspectRatio . AspectRatio = request . AspectRatio ;
}
2014-02-19 05:21:03 +00:00
2016-03-27 21:11:27 +00:00
item . IsLocked = request . LockData ? ? false ;
2014-02-19 05:21:03 +00:00
if ( request . LockedFields ! = null )
2013-06-27 19:06:37 +00:00
{
item . LockedFields = request . LockedFields ;
}
2013-09-13 18:37:44 +00:00
// Only allow this for series. Runtimes for media comes from ffprobe.
if ( item is Series )
{
item . RunTimeTicks = request . RunTimeTicks ;
}
2013-06-27 19:06:37 +00:00
foreach ( var pair in request . ProviderIds . ToList ( ) )
{
if ( string . IsNullOrEmpty ( pair . Value ) )
{
request . ProviderIds . Remove ( pair . Key ) ;
}
}
item . ProviderIds = request . ProviderIds ;
2020-04-05 16:07:43 +00:00
if ( item is Video video )
2013-07-29 12:34:46 +00:00
{
video . Video3DFormat = request . Video3DFormat ;
}
2013-09-03 14:26:17 +00:00
2015-05-18 22:23:03 +00:00
if ( request . AlbumArtists ! = null )
2013-06-27 19:06:37 +00:00
{
2020-04-05 16:07:43 +00:00
if ( item is IHasAlbumArtist hasAlbumArtists )
2015-05-18 22:23:03 +00:00
{
hasAlbumArtists . AlbumArtists = request
. AlbumArtists
. Select ( i = > i . Name )
2017-08-10 18:01:31 +00:00
. ToArray ( ) ;
2015-05-18 22:23:03 +00:00
}
2013-06-27 19:06:37 +00:00
}
2015-05-18 22:23:03 +00:00
if ( request . ArtistItems ! = null )
2015-03-13 17:25:28 +00:00
{
2020-04-05 16:07:43 +00:00
if ( item is IHasArtist hasArtists )
2015-05-18 22:23:03 +00:00
{
hasArtists . Artists = request
. ArtistItems
. Select ( i = > i . Name )
2017-08-24 19:52:19 +00:00
. ToArray ( ) ;
2015-05-18 22:23:03 +00:00
}
2015-03-13 17:25:28 +00:00
}
2015-03-16 01:48:25 +00:00
2020-06-08 18:20:33 +00:00
switch ( item )
2013-07-10 20:10:55 +00:00
{
2020-06-08 18:20:33 +00:00
case Audio song :
song . Album = request . Album ;
break ;
case MusicVideo musicVideo :
musicVideo . Album = request . Album ;
break ;
case Series series :
{
series . Status = GetSeriesStatus ( request ) ;
2013-07-10 20:10:55 +00:00
2020-06-08 18:20:33 +00:00
if ( request . AirDays ! = null )
{
series . AirDays = request . AirDays ;
series . AirTime = request . AirTime ;
}
2017-08-13 20:15:07 +00:00
2020-06-08 18:20:33 +00:00
break ;
2017-08-13 20:15:07 +00:00
}
2013-06-27 19:06:37 +00:00
}
}
2017-02-19 03:46:09 +00:00
private SeriesStatus ? GetSeriesStatus ( BaseItemDto item )
{
if ( string . IsNullOrEmpty ( item . Status ) )
{
return null ;
}
return ( SeriesStatus ) Enum . Parse ( typeof ( SeriesStatus ) , item . Status , true ) ;
}
2020-06-08 18:20:33 +00:00
private DateTime NormalizeDateTime ( DateTime val )
{
return DateTime . SpecifyKind ( val , DateTimeKind . Utc ) ;
}
private List < NameValuePair > GetContentTypeOptions ( bool isForItem )
{
var list = new List < NameValuePair > ( ) ;
if ( isForItem )
{
list . Add ( new NameValuePair
{
Name = "Inherit" ,
Value = string . Empty
} ) ;
}
list . Add ( new NameValuePair
{
Name = "Movies" ,
Value = "movies"
} ) ;
list . Add ( new NameValuePair
{
Name = "Music" ,
Value = "music"
} ) ;
list . Add ( new NameValuePair
{
Name = "Shows" ,
Value = "tvshows"
} ) ;
if ( ! isForItem )
{
list . Add ( new NameValuePair
{
Name = "Books" ,
Value = "books"
} ) ;
}
list . Add ( new NameValuePair
{
Name = "HomeVideos" ,
Value = "homevideos"
} ) ;
list . Add ( new NameValuePair
{
Name = "MusicVideos" ,
Value = "musicvideos"
} ) ;
list . Add ( new NameValuePair
{
Name = "Photos" ,
Value = "photos"
} ) ;
if ( ! isForItem )
{
list . Add ( new NameValuePair
{
Name = "MixedContent" ,
Value = string . Empty
} ) ;
}
foreach ( var val in list )
{
val . Name = _localizationManager . GetLocalizedString ( val . Name ) ;
}
return list ;
}
2013-06-27 19:06:37 +00:00
}
}