mirror of
				https://github.com/jellyfin/jellyfin.git
				synced 2025-10-25 07:49:17 -04:00 
			
		
		
		
	de-normalize item by name data. create counts during library scan for fast access.
This commit is contained in:
		
							parent
							
								
									d078edfb96
								
							
						
					
					
						commit
						740a10a4e3
					
				| @ -22,28 +22,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|     { |     { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |  | ||||||
|     /// Class GetArtistsItemCounts |  | ||||||
|     /// </summary> |  | ||||||
|     [Route("/Artists/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that an artist appears in")] |  | ||||||
|     public class GetArtistsItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public Guid UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The artist name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [Route("/Artists/{Name}", "GET")] |     [Route("/Artists/{Name}", "GET")] | ||||||
|     [Api(Description = "Gets an artist, by name")] |     [Api(Description = "Gets an artist, by name")] | ||||||
|     public class GetArtist : IReturn<BaseItemDto> |     public class GetArtist : IReturn<BaseItemDto> | ||||||
| @ -114,49 +92,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return await DtoService.GetBaseItemDto(item, fields.ToList()).ConfigureAwait(false); |             return await DtoService.GetBaseItemDto(item, fields.ToList()).ConfigureAwait(false); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetArtistsItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugArtistName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => |  | ||||||
|             { |  | ||||||
|                 var song = i as Audio; |  | ||||||
| 
 |  | ||||||
|                 if (song != null) |  | ||||||
|                 { |  | ||||||
|                     return song.HasArtist(name); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 var musicVideo = i as MusicVideo; |  | ||||||
| 
 |  | ||||||
|                 if (musicVideo != null) |  | ||||||
|                 { |  | ||||||
|                     return musicVideo.HasArtist(name); |  | ||||||
|                 } |  | ||||||
|                  |  | ||||||
|                 return false; |  | ||||||
| 
 |  | ||||||
|             }).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 SongCount = items.OfType<Audio>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AlbumCount = items.Select(i => i.Parent).OfType<MusicAlbum>().Distinct().Count(), |  | ||||||
| 
 |  | ||||||
|                 MusicVideoCount = items.OfType<MusicVideo>().Count(i => i.HasArtist(name)) |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the specified request. |         /// Gets the specified request. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -193,7 +128,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|                     return list; |                     return list; | ||||||
|                 }) |                 }) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|                 .Select(name => new IbnStub<Artist>(name, () => itemsList.Where(i => i.HasArtist(name)), GetEntity)); |                 .Select(name => new IbnStub<Artist>(name, GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <typeparam name="TItemType">The type of the T item type.</typeparam> |     /// <typeparam name="TItemType">The type of the T item type.</typeparam> | ||||||
|     public abstract class BaseItemsByNameService<TItemType> : BaseApiService |     public abstract class BaseItemsByNameService<TItemType> : BaseApiService | ||||||
|         where TItemType : BaseItem |         where TItemType : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// The _user manager |         /// The _user manager | ||||||
| @ -38,6 +38,8 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         /// <param name="userManager">The user manager.</param> |         /// <param name="userManager">The user manager.</param> | ||||||
|         /// <param name="libraryManager">The library manager.</param> |         /// <param name="libraryManager">The library manager.</param> | ||||||
|         /// <param name="userDataRepository">The user data repository.</param> |         /// <param name="userDataRepository">The user data repository.</param> | ||||||
|  |         /// <param name="itemRepository">The item repository.</param> | ||||||
|  |         /// <param name="dtoService">The dto service.</param> | ||||||
|         protected BaseItemsByNameService(IUserManager userManager, ILibraryManager libraryManager, IUserDataRepository userDataRepository, IItemRepository itemRepository, IDtoService dtoService) |         protected BaseItemsByNameService(IUserManager userManager, ILibraryManager libraryManager, IUserDataRepository userDataRepository, IItemRepository itemRepository, IDtoService dtoService) | ||||||
|         { |         { | ||||||
|             UserManager = userManager; |             UserManager = userManager; | ||||||
| @ -292,7 +294,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         /// <returns>Task{DtoBaseItem}.</returns> |         /// <returns>Task{DtoBaseItem}.</returns> | ||||||
|         private async Task<BaseItemDto> GetDto(IbnStub<TItemType> stub, User user, List<ItemFields> fields) |         private async Task<BaseItemDto> GetDto(IbnStub<TItemType> stub, User user, List<ItemFields> fields) | ||||||
|         { |         { | ||||||
|             BaseItem item; |             TItemType item; | ||||||
| 
 | 
 | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
| @ -307,14 +309,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             var dto = user == null ? await DtoService.GetBaseItemDto(item, fields).ConfigureAwait(false) : |             var dto = user == null ? await DtoService.GetBaseItemDto(item, fields).ConfigureAwait(false) : | ||||||
|                 await DtoService.GetBaseItemDto(item, fields, user).ConfigureAwait(false); |                 await DtoService.GetBaseItemDto(item, fields, user).ConfigureAwait(false); | ||||||
| 
 | 
 | ||||||
|             if (fields.Contains(ItemFields.ItemCounts)) |  | ||||||
|             { |  | ||||||
|                 var items = stub.Items; |  | ||||||
| 
 |  | ||||||
|                 dto.ChildCount = items.Count; |  | ||||||
|                 dto.RecentlyAddedItemCount = items.Count(i => i.IsRecentlyAdded()); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return dto; |             return dto; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @ -367,9 +361,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|     public class IbnStub<T> |     public class IbnStub<T> | ||||||
|         where T : BaseItem |         where T : BaseItem | ||||||
|     { |     { | ||||||
|         private readonly Func<IEnumerable<BaseItem>> _childItemsFunction; |  | ||||||
|         private List<BaseItem> _childItems; |  | ||||||
| 
 |  | ||||||
|         private readonly Func<string,Task<T>> _itemFunction; |         private readonly Func<string,Task<T>> _itemFunction; | ||||||
|         private Task<T> _itemTask; |         private Task<T> _itemTask; | ||||||
|          |          | ||||||
| @ -377,11 +368,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
| 
 | 
 | ||||||
|         private UserItemData _userData; |         private UserItemData _userData; | ||||||
| 
 | 
 | ||||||
|         public List<BaseItem> Items |  | ||||||
|         { |  | ||||||
|             get { return _childItems ?? (_childItems = _childItemsFunction().ToList()); } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public Task<T> GetItem() |         public Task<T> GetItem() | ||||||
|         { |         { | ||||||
|             return _itemTask ?? (_itemTask = _itemFunction(Name)); |             return _itemTask ?? (_itemTask = _itemFunction(Name)); | ||||||
| @ -394,10 +380,9 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return _userData ?? (_userData = repo.GetUserData(userId, item.GetUserDataKey())); |             return _userData ?? (_userData = repo.GetUserData(userId, item.GetUserDataKey())); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public IbnStub(string name, Func<IEnumerable<BaseItem>> childItems, Func<string,Task<T>> item) |         public IbnStub(string name, Func<string,Task<T>> item) | ||||||
|         { |         { | ||||||
|             Name = name; |             Name = name; | ||||||
|             _childItemsFunction = childItems; |  | ||||||
|             _itemFunction = item; |             _itemFunction = item; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -23,25 +23,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Route("/GameGenres/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that a genre appears in")] |  | ||||||
|     public class GetGameGenreItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")] |  | ||||||
|         public Guid? UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The genre name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [Route("/GameGenres/{Name}", "GET")] |     [Route("/GameGenres/{Name}", "GET")] | ||||||
|     [Api(Description = "Gets a Game genre, by name")] |     [Api(Description = "Gets a Game genre, by name")] | ||||||
|     public class GetGameGenre : IReturn<BaseItemDto> |     public class GetGameGenre : IReturn<BaseItemDto> | ||||||
| @ -127,7 +108,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return itemsList |             return itemsList | ||||||
|                 .SelectMany(i => i.Genres) |                 .SelectMany(i => i.Genres) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|                 .Select(name => new IbnStub<GameGenre>(name, () => itemsList.Where(i => i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)), GetEntity)); |                 .Select(name => new IbnStub<GameGenre>(name, GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -139,26 +120,5 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         { |         { | ||||||
|             return LibraryManager.GetGameGenre(name); |             return LibraryManager.GetGameGenre(name); | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetGameGenreItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugGameGenreName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => i.Genres != null && i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 GameCount = items.OfType<Game>().Count() |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,7 +1,5 @@ | |||||||
| using MediaBrowser.Controller.Dto; | using MediaBrowser.Controller.Dto; | ||||||
| using MediaBrowser.Controller.Entities; | using MediaBrowser.Controller.Entities; | ||||||
| using MediaBrowser.Controller.Entities.Movies; |  | ||||||
| using MediaBrowser.Controller.Entities.TV; |  | ||||||
| using MediaBrowser.Controller.Library; | using MediaBrowser.Controller.Library; | ||||||
| using MediaBrowser.Controller.Persistence; | using MediaBrowser.Controller.Persistence; | ||||||
| using MediaBrowser.Model.Dto; | using MediaBrowser.Model.Dto; | ||||||
| @ -23,25 +21,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|     { |     { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Route("/Genres/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that a genre appears in")] |  | ||||||
|     public class GetGenreItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")] |  | ||||||
|         public Guid? UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The genre name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class GetGenre |     /// Class GetGenre | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @ -133,7 +112,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return itemsList |             return itemsList | ||||||
|                 .SelectMany(i => i.Genres) |                 .SelectMany(i => i.Genres) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|                 .Select(name => new IbnStub<Genre>(name, () => itemsList.Where(i => i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)), GetEntity)); |                 .Select(name => new IbnStub<Genre>(name, GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -145,34 +124,5 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         { |         { | ||||||
|             return LibraryManager.GetGenre(name); |             return LibraryManager.GetGenre(name); | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetGenreItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugGenreName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 TrailerCount = items.OfType<Trailer>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MovieCount = items.OfType<Movie>().Count(), |  | ||||||
| 
 |  | ||||||
|                 SeriesCount = items.OfType<Series>().Count(), |  | ||||||
| 
 |  | ||||||
|                 GameCount = items.OfType<Game>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AdultVideoCount = items.OfType<AdultVideo>().Count() |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -427,28 +427,9 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
| 
 | 
 | ||||||
|                 items = items.Where(i => |                 items = items.Where(i => | ||||||
|                 { |                 { | ||||||
|                     var audio = i as Audio; |                     var audio = i as IHasArtist; | ||||||
| 
 | 
 | ||||||
|                     if (audio != null) |                     return audio != null && artists.Any(audio.HasArtist); | ||||||
|                     { |  | ||||||
|                         return artists.Any(audio.HasArtist); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     var album = i as MusicAlbum; |  | ||||||
| 
 |  | ||||||
|                     if (album != null) |  | ||||||
|                     { |  | ||||||
|                         return artists.Any(album.HasArtist); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     var musicVideo = i as MusicVideo; |  | ||||||
| 
 |  | ||||||
|                     if (musicVideo != null) |  | ||||||
|                     { |  | ||||||
|                         return artists.Any(musicVideo.HasArtist); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     return false; |  | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -23,25 +23,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Route("/MusicGenres/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that a genre appears in")] |  | ||||||
|     public class GetMusicGenreItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")] |  | ||||||
|         public Guid? UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The genre name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [Route("/MusicGenres/{Name}", "GET")] |     [Route("/MusicGenres/{Name}", "GET")] | ||||||
|     [Api(Description = "Gets a music genre, by name")] |     [Api(Description = "Gets a music genre, by name")] | ||||||
|     public class GetMusicGenre : IReturn<BaseItemDto> |     public class GetMusicGenre : IReturn<BaseItemDto> | ||||||
| @ -127,7 +108,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return itemsList |             return itemsList | ||||||
|                 .SelectMany(i => i.Genres) |                 .SelectMany(i => i.Genres) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|                 .Select(name => new IbnStub<MusicGenre>(name, () => itemsList.Where(i => i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)), GetEntity)); |                 .Select(name => new IbnStub<MusicGenre>(name, GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -139,30 +120,5 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         { |         { | ||||||
|             return LibraryManager.GetMusicGenre(name); |             return LibraryManager.GetMusicGenre(name); | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetMusicGenreItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugGenreName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => i.Genres != null && i.Genres.Contains(name, StringComparer.OrdinalIgnoreCase)).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 SongCount = items.OfType<Audio>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AlbumCount = items.OfType<MusicAlbum>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MusicVideoCount = items.OfType<MusicVideo>().Count() |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,8 +1,5 @@ | |||||||
| using MediaBrowser.Controller.Dto; | using MediaBrowser.Controller.Dto; | ||||||
| using MediaBrowser.Controller.Entities; | using MediaBrowser.Controller.Entities; | ||||||
| using MediaBrowser.Controller.Entities.Audio; |  | ||||||
| using MediaBrowser.Controller.Entities.Movies; |  | ||||||
| using MediaBrowser.Controller.Entities.TV; |  | ||||||
| using MediaBrowser.Controller.Library; | using MediaBrowser.Controller.Library; | ||||||
| using MediaBrowser.Controller.Persistence; | using MediaBrowser.Controller.Persistence; | ||||||
| using MediaBrowser.Model.Dto; | using MediaBrowser.Model.Dto; | ||||||
| @ -29,28 +26,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|         public string PersonTypes { get; set; } |         public string PersonTypes { get; set; } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |  | ||||||
|     /// Class GetPersonItemCounts |  | ||||||
|     /// </summary> |  | ||||||
|     [Route("/Persons/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that a person appears in")] |  | ||||||
|     public class GetPersonItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public Guid UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The person name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class GetPerson |     /// Class GetPerson | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @ -136,43 +111,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return ToOptimizedResult(result); |             return ToOptimizedResult(result); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetPersonItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugPersonName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => i.People != null && i.People.Any(p => string.Equals(p.Name, name, StringComparison.OrdinalIgnoreCase))).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 TrailerCount = items.OfType<Trailer>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MovieCount = items.OfType<Movie>().Count(), |  | ||||||
| 
 |  | ||||||
|                 SeriesCount = items.OfType<Series>().Count(), |  | ||||||
| 
 |  | ||||||
|                 GameCount = items.OfType<Game>().Count(), |  | ||||||
| 
 |  | ||||||
|                 SongCount = items.OfType<Audio>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AlbumCount = items.OfType<MusicAlbum>().Count(), |  | ||||||
| 
 |  | ||||||
|                 EpisodeCount = items.OfType<Episode>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MusicVideoCount = items.OfType<MusicVideo>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AdultVideoCount = items.OfType<AdultVideo>().Count() |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets all items. |         /// Gets all items. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -193,15 +131,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|                 .Select(i => i.Name) |                 .Select(i => i.Name) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
| 
 | 
 | ||||||
|                 .Select(name => new IbnStub<Person>(name, () => |                 .Select(name => new IbnStub<Person>(name, GetEntity) | ||||||
|                 { |  | ||||||
|                     if (personTypes.Length == 0) |  | ||||||
|                     { |  | ||||||
|                         return itemsList.Where(i => i.People.Any(p => p.Name.Equals(name, StringComparison.OrdinalIgnoreCase))); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     return itemsList.Where(i => i.People.Any(p => p.Name.Equals(name, StringComparison.OrdinalIgnoreCase) && (personTypes.Contains(p.Type ?? string.Empty, StringComparer.OrdinalIgnoreCase) || personTypes.Contains(p.Role ?? string.Empty, StringComparer.OrdinalIgnoreCase)))); |  | ||||||
|                 }, GetEntity) |  | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,8 +1,5 @@ | |||||||
| using MediaBrowser.Controller.Dto; | using MediaBrowser.Controller.Dto; | ||||||
| using MediaBrowser.Controller.Entities; | using MediaBrowser.Controller.Entities; | ||||||
| using MediaBrowser.Controller.Entities.Audio; |  | ||||||
| using MediaBrowser.Controller.Entities.Movies; |  | ||||||
| using MediaBrowser.Controller.Entities.TV; |  | ||||||
| using MediaBrowser.Controller.Library; | using MediaBrowser.Controller.Library; | ||||||
| using MediaBrowser.Controller.Persistence; | using MediaBrowser.Controller.Persistence; | ||||||
| using MediaBrowser.Model.Dto; | using MediaBrowser.Model.Dto; | ||||||
| @ -24,25 +21,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|     { |     { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Route("/Studios/{Name}/Counts", "GET")] |  | ||||||
|     [Api(Description = "Gets item counts of library items that a studio appears in")] |  | ||||||
|     public class GetStudioItemCounts : IReturn<ItemByNameCounts> |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the user id. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The user id.</value> |  | ||||||
|         [ApiMember(Name = "UserId", Description = "User Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public Guid UserId { get; set; } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets or sets the name. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         [ApiMember(Name = "Name", Description = "The studio name", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")] |  | ||||||
|         public string Name { get; set; } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class GetStudio |     /// Class GetStudio | ||||||
|     /// </summary> |     /// </summary> | ||||||
| @ -109,41 +87,6 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return await DtoService.GetBaseItemDto(item, fields.ToList()).ConfigureAwait(false); |             return await DtoService.GetBaseItemDto(item, fields.ToList()).ConfigureAwait(false); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the specified request. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="request">The request.</param> |  | ||||||
|         /// <returns>System.Object.</returns> |  | ||||||
|         public object Get(GetStudioItemCounts request) |  | ||||||
|         { |  | ||||||
|             var name = DeSlugStudioName(request.Name, LibraryManager); |  | ||||||
| 
 |  | ||||||
|             var items = GetItems(request.UserId).Where(i => i.Studios != null && i.Studios.Contains(name, StringComparer.OrdinalIgnoreCase)).ToList(); |  | ||||||
| 
 |  | ||||||
|             var counts = new ItemByNameCounts |  | ||||||
|             { |  | ||||||
|                 TotalCount = items.Count, |  | ||||||
| 
 |  | ||||||
|                 TrailerCount = items.OfType<Trailer>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MovieCount = items.OfType<Movie>().Count(), |  | ||||||
| 
 |  | ||||||
|                 SeriesCount = items.OfType<Series>().Count(), |  | ||||||
| 
 |  | ||||||
|                 GameCount = items.OfType<Game>().Count(), |  | ||||||
| 
 |  | ||||||
|                 SongCount = items.OfType<Audio>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AlbumCount = items.OfType<MusicAlbum>().Count(), |  | ||||||
| 
 |  | ||||||
|                 MusicVideoCount = items.OfType<MusicVideo>().Count(), |  | ||||||
| 
 |  | ||||||
|                 AdultVideoCount = items.OfType<AdultVideo>().Count() |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             return ToOptimizedResult(counts); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the specified request. |         /// Gets the specified request. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -169,7 +112,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return itemsList |             return itemsList | ||||||
|                 .SelectMany(i => i.Studios) |                 .SelectMany(i => i.Studios) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|                 .Select(name => new IbnStub<Studio>(name, () => itemsList.Where(i => i.Studios.Contains(name, StringComparer.OrdinalIgnoreCase)), GetEntity)); |                 .Select(name => new IbnStub<Studio>(name, GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|  | |||||||
| @ -118,7 +118,7 @@ namespace MediaBrowser.Api.UserLibrary | |||||||
|             return itemsList |             return itemsList | ||||||
|                 .Select(i => i.ProductionYear.Value) |                 .Select(i => i.ProductionYear.Value) | ||||||
|                 .Distinct() |                 .Distinct() | ||||||
|                 .Select(year => new IbnStub<Year>(year.ToString(UsCulture), () => itemsList.Where(i => i.ProductionYear.HasValue && i.ProductionYear.Value == year), GetEntity)); |                 .Select(year => new IbnStub<Year>(year.ToString(UsCulture), GetEntity)); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|  | |||||||
| @ -1,11 +1,20 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities.Audio | namespace MediaBrowser.Controller.Entities.Audio | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class Artist |     /// Class Artist | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Artist : BaseItem, IItemByName |     public class Artist : BaseItem, IItemByName, IHasMusicGenres | ||||||
|     { |     { | ||||||
|  |         public Artist() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         public string LastFmImageUrl { get; set; } |         public string LastFmImageUrl { get; set; } | ||||||
|          |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -17,5 +26,8 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|             return "Artist-" + Name; |             return "Artist-" + Name; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class Audio |     /// Class Audio | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Audio : BaseItem, IHasMediaStreams, IHasAlbumArtist |     public class Audio : BaseItem, IHasMediaStreams, IHasAlbumArtist, IHasArtist, IHasMusicGenres | ||||||
|     { |     { | ||||||
|         public Audio() |         public Audio() | ||||||
|         { |         { | ||||||
|  | |||||||
| @ -5,4 +5,9 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|     { |     { | ||||||
|         string AlbumArtist { get; } |         string AlbumArtist { get; } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public interface IHasArtist | ||||||
|  |     { | ||||||
|  |         bool HasArtist(string name); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -0,0 +1,9 @@ | |||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Controller.Entities.Audio | ||||||
|  | { | ||||||
|  |     public interface IHasMusicGenres | ||||||
|  |     { | ||||||
|  |         List<string> Genres { get; } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -1,4 +1,5 @@ | |||||||
| using System.Linq; | using System; | ||||||
|  | using System.Linq; | ||||||
| using System.Runtime.Serialization; | using System.Runtime.Serialization; | ||||||
| 
 | 
 | ||||||
| namespace MediaBrowser.Controller.Entities.Audio | namespace MediaBrowser.Controller.Entities.Audio | ||||||
| @ -6,8 +7,13 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Class MusicAlbum |     /// Class MusicAlbum | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class MusicAlbum : Folder, IHasAlbumArtist |     public class MusicAlbum : Folder, IHasAlbumArtist, IHasArtist, IHasMusicGenres | ||||||
|     { |     { | ||||||
|  |         public MusicAlbum() | ||||||
|  |         { | ||||||
|  |             Artists = new string[] { }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         public string LastFmImageUrl { get; set; } |         public string LastFmImageUrl { get; set; } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -60,19 +66,13 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|         /// <returns><c>true</c> if the specified artist has artist; otherwise, <c>false</c>.</returns> |         /// <returns><c>true</c> if the specified artist has artist; otherwise, <c>false</c>.</returns> | ||||||
|         public bool HasArtist(string artist) |         public bool HasArtist(string artist) | ||||||
|         { |         { | ||||||
|             return RecursiveChildren.OfType<Audio>().Any(i => i.HasArtist(artist)); |             return string.Equals(AlbumArtist, artist, StringComparison.OrdinalIgnoreCase) | ||||||
|  |                    || Artists.Contains(artist, StringComparer.OrdinalIgnoreCase); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public string AlbumArtist |         public string AlbumArtist { get; set; } | ||||||
|         { | 
 | ||||||
|             get |         public string[] Artists { get; set; } | ||||||
|             { |  | ||||||
|                 return RecursiveChildren |  | ||||||
|                   .OfType<Audio>() |  | ||||||
|                   .Select(i => i.AlbumArtist) |  | ||||||
|                   .FirstOrDefault(i => !string.IsNullOrEmpty(i)); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public class MusicAlbumDisc : Folder |     public class MusicAlbumDisc : Folder | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities.Audio | namespace MediaBrowser.Controller.Entities.Audio | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,6 +9,12 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class MusicGenre : BaseItem, IItemByName |     public class MusicGenre : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public MusicGenre() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -14,5 +23,9 @@ namespace MediaBrowser.Controller.Entities.Audio | |||||||
|         { |         { | ||||||
|             return "MusicGenre-" + Name; |             return "MusicGenre-" + Name; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1537,5 +1537,58 @@ namespace MediaBrowser.Controller.Entities | |||||||
|             // Refresh metadata |             // Refresh metadata | ||||||
|             return RefreshMetadata(CancellationToken.None, forceSave: true); |             return RefreshMetadata(CancellationToken.None, forceSave: true); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates that images within the item are still on the file system | ||||||
|  |         /// </summary> | ||||||
|  |         public void ValidateImages() | ||||||
|  |         { | ||||||
|  |             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below | ||||||
|  |             var deletedKeys = Images | ||||||
|  |                 .ToList() | ||||||
|  |                 .Where(image => !File.Exists(image.Value)) | ||||||
|  |                 .Select(i => i.Key) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             // Now remove them from the dictionary | ||||||
|  |             foreach (var key in deletedKeys) | ||||||
|  |             { | ||||||
|  |                 Images.Remove(key); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates that backdrops within the item are still on the file system | ||||||
|  |         /// </summary> | ||||||
|  |         public void ValidateBackdrops() | ||||||
|  |         { | ||||||
|  |             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below | ||||||
|  |             var deletedImages = BackdropImagePaths | ||||||
|  |                 .Where(path => !File.Exists(path)) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             // Now remove them from the dictionary | ||||||
|  |             foreach (var path in deletedImages) | ||||||
|  |             { | ||||||
|  |                 BackdropImagePaths.Remove(path); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the screenshots. | ||||||
|  |         /// </summary> | ||||||
|  |         public void ValidateScreenshots() | ||||||
|  |         { | ||||||
|  |             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below | ||||||
|  |             var deletedImages = ScreenshotImagePaths | ||||||
|  |                 .Where(path => !File.Exists(path)) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             // Now remove them from the dictionary | ||||||
|  |             foreach (var path in deletedImages) | ||||||
|  |             { | ||||||
|  |                 ScreenshotImagePaths.Remove(path); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,8 +1,17 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     public class GameGenre : BaseItem, IItemByName |     public class GameGenre : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public GameGenre() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -11,5 +20,9 @@ namespace MediaBrowser.Controller.Entities | |||||||
|         { |         { | ||||||
|             return "GameGenre-" + Name; |             return "GameGenre-" + Name; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,6 +9,12 @@ namespace MediaBrowser.Controller.Entities | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Genre : BaseItem, IItemByName |     public class Genre : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public Genre() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -14,5 +23,9 @@ namespace MediaBrowser.Controller.Entities | |||||||
|         { |         { | ||||||
|             return "Genre-" + Name; |             return "Genre-" + Name; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,5 +9,8 @@ namespace MediaBrowser.Controller.Entities | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public interface IItemByName |     public interface IItemByName | ||||||
|     { |     { | ||||||
|  |         ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,9 +1,10 @@ | |||||||
| using MediaBrowser.Model.Entities; | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Model.Entities; | ||||||
| using System; | using System; | ||||||
| 
 | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     public class MusicVideo : Video |     public class MusicVideo : Video, IHasArtist, IHasMusicGenres | ||||||
|     { |     { | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets or sets the artist. |         /// Gets or sets the artist. | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,6 +9,16 @@ namespace MediaBrowser.Controller.Entities | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Person : BaseItem, IItemByName |     public class Person : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public Person() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,6 +9,12 @@ namespace MediaBrowser.Controller.Entities | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Studio : BaseItem, IItemByName |     public class Studio : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public Studio() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -14,5 +23,9 @@ namespace MediaBrowser.Controller.Entities | |||||||
|         { |         { | ||||||
|             return "Studio-" + Name; |             return "Studio-" + Name; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -136,6 +136,11 @@ namespace MediaBrowser.Controller.Entities | |||||||
|             get { return Video3DFormat.HasValue; } |             get { return Video3DFormat.HasValue; } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         public bool IsHd | ||||||
|  |         { | ||||||
|  |             get { return MediaStreams != null && MediaStreams.Any(i => i.Type == MediaStreamType.Video && i.Width.HasValue && i.Width.Value >= 1280); } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the type of the media. |         /// Gets the type of the media. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -1,4 +1,7 @@ | |||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
| namespace MediaBrowser.Controller.Entities | namespace MediaBrowser.Controller.Entities | ||||||
| { | { | ||||||
|     /// <summary> |     /// <summary> | ||||||
| @ -6,6 +9,16 @@ namespace MediaBrowser.Controller.Entities | |||||||
|     /// </summary> |     /// </summary> | ||||||
|     public class Year : BaseItem, IItemByName |     public class Year : BaseItem, IItemByName | ||||||
|     { |     { | ||||||
|  |         public Year() | ||||||
|  |         { | ||||||
|  |             ItemCounts = new ItemByNameCounts(); | ||||||
|  |             UserItemCounts = new Dictionary<Guid, ItemByNameCounts>(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public ItemByNameCounts ItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Dictionary<Guid, ItemByNameCounts> UserItemCounts { get; set; } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the user data key. |         /// Gets the user data key. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -235,6 +235,38 @@ namespace MediaBrowser.Controller.Library | |||||||
|         /// <returns>Task.</returns> |         /// <returns>Task.</returns> | ||||||
|         Task ValidateArtists(CancellationToken cancellationToken, IProgress<double> progress); |         Task ValidateArtists(CancellationToken cancellationToken, IProgress<double> progress); | ||||||
| 
 | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the music genres. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         Task ValidateMusicGenres(CancellationToken cancellationToken, IProgress<double> progress); | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the game genres. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         Task ValidateGameGenres(CancellationToken cancellationToken, IProgress<double> progress); | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the genres. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         Task ValidateGenres(CancellationToken cancellationToken, IProgress<double> progress); | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         /// Validates the studios. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         Task ValidateStudios(CancellationToken cancellationToken, IProgress<double> progress); | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Occurs when [item added]. |         /// Occurs when [item added]. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -74,6 +74,7 @@ | |||||||
|     <Compile Include="Dto\IDtoService.cs" /> |     <Compile Include="Dto\IDtoService.cs" /> | ||||||
|     <Compile Include="Entities\AdultVideo.cs" /> |     <Compile Include="Entities\AdultVideo.cs" /> | ||||||
|     <Compile Include="Entities\Audio\IHasAlbumArtist.cs" /> |     <Compile Include="Entities\Audio\IHasAlbumArtist.cs" /> | ||||||
|  |     <Compile Include="Entities\Audio\IHasMusicGenres.cs" /> | ||||||
|     <Compile Include="Entities\Book.cs" /> |     <Compile Include="Entities\Book.cs" /> | ||||||
|     <Compile Include="Configuration\IServerConfigurationManager.cs" /> |     <Compile Include="Configuration\IServerConfigurationManager.cs" /> | ||||||
|     <Compile Include="Entities\Audio\MusicGenre.cs" /> |     <Compile Include="Entities\Audio\MusicGenre.cs" /> | ||||||
| @ -90,6 +91,7 @@ | |||||||
|     <Compile Include="Localization\ILocalizationManager.cs" /> |     <Compile Include="Localization\ILocalizationManager.cs" /> | ||||||
|     <Compile Include="Notifications\INotificationsRepository.cs" /> |     <Compile Include="Notifications\INotificationsRepository.cs" /> | ||||||
|     <Compile Include="Notifications\NotificationUpdateEventArgs.cs" /> |     <Compile Include="Notifications\NotificationUpdateEventArgs.cs" /> | ||||||
|  |     <Compile Include="Providers\IDynamicInfoProvider.cs" /> | ||||||
|     <Compile Include="Session\ISessionManager.cs" /> |     <Compile Include="Session\ISessionManager.cs" /> | ||||||
|     <Compile Include="Drawing\ImageExtensions.cs" /> |     <Compile Include="Drawing\ImageExtensions.cs" /> | ||||||
|     <Compile Include="Drawing\ImageHeader.cs" /> |     <Compile Include="Drawing\ImageHeader.cs" /> | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								MediaBrowser.Controller/Providers/IDynamicInfoProvider.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								MediaBrowser.Controller/Providers/IDynamicInfoProvider.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  |  | ||||||
|  | namespace MediaBrowser.Controller.Providers | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Marker interface for a provider that always runs | ||||||
|  |     /// </summary> | ||||||
|  |     public interface IDynamicInfoProvider | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -28,6 +28,8 @@ namespace MediaBrowser.Model.ApiClient | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         event EventHandler<HttpResponseEventArgs> HttpResponseReceived; |         event EventHandler<HttpResponseEventArgs> HttpResponseReceived; | ||||||
| 
 | 
 | ||||||
|  |         Task<T> GetAsync<T>(string url, CancellationToken cancellationToken); | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the critic reviews. |         /// Gets the critic reviews. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -36,8 +36,6 @@ namespace MediaBrowser.Model.Dto | |||||||
|         /// <value>The name of the sort.</value> |         /// <value>The name of the sort.</value> | ||||||
|         public string SortName { get; set; } |         public string SortName { get; set; } | ||||||
| 
 | 
 | ||||||
|         public string MainFeaturePlaylistName { get; set; } |  | ||||||
|          |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets or sets the video3 D format. |         /// Gets or sets the video3 D format. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -521,6 +519,48 @@ namespace MediaBrowser.Model.Dto | |||||||
|         /// <value>The locked fields.</value> |         /// <value>The locked fields.</value> | ||||||
|         public List<MetadataFields> LockedFields { get; set; } |         public List<MetadataFields> LockedFields { get; set; } | ||||||
| 
 | 
 | ||||||
|  |         public int? AdultVideoCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the movie count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The movie count.</value> | ||||||
|  |         public int? MovieCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the series count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The series count.</value> | ||||||
|  |         public int? SeriesCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the episode count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The episode count.</value> | ||||||
|  |         public int? EpisodeCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the game count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The game count.</value> | ||||||
|  |         public int? GameCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the trailer count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The trailer count.</value> | ||||||
|  |         public int? TrailerCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the song count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The song count.</value> | ||||||
|  |         public int? SongCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the album count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The album count.</value> | ||||||
|  |         public int? AlbumCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the music video count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The music video count.</value> | ||||||
|  |         public int? MusicVideoCount { get; set; } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets or sets a value indicating whether [enable internet providers]. |         /// Gets or sets a value indicating whether [enable internet providers]. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -11,6 +11,10 @@ namespace MediaBrowser.Model.Dto | |||||||
|         /// </summary> |         /// </summary> | ||||||
|         /// <value>The total count.</value> |         /// <value>The total count.</value> | ||||||
|         public int TotalCount { get; set; } |         public int TotalCount { get; set; } | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets or sets the adult video count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The adult video count.</value> | ||||||
|         public int AdultVideoCount { get; set; } |         public int AdultVideoCount { get; set; } | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets or sets the movie count. |         /// Gets or sets the movie count. | ||||||
|  | |||||||
| @ -98,13 +98,13 @@ namespace MediaBrowser.Providers | |||||||
|             var args = GetResolveArgsContainingImages(item); |             var args = GetResolveArgsContainingImages(item); | ||||||
| 
 | 
 | ||||||
|             // Make sure current image paths still exist |             // Make sure current image paths still exist | ||||||
|             ValidateImages(item); |             item.ValidateImages(); | ||||||
| 
 | 
 | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |             cancellationToken.ThrowIfCancellationRequested(); | ||||||
| 
 | 
 | ||||||
|             // Make sure current backdrop paths still exist |             // Make sure current backdrop paths still exist | ||||||
|             ValidateBackdrops(item); |             item.ValidateBackdrops(); | ||||||
|             ValidateScreenshots(item, args); |             item.ValidateScreenshots(); | ||||||
| 
 | 
 | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |             cancellationToken.ThrowIfCancellationRequested(); | ||||||
| 
 | 
 | ||||||
| @ -128,74 +128,6 @@ namespace MediaBrowser.Providers | |||||||
|             return item.ResolveArgs; |             return item.ResolveArgs; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |  | ||||||
|         /// Validates that images within the item are still on the file system |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="item">The item.</param> |  | ||||||
|         internal static void ValidateImages(BaseItem item) |  | ||||||
|         { |  | ||||||
|             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below |  | ||||||
|             var deletedKeys = item.Images |  | ||||||
|                 .ToList() |  | ||||||
|                 .Where(image => !File.Exists(image.Value)) |  | ||||||
|                 .Select(i => i.Key) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             // Now remove them from the dictionary |  | ||||||
|             foreach (var key in deletedKeys) |  | ||||||
|             { |  | ||||||
|                 item.Images.Remove(key); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Validates that backdrops within the item are still on the file system |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="item">The item.</param> |  | ||||||
|         internal static void ValidateBackdrops(BaseItem item) |  | ||||||
|         { |  | ||||||
|             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below |  | ||||||
|             var deletedImages = item.BackdropImagePaths |  | ||||||
|                 .Where(path => !File.Exists(path)) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             // Now remove them from the dictionary |  | ||||||
|             foreach (var path in deletedImages) |  | ||||||
|             { |  | ||||||
|                 item.BackdropImagePaths.Remove(path); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Validates the screenshots. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="item">The item.</param> |  | ||||||
|         /// <param name="args">The args.</param> |  | ||||||
|         private void ValidateScreenshots(BaseItem item, ItemResolveArgs args) |  | ||||||
|         { |  | ||||||
|             // Only validate paths from the same directory - need to copy to a list because we are going to potentially modify the collection below |  | ||||||
|             var deletedImages = item.ScreenshotImagePaths |  | ||||||
|                 .Where(path => !File.Exists(path)) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             // Now remove them from the dictionary |  | ||||||
|             foreach (var path in deletedImages) |  | ||||||
|             { |  | ||||||
|                 item.ScreenshotImagePaths.Remove(path); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Determines whether [is in same directory] [the specified item]. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="item">The item.</param> |  | ||||||
|         /// <param name="path">The path.</param> |  | ||||||
|         /// <returns><c>true</c> if [is in same directory] [the specified item]; otherwise, <c>false</c>.</returns> |  | ||||||
|         private bool IsInMetaLocation(BaseItem item, string path) |  | ||||||
|         { |  | ||||||
|             return string.Equals(Path.GetDirectoryName(path), item.MetaLocation, StringComparison.OrdinalIgnoreCase); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Gets the image. |         /// Gets the image. | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | |||||||
| @ -70,7 +70,6 @@ | |||||||
|     <Compile Include="Music\AlbumInfoFromSongProvider.cs" /> |     <Compile Include="Music\AlbumInfoFromSongProvider.cs" /> | ||||||
|     <Compile Include="Music\ArtistInfoFromSongProvider.cs" /> |     <Compile Include="Music\ArtistInfoFromSongProvider.cs" /> | ||||||
|     <Compile Include="Music\ArtistProviderFromXml.cs" /> |     <Compile Include="Music\ArtistProviderFromXml.cs" /> | ||||||
|     <Compile Include="Music\ArtistsPostScanTask.cs" /> |  | ||||||
|     <Compile Include="Music\FanArtAlbumProvider.cs" /> |     <Compile Include="Music\FanArtAlbumProvider.cs" /> | ||||||
|     <Compile Include="Music\FanArtArtistByNameProvider.cs" /> |     <Compile Include="Music\FanArtArtistByNameProvider.cs" /> | ||||||
|     <Compile Include="Music\FanArtArtistProvider.cs" /> |     <Compile Include="Music\FanArtArtistProvider.cs" /> | ||||||
| @ -81,6 +80,7 @@ | |||||||
|     <Compile Include="Music\LastfmArtistProvider.cs" /> |     <Compile Include="Music\LastfmArtistProvider.cs" /> | ||||||
|     <Compile Include="Music\LastfmBaseProvider.cs" /> |     <Compile Include="Music\LastfmBaseProvider.cs" /> | ||||||
|     <Compile Include="Music\LastfmHelper.cs" /> |     <Compile Include="Music\LastfmHelper.cs" /> | ||||||
|  |     <Compile Include="Music\MusicAlbumDynamicInfoProvider.cs" /> | ||||||
|     <Compile Include="Music\MusicVideoXmlParser.cs" /> |     <Compile Include="Music\MusicVideoXmlParser.cs" /> | ||||||
|     <Compile Include="Music\SoundtrackPostScanTask.cs" /> |     <Compile Include="Music\SoundtrackPostScanTask.cs" /> | ||||||
|     <Compile Include="Properties\AssemblyInfo.cs" /> |     <Compile Include="Properties\AssemblyInfo.cs" /> | ||||||
| @ -106,9 +106,9 @@ | |||||||
|     <Compile Include="TV\RemoteSeasonProvider.cs" /> |     <Compile Include="TV\RemoteSeasonProvider.cs" /> | ||||||
|     <Compile Include="TV\RemoteSeriesProvider.cs" /> |     <Compile Include="TV\RemoteSeriesProvider.cs" /> | ||||||
|     <Compile Include="TV\SeasonProviderFromXml.cs" /> |     <Compile Include="TV\SeasonProviderFromXml.cs" /> | ||||||
|  |     <Compile Include="TV\SeriesPostScanTask.cs" /> | ||||||
|     <Compile Include="TV\SeriesProviderFromXml.cs" /> |     <Compile Include="TV\SeriesProviderFromXml.cs" /> | ||||||
|     <Compile Include="TV\SeriesXmlParser.cs" /> |     <Compile Include="TV\SeriesXmlParser.cs" /> | ||||||
|     <Compile Include="TV\SeriesPostScanTask.cs" /> |  | ||||||
|     <Compile Include="TV\TvdbPersonImageProvider.cs" /> |     <Compile Include="TV\TvdbPersonImageProvider.cs" /> | ||||||
|     <Compile Include="TV\TvdbPrescanTask.cs" /> |     <Compile Include="TV\TvdbPrescanTask.cs" /> | ||||||
|     <Compile Include="TV\TvdbSeriesImageProvider.cs" /> |     <Compile Include="TV\TvdbSeriesImageProvider.cs" /> | ||||||
|  | |||||||
| @ -1,161 +0,0 @@ | |||||||
| using MediaBrowser.Common.Progress; |  | ||||||
| using MediaBrowser.Controller.Entities; |  | ||||||
| using MediaBrowser.Controller.Entities.Audio; |  | ||||||
| using MediaBrowser.Controller.Library; |  | ||||||
| using MediaBrowser.Model.Entities; |  | ||||||
| using System; |  | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.Globalization; |  | ||||||
| using System.Linq; |  | ||||||
| using System.Threading; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| 
 |  | ||||||
| namespace MediaBrowser.Providers.Music |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Class ArtistsPostScanTask |  | ||||||
|     /// </summary> |  | ||||||
|     public class ArtistsPostScanTask : ILibraryPostScanTask |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// The _library manager |  | ||||||
|         /// </summary> |  | ||||||
|         private readonly ILibraryManager _libraryManager; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Initializes a new instance of the <see cref="ArtistsPostScanTask"/> class. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="libraryManager">The library manager.</param> |  | ||||||
|         public ArtistsPostScanTask(ILibraryManager libraryManager) |  | ||||||
|         { |  | ||||||
|             _libraryManager = libraryManager; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Runs the specified progress. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="progress">The progress.</param> |  | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |  | ||||||
|         /// <returns>Task.</returns> |  | ||||||
|         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) |  | ||||||
|         { |  | ||||||
|             var allItems = _libraryManager.RootFolder.RecursiveChildren.ToList(); |  | ||||||
| 
 |  | ||||||
|             var allArtists = await GetAllArtists(allItems).ConfigureAwait(false); |  | ||||||
| 
 |  | ||||||
|             progress.Report(10); |  | ||||||
| 
 |  | ||||||
|             var allMusicArtists = allItems.OfType<MusicArtist>().ToList(); |  | ||||||
|             var allSongs = allItems.OfType<Audio>().ToList(); |  | ||||||
| 
 |  | ||||||
|             var numComplete = 0; |  | ||||||
| 
 |  | ||||||
|             foreach (var artist in allArtists) |  | ||||||
|             { |  | ||||||
|                 var musicArtist = FindMusicArtist(artist, allMusicArtists); |  | ||||||
| 
 |  | ||||||
|                 if (musicArtist != null) |  | ||||||
|                 { |  | ||||||
|                     MergeImages(musicArtist.Images, artist.Images); |  | ||||||
| 
 |  | ||||||
|                     // Merge backdrops |  | ||||||
|                     var backdrops = musicArtist.BackdropImagePaths.ToList(); |  | ||||||
|                     backdrops.InsertRange(0, artist.BackdropImagePaths); |  | ||||||
|                     artist.BackdropImagePaths = backdrops.Distinct(StringComparer.OrdinalIgnoreCase) |  | ||||||
|                         .ToList(); |  | ||||||
| 
 |  | ||||||
|                     ImageFromMediaLocationProvider.ValidateImages(artist); |  | ||||||
|                     ImageFromMediaLocationProvider.ValidateBackdrops(artist); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 if (!artist.LockedFields.Contains(MetadataFields.Genres)) |  | ||||||
|                 { |  | ||||||
|                     // Avoid implicitly captured closure |  | ||||||
|                     var artist1 = artist; |  | ||||||
| 
 |  | ||||||
|                     artist.Genres = allSongs.Where(i => i.HasArtist(artist1.Name)) |  | ||||||
|                         .SelectMany(i => i.Genres) |  | ||||||
|                         .Distinct(StringComparer.OrdinalIgnoreCase) |  | ||||||
|                         .ToList(); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 numComplete++; |  | ||||||
|                 double percent = numComplete; |  | ||||||
|                 percent /= allArtists.Length; |  | ||||||
|                 percent *= 5; |  | ||||||
| 
 |  | ||||||
|                 progress.Report(10 + percent); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var innerProgress = new ActionableProgress<double>(); |  | ||||||
| 
 |  | ||||||
|             innerProgress.RegisterAction(pct => progress.Report(15 + pct * .85)); |  | ||||||
| 
 |  | ||||||
|             await _libraryManager.ValidateArtists(cancellationToken, innerProgress).ConfigureAwait(false); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private void MergeImages(Dictionary<ImageType, string> source, Dictionary<ImageType, string> target) |  | ||||||
|         { |  | ||||||
|             foreach (var key in source.Keys |  | ||||||
|                 .ToList() |  | ||||||
|                 .Where(k => !target.ContainsKey(k))) |  | ||||||
|             { |  | ||||||
|                 string path; |  | ||||||
| 
 |  | ||||||
|                 if (source.TryGetValue(key, out path)) |  | ||||||
|                 { |  | ||||||
|                     target[key] = path; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets all artists. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="allItems">All items.</param> |  | ||||||
|         /// <returns>Task{Artist[]}.</returns> |  | ||||||
|         private Task<Artist[]> GetAllArtists(IEnumerable<BaseItem> allItems) |  | ||||||
|         { |  | ||||||
|             var itemsList = allItems.OfType<Audio>().ToList(); |  | ||||||
| 
 |  | ||||||
|             var tasks = itemsList |  | ||||||
|                 .SelectMany(i => |  | ||||||
|                 { |  | ||||||
|                     var list = new List<string>(); |  | ||||||
| 
 |  | ||||||
|                     if (!string.IsNullOrEmpty(i.AlbumArtist)) |  | ||||||
|                     { |  | ||||||
|                         list.Add(i.AlbumArtist); |  | ||||||
|                     } |  | ||||||
|                     list.AddRange(i.Artists); |  | ||||||
| 
 |  | ||||||
|                     return list; |  | ||||||
|                 }) |  | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |  | ||||||
|                 .Select(i => _libraryManager.GetArtist(i)); |  | ||||||
| 
 |  | ||||||
|             return Task.WhenAll(tasks); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Finds the music artist. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="artist">The artist.</param> |  | ||||||
|         /// <param name="allMusicArtists">All music artists.</param> |  | ||||||
|         /// <returns>MusicArtist.</returns> |  | ||||||
|         private static MusicArtist FindMusicArtist(Artist artist, IEnumerable<MusicArtist> allMusicArtists) |  | ||||||
|         { |  | ||||||
|             var musicBrainzId = artist.GetProviderId(MetadataProviders.Musicbrainz); |  | ||||||
| 
 |  | ||||||
|             return allMusicArtists.FirstOrDefault(i => |  | ||||||
|             { |  | ||||||
|                 if (!string.IsNullOrWhiteSpace(musicBrainzId) && string.Equals(musicBrainzId, i.GetProviderId(MetadataProviders.Musicbrainz), StringComparison.OrdinalIgnoreCase)) |  | ||||||
|                 { |  | ||||||
|                     return true; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 return string.Compare(i.Name, artist.Name, CultureInfo.CurrentCulture, CompareOptions.IgnoreNonSpace | CompareOptions.IgnoreCase | CompareOptions.IgnoreSymbols) == 0; |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -0,0 +1,85 @@ | |||||||
|  | using MediaBrowser.Controller.Configuration; | ||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Controller.Providers; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Providers.Music | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class MusicAlbumDynamicInfoProvider | ||||||
|  |     /// </summary> | ||||||
|  |     public class MusicAlbumDynamicInfoProvider : BaseMetadataProvider, IDynamicInfoProvider | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="BaseMetadataProvider" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="logManager">The log manager.</param> | ||||||
|  |         /// <param name="configurationManager">The configuration manager.</param> | ||||||
|  |         public MusicAlbumDynamicInfoProvider(ILogManager logManager, IServerConfigurationManager configurationManager) | ||||||
|  |             : base(logManager, configurationManager) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Supportses the specified item. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="item">The item.</param> | ||||||
|  |         /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns> | ||||||
|  |         public override bool Supports(BaseItem item) | ||||||
|  |         { | ||||||
|  |             return item is MusicAlbum; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Needses the refresh internal. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="item">The item.</param> | ||||||
|  |         /// <param name="providerInfo">The provider info.</param> | ||||||
|  |         /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns> | ||||||
|  |         protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo) | ||||||
|  |         { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Fetches metadata and returns true or false indicating if any work that requires persistence was done | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="item">The item.</param> | ||||||
|  |         /// <param name="force">if set to <c>true</c> [force].</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task{System.Boolean}.</returns> | ||||||
|  |         public override Task<bool> FetchAsync(BaseItem item, bool force, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var album = (MusicAlbum)item; | ||||||
|  | 
 | ||||||
|  |             var songs = album.RecursiveChildren | ||||||
|  |                              .OfType<Audio>() | ||||||
|  |                              .ToList(); | ||||||
|  | 
 | ||||||
|  |             album.AlbumArtist = songs | ||||||
|  |                 .Select(i => i.AlbumArtist) | ||||||
|  |                 .FirstOrDefault(i => !string.IsNullOrEmpty(i)); | ||||||
|  | 
 | ||||||
|  |             album.Artists = songs.SelectMany(i => i.Artists) | ||||||
|  |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                 .ToArray(); | ||||||
|  | 
 | ||||||
|  |             // Don't save to the db | ||||||
|  |             return FalseTaskResult; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the priority. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <value>The priority.</value> | ||||||
|  |         public override MetadataProviderPriority Priority | ||||||
|  |         { | ||||||
|  |             get { return MetadataProviderPriority.Last; } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -107,6 +107,15 @@ namespace MediaBrowser.Server.Implementations.Dto | |||||||
|                     .ToArray(); |                     .ToArray(); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if (fields.Contains(ItemFields.ItemCounts)) | ||||||
|  |             { | ||||||
|  |                 var itemByName = item as IItemByName; | ||||||
|  |                 if (itemByName != null) | ||||||
|  |                 { | ||||||
|  |                     AttachItemByNameCounts(dto, itemByName, user); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             // Make sure all the tasks we kicked off have completed. |             // Make sure all the tasks we kicked off have completed. | ||||||
|             if (tasks.Count > 0) |             if (tasks.Count > 0) | ||||||
|             { |             { | ||||||
| @ -116,6 +125,41 @@ namespace MediaBrowser.Server.Implementations.Dto | |||||||
|             return dto; |             return dto; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Attaches the item by name counts. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="dto">The dto.</param> | ||||||
|  |         /// <param name="item">The item.</param> | ||||||
|  |         /// <param name="user">The user.</param> | ||||||
|  |         private void AttachItemByNameCounts(BaseItemDto dto, IItemByName item, User user) | ||||||
|  |         { | ||||||
|  |             ItemByNameCounts counts; | ||||||
|  | 
 | ||||||
|  |             if (user == null) | ||||||
|  |             { | ||||||
|  |                 counts = item.ItemCounts; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 if (!item.UserItemCounts.TryGetValue(user.Id, out counts)) | ||||||
|  |                 { | ||||||
|  |                     counts = new ItemByNameCounts(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             dto.ChildCount = counts.TotalCount; | ||||||
|  | 
 | ||||||
|  |             dto.AdultVideoCount = counts.AdultVideoCount; | ||||||
|  |             dto.AlbumCount = counts.AlbumCount; | ||||||
|  |             dto.EpisodeCount = counts.EpisodeCount; | ||||||
|  |             dto.GameCount = counts.GameCount; | ||||||
|  |             dto.MovieCount = counts.MovieCount; | ||||||
|  |             dto.MusicVideoCount = counts.MusicVideoCount; | ||||||
|  |             dto.SeriesCount = counts.SeriesCount; | ||||||
|  |             dto.SongCount = counts.SongCount; | ||||||
|  |             dto.TrailerCount = counts.TrailerCount; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Attaches the user specific info. |         /// Attaches the user specific info. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -380,7 +424,9 @@ namespace MediaBrowser.Server.Implementations.Dto | |||||||
|                 _logger.ErrorException("Error getting {0} image info for {1}", ex, type, path); |                 _logger.ErrorException("Error getting {0} image info for {1}", ex, type, path); | ||||||
|                 return null; |                 return null; | ||||||
|             } |             } | ||||||
|         }        /// <summary> |         }         | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|         /// Attaches People DTO's to a DTOBaseItem |         /// Attaches People DTO's to a DTOBaseItem | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         /// <param name="dto">The dto.</param> |         /// <param name="dto">The dto.</param> | ||||||
| @ -913,12 +959,7 @@ namespace MediaBrowser.Server.Implementations.Dto | |||||||
| 
 | 
 | ||||||
|             if (album != null) |             if (album != null) | ||||||
|             { |             { | ||||||
|                 var songs = album.RecursiveChildren.OfType<Audio>().ToList(); |                 dto.Artists = album.Artists; | ||||||
| 
 |  | ||||||
|                 dto.Artists = |  | ||||||
|                     songs.SelectMany(i => i.Artists) |  | ||||||
|                          .Distinct(StringComparer.OrdinalIgnoreCase) |  | ||||||
|                          .ToArray(); |  | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             var hasAlbumArtist = item as IHasAlbumArtist; |             var hasAlbumArtist = item as IHasAlbumArtist; | ||||||
| @ -935,7 +976,6 @@ namespace MediaBrowser.Server.Implementations.Dto | |||||||
|                 dto.VideoType = video.VideoType; |                 dto.VideoType = video.VideoType; | ||||||
|                 dto.Video3DFormat = video.Video3DFormat; |                 dto.Video3DFormat = video.Video3DFormat; | ||||||
|                 dto.IsoType = video.IsoType; |                 dto.IsoType = video.IsoType; | ||||||
|                 dto.MainFeaturePlaylistName = video.MainFeaturePlaylistName; |  | ||||||
| 
 | 
 | ||||||
|                 dto.PartCount = video.AdditionalPartIds.Count + 1; |                 dto.PartCount = video.AdditionalPartIds.Count + 1; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -81,6 +81,10 @@ namespace MediaBrowser.Server.Implementations.HttpServer | |||||||
|                 { |                 { | ||||||
|                     bytes = await ReceiveBytesAsync(CancellationToken.None).ConfigureAwait(false); |                     bytes = await ReceiveBytesAsync(CancellationToken.None).ConfigureAwait(false); | ||||||
|                 } |                 } | ||||||
|  |                 catch (OperationCanceledException) | ||||||
|  |                 { | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|                 catch (WebSocketException ex) |                 catch (WebSocketException ex) | ||||||
|                 { |                 { | ||||||
|                     _logger.ErrorException("Error receiving web socket message", ex); |                     _logger.ErrorException("Error receiving web socket message", ex); | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ using MediaBrowser.Controller.Sorting; | |||||||
| using MediaBrowser.Model.Configuration; | using MediaBrowser.Model.Configuration; | ||||||
| using MediaBrowser.Model.Entities; | using MediaBrowser.Model.Entities; | ||||||
| using MediaBrowser.Model.Logging; | using MediaBrowser.Model.Logging; | ||||||
|  | using MediaBrowser.Server.Implementations.Library.Validators; | ||||||
| using MediaBrowser.Server.Implementations.ScheduledTasks; | using MediaBrowser.Server.Implementations.ScheduledTasks; | ||||||
| using MoreLinq; | using MoreLinq; | ||||||
| using System; | using System; | ||||||
| @ -597,11 +598,11 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <param name="name">The name.</param> |         /// <param name="name">The name.</param> | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|         /// <param name="forceCreation">if set to <c>true</c> [force creation].</param> |         /// <param name="refreshMetadata">if set to <c>true</c> [force creation].</param> | ||||||
|         /// <returns>Task{Person}.</returns> |         /// <returns>Task{Person}.</returns> | ||||||
|         private Task<Person> GetPerson(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool forceCreation = false) |         private Task<Person> GetPerson(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<Person>(ConfigurationManager.ApplicationPaths.PeoplePath, name, cancellationToken, allowSlowProviders, forceCreation); |             return GetItemByName<Person>(ConfigurationManager.ApplicationPaths.PeoplePath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -612,7 +613,20 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <returns>Task{Studio}.</returns> |         /// <returns>Task{Studio}.</returns> | ||||||
|         public Task<Studio> GetStudio(string name, bool allowSlowProviders = false) |         public Task<Studio> GetStudio(string name, bool allowSlowProviders = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<Studio>(ConfigurationManager.ApplicationPaths.StudioPath, name, CancellationToken.None, allowSlowProviders); |             return GetStudio(name, CancellationToken.None, allowSlowProviders); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the studio. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="name">The name.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|  |         /// <param name="refreshMetadata">if set to <c>true</c> [refresh metadata].</param> | ||||||
|  |         /// <returns>Task{Studio}.</returns> | ||||||
|  |         internal Task<Studio> GetStudio(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|  |         { | ||||||
|  |             return GetItemByName<Studio>(ConfigurationManager.ApplicationPaths.StudioPath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -623,7 +637,20 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <returns>Task{Genre}.</returns> |         /// <returns>Task{Genre}.</returns> | ||||||
|         public Task<Genre> GetGenre(string name, bool allowSlowProviders = false) |         public Task<Genre> GetGenre(string name, bool allowSlowProviders = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<Genre>(ConfigurationManager.ApplicationPaths.GenrePath, name, CancellationToken.None, allowSlowProviders); |             return GetGenre(name, CancellationToken.None, allowSlowProviders); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the genre. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="name">The name.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|  |         /// <param name="refreshMetadata">if set to <c>true</c> [refresh metadata].</param> | ||||||
|  |         /// <returns>Task{Genre}.</returns> | ||||||
|  |         internal Task<Genre> GetGenre(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|  |         { | ||||||
|  |             return GetItemByName<Genre>(ConfigurationManager.ApplicationPaths.GenrePath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -634,7 +661,20 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <returns>Task{MusicGenre}.</returns> |         /// <returns>Task{MusicGenre}.</returns> | ||||||
|         public Task<MusicGenre> GetMusicGenre(string name, bool allowSlowProviders = false) |         public Task<MusicGenre> GetMusicGenre(string name, bool allowSlowProviders = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<MusicGenre>(ConfigurationManager.ApplicationPaths.MusicGenrePath, name, CancellationToken.None, allowSlowProviders); |             return GetMusicGenre(name, CancellationToken.None, allowSlowProviders); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the music genre. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="name">The name.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|  |         /// <param name="refreshMetadata">if set to <c>true</c> [refresh metadata].</param> | ||||||
|  |         /// <returns>Task{MusicGenre}.</returns> | ||||||
|  |         internal Task<MusicGenre> GetMusicGenre(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|  |         { | ||||||
|  |             return GetItemByName<MusicGenre>(ConfigurationManager.ApplicationPaths.MusicGenrePath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -645,7 +685,20 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <returns>Task{GameGenre}.</returns> |         /// <returns>Task{GameGenre}.</returns> | ||||||
|         public Task<GameGenre> GetGameGenre(string name, bool allowSlowProviders = false) |         public Task<GameGenre> GetGameGenre(string name, bool allowSlowProviders = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<GameGenre>(ConfigurationManager.ApplicationPaths.GameGenrePath, name, CancellationToken.None, allowSlowProviders); |             return GetGameGenre(name, CancellationToken.None, allowSlowProviders); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the game genre. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="name">The name.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|  |         /// <param name="refreshMetadata">if set to <c>true</c> [refresh metadata].</param> | ||||||
|  |         /// <returns>Task{GameGenre}.</returns> | ||||||
|  |         internal Task<GameGenre> GetGameGenre(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|  |         { | ||||||
|  |             return GetItemByName<GameGenre>(ConfigurationManager.ApplicationPaths.GameGenrePath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -665,11 +718,11 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <param name="name">The name.</param> |         /// <param name="name">The name.</param> | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|         /// <param name="forceCreation">if set to <c>true</c> [force creation].</param> |         /// <param name="refreshMetadata">if set to <c>true</c> [force creation].</param> | ||||||
|         /// <returns>Task{Artist}.</returns> |         /// <returns>Task{Artist}.</returns> | ||||||
|         private Task<Artist> GetArtist(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool forceCreation = false) |         internal Task<Artist> GetArtist(string name, CancellationToken cancellationToken, bool allowSlowProviders = false, bool refreshMetadata = false) | ||||||
|         { |         { | ||||||
|             return GetItemByName<Artist>(ConfigurationManager.ApplicationPaths.ArtistsPath, name, cancellationToken, allowSlowProviders, forceCreation); |             return GetItemByName<Artist>(ConfigurationManager.ApplicationPaths.ArtistsPath, name, cancellationToken, allowSlowProviders, refreshMetadata); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -707,11 +760,11 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <param name="name">The name.</param> |         /// <param name="name">The name.</param> | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> |         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> | ||||||
|         /// <param name="forceCreation">if set to <c>true</c> [force creation].</param> |         /// <param name="refreshMetadata">if set to <c>true</c> [force creation].</param> | ||||||
|         /// <returns>Task{``0}.</returns> |         /// <returns>Task{``0}.</returns> | ||||||
|         /// <exception cref="System.ArgumentNullException"> |         /// <exception cref="System.ArgumentNullException"> | ||||||
|         /// </exception> |         /// </exception> | ||||||
|         private async Task<T> GetItemByName<T>(string path, string name, CancellationToken cancellationToken, bool allowSlowProviders = true, bool forceCreation = false) |         private async Task<T> GetItemByName<T>(string path, string name, CancellationToken cancellationToken, bool allowSlowProviders = true, bool refreshMetadata = false) | ||||||
|             where T : BaseItem, new() |             where T : BaseItem, new() | ||||||
|         { |         { | ||||||
|             if (string.IsNullOrEmpty(path)) |             if (string.IsNullOrEmpty(path)) | ||||||
| @ -730,11 +783,25 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
| 
 | 
 | ||||||
|             if (!_itemsByName.TryGetValue(key, out obj)) |             if (!_itemsByName.TryGetValue(key, out obj)) | ||||||
|             { |             { | ||||||
|                 obj = await CreateItemByName<T>(path, name, cancellationToken, allowSlowProviders).ConfigureAwait(false); |                 var tuple = CreateItemByName<T>(path, name, cancellationToken); | ||||||
|  | 
 | ||||||
|  |                 obj = tuple.Item2; | ||||||
| 
 | 
 | ||||||
|                 _itemsByName.AddOrUpdate(key, obj, (keyName, oldValue) => obj); |                 _itemsByName.AddOrUpdate(key, obj, (keyName, oldValue) => obj); | ||||||
|  | 
 | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await obj.RefreshMetadata(cancellationToken, tuple.Item1, allowSlowProviders: allowSlowProviders).ConfigureAwait(false); | ||||||
|                 } |                 } | ||||||
|             else if (forceCreation) |                 catch (OperationCanceledException) | ||||||
|  |                 { | ||||||
|  |                     BaseItem removed; | ||||||
|  |                     _itemsByName.TryRemove(key, out removed); | ||||||
|  | 
 | ||||||
|  |                     throw; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else if (refreshMetadata) | ||||||
|             { |             { | ||||||
|                 await obj.RefreshMetadata(cancellationToken, false, allowSlowProviders: allowSlowProviders).ConfigureAwait(false); |                 await obj.RefreshMetadata(cancellationToken, false, allowSlowProviders: allowSlowProviders).ConfigureAwait(false); | ||||||
|             } |             } | ||||||
| @ -749,10 +816,9 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <param name="path">The path.</param> |         /// <param name="path">The path.</param> | ||||||
|         /// <param name="name">The name.</param> |         /// <param name="name">The name.</param> | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|         /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param> |  | ||||||
|         /// <returns>Task{``0}.</returns> |         /// <returns>Task{``0}.</returns> | ||||||
|         /// <exception cref="System.IO.IOException">Path not created:  + path</exception> |         /// <exception cref="System.IO.IOException">Path not created:  + path</exception> | ||||||
|         private async Task<T> CreateItemByName<T>(string path, string name, CancellationToken cancellationToken, bool allowSlowProviders = true) |         private Tuple<bool, T> CreateItemByName<T>(string path, string name, CancellationToken cancellationToken) | ||||||
|             where T : BaseItem, new() |             where T : BaseItem, new() | ||||||
|         { |         { | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |             cancellationToken.ThrowIfCancellationRequested(); | ||||||
| @ -783,6 +849,7 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|             var id = path.GetMBId(type); |             var id = path.GetMBId(type); | ||||||
| 
 | 
 | ||||||
|             var item = RetrieveItem(id) as T; |             var item = RetrieveItem(id) as T; | ||||||
|  | 
 | ||||||
|             if (item == null) |             if (item == null) | ||||||
|             { |             { | ||||||
|                 item = new T |                 item = new T | ||||||
| @ -796,16 +863,10 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|                 isNew = true; |                 isNew = true; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |  | ||||||
| 
 |  | ||||||
|             // Set this now so we don't cause additional file system access during provider executions |             // Set this now so we don't cause additional file system access during provider executions | ||||||
|             item.ResetResolveArgs(fileInfo); |             item.ResetResolveArgs(fileInfo); | ||||||
| 
 | 
 | ||||||
|             await item.RefreshMetadata(cancellationToken, isNew, allowSlowProviders: allowSlowProviders).ConfigureAwait(false); |             return new Tuple<bool,T>(isNew, item); | ||||||
| 
 |  | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |  | ||||||
| 
 |  | ||||||
|             return item; |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -884,75 +945,53 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|         /// <param name="progress">The progress.</param> |         /// <param name="progress">The progress.</param> | ||||||
|         /// <returns>Task.</returns> |         /// <returns>Task.</returns> | ||||||
|         public async Task ValidateArtists(CancellationToken cancellationToken, IProgress<double> progress) |         public Task ValidateArtists(CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|         { |         { | ||||||
|             const int maxTasks = 25; |             return new ArtistsValidator(this, _userManager, _logger).Run(progress, cancellationToken); | ||||||
| 
 |  | ||||||
|             var tasks = new List<Task>(); |  | ||||||
| 
 |  | ||||||
|             var artists = RootFolder.RecursiveChildren |  | ||||||
|                 .OfType<Audio>() |  | ||||||
|                 .SelectMany(c => |  | ||||||
|                 { |  | ||||||
|                     var list = new List<string>(); |  | ||||||
| 
 |  | ||||||
|                     if (!string.IsNullOrEmpty(c.AlbumArtist)) |  | ||||||
|                     { |  | ||||||
|                         list.Add(c.AlbumArtist); |  | ||||||
|                     } |  | ||||||
|                     list.AddRange(c.Artists); |  | ||||||
| 
 |  | ||||||
|                     return list; |  | ||||||
|                 }) |  | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |  | ||||||
|                 .ToList(); |  | ||||||
| 
 |  | ||||||
|             var numComplete = 0; |  | ||||||
| 
 |  | ||||||
|             foreach (var artist in artists) |  | ||||||
|             { |  | ||||||
|                 if (tasks.Count > maxTasks) |  | ||||||
|                 { |  | ||||||
|                     await Task.WhenAll(tasks).ConfigureAwait(false); |  | ||||||
|                     tasks.Clear(); |  | ||||||
| 
 |  | ||||||
|                     // Safe cancellation point, when there are no pending tasks |  | ||||||
|                     cancellationToken.ThrowIfCancellationRequested(); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|                 // Avoid accessing the foreach variable within the closure |         /// <summary> | ||||||
|                 var currentArtist = artist; |         /// Validates the music genres. | ||||||
| 
 |         /// </summary> | ||||||
|                 tasks.Add(Task.Run(async () => |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task ValidateMusicGenres(CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|         { |         { | ||||||
|                     cancellationToken.ThrowIfCancellationRequested(); |             return new MusicGenresValidator(this, _userManager, _logger).Run(progress, cancellationToken); | ||||||
| 
 |  | ||||||
|                     try |  | ||||||
|                     { |  | ||||||
|                         await GetArtist(currentArtist, cancellationToken, true, true).ConfigureAwait(false); |  | ||||||
|                     } |  | ||||||
|                     catch (IOException ex) |  | ||||||
|                     { |  | ||||||
|                         _logger.ErrorException("Error validating Artist {0}", ex, currentArtist); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|                     // Update progress |         /// <summary> | ||||||
|                     lock (progress) |         /// Validates the game genres. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task ValidateGameGenres(CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|         { |         { | ||||||
|                         numComplete++; |             return new GameGenresValidator(this, _userManager, _logger).Run(progress, cancellationToken); | ||||||
|                         double percent = numComplete; |  | ||||||
|                         percent /= artists.Count; |  | ||||||
| 
 |  | ||||||
|                         progress.Report(100 * percent); |  | ||||||
|                     } |  | ||||||
|                 })); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|             await Task.WhenAll(tasks).ConfigureAwait(false); |         /// <summary> | ||||||
|  |         /// Validates the studios. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task ValidateStudios(CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|  |         { | ||||||
|  |             return new StudiosValidator(this, _userManager, _logger).Run(progress, cancellationToken); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|             progress.Report(100); |         /// <summary> | ||||||
| 
 |         /// Validates the genres. | ||||||
|             _logger.Info("Artist validation complete"); |         /// </summary> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task ValidateGenres(CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|  |         { | ||||||
|  |             return new GenresValidator(this, _userManager, _logger).Run(progress, cancellationToken); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |         /// <summary> | ||||||
| @ -1000,12 +1039,12 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
| 
 | 
 | ||||||
|             var innerProgress = new ActionableProgress<double>(); |             var innerProgress = new ActionableProgress<double>(); | ||||||
| 
 | 
 | ||||||
|             innerProgress.RegisterAction(pct => progress.Report(15 + pct * .65)); |             innerProgress.RegisterAction(pct => progress.Report(15 + pct * .6)); | ||||||
| 
 | 
 | ||||||
|             // Now validate the entire media library |             // Now validate the entire media library | ||||||
|             await RootFolder.ValidateChildren(innerProgress, cancellationToken, recursive: true).ConfigureAwait(false); |             await RootFolder.ValidateChildren(innerProgress, cancellationToken, recursive: true).ConfigureAwait(false); | ||||||
| 
 | 
 | ||||||
|             progress.Report(80); |             progress.Report(75); | ||||||
| 
 | 
 | ||||||
|             // Run post-scan tasks |             // Run post-scan tasks | ||||||
|             await RunPostScanTasks(progress, cancellationToken).ConfigureAwait(false); |             await RunPostScanTasks(progress, cancellationToken).ConfigureAwait(false); | ||||||
| @ -1078,7 +1117,7 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|                         double percent = progressDictionary.Values.Sum(); |                         double percent = progressDictionary.Values.Sum(); | ||||||
|                         percent /= postscanTasks.Count; |                         percent /= postscanTasks.Count; | ||||||
| 
 | 
 | ||||||
|                         progress.Report(80 + percent * .2); |                         progress.Report(75 + percent * .25); | ||||||
|                     } |                     } | ||||||
|                 }); |                 }); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -155,7 +155,7 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             // Find genres, from non-audio items |             // Find genres, from non-audio items | ||||||
|             var genres = items.Where(i => !(i is Audio) && !(i is MusicAlbum) && !(i is MusicArtist) && !(i is MusicVideo) && !(i is Game)) |             var genres = items.Where(i => !(i is IHasMusicGenres) && !(i is Game)) | ||||||
|                 .SelectMany(i => i.Genres) |                 .SelectMany(i => i.Genres) | ||||||
|                 .Where(i => !string.IsNullOrEmpty(i)) |                 .Where(i => !string.IsNullOrEmpty(i)) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
| @ -181,7 +181,7 @@ namespace MediaBrowser.Server.Implementations.Library | |||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             // Find music genres |             // Find music genres | ||||||
|             var musicGenres = items.Where(i => (i is Audio) || (i is MusicAlbum) || (i is MusicArtist) || (i is MusicVideo)) |             var musicGenres = items.Where(i => i is IHasMusicGenres) | ||||||
|                 .SelectMany(i => i.Genres) |                 .SelectMany(i => i.Genres) | ||||||
|                 .Where(i => !string.IsNullOrEmpty(i)) |                 .Where(i => !string.IsNullOrEmpty(i)) | ||||||
|                 .Distinct(StringComparer.OrdinalIgnoreCase) |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  | |||||||
| @ -0,0 +1,38 @@ | |||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using System; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class ArtistsPostScanTask | ||||||
|  |     /// </summary> | ||||||
|  |     public class ArtistsPostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="ArtistsPostScanTask" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         public ArtistsPostScanTask(ILibraryManager libraryManager) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             return _libraryManager.ValidateArtists(cancellationToken, progress); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,285 @@ | |||||||
|  | using MediaBrowser.Common.Progress; | ||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using MediaBrowser.Model.Entities; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Concurrent; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Globalization; | ||||||
|  | using System.IO; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class ArtistsValidator | ||||||
|  |     /// </summary> | ||||||
|  |     public class ArtistsValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly LibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="ArtistsPostScanTask" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         /// <param name="userManager">The user manager.</param> | ||||||
|  |         /// <param name="logger">The logger.</param> | ||||||
|  |         public ArtistsValidator(LibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren.ToList(); | ||||||
|  | 
 | ||||||
|  |             var allMusicArtists = allItems.OfType<MusicArtist>().ToList(); | ||||||
|  |             var allSongs = allItems.OfType<Audio>().ToList(); | ||||||
|  | 
 | ||||||
|  |             var innerProgress = new ActionableProgress<double>(); | ||||||
|  | 
 | ||||||
|  |             innerProgress.RegisterAction(pct => progress.Report(pct * .8)); | ||||||
|  |              | ||||||
|  |             var allArtists = await GetAllArtists(allSongs, cancellationToken, innerProgress).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             progress.Report(80); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<IHasArtist>>(i.Id, i.RootFolder.GetRecursiveChildren(i).OfType<IHasArtist>().ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             foreach (var artist in allArtists) | ||||||
|  |             { | ||||||
|  |                 cancellationToken.ThrowIfCancellationRequested(); | ||||||
|  |                  | ||||||
|  |                 artist.ValidateImages(); | ||||||
|  |                 artist.ValidateBackdrops(); | ||||||
|  | 
 | ||||||
|  |                 var musicArtist = FindMusicArtist(artist, allMusicArtists); | ||||||
|  | 
 | ||||||
|  |                 if (musicArtist != null) | ||||||
|  |                 { | ||||||
|  |                     MergeImages(musicArtist.Images, artist.Images); | ||||||
|  | 
 | ||||||
|  |                     // Merge backdrops | ||||||
|  |                     var backdrops = musicArtist.BackdropImagePaths.ToList(); | ||||||
|  |                     backdrops.InsertRange(0, artist.BackdropImagePaths); | ||||||
|  |                     artist.BackdropImagePaths = backdrops.Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                         .ToList(); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 if (!artist.LockedFields.Contains(MetadataFields.Genres)) | ||||||
|  |                 { | ||||||
|  |                     // Avoid implicitly captured closure | ||||||
|  |                     var artist1 = artist; | ||||||
|  | 
 | ||||||
|  |                     artist.Genres = allSongs.Where(i => i.HasArtist(artist1.Name)) | ||||||
|  |                         .SelectMany(i => i.Genres) | ||||||
|  |                         .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                         .ToList(); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // Populate counts of items | ||||||
|  |                 SetItemCounts(artist, null, allItems.OfType<IHasArtist>()); | ||||||
|  | 
 | ||||||
|  |                 foreach (var lib in userLibraries) | ||||||
|  |                 { | ||||||
|  |                     SetItemCounts(artist, lib.Item1, lib.Item2); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= allArtists.Count; | ||||||
|  |                 percent *= 20; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(80 + percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Sets the item counts. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="artist">The artist.</param> | ||||||
|  |         /// <param name="userId">The user id.</param> | ||||||
|  |         /// <param name="allItems">All items.</param> | ||||||
|  |         private void SetItemCounts(Artist artist, Guid? userId, IEnumerable<IHasArtist> allItems) | ||||||
|  |         { | ||||||
|  |             var name = artist.Name; | ||||||
|  | 
 | ||||||
|  |             var items = allItems | ||||||
|  |                 .Where(i => i.HasArtist(name)) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var counts = new ItemByNameCounts | ||||||
|  |             { | ||||||
|  |                 TotalCount = items.Count, | ||||||
|  | 
 | ||||||
|  |                 SongCount = items.OfType<Audio>().Count(), | ||||||
|  | 
 | ||||||
|  |                 AlbumCount = items.OfType<MusicAlbum>().Count(), | ||||||
|  | 
 | ||||||
|  |                 MusicVideoCount = items.OfType<MusicVideo>().Count() | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             if (userId.HasValue) | ||||||
|  |             { | ||||||
|  |                 artist.UserItemCounts[userId.Value] = counts; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 artist.ItemCounts = counts; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Merges the images. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="source">The source.</param> | ||||||
|  |         /// <param name="target">The target.</param> | ||||||
|  |         private void MergeImages(Dictionary<ImageType, string> source, Dictionary<ImageType, string> target) | ||||||
|  |         { | ||||||
|  |             foreach (var key in source.Keys | ||||||
|  |                 .ToList() | ||||||
|  |                 .Where(k => !target.ContainsKey(k))) | ||||||
|  |             { | ||||||
|  |                 string path; | ||||||
|  | 
 | ||||||
|  |                 if (source.TryGetValue(key, out path)) | ||||||
|  |                 { | ||||||
|  |                     target[key] = path; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets all artists. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="allSongs">All songs.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <returns>Task{Artist[]}.</returns> | ||||||
|  |         private async Task<List<Artist>> GetAllArtists(IEnumerable<Audio> allSongs, CancellationToken cancellationToken, IProgress<double> progress) | ||||||
|  |         { | ||||||
|  |             var allArtists = allSongs | ||||||
|  |                 .SelectMany(i => | ||||||
|  |                 { | ||||||
|  |                     var list = new List<string>(); | ||||||
|  | 
 | ||||||
|  |                     if (!string.IsNullOrEmpty(i.AlbumArtist)) | ||||||
|  |                     { | ||||||
|  |                         list.Add(i.AlbumArtist); | ||||||
|  |                     } | ||||||
|  |                     list.AddRange(i.Artists); | ||||||
|  | 
 | ||||||
|  |                     return list; | ||||||
|  |                 }) | ||||||
|  |                 .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             const int maxTasks = 5; | ||||||
|  | 
 | ||||||
|  |             var tasks = new List<Task>(); | ||||||
|  | 
 | ||||||
|  |             var returnArtists = new ConcurrentBag<Artist>(); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var artist in allArtists) | ||||||
|  |             { | ||||||
|  |                 if (tasks.Count > maxTasks) | ||||||
|  |                 { | ||||||
|  |                     await Task.WhenAll(tasks).ConfigureAwait(false); | ||||||
|  |                     tasks.Clear(); | ||||||
|  | 
 | ||||||
|  |                     // Safe cancellation point, when there are no pending tasks | ||||||
|  |                     cancellationToken.ThrowIfCancellationRequested(); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // Avoid accessing the foreach variable within the closure | ||||||
|  |                 var currentArtist = artist; | ||||||
|  | 
 | ||||||
|  |                 tasks.Add(Task.Run(async () => | ||||||
|  |                 { | ||||||
|  |                     cancellationToken.ThrowIfCancellationRequested(); | ||||||
|  | 
 | ||||||
|  |                     try | ||||||
|  |                     { | ||||||
|  |                         var artistItem = await _libraryManager.GetArtist(currentArtist, cancellationToken, true, true) | ||||||
|  |                             .ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |                         returnArtists.Add(artistItem); | ||||||
|  |                     } | ||||||
|  |                     catch (IOException ex) | ||||||
|  |                     { | ||||||
|  |                         _logger.ErrorException("Error validating Artist {0}", ex, currentArtist); | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     // Update progress | ||||||
|  |                     lock (progress) | ||||||
|  |                     { | ||||||
|  |                         numComplete++; | ||||||
|  |                         double percent = numComplete; | ||||||
|  |                         percent /= allArtists.Count; | ||||||
|  | 
 | ||||||
|  |                         progress.Report(100 * percent); | ||||||
|  |                     } | ||||||
|  |                 })); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             await Task.WhenAll(tasks).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             return returnArtists.ToList(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Finds the music artist. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="artist">The artist.</param> | ||||||
|  |         /// <param name="allMusicArtists">All music artists.</param> | ||||||
|  |         /// <returns>MusicArtist.</returns> | ||||||
|  |         private static MusicArtist FindMusicArtist(Artist artist, IEnumerable<MusicArtist> allMusicArtists) | ||||||
|  |         { | ||||||
|  |             var musicBrainzId = artist.GetProviderId(MetadataProviders.Musicbrainz); | ||||||
|  | 
 | ||||||
|  |             return allMusicArtists.FirstOrDefault(i => | ||||||
|  |             { | ||||||
|  |                 if (!string.IsNullOrWhiteSpace(musicBrainzId) && string.Equals(musicBrainzId, i.GetProviderId(MetadataProviders.Musicbrainz), StringComparison.OrdinalIgnoreCase)) | ||||||
|  |                 { | ||||||
|  |                     return true; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 return string.Compare(i.Name, artist.Name, CultureInfo.CurrentCulture, CompareOptions.IgnoreNonSpace | CompareOptions.IgnoreCase | CompareOptions.IgnoreSymbols) == 0; | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,155 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Controller.Entities.Movies; | ||||||
|  | using MediaBrowser.Controller.Entities.TV; | ||||||
|  | using MediaBrowser.Model.Dto; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class CountHelpers | ||||||
|  |     /// </summary> | ||||||
|  |     internal static class CountHelpers | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// Adds to dictionary. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="item">The item.</param> | ||||||
|  |         /// <param name="counts">The counts.</param> | ||||||
|  |         internal static void AddToDictionary(BaseItem item, Dictionary<string, int> counts) | ||||||
|  |         { | ||||||
|  |             if (item is Movie) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Movie"); | ||||||
|  |             } | ||||||
|  |             else if (item is Trailer) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Trailer"); | ||||||
|  |             } | ||||||
|  |             else if (item is Series) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Series"); | ||||||
|  |             } | ||||||
|  |             else if (item is Game) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Game"); | ||||||
|  |             } | ||||||
|  |             else if (item is Audio) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Audio"); | ||||||
|  |             } | ||||||
|  |             else if (item is MusicAlbum) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "MusicAlbum"); | ||||||
|  |             } | ||||||
|  |             else if (item is Episode) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "Episode"); | ||||||
|  |             } | ||||||
|  |             else if (item is MusicVideo) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "MusicVideo"); | ||||||
|  |             } | ||||||
|  |             else if (item is AdultVideo) | ||||||
|  |             { | ||||||
|  |                 IncrementCount(counts, "AdultVideo"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             IncrementCount(counts, "Total"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Increments the count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="counts">The counts.</param> | ||||||
|  |         /// <param name="key">The key.</param> | ||||||
|  |         internal static void IncrementCount(Dictionary<string, int> counts, string key) | ||||||
|  |         { | ||||||
|  |             int count; | ||||||
|  | 
 | ||||||
|  |             if (counts.TryGetValue(key, out count)) | ||||||
|  |             { | ||||||
|  |                 count++; | ||||||
|  |                 counts[key] = count; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 counts.Add(key, 1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the counts. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="counts">The counts.</param> | ||||||
|  |         /// <returns>ItemByNameCounts.</returns> | ||||||
|  |         internal static ItemByNameCounts GetCounts(Dictionary<string, int> counts) | ||||||
|  |         { | ||||||
|  |             return new ItemByNameCounts | ||||||
|  |             { | ||||||
|  |                 AdultVideoCount = GetCount(counts, "AdultVideo"), | ||||||
|  |                 AlbumCount = GetCount(counts, "MusicAlbum"), | ||||||
|  |                 EpisodeCount = GetCount(counts, "Episode"), | ||||||
|  |                 GameCount = GetCount(counts, "Game"), | ||||||
|  |                 MovieCount = GetCount(counts, "Movie"), | ||||||
|  |                 MusicVideoCount = GetCount(counts, "MusicVideo"), | ||||||
|  |                 SeriesCount = GetCount(counts, "Series"), | ||||||
|  |                 SongCount = GetCount(counts, "Audio"), | ||||||
|  |                 TrailerCount = GetCount(counts, "Trailer"), | ||||||
|  |                 TotalCount = GetCount(counts, "Total") | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Gets the count. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="counts">The counts.</param> | ||||||
|  |         /// <param name="key">The key.</param> | ||||||
|  |         /// <returns>System.Int32.</returns> | ||||||
|  |         internal static int GetCount(Dictionary<string, int> counts, string key) | ||||||
|  |         { | ||||||
|  |             int count; | ||||||
|  | 
 | ||||||
|  |             if (counts.TryGetValue(key, out count)) | ||||||
|  |             { | ||||||
|  |                 return count; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Sets the item counts. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="userId">The user id.</param> | ||||||
|  |         /// <param name="media">The media.</param> | ||||||
|  |         /// <param name="names">The names.</param> | ||||||
|  |         /// <param name="masterDictionary">The master dictionary.</param> | ||||||
|  |         internal static void SetItemCounts(Guid? userId, BaseItem media, List<string> names, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 Dictionary<Guid, Dictionary<string, int>> libraryCounts; | ||||||
|  | 
 | ||||||
|  |                 if (!masterDictionary.TryGetValue(name, out libraryCounts)) | ||||||
|  |                 { | ||||||
|  |                     libraryCounts = new Dictionary<Guid, Dictionary<string, int>>(); | ||||||
|  |                     masterDictionary.Add(name, libraryCounts); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 var userLibId = userId ?? Guid.Empty; | ||||||
|  |                 Dictionary<string, int> userDictionary; | ||||||
|  | 
 | ||||||
|  |                 if (!libraryCounts.TryGetValue(userLibId, out userDictionary)) | ||||||
|  |                 { | ||||||
|  |                     userDictionary = new Dictionary<string, int>(); | ||||||
|  |                     libraryCounts.Add(userLibId, userDictionary); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 AddToDictionary(media, userDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,45 @@ | |||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using System; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class GameGenresPostScanTask | ||||||
|  |     /// </summary> | ||||||
|  |     public class GameGenresPostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="GameGenresPostScanTask"/> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         /// <param name="userManager">The user manager.</param> | ||||||
|  |         public GameGenresPostScanTask(ILibraryManager libraryManager, IUserManager userManager) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             return _libraryManager.ValidateGameGenres(cancellationToken, progress); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,132 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     class GameGenresValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly LibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public GameGenresValidator(LibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren.OfType<Game>().ToList(); | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<Game>>(i.Id, i.RootFolder.GetRecursiveChildren(i).OfType<Game>().ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var allLibraryItems = allItems; | ||||||
|  | 
 | ||||||
|  |             var masterDictionary = new Dictionary<string, Dictionary<Guid, Dictionary<string, int>>>(StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |             // Populate counts of items | ||||||
|  |             SetItemCounts(null, allLibraryItems, masterDictionary); | ||||||
|  | 
 | ||||||
|  |             progress.Report(2); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var lib in userLibraries) | ||||||
|  |             { | ||||||
|  |                 SetItemCounts(lib.Item1, lib.Item2, masterDictionary); | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= userLibraries.Count; | ||||||
|  |                 percent *= 8; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(10); | ||||||
|  | 
 | ||||||
|  |             var names = masterDictionary.Keys.ToList(); | ||||||
|  |             numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await UpdateItemByNameCounts(name, cancellationToken, masterDictionary[name]).ConfigureAwait(false); | ||||||
|  |                 } | ||||||
|  |                 catch (Exception ex) | ||||||
|  |                 { | ||||||
|  |                     _logger.ErrorException("Error updating counts for {0}", ex, name); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= names.Count; | ||||||
|  |                 percent *= 90; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent + 10); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task UpdateItemByNameCounts(string name, CancellationToken cancellationToken, Dictionary<Guid, Dictionary<string, int>> counts) | ||||||
|  |         { | ||||||
|  |             var itemByName = await _libraryManager.GetGameGenre(name, cancellationToken, true, true).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             foreach (var libraryId in counts.Keys.ToList()) | ||||||
|  |             { | ||||||
|  |                 var itemCounts = CountHelpers.GetCounts(counts[libraryId]); | ||||||
|  | 
 | ||||||
|  |                 if (libraryId == Guid.Empty) | ||||||
|  |                 { | ||||||
|  |                     itemByName.ItemCounts = itemCounts; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     itemByName.UserItemCounts[libraryId] = itemCounts; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void SetItemCounts(Guid? userId, IEnumerable<BaseItem> allItems, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var media in allItems) | ||||||
|  |             { | ||||||
|  |                 var names = media | ||||||
|  |                     .Genres | ||||||
|  |                     .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                     .ToList(); | ||||||
|  | 
 | ||||||
|  |                 CountHelpers.SetItemCounts(userId, media, names, masterDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,42 @@ | |||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using System; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     public class GenresPostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="ArtistsPostScanTask" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         /// <param name="userManager">The user manager.</param> | ||||||
|  |         public GenresPostScanTask(ILibraryManager libraryManager, IUserManager userManager) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             return _libraryManager.ValidateGenres(cancellationToken, progress); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,135 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     class GenresValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly LibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public GenresValidator(LibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren | ||||||
|  |                 .Where(i => !(i is IHasMusicGenres) && !(i is Game)) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<BaseItem>>(i.Id, i.RootFolder.GetRecursiveChildren(i).Where(m => !(m is IHasMusicGenres) && !(m is Game)).ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var allLibraryItems = allItems; | ||||||
|  | 
 | ||||||
|  |             var masterDictionary = new Dictionary<string, Dictionary<Guid, Dictionary<string, int>>>(StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |             // Populate counts of items | ||||||
|  |             SetItemCounts(null, allLibraryItems, masterDictionary); | ||||||
|  | 
 | ||||||
|  |             progress.Report(2); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var lib in userLibraries) | ||||||
|  |             { | ||||||
|  |                 SetItemCounts(lib.Item1, lib.Item2, masterDictionary); | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= userLibraries.Count; | ||||||
|  |                 percent *= 8; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(10); | ||||||
|  | 
 | ||||||
|  |             var names = masterDictionary.Keys.ToList(); | ||||||
|  |             numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await UpdateItemByNameCounts(name, cancellationToken, masterDictionary[name]).ConfigureAwait(false); | ||||||
|  |                 } | ||||||
|  |                 catch (Exception ex) | ||||||
|  |                 { | ||||||
|  |                     _logger.ErrorException("Error updating counts for {0}", ex, name); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= names.Count; | ||||||
|  |                 percent *= 90; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent + 10); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task UpdateItemByNameCounts(string name, CancellationToken cancellationToken, Dictionary<Guid, Dictionary<string, int>> counts) | ||||||
|  |         { | ||||||
|  |             var itemByName = await _libraryManager.GetGenre(name, cancellationToken, true, true).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             foreach (var libraryId in counts.Keys.ToList()) | ||||||
|  |             { | ||||||
|  |                 var itemCounts = CountHelpers.GetCounts(counts[libraryId]); | ||||||
|  | 
 | ||||||
|  |                 if (libraryId == Guid.Empty) | ||||||
|  |                 { | ||||||
|  |                     itemByName.ItemCounts = itemCounts; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     itemByName.UserItemCounts[libraryId] = itemCounts; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void SetItemCounts(Guid? userId, IEnumerable<BaseItem> allItems, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var media in allItems) | ||||||
|  |             { | ||||||
|  |                 var names = media | ||||||
|  |                     .Genres | ||||||
|  |                     .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                     .ToList(); | ||||||
|  | 
 | ||||||
|  |                 CountHelpers.SetItemCounts(userId, media, names, masterDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,45 @@ | |||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using System; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class MusicGenresPostScanTask | ||||||
|  |     /// </summary> | ||||||
|  |     public class MusicGenresPostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="ArtistsPostScanTask" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         /// <param name="userManager">The user manager.</param> | ||||||
|  |         public MusicGenresPostScanTask(ILibraryManager libraryManager, IUserManager userManager) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             return _libraryManager.ValidateMusicGenres(cancellationToken, progress); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,135 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Entities.Audio; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     class MusicGenresValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly LibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public MusicGenresValidator(LibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren | ||||||
|  |                 .Where(i => i is IHasMusicGenres) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<BaseItem>>(i.Id, i.RootFolder.GetRecursiveChildren(i).Where(m => m is IHasMusicGenres).ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var allLibraryItems = allItems; | ||||||
|  | 
 | ||||||
|  |             var masterDictionary = new Dictionary<string, Dictionary<Guid, Dictionary<string, int>>>(StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |             // Populate counts of items | ||||||
|  |             SetItemCounts(null, allLibraryItems, masterDictionary); | ||||||
|  | 
 | ||||||
|  |             progress.Report(2); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var lib in userLibraries) | ||||||
|  |             { | ||||||
|  |                 SetItemCounts(lib.Item1, lib.Item2, masterDictionary); | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= userLibraries.Count; | ||||||
|  |                 percent *= 8; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(10); | ||||||
|  | 
 | ||||||
|  |             var names = masterDictionary.Keys.ToList(); | ||||||
|  |             numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await UpdateItemByNameCounts(name, cancellationToken, masterDictionary[name]).ConfigureAwait(false); | ||||||
|  |                 } | ||||||
|  |                 catch (Exception ex) | ||||||
|  |                 { | ||||||
|  |                     _logger.ErrorException("Error updating counts for {0}", ex, name); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= names.Count; | ||||||
|  |                 percent *= 90; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent + 10); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task UpdateItemByNameCounts(string name, CancellationToken cancellationToken, Dictionary<Guid, Dictionary<string, int>> counts) | ||||||
|  |         { | ||||||
|  |             var itemByName = await _libraryManager.GetMusicGenre(name, cancellationToken, true, true).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             foreach (var libraryId in counts.Keys.ToList()) | ||||||
|  |             { | ||||||
|  |                 var itemCounts = CountHelpers.GetCounts(counts[libraryId]); | ||||||
|  | 
 | ||||||
|  |                 if (libraryId == Guid.Empty) | ||||||
|  |                 { | ||||||
|  |                     itemByName.ItemCounts = itemCounts; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     itemByName.UserItemCounts[libraryId] = itemCounts; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void SetItemCounts(Guid? userId, IEnumerable<BaseItem> allItems, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var media in allItems) | ||||||
|  |             { | ||||||
|  |                 var names = media | ||||||
|  |                     .Genres | ||||||
|  |                     .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                     .ToList(); | ||||||
|  | 
 | ||||||
|  |                 CountHelpers.SetItemCounts(userId, media, names, masterDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,137 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     class PeoplePostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public PeoplePostScanTask(ILibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren.ToList(); | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<BaseItem>>(i.Id, i.RootFolder.GetRecursiveChildren(i).ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var allLibraryItems = allItems; | ||||||
|  | 
 | ||||||
|  |             var masterDictionary = new Dictionary<string, Dictionary<Guid, Dictionary<string, int>>>(StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |             // Populate counts of items | ||||||
|  |             SetItemCounts(null, allLibraryItems, masterDictionary); | ||||||
|  | 
 | ||||||
|  |             progress.Report(2); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var lib in userLibraries) | ||||||
|  |             { | ||||||
|  |                 cancellationToken.ThrowIfCancellationRequested(); | ||||||
|  | 
 | ||||||
|  |                 SetItemCounts(lib.Item1, lib.Item2, masterDictionary); | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= userLibraries.Count; | ||||||
|  |                 percent *= 8; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(10); | ||||||
|  | 
 | ||||||
|  |             var names = masterDictionary.Keys.ToList(); | ||||||
|  |             numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 cancellationToken.ThrowIfCancellationRequested(); | ||||||
|  |                  | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await UpdateItemByNameCounts(name, masterDictionary[name]).ConfigureAwait(false); | ||||||
|  |                 } | ||||||
|  |                 catch (Exception ex) | ||||||
|  |                 { | ||||||
|  |                     _logger.ErrorException("Error updating counts for {0}", ex, name); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= names.Count; | ||||||
|  |                 percent *= 90; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent + 10); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task UpdateItemByNameCounts(string name, Dictionary<Guid, Dictionary<string, int>> counts) | ||||||
|  |         { | ||||||
|  |             var itemByName = await _libraryManager.GetPerson(name).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             foreach (var libraryId in counts.Keys.ToList()) | ||||||
|  |             { | ||||||
|  |                 var itemCounts = CountHelpers.GetCounts(counts[libraryId]); | ||||||
|  | 
 | ||||||
|  |                 if (libraryId == Guid.Empty) | ||||||
|  |                 { | ||||||
|  |                     itemByName.ItemCounts = itemCounts; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     itemByName.UserItemCounts[libraryId] = itemCounts; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void SetItemCounts(Guid? userId, IEnumerable<BaseItem> allItems, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var media in allItems) | ||||||
|  |             { | ||||||
|  |                 var names = media | ||||||
|  |                     .People.Select(i => i.Name) | ||||||
|  |                     .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                     .ToList(); | ||||||
|  | 
 | ||||||
|  |                 CountHelpers.SetItemCounts(userId, media, names, masterDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,45 @@ | |||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using System; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     /// <summary> | ||||||
|  |     /// Class MusicGenresPostScanTask | ||||||
|  |     /// </summary> | ||||||
|  |     public class StudiosPostScanTask : ILibraryPostScanTask | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Initializes a new instance of the <see cref="ArtistsPostScanTask" /> class. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="libraryManager">The library manager.</param> | ||||||
|  |         /// <param name="userManager">The user manager.</param> | ||||||
|  |         public StudiosPostScanTask(ILibraryManager libraryManager, IUserManager userManager) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             return _libraryManager.ValidateStudios(cancellationToken, progress); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -0,0 +1,132 @@ | |||||||
|  | using MediaBrowser.Controller.Entities; | ||||||
|  | using MediaBrowser.Controller.Library; | ||||||
|  | using MediaBrowser.Model.Logging; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Threading; | ||||||
|  | using System.Threading.Tasks; | ||||||
|  | 
 | ||||||
|  | namespace MediaBrowser.Server.Implementations.Library.Validators | ||||||
|  | { | ||||||
|  |     class StudiosValidator | ||||||
|  |     { | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _library manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly LibraryManager _libraryManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _user manager | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly IUserManager _userManager; | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// The _logger | ||||||
|  |         /// </summary> | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public StudiosValidator(LibraryManager libraryManager, IUserManager userManager, ILogger logger) | ||||||
|  |         { | ||||||
|  |             _libraryManager = libraryManager; | ||||||
|  |             _userManager = userManager; | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         /// <summary> | ||||||
|  |         /// Runs the specified progress. | ||||||
|  |         /// </summary> | ||||||
|  |         /// <param name="progress">The progress.</param> | ||||||
|  |         /// <param name="cancellationToken">The cancellation token.</param> | ||||||
|  |         /// <returns>Task.</returns> | ||||||
|  |         public async Task Run(IProgress<double> progress, CancellationToken cancellationToken) | ||||||
|  |         { | ||||||
|  |             var allItems = _libraryManager.RootFolder.RecursiveChildren.ToList(); | ||||||
|  | 
 | ||||||
|  |             var userLibraries = _userManager.Users | ||||||
|  |                 .Select(i => new Tuple<Guid, List<BaseItem>>(i.Id, i.RootFolder.GetRecursiveChildren(i).ToList())) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             var allLibraryItems = allItems; | ||||||
|  | 
 | ||||||
|  |             var masterDictionary = new Dictionary<string, Dictionary<Guid, Dictionary<string, int>>>(StringComparer.OrdinalIgnoreCase); | ||||||
|  | 
 | ||||||
|  |             // Populate counts of items | ||||||
|  |             SetItemCounts(null, allLibraryItems, masterDictionary); | ||||||
|  | 
 | ||||||
|  |             progress.Report(2); | ||||||
|  | 
 | ||||||
|  |             var numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var lib in userLibraries) | ||||||
|  |             { | ||||||
|  |                 SetItemCounts(lib.Item1, lib.Item2, masterDictionary); | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= userLibraries.Count; | ||||||
|  |                 percent *= 8; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(10); | ||||||
|  | 
 | ||||||
|  |             var names = masterDictionary.Keys.ToList(); | ||||||
|  |             numComplete = 0; | ||||||
|  | 
 | ||||||
|  |             foreach (var name in names) | ||||||
|  |             { | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     await UpdateItemByNameCounts(name, cancellationToken, masterDictionary[name]).ConfigureAwait(false); | ||||||
|  |                 } | ||||||
|  |                 catch (Exception ex) | ||||||
|  |                 { | ||||||
|  |                     _logger.ErrorException("Error updating counts for {0}", ex, name); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 numComplete++; | ||||||
|  |                 double percent = numComplete; | ||||||
|  |                 percent /= names.Count; | ||||||
|  |                 percent *= 90; | ||||||
|  | 
 | ||||||
|  |                 progress.Report(percent + 10); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             progress.Report(100); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task UpdateItemByNameCounts(string name, CancellationToken cancellationToken, Dictionary<Guid, Dictionary<string, int>> counts) | ||||||
|  |         { | ||||||
|  |             var itemByName = await _libraryManager.GetStudio(name, cancellationToken, true, true).ConfigureAwait(false); | ||||||
|  | 
 | ||||||
|  |             foreach (var libraryId in counts.Keys.ToList()) | ||||||
|  |             { | ||||||
|  |                 var itemCounts = CountHelpers.GetCounts(counts[libraryId]); | ||||||
|  | 
 | ||||||
|  |                 if (libraryId == Guid.Empty) | ||||||
|  |                 { | ||||||
|  |                     itemByName.ItemCounts = itemCounts; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     itemByName.UserItemCounts[libraryId] = itemCounts; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void SetItemCounts(Guid? userId, IEnumerable<BaseItem> allItems, Dictionary<string, Dictionary<Guid, Dictionary<string, int>>> masterDictionary) | ||||||
|  |         { | ||||||
|  |             foreach (var media in allItems) | ||||||
|  |             { | ||||||
|  |                 var names = media | ||||||
|  |                     .Studios | ||||||
|  |                     .Distinct(StringComparer.OrdinalIgnoreCase) | ||||||
|  |                     .ToList(); | ||||||
|  | 
 | ||||||
|  |                 CountHelpers.SetItemCounts(userId, media, names, masterDictionary); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @ -146,6 +146,18 @@ | |||||||
|     <Compile Include="Library\Resolvers\TV\SeriesResolver.cs" /> |     <Compile Include="Library\Resolvers\TV\SeriesResolver.cs" /> | ||||||
|     <Compile Include="Library\Resolvers\VideoResolver.cs" /> |     <Compile Include="Library\Resolvers\VideoResolver.cs" /> | ||||||
|     <Compile Include="Library\UserManager.cs" /> |     <Compile Include="Library\UserManager.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\ArtistsPostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\ArtistsValidator.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\CountHelpers.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\GameGenresPostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\GameGenresValidator.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\GenresPostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\GenresValidator.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\MusicGenresPostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\MusicGenresValidator.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\PeoplePostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\StudiosPostScanTask.cs" /> | ||||||
|  |     <Compile Include="Library\Validators\StudiosValidator.cs" /> | ||||||
|     <Compile Include="Localization\LocalizationManager.cs" /> |     <Compile Include="Localization\LocalizationManager.cs" /> | ||||||
|     <Compile Include="MediaEncoder\MediaEncoder.cs" /> |     <Compile Include="MediaEncoder\MediaEncoder.cs" /> | ||||||
|     <Compile Include="Persistence\SqliteChapterRepository.cs" /> |     <Compile Include="Persistence\SqliteChapterRepository.cs" /> | ||||||
| @ -155,7 +167,6 @@ | |||||||
|     <Compile Include="Properties\AssemblyInfo.cs" /> |     <Compile Include="Properties\AssemblyInfo.cs" /> | ||||||
|     <Compile Include="Providers\ImageSaver.cs" /> |     <Compile Include="Providers\ImageSaver.cs" /> | ||||||
|     <Compile Include="Providers\ProviderManager.cs" /> |     <Compile Include="Providers\ProviderManager.cs" /> | ||||||
|     <Compile Include="ScheduledTasks\ArtistValidationTask.cs" /> |  | ||||||
|     <Compile Include="ScheduledTasks\PeopleValidationTask.cs" /> |     <Compile Include="ScheduledTasks\PeopleValidationTask.cs" /> | ||||||
|     <Compile Include="ScheduledTasks\ChapterImagesTask.cs" /> |     <Compile Include="ScheduledTasks\ChapterImagesTask.cs" /> | ||||||
|     <Compile Include="ScheduledTasks\PluginUpdateTask.cs" /> |     <Compile Include="ScheduledTasks\PluginUpdateTask.cs" /> | ||||||
|  | |||||||
| @ -189,7 +189,11 @@ namespace MediaBrowser.Server.Implementations.Providers | |||||||
| 
 | 
 | ||||||
|             cancellationToken.ThrowIfCancellationRequested(); |             cancellationToken.ThrowIfCancellationRequested(); | ||||||
| 
 | 
 | ||||||
|  |             // Don't clog up the log with these providers | ||||||
|  |             if (!(provider is IDynamicInfoProvider)) | ||||||
|  |             { | ||||||
|                 _logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name ?? "--Unknown--"); |                 _logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name ?? "--Unknown--"); | ||||||
|  |             } | ||||||
| 
 | 
 | ||||||
|             // This provides the ability to cancel just this one provider |             // This provides the ability to cancel just this one provider | ||||||
|             var innerCancellationTokenSource = new CancellationTokenSource(); |             var innerCancellationTokenSource = new CancellationTokenSource(); | ||||||
|  | |||||||
| @ -1,81 +0,0 @@ | |||||||
| using MediaBrowser.Common.ScheduledTasks; |  | ||||||
| using MediaBrowser.Controller.Library; |  | ||||||
| using System; |  | ||||||
| using System.Collections.Generic; |  | ||||||
| using System.Threading; |  | ||||||
| using System.Threading.Tasks; |  | ||||||
| 
 |  | ||||||
| namespace MediaBrowser.Server.Implementations.ScheduledTasks |  | ||||||
| { |  | ||||||
|     public class ArtistValidationTask |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// The _library manager |  | ||||||
|         /// </summary> |  | ||||||
|         private readonly ILibraryManager _libraryManager; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Initializes a new instance of the <see cref="PeopleValidationTask" /> class. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="libraryManager">The library manager.</param> |  | ||||||
|         public ArtistValidationTask(ILibraryManager libraryManager) |  | ||||||
|         { |  | ||||||
|             _libraryManager = libraryManager; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Creates the triggers that define when the task will run |  | ||||||
|         /// </summary> |  | ||||||
|         /// <returns>IEnumerable{BaseTaskTrigger}.</returns> |  | ||||||
|         public IEnumerable<ITaskTrigger> GetDefaultTriggers() |  | ||||||
|         { |  | ||||||
|             return new ITaskTrigger[] |  | ||||||
|                 { |  | ||||||
|                     new DailyTrigger { TimeOfDay = TimeSpan.FromHours(5) }, |  | ||||||
| 
 |  | ||||||
|                     new IntervalTrigger{ Interval = TimeSpan.FromHours(12)} |  | ||||||
|                 }; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Returns the task to be executed |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="cancellationToken">The cancellation token.</param> |  | ||||||
|         /// <param name="progress">The progress.</param> |  | ||||||
|         /// <returns>Task.</returns> |  | ||||||
|         public Task Execute(CancellationToken cancellationToken, IProgress<double> progress) |  | ||||||
|         { |  | ||||||
|             return _libraryManager.ValidateArtists(cancellationToken, progress); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the name of the task |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The name.</value> |  | ||||||
|         public string Name |  | ||||||
|         { |  | ||||||
|             get { return "Refresh music artists"; } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the description. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The description.</value> |  | ||||||
|         public string Description |  | ||||||
|         { |  | ||||||
|             get { return "Updates metadata for music artists in your media library."; } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Gets the category. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <value>The category.</value> |  | ||||||
|         public string Category |  | ||||||
|         { |  | ||||||
|             get |  | ||||||
|             { |  | ||||||
|                 return "Library"; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -85,11 +85,6 @@ namespace MediaBrowser.Server.Implementations.Session | |||||||
|             get { return _activeConnections.Values.OrderByDescending(c => c.LastActivityDate).ToList(); } |             get { return _activeConnections.Values.OrderByDescending(c => c.LastActivityDate).ToList(); } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         /// <summary> |  | ||||||
|         /// The _true task result |  | ||||||
|         /// </summary> |  | ||||||
|         private readonly Task _trueTaskResult = Task.FromResult(true); |  | ||||||
| 
 |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         /// Logs the user activity. |         /// Logs the user activity. | ||||||
|         /// </summary> |         /// </summary> | ||||||
| @ -339,6 +334,7 @@ namespace MediaBrowser.Server.Implementations.Session | |||||||
|                 // If the client isn't able to report this, then we'll just have to make an assumption |                 // If the client isn't able to report this, then we'll just have to make an assumption | ||||||
|                 data.PlayCount++; |                 data.PlayCount++; | ||||||
|                 data.Played = true; |                 data.Played = true; | ||||||
|  |                 data.PlaybackPositionTicks = 0; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             await _userDataRepository.SaveUserData(user.Id, key, data, CancellationToken.None).ConfigureAwait(false); |             await _userDataRepository.SaveUserData(user.Id, key, data, CancellationToken.None).ConfigureAwait(false); | ||||||
|  | |||||||
| @ -336,11 +336,6 @@ | |||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <Resource Include="Resources\Images\audio.png" /> |     <Resource Include="Resources\Images\audio.png" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |  | ||||||
|     <Resource Include="Resources\Images\starEmpty.png" /> |  | ||||||
|     <Resource Include="Resources\Images\starFull.png" /> |  | ||||||
|     <Resource Include="Resources\Images\starHalf.png" /> |  | ||||||
|   </ItemGroup> |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <Resource Include="Resources\Images\artist.png" /> |     <Resource Include="Resources\Images\artist.png" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
| @ -358,8 +353,6 @@ | |||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <Resource Include="Resources\Images\folder.jpg" /> |     <Resource Include="Resources\Images\folder.jpg" /> | ||||||
|     <Resource Include="Resources\Images\mblogoblackfull.png" /> |  | ||||||
|     <Resource Include="Resources\Images\mblogowhitefull.png" /> |  | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <BootstrapperPackage Include=".NETFramework,Version=v4.5"> |     <BootstrapperPackage Include=".NETFramework,Version=v4.5"> | ||||||
|  | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 58 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 58 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 1.8 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.4 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.0 KiB | 
| @ -3171,144 +3171,6 @@ MediaBrowser.ApiClient = function ($, navigator, JSON, WebSocket, setTimeout, wi | |||||||
|             }); |             }); | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         /** |  | ||||||
|             Gets a variety of item counts that a person appears in |  | ||||||
|         */ |  | ||||||
|         self.getPersonItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("Persons/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|             Gets a variety of item counts that a genre appears in |  | ||||||
|         */ |  | ||||||
|         self.getGenreItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("Genres/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         self.getMusicGenreItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("MusicGenres/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         self.getGameGenreItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("GameGenres/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|             Gets a variety of item counts that an artist appears in |  | ||||||
|         */ |  | ||||||
|         self.getArtistItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("Artists/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         /** |  | ||||||
|             Gets a variety of item counts that a studio appears in |  | ||||||
|         */ |  | ||||||
|         self.getStudioItemCounts = function (userId, name) { |  | ||||||
| 
 |  | ||||||
|             if (!userId) { |  | ||||||
|                 throw new Error("null userId"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!name) { |  | ||||||
|                 throw new Error("null name"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var url = self.getUrl("Studios/" + self.encodeName(name) + "/Counts", { |  | ||||||
|                 userId: userId |  | ||||||
|             }); |  | ||||||
| 
 |  | ||||||
|             return self.ajax({ |  | ||||||
|                 type: "GET", |  | ||||||
|                 url: url, |  | ||||||
|                 dataType: "json" |  | ||||||
|             }); |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         /** |         /** | ||||||
|          * Clears a user's personal rating for an item |          * Clears a user's personal rating for an item | ||||||
|          * @param {String} userId |          * @param {String} userId | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| <?xml version="1.0" encoding="utf-8"?> | <?xml version="1.0" encoding="utf-8"?> | ||||||
| <packages> | <packages> | ||||||
|   <package id="MediaBrowser.ApiClient.Javascript" version="3.0.174" targetFramework="net45" /> |   <package id="MediaBrowser.ApiClient.Javascript" version="3.0.175" targetFramework="net45" /> | ||||||
|   <package id="ServiceStack.Common" version="3.9.58" targetFramework="net45" /> |   <package id="ServiceStack.Common" version="3.9.58" targetFramework="net45" /> | ||||||
|   <package id="ServiceStack.Text" version="3.9.58" targetFramework="net45" /> |   <package id="ServiceStack.Text" version="3.9.58" targetFramework="net45" /> | ||||||
| </packages> | </packages> | ||||||
| @ -237,4 +237,7 @@ Global | |||||||
| 	GlobalSection(SolutionProperties) = preSolution | 	GlobalSection(SolutionProperties) = preSolution | ||||||
| 		HideSolutionNode = FALSE | 		HideSolutionNode = FALSE | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
|  | 	GlobalSection(Performance) = preSolution | ||||||
|  | 		HasPerformanceSessions = true | ||||||
|  | 	EndGlobalSection | ||||||
| EndGlobal | EndGlobal | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user