| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538 | using System;using System.Collections.Generic;using System.ComponentModel.DataAnnotations;using System.Linq;using System.Threading;using System.Threading.Tasks;using Jellyfin.Api.Constants;using Jellyfin.Api.Extensions;using Jellyfin.Api.Helpers;using Jellyfin.Data.Enums;using MediaBrowser.Common.Api;using MediaBrowser.Controller.Configuration;using MediaBrowser.Controller.Entities;using MediaBrowser.Controller.Entities.Audio;using MediaBrowser.Controller.Entities.TV;using MediaBrowser.Controller.Library;using MediaBrowser.Controller.LiveTv;using MediaBrowser.Controller.Providers;using MediaBrowser.Model.Dto;using MediaBrowser.Model.Entities;using MediaBrowser.Model.Globalization;using MediaBrowser.Model.IO;using Microsoft.AspNetCore.Authorization;using Microsoft.AspNetCore.Http;using Microsoft.AspNetCore.Mvc;namespace Jellyfin.Api.Controllers;/// <summary>/// Item update controller./// </summary>[Route("")][Authorize(Policy = Policies.RequiresElevation)]public class ItemUpdateController : BaseJellyfinApiController{    private readonly ILibraryManager _libraryManager;    private readonly IProviderManager _providerManager;    private readonly ILocalizationManager _localizationManager;    private readonly IFileSystem _fileSystem;    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(        IFileSystem fileSystem,        ILibraryManager libraryManager,        IProviderManager providerManager,        ILocalizationManager localizationManager,        IServerConfigurationManager serverConfigurationManager)    {        _libraryManager = libraryManager;        _providerManager = providerManager;        _localizationManager = localizationManager;        _fileSystem = fileSystem;        _serverConfigurationManager = serverConfigurationManager;    }    /// <summary>    /// Updates an item.    /// </summary>    /// <param name="itemId">The item id.</param>    /// <param name="request">The new item properties.</param>    /// <response code="204">Item updated.</response>    /// <response code="404">Item not found.</response>    /// <returns>An <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if the item could not be found.</returns>    [HttpPost("Items/{itemId}")]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public async Task<ActionResult> UpdateItem([FromRoute, Required] Guid itemId, [FromBody, Required] BaseItemDto request)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        var newLockData = request.LockData ?? false;        var isLockedChanged = item.IsLocked != newLockData;        var series = item as Series;        var displayOrderChanged = series is not null && !string.Equals(            series.DisplayOrder ?? string.Empty,            request.DisplayOrder ?? string.Empty,            StringComparison.OrdinalIgnoreCase);        // Do this first so that metadata savers can pull the updates from the database.        if (request.People is not null)        {            _libraryManager.UpdatePeople(                item,                request.People.Select(x => new PersonInfo                {                    Name = x.Name,                    Role = x.Role,                    Type = x.Type                }).ToList());        }        await UpdateItem(request, item).ConfigureAwait(false);        item.OnMetadataChanged();        await item.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);        if (isLockedChanged && item.IsFolder)        {            var folder = (Folder)item;            foreach (var child in folder.GetRecursiveChildren())            {                child.IsLocked = newLockData;                await child.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);            }        }        if (displayOrderChanged)        {            _providerManager.QueueRefresh(                series!.Id,                new MetadataRefreshOptions(new DirectoryService(_fileSystem))                {                    MetadataRefreshMode = MetadataRefreshMode.FullRefresh,                    ImageRefreshMode = MetadataRefreshMode.FullRefresh,                    ReplaceAllMetadata = true                },                RefreshPriority.High);        }        return NoContent();    }    /// <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>    [HttpGet("Items/{itemId}/MetadataEditor")]    [ProducesResponseType(StatusCodes.Status200OK)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public ActionResult<MetadataEditorInfo> GetMetadataEditorInfo([FromRoute, Required] Guid itemId)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is null)        {            return NotFound();        }        var info = new MetadataEditorInfo        {            ParentalRatingOptions = _localizationManager.GetParentalRatings().ToList(),            ExternalIdInfos = _providerManager.GetExternalIdInfos(item).ToArray(),            Countries = _localizationManager.GetCountries().ToArray(),            Cultures = _localizationManager.GetCultures().ToArray()        };        if (!item.IsVirtualItem            && item is not ICollectionFolder            && item is not UserView            && item is not AggregateFolder            && item is not LiveTvChannel            && item is not IItemByName            && item.SourceType == SourceType.Library)        {            var inheritedContentType = _libraryManager.GetInheritedContentType(item);            var configuredContentType = _libraryManager.GetConfiguredContentType(item);            if (inheritedContentType is null || configuredContentType is not null)            {                info.ContentTypeOptions = GetContentTypeOptions(true).ToArray();                info.ContentType = configuredContentType;                if (inheritedContentType is null || inheritedContentType == CollectionType.tvshows)                {                    info.ContentTypeOptions = info.ContentTypeOptions                        .Where(i => string.IsNullOrWhiteSpace(i.Value)                                    || string.Equals(i.Value, "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>    /// <response code="204">Item content type updated.</response>    /// <response code="404">Item not found.</response>    /// <returns>An <see cref="NoContentResult"/> on success, or a <see cref="NotFoundResult"/> if the item could not be found.</returns>    [HttpPost("Items/{itemId}/ContentType")]    [ProducesResponseType(StatusCodes.Status204NoContent)]    [ProducesResponseType(StatusCodes.Status404NotFound)]    public ActionResult UpdateItemContentType([FromRoute, Required] Guid itemId, [FromQuery] string? contentType)    {        var item = _libraryManager.GetItemById<BaseItem>(itemId, User.GetUserId());        if (item is 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();        return NoContent();    }    private async Task UpdateItem(BaseItemDto request, BaseItem item)    {        item.Name = request.Name;        item.ForcedSortName = request.ForcedSortName;        item.OriginalTitle = string.IsNullOrWhiteSpace(request.OriginalTitle) ? null : request.OriginalTitle;        item.CriticRating = request.CriticRating;        item.CommunityRating = request.CommunityRating;        item.IndexNumber = request.IndexNumber;        item.ParentIndexNumber = request.ParentIndexNumber;        item.Overview = request.Overview;        item.Genres = request.Genres;        if (item is Episode episode)        {            episode.AirsAfterSeasonNumber = request.AirsAfterSeasonNumber;            episode.AirsBeforeEpisodeNumber = request.AirsBeforeEpisodeNumber;            episode.AirsBeforeSeasonNumber = request.AirsBeforeSeasonNumber;        }        if (request.Height is not null && item is LiveTvChannel channel)        {            channel.Height = request.Height.Value;        }        if (request.Taglines is not null)        {            item.Tagline = request.Taglines.FirstOrDefault();        }        if (request.Studios is not null)        {            item.Studios = Array.ConvertAll(request.Studios, x => x.Name);        }        if (request.DateCreated.HasValue)        {            item.DateCreated = NormalizeDateTime(request.DateCreated.Value);        }        item.EndDate = request.EndDate.HasValue ? NormalizeDateTime(request.EndDate.Value) : null;        item.PremiereDate = request.PremiereDate.HasValue ? NormalizeDateTime(request.PremiereDate.Value) : null;        item.ProductionYear = request.ProductionYear;        request.OfficialRating = string.IsNullOrWhiteSpace(request.OfficialRating) ? null : request.OfficialRating;        item.OfficialRating = request.OfficialRating;        item.CustomRating = request.CustomRating;        var currentTags = item.Tags;        var newTags = request.Tags;        var removedTags = currentTags.Except(newTags).ToList();        var addedTags = newTags.Except(currentTags).ToList();        item.Tags = newTags;        if (item is Series rseries)        {            foreach (var season in rseries.Children.OfType<Season>())            {                if (!season.LockedFields.Contains(MetadataField.OfficialRating))                {                    season.OfficialRating = request.OfficialRating;                }                season.CustomRating = request.CustomRating;                if (!season.LockedFields.Contains(MetadataField.Tags))                {                    season.Tags = season.Tags.Concat(addedTags).Except(removedTags).Distinct().ToArray();                }                season.OnMetadataChanged();                await season.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);                foreach (var ep in season.Children.OfType<Episode>())                {                    if (!ep.LockedFields.Contains(MetadataField.OfficialRating))                    {                        ep.OfficialRating = request.OfficialRating;                    }                    ep.CustomRating = request.CustomRating;                    if (!ep.LockedFields.Contains(MetadataField.Tags))                    {                        ep.Tags = ep.Tags.Concat(addedTags).Except(removedTags).Distinct().ToArray();                    }                    ep.OnMetadataChanged();                    await ep.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);                }            }        }        else if (item is Season season)        {            foreach (var ep in season.Children.OfType<Episode>())            {                if (!ep.LockedFields.Contains(MetadataField.OfficialRating))                {                    ep.OfficialRating = request.OfficialRating;                }                ep.CustomRating = request.CustomRating;                if (!ep.LockedFields.Contains(MetadataField.Tags))                {                    ep.Tags = ep.Tags.Concat(addedTags).Except(removedTags).Distinct().ToArray();                }                ep.OnMetadataChanged();                await ep.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);            }        }        else if (item is MusicAlbum album)        {            foreach (BaseItem track in album.Children)            {                if (!track.LockedFields.Contains(MetadataField.OfficialRating))                {                    track.OfficialRating = request.OfficialRating;                }                track.CustomRating = request.CustomRating;                if (!track.LockedFields.Contains(MetadataField.Tags))                {                    track.Tags = track.Tags.Concat(addedTags).Except(removedTags).Distinct().ToArray();                }                track.OnMetadataChanged();                await track.UpdateToRepositoryAsync(ItemUpdateType.MetadataEdit, CancellationToken.None).ConfigureAwait(false);            }        }        if (request.ProductionLocations is not null)        {            item.ProductionLocations = request.ProductionLocations;        }        item.PreferredMetadataCountryCode = request.PreferredMetadataCountryCode;        item.PreferredMetadataLanguage = request.PreferredMetadataLanguage;        if (item is IHasDisplayOrder hasDisplayOrder)        {            hasDisplayOrder.DisplayOrder = request.DisplayOrder;        }        if (item is IHasAspectRatio hasAspectRatio)        {            hasAspectRatio.AspectRatio = request.AspectRatio;        }        item.IsLocked = request.LockData ?? false;        if (request.LockedFields is not null)        {            item.LockedFields = request.LockedFields;        }        // Only allow this for series. Runtimes for media comes from ffprobe.        if (item is Series)        {            item.RunTimeTicks = request.RunTimeTicks;        }        foreach (var pair in request.ProviderIds.ToList())        {            if (string.IsNullOrEmpty(pair.Value))            {                request.ProviderIds.Remove(pair.Key);            }        }        item.ProviderIds = request.ProviderIds;        if (item is Video video)        {            video.Video3DFormat = request.Video3DFormat;        }        if (request.AlbumArtists is not null)        {            if (item is IHasAlbumArtist hasAlbumArtists)            {                hasAlbumArtists.AlbumArtists = Array.ConvertAll(request.AlbumArtists, i => i.Name);            }        }        if (request.ArtistItems is not null)        {            if (item is IHasArtist hasArtists)            {                hasArtists.Artists = Array.ConvertAll(request.ArtistItems, i => i.Name);            }        }        switch (item)        {            case Audio song:                song.Album = request.Album;                break;            case MusicVideo musicVideo:                musicVideo.Album = request.Album;                break;            case Series series:                {                    series.Status = GetSeriesStatus(request);                    if (request.AirDays is not null)                    {                        series.AirDays = request.AirDays;                        series.AirTime = request.AirTime;                    }                    break;                }        }    }    private SeriesStatus? GetSeriesStatus(BaseItemDto item)    {        if (string.IsNullOrEmpty(item.Status))        {            return null;        }        return (SeriesStatus)Enum.Parse(typeof(SeriesStatus), item.Status, true);    }    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;    }}
 |