| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 | 
							- using System;
 
- using System.Collections.Generic;
 
- using MediaBrowser.Controller.Entities;
 
- using MediaBrowser.Controller.Entities.Audio;
 
- using MediaBrowser.Controller.Entities.Movies;
 
- using MediaBrowser.Controller.Entities.TV;
 
- using MediaBrowser.Controller.Providers;
 
- using MediaBrowser.Model.Entities;
 
- using MediaBrowser.Providers.Manager;
 
- using Xunit;
 
- namespace Jellyfin.Providers.Tests.Manager
 
- {
 
-     public class MetadataServiceTests
 
-     {
 
-         [Theory]
 
-         [InlineData(false, false)]
 
-         [InlineData(true, false)]
 
-         [InlineData(true, true)]
 
-         public void MergeBaseItemData_MergeMetadataSettings_MergesWhenSet(bool mergeMetadataSettings, bool defaultDate)
 
-         {
 
-             var newLocked = new[] { MetadataField.Cast };
 
-             var newString = "new";
 
-             var newDate = DateTime.Now;
 
-             var oldLocked = new[] { MetadataField.Genres };
 
-             var oldString = "old";
 
-             var oldDate = DateTime.UnixEpoch;
 
-             var source = new MetadataResult<Movie>
 
-             {
 
-                 Item = new Movie
 
-                 {
 
-                     LockedFields = newLocked,
 
-                     IsLocked = true,
 
-                     PreferredMetadataCountryCode = newString,
 
-                     PreferredMetadataLanguage = newString,
 
-                     DateCreated = newDate
 
-                 }
 
-             };
 
-             if (defaultDate)
 
-             {
 
-                 source.Item.DateCreated = default;
 
-             }
 
-             var target = new MetadataResult<Movie>
 
-             {
 
-                 Item = new Movie
 
-                 {
 
-                     LockedFields = oldLocked,
 
-                     IsLocked = false,
 
-                     PreferredMetadataCountryCode = oldString,
 
-                     PreferredMetadataLanguage = oldString,
 
-                     DateCreated = oldDate
 
-                 }
 
-             };
 
-             MetadataService<Movie, MovieInfo>.MergeBaseItemData(source, target, Array.Empty<MetadataField>(), true, mergeMetadataSettings);
 
-             if (mergeMetadataSettings)
 
-             {
 
-                 Assert.Equal(newLocked, target.Item.LockedFields);
 
-                 Assert.True(target.Item.IsLocked);
 
-                 Assert.Equal(newString, target.Item.PreferredMetadataCountryCode);
 
-                 Assert.Equal(newString, target.Item.PreferredMetadataLanguage);
 
-                 Assert.Equal(defaultDate ? oldDate : newDate, target.Item.DateCreated);
 
-             }
 
-             else
 
-             {
 
-                 Assert.Equal(oldLocked, target.Item.LockedFields);
 
-                 Assert.False(target.Item.IsLocked);
 
-                 Assert.Equal(oldString, target.Item.PreferredMetadataCountryCode);
 
-                 Assert.Equal(oldString, target.Item.PreferredMetadataLanguage);
 
-                 Assert.Equal(oldDate, target.Item.DateCreated);
 
-             }
 
-         }
 
-         [Theory]
 
-         [InlineData("Name", MetadataField.Name, false)]
 
-         [InlineData("OriginalTitle", null, false)]
 
-         [InlineData("OfficialRating", MetadataField.OfficialRating)]
 
-         [InlineData("CustomRating")]
 
-         [InlineData("Tagline")]
 
-         [InlineData("Overview", MetadataField.Overview)]
 
-         [InlineData("DisplayOrder", null, false)]
 
-         [InlineData("ForcedSortName", null, false)]
 
-         public void MergeBaseItemData_StringField_ReplacesAppropriately(string propName, MetadataField? lockField = null, bool replacesWithEmpty = true)
 
-         {
 
-             var oldValue = "Old";
 
-             var newValue = "New";
 
-             // Use type Series to hit DisplayOrder
 
-             Assert.False(TestMergeBaseItemData<Series, SeriesInfo>(propName, oldValue, newValue, null, false, out _));
 
-             if (lockField is not null)
 
-             {
 
-                 Assert.False(TestMergeBaseItemData<Series, SeriesInfo>(propName, oldValue, newValue, lockField, true, out _));
 
-                 Assert.False(TestMergeBaseItemData<Series, SeriesInfo>(propName, null, newValue, lockField, false, out _));
 
-                 Assert.False(TestMergeBaseItemData<Series, SeriesInfo>(propName, string.Empty, newValue, lockField, false, out _));
 
-             }
 
-             Assert.True(TestMergeBaseItemData<Series, SeriesInfo>(propName, oldValue, newValue, null, true, out _));
 
-             Assert.True(TestMergeBaseItemData<Series, SeriesInfo>(propName, null, newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Series, SeriesInfo>(propName, string.Empty, newValue, null, false, out _));
 
-             var replacedWithEmpty = TestMergeBaseItemData<Series, SeriesInfo>(propName, oldValue, string.Empty, null, true, out _);
 
-             Assert.Equal(replacesWithEmpty, replacedWithEmpty);
 
-         }
 
-         [Theory]
 
-         [InlineData("Genres", MetadataField.Genres)]
 
-         [InlineData("Studios", MetadataField.Studios)]
 
-         [InlineData("Tags", MetadataField.Tags)]
 
-         [InlineData("ProductionLocations", MetadataField.ProductionLocations)]
 
-         [InlineData("AlbumArtists")]
 
-         public void MergeBaseItemData_StringArrayField_ReplacesAppropriately(string propName, MetadataField? lockField = null)
 
-         {
 
-             // Note that arrays are replaced, not merged
 
-             var oldValue = new[] { "Old" };
 
-             var newValue = new[] { "New" };
 
-             // Use type Audio to hit AlbumArtists
 
-             Assert.False(TestMergeBaseItemData<Audio, SongInfo>(propName, oldValue, newValue, null, false, out _));
 
-             if (lockField is not null)
 
-             {
 
-                 Assert.False(TestMergeBaseItemData<Audio, SongInfo>(propName, oldValue, newValue, lockField, true, out _));
 
-                 Assert.False(TestMergeBaseItemData<Audio, SongInfo>(propName, Array.Empty<string>(), newValue, lockField, false, out _));
 
-             }
 
-             Assert.True(TestMergeBaseItemData<Audio, SongInfo>(propName, oldValue, newValue, null, true, out _));
 
-             Assert.True(TestMergeBaseItemData<Audio, SongInfo>(propName, Array.Empty<string>(), newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Audio, SongInfo>(propName, oldValue, Array.Empty<string>(), null, true, out _));
 
-         }
 
-         public static TheoryData<string, object, object> MergeBaseItemData_SimpleField_ReplacesAppropriately_TestData()
 
-             => new()
 
-             {
 
-                 { "IndexNumber", 1, 2 },
 
-                 { "ParentIndexNumber", 1, 2 },
 
-                 { "ProductionYear", 1, 2 },
 
-                 { "CommunityRating", 1.0f, 2.0f },
 
-                 { "CriticRating", 1.0f, 2.0f },
 
-                 { "EndDate", DateTime.UnixEpoch, DateTime.Now },
 
-                 { "PremiereDate", DateTime.UnixEpoch, DateTime.Now },
 
-                 { "Video3DFormat", Video3DFormat.HalfSideBySide, Video3DFormat.FullSideBySide }
 
-             };
 
-         [Theory]
 
-         [MemberData(nameof(MergeBaseItemData_SimpleField_ReplacesAppropriately_TestData))]
 
-         public void MergeBaseItemData_SimpleField_ReplacesAppropriately(string propName, object oldValue, object newValue)
 
-         {
 
-             // Use type Movie to allow testing of Video3DFormat
 
-             Assert.False(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, newValue, null, true, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, null, newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, null, null, true, out _));
 
-         }
 
-         [Fact]
 
-         public void MergeBaseItemData_MergeTrailers_ReplacesAppropriately()
 
-         {
 
-             string propName = "RemoteTrailers";
 
-             var oldValue = new[]
 
-             {
 
-                 new MediaUrl
 
-                 {
 
-                     Name = "Name 1",
 
-                     Url = "URL 1"
 
-                 }
 
-             };
 
-             var newValue = new[]
 
-             {
 
-                 new MediaUrl
 
-                 {
 
-                     Name = "Name 2",
 
-                     Url = "URL 2"
 
-                 }
 
-             };
 
-             Assert.False(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, newValue, null, true, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, Array.Empty<MediaUrl>(), newValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemData<Movie, MovieInfo>(propName, oldValue, Array.Empty<MediaUrl>(), null, true, out _));
 
-         }
 
-         [Fact]
 
-         public void MergeBaseItemData_ProviderIds_MergesAppropriately()
 
-         {
 
-             var propName = "ProviderIds";
 
-             var oldValue = new Dictionary<string, string>
 
-             {
 
-                 { "provider 1", "id 1" }
 
-             };
 
-             // overwrite provider id
 
-             var overwriteNewValue = new Dictionary<string, string>
 
-             {
 
-                 { "provider 1", "id 2" }
 
-             };
 
-             Assert.False(TestMergeBaseItemData<Movie, MovieInfo>(propName, new Dictionary<string, string>(oldValue), overwriteNewValue, null, false, out _));
 
-             TestMergeBaseItemData<Movie, MovieInfo>(propName, new Dictionary<string, string>(oldValue), overwriteNewValue, null, true, out var overwritten);
 
-             Assert.Equal(overwriteNewValue, overwritten);
 
-             // merge without overwriting
 
-             var mergeNewValue = new Dictionary<string, string>
 
-             {
 
-                 { "provider 1", "id 2" },
 
-                 { "provider 2", "id 3" }
 
-             };
 
-             TestMergeBaseItemData<Movie, MovieInfo>(propName, new Dictionary<string, string>(oldValue), mergeNewValue, null, false, out var merged);
 
-             var actual = (Dictionary<string, string>)merged!;
 
-             Assert.Equal("id 1", actual["provider 1"]);
 
-             Assert.Equal("id 3", actual["provider 2"]);
 
-             // empty source results in no change
 
-             TestMergeBaseItemData<Movie, MovieInfo>(propName, new Dictionary<string, string>(oldValue), new Dictionary<string, string>(), null, true, out var notOverwritten);
 
-             Assert.Equal(oldValue, notOverwritten);
 
-         }
 
-         [Fact]
 
-         public void MergeBaseItemData_MergePeople_MergesAppropriately()
 
-         {
 
-             // PersonInfo in list is changed by merge, create new for every call
 
-             List<PersonInfo> GetOldValue()
 
-                 => new()
 
-                 {
 
-                     new PersonInfo
 
-                     {
 
-                         Name = "Name 1",
 
-                         ProviderIds = new Dictionary<string, string>
 
-                         {
 
-                             { "Provider 1", "1234" }
 
-                         }
 
-                     }
 
-                 };
 
-             // overwrite provider id
 
-             var overwriteNewValue = new List<PersonInfo>
 
-             {
 
-                 new()
 
-                 {
 
-                     Name = "Name 2"
 
-                 }
 
-             };
 
-             Assert.False(TestMergeBaseItemDataPerson(GetOldValue(), overwriteNewValue, null, false, out var result));
 
-             // People not already in target are not merged into it from source
 
-             List<PersonInfo> actual = (List<PersonInfo>)result!;
 
-             Assert.Single(actual);
 
-             Assert.Equal("Name 1", actual[0].Name);
 
-             Assert.True(TestMergeBaseItemDataPerson(GetOldValue(), overwriteNewValue, null, true, out _));
 
-             Assert.True(TestMergeBaseItemDataPerson(new List<PersonInfo>(), overwriteNewValue, null, false, out _));
 
-             Assert.True(TestMergeBaseItemDataPerson(null, overwriteNewValue, null, false, out _));
 
-             Assert.False(TestMergeBaseItemDataPerson(GetOldValue(), overwriteNewValue, MetadataField.Cast, true, out _));
 
-             // providers merge but don't overwrite existing keys
 
-             var mergeNewValue = new List<PersonInfo>
 
-             {
 
-                 new()
 
-                 {
 
-                     Name = "Name 1",
 
-                     ProviderIds = new Dictionary<string, string>
 
-                     {
 
-                         { "Provider 1", "5678" },
 
-                         { "Provider 2", "5678" }
 
-                     }
 
-                 }
 
-             };
 
-             TestMergeBaseItemDataPerson(GetOldValue(), mergeNewValue, null, false, out result);
 
-             actual = (List<PersonInfo>)result!;
 
-             Assert.Single(actual);
 
-             Assert.Equal("Name 1", actual[0].Name);
 
-             Assert.Equal(2, actual[0].ProviderIds.Count);
 
-             Assert.Equal("1234", actual[0].ProviderIds["Provider 1"]);
 
-             Assert.Equal("5678", actual[0].ProviderIds["Provider 2"]);
 
-             // picture adds if missing but won't overwrite (forcing overwrites entire list, not entries in merged PersonInfo)
 
-             var mergePicture1 = new List<PersonInfo>
 
-             {
 
-                 new()
 
-                 {
 
-                     Name = "Name 1",
 
-                     ImageUrl = "URL 1"
 
-                 }
 
-             };
 
-             TestMergeBaseItemDataPerson(GetOldValue(), mergePicture1, null, false, out result);
 
-             actual = (List<PersonInfo>)result!;
 
-             Assert.Single(actual);
 
-             Assert.Equal("Name 1", actual[0].Name);
 
-             Assert.Equal("URL 1", actual[0].ImageUrl);
 
-             var mergePicture2 = new List<PersonInfo>
 
-             {
 
-                 new()
 
-                 {
 
-                     Name = "Name 1",
 
-                     ImageUrl = "URL 2"
 
-                 }
 
-             };
 
-             TestMergeBaseItemDataPerson(mergePicture1, mergePicture2, null, false, out result);
 
-             actual = (List<PersonInfo>)result!;
 
-             Assert.Single(actual);
 
-             Assert.Equal("Name 1", actual[0].Name);
 
-             Assert.Equal("URL 1", actual[0].ImageUrl);
 
-             // empty source can be forced to overwrite a target with data
 
-             Assert.True(TestMergeBaseItemDataPerson(GetOldValue(), new List<PersonInfo>(), null, true, out _));
 
-         }
 
-         private static bool TestMergeBaseItemDataPerson(List<PersonInfo>? oldValue, List<PersonInfo>? newValue, MetadataField? lockField, bool replaceData, out object? actualValue)
 
-         {
 
-             var source = new MetadataResult<Movie>
 
-             {
 
-                 Item = new Movie(),
 
-                 People = newValue
 
-             };
 
-             var target = new MetadataResult<Movie>
 
-             {
 
-                 Item = new Movie(),
 
-                 People = oldValue
 
-             };
 
-             var lockedFields = lockField is null ? Array.Empty<MetadataField>() : new[] { (MetadataField)lockField };
 
-             MetadataService<Movie, MovieInfo>.MergeBaseItemData(source, target, lockedFields, replaceData, false);
 
-             actualValue = target.People;
 
-             return newValue?.Equals(actualValue) ?? actualValue is null;
 
-         }
 
-         /// <summary>
 
-         /// Makes a call to <see cref="MetadataService{TItemType,TIdType}.MergeBaseItemData"/> with the provided parameters and returns whether the target changed or not.
 
-         ///
 
-         /// Reflection is used to allow testing of all fields using the same logic, rather than relying on copy/pasting test code for each field.
 
-         /// </summary>
 
-         /// <param name="propName">The property to test.</param>
 
-         /// <param name="oldValue">The initial value in the target object.</param>
 
-         /// <param name="newValue">The initial value in the source object.</param>
 
-         /// <param name="lockField">The metadata field that locks this property if the field should be locked, or <c>null</c> to leave unlocked.</param>
 
-         /// <param name="replaceData">Passed through to <see cref="MetadataService{TItemType,TIdType}.MergeBaseItemData"/>.</param>
 
-         /// <param name="actualValue">The resulting value set to the target.</param>
 
-         /// <typeparam name="TItemType">The <see cref="BaseItem"/> type to test on.</typeparam>
 
-         /// <typeparam name="TIdType">The <see cref="BaseItem"/> info type.</typeparam>
 
-         /// <returns><c>true</c> if the property on the target updates to match the source value when<see cref="MetadataService{TItemType,TIdType}.MergeBaseItemData"/> is called.</returns>
 
-         private static bool TestMergeBaseItemData<TItemType, TIdType>(string propName, object? oldValue, object? newValue, MetadataField? lockField, bool replaceData, out object? actualValue)
 
-             where TItemType : BaseItem, IHasLookupInfo<TIdType>, new()
 
-             where TIdType : ItemLookupInfo, new()
 
-         {
 
-             var property = typeof(TItemType).GetProperty(propName)!;
 
-             var source = new MetadataResult<TItemType>
 
-             {
 
-                 Item = new TItemType()
 
-             };
 
-             property.SetValue(source.Item, newValue);
 
-             var target = new MetadataResult<TItemType>
 
-             {
 
-                 Item = new TItemType()
 
-             };
 
-             property.SetValue(target.Item, oldValue);
 
-             var lockedFields = lockField is null ? Array.Empty<MetadataField>() : new[] { (MetadataField)lockField };
 
-             // generic type doesn't actually matter to call the static method, just has to be filled in
 
-             MetadataService<TItemType, TIdType>.MergeBaseItemData(source, target, lockedFields, replaceData, false);
 
-             actualValue = property.GetValue(target.Item);
 
-             return newValue?.Equals(actualValue) ?? actualValue is null;
 
-         }
 
-     }
 
- }
 
 
  |