aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorEric Reed <ebr@mediabrowser3.com>2014-01-29 12:22:53 -0500
committerEric Reed <ebr@mediabrowser3.com>2014-01-29 12:22:53 -0500
commitab5145bcd794ee579e0064c9c0f9d15b13f72fb4 (patch)
treed68cdabe4302cc4dda8194ee840060f48d9fa714
parent1f31c8dbfca9ca9134d9ee779256c435f913689b (diff)
parentc0f606683a045e463f518ec466b9fc9a85f8d4fd (diff)
Merge branch 'master' of https://github.com/MediaBrowser/MediaBrowser
-rw-r--r--MediaBrowser.Api/Images/ImageService.cs9
-rw-r--r--MediaBrowser.Api/Images/RemoteImageService.cs22
-rw-r--r--MediaBrowser.Api/ItemRefreshService.cs49
-rw-r--r--MediaBrowser.Api/Library/LibraryStructureService.cs63
-rw-r--r--MediaBrowser.Controller/Drawing/ImageExtensions.cs8
-rw-r--r--MediaBrowser.Controller/Drawing/ImageFormat.cs11
-rw-r--r--MediaBrowser.Controller/Entities/BaseItem.cs122
-rw-r--r--MediaBrowser.Controller/Entities/Folder.cs13
-rw-r--r--MediaBrowser.Controller/Entities/IHasImages.cs27
-rw-r--r--MediaBrowser.Controller/Entities/IHasScreenshots.cs6
-rw-r--r--MediaBrowser.Controller/Entities/Movies/Movie.cs19
-rw-r--r--MediaBrowser.Controller/Entities/User.cs20
-rw-r--r--MediaBrowser.Controller/Entities/Video.cs16
-rw-r--r--MediaBrowser.Controller/IO/IDirectoryWatchers.cs29
-rw-r--r--MediaBrowser.Controller/Library/ILibraryMonitor.cs36
-rw-r--r--MediaBrowser.Controller/LiveTv/ILiveTvRecording.cs5
-rw-r--r--MediaBrowser.Controller/LiveTv/StreamResponseInfo.cs5
-rw-r--r--MediaBrowser.Controller/MediaBrowser.Controller.csproj12
-rw-r--r--MediaBrowser.Controller/Persistence/IItemRepository.cs17
-rw-r--r--MediaBrowser.Controller/Providers/IHasMetadata.cs31
-rw-r--r--MediaBrowser.Controller/Providers/IImageProvider.cs28
-rw-r--r--MediaBrowser.Controller/Providers/ILocalImageProvider.cs66
-rw-r--r--MediaBrowser.Controller/Providers/IMetadataProvider.cs68
-rw-r--r--MediaBrowser.Controller/Providers/IMetadataService.cs38
-rw-r--r--MediaBrowser.Controller/Providers/IProviderManager.cs20
-rw-r--r--MediaBrowser.Controller/Providers/IProviderRepository.cs48
-rw-r--r--MediaBrowser.Controller/Providers/IRemoteImageProvider.cs48
-rw-r--r--MediaBrowser.Controller/Providers/ItemId.cs35
-rw-r--r--MediaBrowser.Controller/Providers/MetadataRefreshOptions.cs49
-rw-r--r--MediaBrowser.Controller/Providers/MetadataStatus.cs122
-rw-r--r--MediaBrowser.Model/Configuration/ServerConfiguration.cs4
-rw-r--r--MediaBrowser.Model/Dto/BaseItemDto.cs6
-rw-r--r--MediaBrowser.Model/Entities/IHasProviderIds.cs11
-rw-r--r--MediaBrowser.Model/Providers/ImageProviderInfo.cs6
-rw-r--r--MediaBrowser.Model/Querying/ItemFields.cs5
-rw-r--r--MediaBrowser.Providers/All/LocalImageProvider.cs327
-rw-r--r--MediaBrowser.Providers/BaseXmlProvider.cs34
-rw-r--r--MediaBrowser.Providers/CollectionFolderImageProvider.cs4
-rw-r--r--MediaBrowser.Providers/GameGenres/GameGenreImageProvider.cs (renamed from MediaBrowser.Providers/ImagesByName/GameGenresManualImageProvider.cs)29
-rw-r--r--MediaBrowser.Providers/GameGenres/GameGenreMetadataService.cs42
-rw-r--r--MediaBrowser.Providers/Genres/GenreImageProvider.cs (renamed from MediaBrowser.Providers/ImagesByName/GenresManualImageProvider.cs)30
-rw-r--r--MediaBrowser.Providers/Genres/GenreMetadataService.cs42
-rw-r--r--MediaBrowser.Providers/ImageFromMediaLocationProvider.cs11
-rw-r--r--MediaBrowser.Providers/ImagesByName/GameGenreImageProvider.cs160
-rw-r--r--MediaBrowser.Providers/ImagesByName/GenreImageProvider.cs160
-rw-r--r--MediaBrowser.Providers/ImagesByName/MusicGenreImageProvider.cs161
-rw-r--r--MediaBrowser.Providers/ImagesByName/StudioImageProvider.cs160
-rw-r--r--MediaBrowser.Providers/LiveTv/ChannelMetadataService.cs37
-rw-r--r--MediaBrowser.Providers/LiveTv/ChannelProviderFromXml.cs91
-rw-r--r--MediaBrowser.Providers/LiveTv/ChannelXmlProvider.cs59
-rw-r--r--MediaBrowser.Providers/LiveTv/ProgramMetadataService.cs41
-rw-r--r--MediaBrowser.Providers/Manager/ImageSaver.cs (renamed from MediaBrowser.Server.Implementations/Providers/ImageSaver.cs)27
-rw-r--r--MediaBrowser.Providers/Manager/ItemImageProvider.cs435
-rw-r--r--MediaBrowser.Providers/Manager/MetadataService.cs358
-rw-r--r--MediaBrowser.Providers/Manager/ProviderManager.cs (renamed from MediaBrowser.Server.Implementations/Providers/ProviderManager.cs)99
-rw-r--r--MediaBrowser.Providers/MediaBrowser.Providers.csproj39
-rw-r--r--MediaBrowser.Providers/Movies/ManualFanartMovieImageProvider.cs35
-rw-r--r--MediaBrowser.Providers/Movies/ManualMovieDbImageProvider.cs32
-rw-r--r--MediaBrowser.Providers/Movies/MovieDbPersonImageProvider.cs207
-rw-r--r--MediaBrowser.Providers/Movies/MovieDbPersonProvider.cs440
-rw-r--r--MediaBrowser.Providers/Movies/MovieDbProvider.cs3
-rw-r--r--MediaBrowser.Providers/Movies/PersonProviderFromXml.cs89
-rw-r--r--MediaBrowser.Providers/Music/ManualFanartAlbumProvider.cs32
-rw-r--r--MediaBrowser.Providers/Music/ManualFanartArtistProvider.cs35
-rw-r--r--MediaBrowser.Providers/Music/ManualLastFmImageProvider.cs37
-rw-r--r--MediaBrowser.Providers/Music/MusicBrainzAlbumProvider.cs15
-rw-r--r--MediaBrowser.Providers/MusicGenres/MusicGenreImageProvider.cs (renamed from MediaBrowser.Providers/ImagesByName/MusicGenresManualImageProvider.cs)29
-rw-r--r--MediaBrowser.Providers/MusicGenres/MusicGenreMetadataService.cs42
-rw-r--r--MediaBrowser.Providers/People/MovieDbPersonImageProvider.cs (renamed from MediaBrowser.Providers/Movies/ManualMovieDbPersonImageProvider.cs)32
-rw-r--r--MediaBrowser.Providers/People/MovieDbPersonProvider.cs289
-rw-r--r--MediaBrowser.Providers/People/PersonMetadataService.cs47
-rw-r--r--MediaBrowser.Providers/People/PersonXmlProvider.cs59
-rw-r--r--MediaBrowser.Providers/People/TvdbPersonImageProvider.cs (renamed from MediaBrowser.Providers/TV/ManualTvdbPersonImageProvider.cs)34
-rw-r--r--MediaBrowser.Providers/ProviderUtils.cs126
-rw-r--r--MediaBrowser.Providers/Studios/StudioMetadataService.cs41
-rw-r--r--MediaBrowser.Providers/Studios/StudiosImageProvider.cs (renamed from MediaBrowser.Providers/ImagesByName/StudiosManualImageProvider.cs)29
-rw-r--r--MediaBrowser.Providers/TV/ManualFanartSeasonProvider.cs32
-rw-r--r--MediaBrowser.Providers/TV/ManualFanartSeriesProvider.cs36
-rw-r--r--MediaBrowser.Providers/TV/ManualTvdbEpisodeImageProvider.cs29
-rw-r--r--MediaBrowser.Providers/TV/ManualTvdbSeasonImageProvider.cs33
-rw-r--r--MediaBrowser.Providers/TV/ManualTvdbSeriesImageProvider.cs33
-rw-r--r--MediaBrowser.Providers/TV/TvdbPersonImageProvider.cs98
-rw-r--r--MediaBrowser.Providers/VirtualItemImageValidator.cs8
-rw-r--r--MediaBrowser.Server.Implementations/Drawing/ImageProcessor.cs12
-rw-r--r--MediaBrowser.Server.Implementations/Dto/DtoService.cs5
-rw-r--r--MediaBrowser.Server.Implementations/FileOrganization/EpisodeFileOrganizer.cs20
-rw-r--r--MediaBrowser.Server.Implementations/FileOrganization/FileOrganizationService.cs16
-rw-r--r--MediaBrowser.Server.Implementations/FileOrganization/OrganizerScheduledTask.cs8
-rw-r--r--MediaBrowser.Server.Implementations/FileOrganization/TvFolderOrganizer.cs8
-rw-r--r--MediaBrowser.Server.Implementations/IO/LibraryMonitor.cs (renamed from MediaBrowser.Server.Implementations/IO/DirectoryWatchers.cs)194
-rw-r--r--MediaBrowser.Server.Implementations/Library/LibraryManager.cs16
-rw-r--r--MediaBrowser.Server.Implementations/Library/ResolverHelper.cs6
-rw-r--r--MediaBrowser.Server.Implementations/Library/UserManager.cs7
-rw-r--r--MediaBrowser.Server.Implementations/Library/Validators/GenresPostScanTask.cs1
-rw-r--r--MediaBrowser.Server.Implementations/Library/Validators/PeoplePostScanTask.cs10
-rw-r--r--MediaBrowser.Server.Implementations/LiveTv/ChannelImageProvider.cs129
-rw-r--r--MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs22
-rw-r--r--MediaBrowser.Server.Implementations/LiveTv/ProgramImageProvider.cs129
-rw-r--r--MediaBrowser.Server.Implementations/LiveTv/RecordingImageProvider.cs2
-rw-r--r--MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj4
-rw-r--r--MediaBrowser.Server.Implementations/Persistence/SqliteItemRepository.cs22
-rw-r--r--MediaBrowser.Server.Implementations/Persistence/SqliteProviderInfoRepository.cs212
-rw-r--r--MediaBrowser.ServerApplication/ApplicationHost.cs23
-rw-r--r--MediaBrowser.ServerApplication/LibraryExplorer.xaml.cs7
104 files changed, 3873 insertions, 2322 deletions
diff --git a/MediaBrowser.Api/Images/ImageService.cs b/MediaBrowser.Api/Images/ImageService.cs
index d7b59c920..c4981a7fa 100644
--- a/MediaBrowser.Api/Images/ImageService.cs
+++ b/MediaBrowser.Api/Images/ImageService.cs
@@ -66,7 +66,7 @@ namespace MediaBrowser.Api.Images
[ApiMember(Name = "Id", Description = "Item Id", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
public string Id { get; set; }
}
-
+
/// <summary>
/// Class UpdateItemImageIndex
/// </summary>
@@ -799,7 +799,12 @@ namespace MediaBrowser.Api.Images
await _providerManager.SaveImage(entity, memoryStream, mimeType, imageType, null, null, CancellationToken.None).ConfigureAwait(false);
- await entity.RefreshMetadata(CancellationToken.None, forceRefresh: true, forceSave: true, allowSlowProviders: false).ConfigureAwait(false);
+ await entity.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ImageRefreshMode = MetadataRefreshMode.None,
+ ForceSave = true
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
}
}
diff --git a/MediaBrowser.Api/Images/RemoteImageService.cs b/MediaBrowser.Api/Images/RemoteImageService.cs
index c18954e50..b84a8f4f7 100644
--- a/MediaBrowser.Api/Images/RemoteImageService.cs
+++ b/MediaBrowser.Api/Images/RemoteImageService.cs
@@ -9,13 +9,13 @@ using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
using ServiceStack;
+using ServiceStack.Text.Controller;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-using ServiceStack.Text.Controller;
namespace MediaBrowser.Api.Images
{
@@ -193,12 +193,7 @@ namespace MediaBrowser.Api.Images
private List<ImageProviderInfo> GetImageProviders(BaseItem item)
{
- return _providerManager.GetImageProviders(item).Select(i => new ImageProviderInfo
- {
- Name = i.Name,
- Priority = i.Priority
-
- }).ToList();
+ return _providerManager.GetImageProviderInfo(item).ToList();
}
public object Get(GetRemoteImages request)
@@ -229,7 +224,9 @@ namespace MediaBrowser.Api.Images
var result = new RemoteImageResult
{
TotalRecordCount = imagesList.Count,
- Providers = _providerManager.GetImageProviders(item).Select(i => i.Name).ToList()
+ Providers = images.Select(i => i.ProviderName)
+ .Distinct(StringComparer.OrdinalIgnoreCase)
+ .ToList()
};
if (request.StartIndex.HasValue)
@@ -284,8 +281,13 @@ namespace MediaBrowser.Api.Images
{
await _providerManager.SaveImage(item, request.ImageUrl, null, request.Type, null, CancellationToken.None).ConfigureAwait(false);
- await item.RefreshMetadata(CancellationToken.None, forceSave: true, allowSlowProviders: false)
- .ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = true,
+ ImageRefreshMode = MetadataRefreshMode.None,
+ MetadataRefreshMode = MetadataRefreshMode.None
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
/// <summary>
diff --git a/MediaBrowser.Api/ItemRefreshService.cs b/MediaBrowser.Api/ItemRefreshService.cs
index 1b8b49f98..a0055f4e6 100644
--- a/MediaBrowser.Api/ItemRefreshService.cs
+++ b/MediaBrowser.Api/ItemRefreshService.cs
@@ -2,6 +2,7 @@
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
using ServiceStack;
using System;
using System.Linq;
@@ -131,7 +132,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(cancellationToken, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -152,7 +157,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -173,7 +182,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -194,7 +207,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -215,7 +232,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -236,7 +257,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
}
catch (Exception ex)
{
@@ -266,7 +291,11 @@ namespace MediaBrowser.Api
try
{
- await item.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
if (item.IsFolder)
{
@@ -301,7 +330,11 @@ namespace MediaBrowser.Api
{
foreach (var child in collectionFolder.Children.ToList())
{
- await child.RefreshMetadata(CancellationToken.None, forceRefresh: request.Forced).ConfigureAwait(false);
+ await child.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = request.Forced,
+
+ }, CancellationToken.None).ConfigureAwait(false);
if (child.IsFolder)
{
diff --git a/MediaBrowser.Api/Library/LibraryStructureService.cs b/MediaBrowser.Api/Library/LibraryStructureService.cs
index 775907379..8ea472da3 100644
--- a/MediaBrowser.Api/Library/LibraryStructureService.cs
+++ b/MediaBrowser.Api/Library/LibraryStructureService.cs
@@ -167,6 +167,17 @@ namespace MediaBrowser.Api.Library
public bool RefreshLibrary { get; set; }
}
+ [Route("/Library/Changes/Path", "POST")]
+ public class ReportChangedPath : IReturnVoid
+ {
+ /// <summary>
+ /// Gets or sets the name.
+ /// </summary>
+ /// <value>The name.</value>
+ [ApiMember(Name = "Path", Description = "The path that was changed.", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "POST")]
+ public string Path { get; set; }
+ }
+
/// <summary>
/// Class LibraryStructureService
/// </summary>
@@ -187,7 +198,7 @@ namespace MediaBrowser.Api.Library
/// </summary>
private readonly ILibraryManager _libraryManager;
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly IFileSystem _fileSystem;
private readonly ILogger _logger;
@@ -199,7 +210,7 @@ namespace MediaBrowser.Api.Library
/// <param name="userManager">The user manager.</param>
/// <param name="libraryManager">The library manager.</param>
/// <exception cref="System.ArgumentNullException">appPaths</exception>
- public LibraryStructureService(IServerApplicationPaths appPaths, IUserManager userManager, ILibraryManager libraryManager, IDirectoryWatchers directoryWatchers, IFileSystem fileSystem, ILogger logger)
+ public LibraryStructureService(IServerApplicationPaths appPaths, IUserManager userManager, ILibraryManager libraryManager, ILibraryMonitor libraryMonitor, IFileSystem fileSystem, ILogger logger)
{
if (appPaths == null)
{
@@ -209,12 +220,27 @@ namespace MediaBrowser.Api.Library
_userManager = userManager;
_appPaths = appPaths;
_libraryManager = libraryManager;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_fileSystem = fileSystem;
_logger = logger;
}
/// <summary>
+ /// Posts the specified request.
+ /// </summary>
+ /// <param name="request">The request.</param>
+ /// <exception cref="System.ArgumentException">Please supply a Path</exception>
+ public void Post(ReportChangedPath request)
+ {
+ if (string.IsNullOrEmpty(request.Path))
+ {
+ throw new ArgumentException("Please supply a Path");
+ }
+
+ _libraryMonitor.ReportFileSystemChanged(request.Path);
+ }
+
+ /// <summary>
/// Gets the specified request.
/// </summary>
/// <param name="request">The request.</param>
@@ -270,8 +296,7 @@ namespace MediaBrowser.Api.Library
throw new ArgumentException("There is already a media collection with the name " + name + ".");
}
- _directoryWatchers.Stop();
- _directoryWatchers.TemporarilyIgnore(virtualFolderPath);
+ _libraryMonitor.Stop();
try
{
@@ -294,10 +319,8 @@ namespace MediaBrowser.Api.Library
// No need to start if scanning the library because it will handle it
if (!request.RefreshLibrary)
{
- _directoryWatchers.Start();
+ _libraryMonitor.Start();
}
-
- _directoryWatchers.RemoveTempIgnore(virtualFolderPath);
}
if (request.RefreshLibrary)
@@ -348,9 +371,7 @@ namespace MediaBrowser.Api.Library
throw new ArgumentException("There is already a media collection with the name " + newPath + ".");
}
- _directoryWatchers.Stop();
- _directoryWatchers.TemporarilyIgnore(currentPath);
- _directoryWatchers.TemporarilyIgnore(newPath);
+ _libraryMonitor.Stop();
try
{
@@ -376,11 +397,8 @@ namespace MediaBrowser.Api.Library
// No need to start if scanning the library because it will handle it
if (!request.RefreshLibrary)
{
- _directoryWatchers.Start();
+ _libraryMonitor.Start();
}
-
- _directoryWatchers.RemoveTempIgnore(currentPath);
- _directoryWatchers.RemoveTempIgnore(newPath);
}
if (request.RefreshLibrary)
@@ -420,8 +438,7 @@ namespace MediaBrowser.Api.Library
throw new DirectoryNotFoundException("The media folder does not exist");
}
- _directoryWatchers.Stop();
- _directoryWatchers.TemporarilyIgnore(path);
+ _libraryMonitor.Stop();
try
{
@@ -437,10 +454,8 @@ namespace MediaBrowser.Api.Library
// No need to start if scanning the library because it will handle it
if (!request.RefreshLibrary)
{
- _directoryWatchers.Start();
+ _libraryMonitor.Start();
}
-
- _directoryWatchers.RemoveTempIgnore(path);
}
if (request.RefreshLibrary)
@@ -460,7 +475,7 @@ namespace MediaBrowser.Api.Library
throw new ArgumentNullException("request");
}
- _directoryWatchers.Stop();
+ _libraryMonitor.Stop();
try
{
@@ -485,7 +500,7 @@ namespace MediaBrowser.Api.Library
// No need to start if scanning the library because it will handle it
if (!request.RefreshLibrary)
{
- _directoryWatchers.Start();
+ _libraryMonitor.Start();
}
}
@@ -506,7 +521,7 @@ namespace MediaBrowser.Api.Library
throw new ArgumentNullException("request");
}
- _directoryWatchers.Stop();
+ _libraryMonitor.Stop();
try
{
@@ -531,7 +546,7 @@ namespace MediaBrowser.Api.Library
// No need to start if scanning the library because it will handle it
if (!request.RefreshLibrary)
{
- _directoryWatchers.Start();
+ _libraryMonitor.Start();
}
}
diff --git a/MediaBrowser.Controller/Drawing/ImageExtensions.cs b/MediaBrowser.Controller/Drawing/ImageExtensions.cs
index 9dc58b3d2..c7e1968e7 100644
--- a/MediaBrowser.Controller/Drawing/ImageExtensions.cs
+++ b/MediaBrowser.Controller/Drawing/ImageExtensions.cs
@@ -18,17 +18,17 @@ namespace MediaBrowser.Controller.Drawing
/// <param name="image">The image.</param>
/// <param name="toStream">To stream.</param>
/// <param name="quality">The quality.</param>
- public static void Save(this Image image, ImageFormat outputFormat, Stream toStream, int quality)
+ public static void Save(this Image image, System.Drawing.Imaging.ImageFormat outputFormat, Stream toStream, int quality)
{
// Use special save methods for jpeg and png that will result in a much higher quality image
// All other formats use the generic Image.Save
- if (ImageFormat.Jpeg.Equals(outputFormat))
+ if (System.Drawing.Imaging.ImageFormat.Jpeg.Equals(outputFormat))
{
SaveAsJpeg(image, toStream, quality);
}
- else if (ImageFormat.Png.Equals(outputFormat))
+ else if (System.Drawing.Imaging.ImageFormat.Png.Equals(outputFormat))
{
- image.Save(toStream, ImageFormat.Png);
+ image.Save(toStream, System.Drawing.Imaging.ImageFormat.Png);
}
else
{
diff --git a/MediaBrowser.Controller/Drawing/ImageFormat.cs b/MediaBrowser.Controller/Drawing/ImageFormat.cs
new file mode 100644
index 000000000..f78562556
--- /dev/null
+++ b/MediaBrowser.Controller/Drawing/ImageFormat.cs
@@ -0,0 +1,11 @@
+
+namespace MediaBrowser.Controller.Drawing
+{
+ public enum ImageFormat
+ {
+ Jpg,
+ Png,
+ Gif,
+ Bmp
+ }
+}
diff --git a/MediaBrowser.Controller/Entities/BaseItem.cs b/MediaBrowser.Controller/Entities/BaseItem.cs
index 7840fb3f0..06ebe8905 100644
--- a/MediaBrowser.Controller/Entities/BaseItem.cs
+++ b/MediaBrowser.Controller/Entities/BaseItem.cs
@@ -23,7 +23,7 @@ namespace MediaBrowser.Controller.Entities
/// <summary>
/// Class BaseItem
/// </summary>
- public abstract class BaseItem : IHasProviderIds, ILibraryItem, IHasImages, IHasUserData
+ public abstract class BaseItem : IHasProviderIds, ILibraryItem, IHasImages, IHasUserData, IHasMetadata
{
protected BaseItem()
{
@@ -364,11 +364,16 @@ namespace MediaBrowser.Controller.Entities
}
}
+ private string _forcedSortName;
/// <summary>
/// Gets or sets the name of the forced sort.
/// </summary>
/// <value>The name of the forced sort.</value>
- public string ForcedSortName { get; set; }
+ public string ForcedSortName
+ {
+ get { return _forcedSortName; }
+ set { _forcedSortName = value; _sortName = null; }
+ }
private string _sortName;
/// <summary>
@@ -767,25 +772,35 @@ namespace MediaBrowser.Controller.Entities
}).ToList();
}
+ public Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool resetResolveArgs = true)
+ {
+ return RefreshMetadata(new MetadataRefreshOptions { ResetResolveArgs = resetResolveArgs }, cancellationToken);
+ }
+
/// <summary>
/// Overrides the base implementation to refresh metadata for local trailers
/// </summary>
+ /// <param name="options">The options.</param>
/// <param name="cancellationToken">The cancellation token.</param>
- /// <param name="forceSave">if set to <c>true</c> [is new item].</param>
- /// <param name="forceRefresh">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
- /// <param name="resetResolveArgs">if set to <c>true</c> [reset resolve args].</param>
/// <returns>true if a provider reports we changed</returns>
- public virtual async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
+ public async Task<bool> RefreshMetadata(MetadataRefreshOptions options, CancellationToken cancellationToken)
{
- if (resetResolveArgs)
+ if (options.ResetResolveArgs)
{
// Reload this
ResetResolveArgs();
}
+ await ProviderManager.RefreshMetadata(this, options, cancellationToken).ConfigureAwait(false);
+
+ return false;
+ }
+
+ [Obsolete]
+ public virtual async Task<bool> RefreshMetadataDirect(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
+ {
// Refresh for the item
- var itemRefreshTask = ProviderManager.ExecuteMetadataProviders(this, cancellationToken, forceRefresh, allowSlowProviders);
+ var itemRefreshTask = ProviderManager.ExecuteMetadataProviders(this, cancellationToken, forceRefresh);
cancellationToken.ThrowIfCancellationRequested();
@@ -800,15 +815,15 @@ namespace MediaBrowser.Controller.Entities
var hasThemeMedia = this as IHasThemeMedia;
if (hasThemeMedia != null)
{
- themeSongsChanged = await RefreshThemeSongs(hasThemeMedia, cancellationToken, forceSave, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ themeSongsChanged = await RefreshThemeSongs(hasThemeMedia, cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
- themeVideosChanged = await RefreshThemeVideos(hasThemeMedia, cancellationToken, forceSave, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ themeVideosChanged = await RefreshThemeVideos(hasThemeMedia, cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
}
var hasTrailers = this as IHasTrailers;
if (hasTrailers != null)
{
- localTrailersChanged = await RefreshLocalTrailers(hasTrailers, cancellationToken, forceSave, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ localTrailersChanged = await RefreshLocalTrailers(hasTrailers, cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
}
}
@@ -829,14 +844,20 @@ namespace MediaBrowser.Controller.Entities
return changed;
}
- private async Task<bool> RefreshLocalTrailers(IHasTrailers item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true)
+ private async Task<bool> RefreshLocalTrailers(IHasTrailers item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
var newItems = LoadLocalTrailers().ToList();
var newItemIds = newItems.Select(i => i.Id).ToList();
var itemsChanged = !item.LocalTrailerIds.SequenceEqual(newItemIds);
- var tasks = newItems.Select(i => i.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs: false));
+ var tasks = newItems.Select(i => i.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = forceSave,
+ ReplaceAllMetadata = forceRefresh,
+ ResetResolveArgs = false
+
+ }, cancellationToken));
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
@@ -845,14 +866,20 @@ namespace MediaBrowser.Controller.Entities
return itemsChanged || results.Contains(true);
}
- private async Task<bool> RefreshThemeVideos(IHasThemeMedia item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true)
+ private async Task<bool> RefreshThemeVideos(IHasThemeMedia item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
var newThemeVideos = LoadThemeVideos().ToList();
var newThemeVideoIds = newThemeVideos.Select(i => i.Id).ToList();
var themeVideosChanged = !item.ThemeVideoIds.SequenceEqual(newThemeVideoIds);
- var tasks = newThemeVideos.Select(i => i.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs: false));
+ var tasks = newThemeVideos.Select(i => i.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = forceSave,
+ ReplaceAllMetadata = forceRefresh,
+ ResetResolveArgs = false
+
+ }, cancellationToken));
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
@@ -864,14 +891,20 @@ namespace MediaBrowser.Controller.Entities
/// <summary>
/// Refreshes the theme songs.
/// </summary>
- private async Task<bool> RefreshThemeSongs(IHasThemeMedia item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true)
+ private async Task<bool> RefreshThemeSongs(IHasThemeMedia item, CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
var newThemeSongs = LoadThemeSongs().ToList();
var newThemeSongIds = newThemeSongs.Select(i => i.Id).ToList();
var themeSongsChanged = !item.ThemeSongIds.SequenceEqual(newThemeSongIds);
- var tasks = newThemeSongs.Select(i => i.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs: false));
+ var tasks = newThemeSongs.Select(i => i.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = forceSave,
+ ReplaceAllMetadata = forceRefresh,
+ ResetResolveArgs = false
+
+ }, cancellationToken));
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
@@ -1456,7 +1489,13 @@ namespace MediaBrowser.Controller.Entities
// Refresh metadata
// Need to disable slow providers or the image might get re-downloaded
- return RefreshMetadata(CancellationToken.None, forceSave: true, allowSlowProviders: false);
+ return RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = true,
+ ImageRefreshMode = MetadataRefreshMode.None,
+ MetadataRefreshMode = MetadataRefreshMode.None
+
+ }, CancellationToken.None);
}
/// <summary>
@@ -1482,8 +1521,10 @@ namespace MediaBrowser.Controller.Entities
/// <summary>
/// Validates that images within the item are still on the file system
/// </summary>
- public void ValidateImages()
+ public bool ValidateImages()
{
+ var changed = false;
+
// 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
.Where(image => !File.Exists(image.Value))
@@ -1494,14 +1535,28 @@ namespace MediaBrowser.Controller.Entities
foreach (var key in deletedKeys)
{
Images.Remove(key);
+ changed = true;
}
+
+ if (ValidateBackdrops())
+ {
+ changed = true;
+ }
+ if (ValidateScreenshots())
+ {
+ changed = true;
+ }
+
+ return changed;
}
/// <summary>
/// Validates that backdrops within the item are still on the file system
/// </summary>
- public void ValidateBackdrops()
+ private bool ValidateBackdrops()
{
+ var changed = false;
+
// 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))
@@ -1513,7 +1568,11 @@ namespace MediaBrowser.Controller.Entities
BackdropImagePaths.Remove(path);
RemoveImageSourceForPath(path);
+
+ changed = true;
}
+
+ return changed;
}
/// <summary>
@@ -1593,9 +1652,16 @@ namespace MediaBrowser.Controller.Entities
/// <summary>
/// Validates the screenshots.
/// </summary>
- public void ValidateScreenshots()
+ private bool ValidateScreenshots()
{
- var hasScreenshots = (IHasScreenshots)this;
+ var changed = false;
+
+ var hasScreenshots = this as IHasScreenshots;
+
+ if (hasScreenshots == null)
+ {
+ return changed;
+ }
// 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 = hasScreenshots.ScreenshotImagePaths
@@ -1606,7 +1672,10 @@ namespace MediaBrowser.Controller.Entities
foreach (var path in deletedImages)
{
hasScreenshots.ScreenshotImagePaths.Remove(path);
+ changed = true;
}
+
+ return changed;
}
/// <summary>
@@ -1699,7 +1768,12 @@ namespace MediaBrowser.Controller.Entities
FileSystem.SwapFiles(file1, file2);
// Directory watchers should repeat this, but do a quick refresh first
- return RefreshMetadata(CancellationToken.None, forceSave: true, allowSlowProviders: false);
+ return RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = true,
+ MetadataRefreshMode = MetadataRefreshMode.None
+
+ }, CancellationToken.None);
}
public virtual bool IsPlayed(User user)
diff --git a/MediaBrowser.Controller/Entities/Folder.cs b/MediaBrowser.Controller/Entities/Folder.cs
index a0fefeac7..a4257b2a5 100644
--- a/MediaBrowser.Controller/Entities/Folder.cs
+++ b/MediaBrowser.Controller/Entities/Folder.cs
@@ -3,6 +3,7 @@ using MediaBrowser.Common.Progress;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Localization;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Controller.Resolvers;
using MediaBrowser.Model.Entities;
using MoreLinq;
@@ -535,7 +536,13 @@ namespace MediaBrowser.Controller.Entities
try
{
//refresh it
- await child.RefreshMetadata(cancellationToken, forceSave: currentTuple.Item2, forceRefresh: forceRefreshMetadata, resetResolveArgs: false).ConfigureAwait(false);
+ await child.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = currentTuple.Item2,
+ ReplaceAllMetadata = forceRefreshMetadata,
+ ResetResolveArgs = false
+
+ }, cancellationToken).ConfigureAwait(false);
}
catch (IOException ex)
{
@@ -907,9 +914,9 @@ namespace MediaBrowser.Controller.Entities
return item;
}
- public override async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
+ public override async Task<bool> RefreshMetadataDirect(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
- var changed = await base.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs).ConfigureAwait(false);
+ var changed = await base.RefreshMetadataDirect(cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
return (SupportsShortcutChildren && LocationType == LocationType.FileSystem && RefreshLinkedChildren()) || changed;
}
diff --git a/MediaBrowser.Controller/Entities/IHasImages.cs b/MediaBrowser.Controller/Entities/IHasImages.cs
index a7cd76a66..784337e3b 100644
--- a/MediaBrowser.Controller/Entities/IHasImages.cs
+++ b/MediaBrowser.Controller/Entities/IHasImages.cs
@@ -1,5 +1,6 @@
using MediaBrowser.Model.Entities;
using System;
+using System.Collections.Generic;
using System.Threading.Tasks;
namespace MediaBrowser.Controller.Entities
@@ -10,7 +11,7 @@ namespace MediaBrowser.Controller.Entities
/// Gets the name.
/// </summary>
/// <value>The name.</value>
- string Name { get; }
+ string Name { get; set; }
/// <summary>
/// Gets the path.
@@ -25,6 +26,12 @@ namespace MediaBrowser.Controller.Entities
Guid Id { get; }
/// <summary>
+ /// Gets the type of the location.
+ /// </summary>
+ /// <value>The type of the location.</value>
+ LocationType LocationType { get; }
+
+ /// <summary>
/// Gets the image path.
/// </summary>
/// <param name="imageType">Type of the image.</param>
@@ -81,6 +88,24 @@ namespace MediaBrowser.Controller.Entities
/// </summary>
/// <returns>System.String.</returns>
string GetPreferredMetadataLanguage();
+
+ /// <summary>
+ /// Validates the images and returns true or false indicating if any were removed.
+ /// </summary>
+ bool ValidateImages();
+
+ /// <summary>
+ /// Gets or sets the backdrop image paths.
+ /// </summary>
+ /// <value>The backdrop image paths.</value>
+ List<string> BackdropImagePaths { get; set; }
+
+ /// <summary>
+ /// Determines whether [contains image with source URL] [the specified URL].
+ /// </summary>
+ /// <param name="url">The URL.</param>
+ /// <returns><c>true</c> if [contains image with source URL] [the specified URL]; otherwise, <c>false</c>.</returns>
+ bool ContainsImageWithSourceUrl(string url);
}
public static class HasImagesExtensions
diff --git a/MediaBrowser.Controller/Entities/IHasScreenshots.cs b/MediaBrowser.Controller/Entities/IHasScreenshots.cs
index 2276c707a..70d154a95 100644
--- a/MediaBrowser.Controller/Entities/IHasScreenshots.cs
+++ b/MediaBrowser.Controller/Entities/IHasScreenshots.cs
@@ -14,8 +14,10 @@ namespace MediaBrowser.Controller.Entities
List<string> ScreenshotImagePaths { get; set; }
/// <summary>
- /// Validates the screenshots.
+ /// Determines whether [contains image with source URL] [the specified URL].
/// </summary>
- void ValidateScreenshots();
+ /// <param name="url">The URL.</param>
+ /// <returns><c>true</c> if [contains image with source URL] [the specified URL]; otherwise, <c>false</c>.</returns>
+ bool ContainsImageWithSourceUrl(string url);
}
}
diff --git a/MediaBrowser.Controller/Entities/Movies/Movie.cs b/MediaBrowser.Controller/Entities/Movies/Movie.cs
index 2b252a6c2..dbbe5ce01 100644
--- a/MediaBrowser.Controller/Entities/Movies/Movie.cs
+++ b/MediaBrowser.Controller/Entities/Movies/Movie.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Model.Configuration;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Entities;
using System;
using System.Collections.Generic;
@@ -108,13 +109,11 @@ namespace MediaBrowser.Controller.Entities.Movies
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="forceSave">if set to <c>true</c> [is new item].</param>
/// <param name="forceRefresh">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
- /// <param name="resetResolveArgs">The reset resolve args.</param>
/// <returns>Task{System.Boolean}.</returns>
- public override async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
+ public override async Task<bool> RefreshMetadataDirect(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
// Kick off a task to refresh the main item
- var result = await base.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs).ConfigureAwait(false);
+ var result = await base.RefreshMetadataDirect(cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
var specialFeaturesChanged = false;
@@ -122,7 +121,7 @@ namespace MediaBrowser.Controller.Entities.Movies
// In other words, it must be part of the Parent/Child tree
if (LocationType == LocationType.FileSystem && Parent != null && !IsInMixedFolder)
{
- specialFeaturesChanged = await RefreshSpecialFeatures(cancellationToken, forceSave, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ specialFeaturesChanged = await RefreshSpecialFeatures(cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
}
return specialFeaturesChanged || result;
@@ -135,7 +134,13 @@ namespace MediaBrowser.Controller.Entities.Movies
var itemsChanged = !SpecialFeatureIds.SequenceEqual(newItemIds);
- var tasks = newItems.Select(i => i.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs: false));
+ var tasks = newItems.Select(i => i.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = forceSave,
+ ReplaceAllMetadata = forceRefresh,
+ ResetResolveArgs = false
+
+ }, cancellationToken));
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
diff --git a/MediaBrowser.Controller/Entities/User.cs b/MediaBrowser.Controller/Entities/User.cs
index 466e709dd..c109e1d0c 100644
--- a/MediaBrowser.Controller/Entities/User.cs
+++ b/MediaBrowser.Controller/Entities/User.cs
@@ -1,5 +1,6 @@
using MediaBrowser.Common.Configuration;
using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Serialization;
using System;
@@ -212,7 +213,12 @@ namespace MediaBrowser.Controller.Entities
// Kick off a task to validate the media library
Task.Run(() => ValidateMediaLibrary(new Progress<double>(), CancellationToken.None));
- return RefreshMetadata(CancellationToken.None, forceSave: true, forceRefresh: true);
+ return RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = true,
+ ReplaceAllMetadata = true
+
+ }, CancellationToken.None);
}
/// <summary>
@@ -275,17 +281,13 @@ namespace MediaBrowser.Controller.Entities
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="forceSave">if set to <c>true</c> [is new item].</param>
/// <param name="forceRefresh">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
/// <returns>true if a provider reports we changed</returns>
- public override async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
+ public override async Task<bool> RefreshMetadataDirect(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
- if (resetResolveArgs)
- {
- // Reload this
- ResetResolveArgs();
- }
+ // Reload this
+ ResetResolveArgs();
- var updateReason = await ProviderManager.ExecuteMetadataProviders(this, cancellationToken, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ var updateReason = await ProviderManager.ExecuteMetadataProviders(this, cancellationToken, forceRefresh).ConfigureAwait(false);
var changed = updateReason.HasValue;
diff --git a/MediaBrowser.Controller/Entities/Video.cs b/MediaBrowser.Controller/Entities/Video.cs
index e663c8353..9c9466766 100644
--- a/MediaBrowser.Controller/Entities/Video.cs
+++ b/MediaBrowser.Controller/Entities/Video.cs
@@ -1,4 +1,5 @@
using MediaBrowser.Controller.Persistence;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Controller.Resolvers;
using MediaBrowser.Model.Entities;
using System;
@@ -164,13 +165,11 @@ namespace MediaBrowser.Controller.Entities
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="forceSave">if set to <c>true</c> [is new item].</param>
/// <param name="forceRefresh">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
- /// <param name="resetResolveArgs">The reset resolve args.</param>
/// <returns>true if a provider reports we changed</returns>
- public override async Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true)
+ public override async Task<bool> RefreshMetadataDirect(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false)
{
// Kick off a task to refresh the main item
- var result = await base.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders, resetResolveArgs).ConfigureAwait(false);
+ var result = await base.RefreshMetadataDirect(cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
var additionalPartsChanged = false;
@@ -181,7 +180,7 @@ namespace MediaBrowser.Controller.Entities
{
try
{
- additionalPartsChanged = await RefreshAdditionalParts(cancellationToken, forceSave, forceRefresh, allowSlowProviders).ConfigureAwait(false);
+ additionalPartsChanged = await RefreshAdditionalParts(cancellationToken, forceSave, forceRefresh).ConfigureAwait(false);
}
catch (IOException ex)
{
@@ -208,7 +207,12 @@ namespace MediaBrowser.Controller.Entities
var itemsChanged = !AdditionalPartIds.SequenceEqual(newItemIds);
- var tasks = newItems.Select(i => i.RefreshMetadata(cancellationToken, forceSave, forceRefresh, allowSlowProviders));
+ var tasks = newItems.Select(i => i.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = forceSave,
+ ReplaceAllMetadata = forceRefresh
+
+ }, cancellationToken));
var results = await Task.WhenAll(tasks).ConfigureAwait(false);
diff --git a/MediaBrowser.Controller/IO/IDirectoryWatchers.cs b/MediaBrowser.Controller/IO/IDirectoryWatchers.cs
deleted file mode 100644
index 9a43ee8ac..000000000
--- a/MediaBrowser.Controller/IO/IDirectoryWatchers.cs
+++ /dev/null
@@ -1,29 +0,0 @@
-using System;
-
-namespace MediaBrowser.Controller.IO
-{
- public interface IDirectoryWatchers : IDisposable
- {
- /// <summary>
- /// Add the path to our temporary ignore list. Use when writing to a path within our listening scope.
- /// </summary>
- /// <param name="path">The path.</param>
- void TemporarilyIgnore(string path);
-
- /// <summary>
- /// Removes the temp ignore.
- /// </summary>
- /// <param name="path">The path.</param>
- void RemoveTempIgnore(string path);
-
- /// <summary>
- /// Starts this instance.
- /// </summary>
- void Start();
-
- /// <summary>
- /// Stops this instance.
- /// </summary>
- void Stop();
- }
-} \ No newline at end of file
diff --git a/MediaBrowser.Controller/Library/ILibraryMonitor.cs b/MediaBrowser.Controller/Library/ILibraryMonitor.cs
new file mode 100644
index 000000000..918382f04
--- /dev/null
+++ b/MediaBrowser.Controller/Library/ILibraryMonitor.cs
@@ -0,0 +1,36 @@
+using System;
+
+namespace MediaBrowser.Controller.Library
+{
+ public interface ILibraryMonitor : IDisposable
+ {
+ /// <summary>
+ /// Starts this instance.
+ /// </summary>
+ void Start();
+
+ /// <summary>
+ /// Stops this instance.
+ /// </summary>
+ void Stop();
+
+ /// <summary>
+ /// Reports the file system change beginning.
+ /// </summary>
+ /// <param name="path">The path.</param>
+ void ReportFileSystemChangeBeginning(string path);
+
+ /// <summary>
+ /// Reports the file system change complete.
+ /// </summary>
+ /// <param name="path">The path.</param>
+ /// <param name="refreshPath">if set to <c>true</c> [refresh path].</param>
+ void ReportFileSystemChangeComplete(string path, bool refreshPath);
+
+ /// <summary>
+ /// Reports the file system changed.
+ /// </summary>
+ /// <param name="path">The path.</param>
+ void ReportFileSystemChanged(string path);
+ }
+} \ No newline at end of file
diff --git a/MediaBrowser.Controller/LiveTv/ILiveTvRecording.cs b/MediaBrowser.Controller/LiveTv/ILiveTvRecording.cs
index d9bceb6ca..c94a25a30 100644
--- a/MediaBrowser.Controller/LiveTv/ILiveTvRecording.cs
+++ b/MediaBrowser.Controller/LiveTv/ILiveTvRecording.cs
@@ -1,4 +1,5 @@
using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using System.Threading;
using System.Threading.Tasks;
@@ -11,8 +12,6 @@ namespace MediaBrowser.Controller.LiveTv
string MediaType { get; }
- LocationType LocationType { get; }
-
RecordingInfo RecordingInfo { get; set; }
string GetClientTypeName();
@@ -21,6 +20,6 @@ namespace MediaBrowser.Controller.LiveTv
bool IsParentalAllowed(User user);
- Task<bool> RefreshMetadata(CancellationToken cancellationToken, bool forceSave = false, bool forceRefresh = false, bool allowSlowProviders = true, bool resetResolveArgs = true);
+ Task<bool> RefreshMetadata(MetadataRefreshOptions options, CancellationToken cancellationToken);
}
}
diff --git a/MediaBrowser.Controller/LiveTv/StreamResponseInfo.cs b/MediaBrowser.Controller/LiveTv/StreamResponseInfo.cs
index c3b438c5e..b133874d0 100644
--- a/MediaBrowser.Controller/LiveTv/StreamResponseInfo.cs
+++ b/MediaBrowser.Controller/LiveTv/StreamResponseInfo.cs
@@ -1,4 +1,5 @@
-using System.IO;
+using MediaBrowser.Controller.Drawing;
+using System.IO;
namespace MediaBrowser.Controller.LiveTv
{
@@ -14,6 +15,6 @@ namespace MediaBrowser.Controller.LiveTv
/// Gets or sets the type of the MIME.
/// </summary>
/// <value>The type of the MIME.</value>
- public string MimeType { get; set; }
+ public ImageFormat Format { get; set; }
}
}
diff --git a/MediaBrowser.Controller/MediaBrowser.Controller.csproj b/MediaBrowser.Controller/MediaBrowser.Controller.csproj
index 56ac695a2..ee8bb2761 100644
--- a/MediaBrowser.Controller/MediaBrowser.Controller.csproj
+++ b/MediaBrowser.Controller/MediaBrowser.Controller.csproj
@@ -69,6 +69,7 @@
<Link>Properties\SharedVersion.cs</Link>
</Compile>
<Compile Include="Drawing\IImageProcessor.cs" />
+ <Compile Include="Drawing\ImageFormat.cs" />
<Compile Include="Drawing\ImageProcessingOptions.cs" />
<Compile Include="Dto\IDtoService.cs" />
<Compile Include="Entities\AdultVideo.cs" />
@@ -143,8 +144,17 @@
<Compile Include="Persistence\IFileOrganizationRepository.cs" />
<Compile Include="Persistence\MediaStreamQuery.cs" />
<Compile Include="Providers\IDynamicInfoProvider.cs" />
+ <Compile Include="Providers\IHasMetadata.cs" />
<Compile Include="Providers\IImageProvider.cs" />
+ <Compile Include="Providers\IProviderRepository.cs" />
+ <Compile Include="Providers\IRemoteImageProvider.cs" />
+ <Compile Include="Providers\ILocalImageProvider.cs" />
+ <Compile Include="Providers\IMetadataProvider.cs" />
+ <Compile Include="Providers\IMetadataService.cs" />
+ <Compile Include="Providers\ItemId.cs" />
+ <Compile Include="Providers\MetadataRefreshOptions.cs" />
<Compile Include="Providers\NameParser.cs" />
+ <Compile Include="Providers\MetadataStatus.cs" />
<Compile Include="Session\ISessionManager.cs" />
<Compile Include="Drawing\ImageExtensions.cs" />
<Compile Include="Entities\AggregateFolder.cs" />
@@ -174,7 +184,7 @@
<Compile Include="Entities\Video.cs" />
<Compile Include="Entities\CollectionFolder.cs" />
<Compile Include="Entities\Year.cs" />
- <Compile Include="IO\IDirectoryWatchers.cs" />
+ <Compile Include="Library\ILibraryMonitor.cs" />
<Compile Include="IServerApplicationHost.cs" />
<Compile Include="IServerApplicationPaths.cs" />
<Compile Include="Library\SearchHintInfo.cs" />
diff --git a/MediaBrowser.Controller/Persistence/IItemRepository.cs b/MediaBrowser.Controller/Persistence/IItemRepository.cs
index 3affe48e7..3a5cb4e87 100644
--- a/MediaBrowser.Controller/Persistence/IItemRepository.cs
+++ b/MediaBrowser.Controller/Persistence/IItemRepository.cs
@@ -1,5 +1,4 @@
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using System;
using System.Collections.Generic;
@@ -112,22 +111,6 @@ namespace MediaBrowser.Controller.Persistence
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>Task.</returns>
Task SaveMediaStreams(Guid id, IEnumerable<MediaStream> streams, CancellationToken cancellationToken);
-
- /// <summary>
- /// Gets the provider history.
- /// </summary>
- /// <param name="itemId">The item identifier.</param>
- /// <returns>IEnumerable{BaseProviderInfo}.</returns>
- IEnumerable<BaseProviderInfo> GetProviderHistory(Guid itemId);
-
- /// <summary>
- /// Saves the provider history.
- /// </summary>
- /// <param name="id">The identifier.</param>
- /// <param name="history">The history.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- Task SaveProviderHistory(Guid id, IEnumerable<BaseProviderInfo> history, CancellationToken cancellationToken);
}
}
diff --git a/MediaBrowser.Controller/Providers/IHasMetadata.cs b/MediaBrowser.Controller/Providers/IHasMetadata.cs
new file mode 100644
index 000000000..33c5184b8
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/IHasMetadata.cs
@@ -0,0 +1,31 @@
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Model.Entities;
+using System;
+using System.Collections.Generic;
+
+namespace MediaBrowser.Controller.Providers
+{
+ /// <summary>
+ /// Interface IHasMetadata
+ /// </summary>
+ public interface IHasMetadata : IHasImages, IHasProviderIds
+ {
+ /// <summary>
+ /// Gets the preferred metadata country code.
+ /// </summary>
+ /// <returns>System.String.</returns>
+ string GetPreferredMetadataCountryCode();
+
+ /// <summary>
+ /// Gets the locked fields.
+ /// </summary>
+ /// <value>The locked fields.</value>
+ List<MetadataFields> LockedFields { get; }
+
+ /// <summary>
+ /// Gets or sets the date last saved.
+ /// </summary>
+ /// <value>The date last saved.</value>
+ DateTime DateLastSaved { get; set; }
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/IImageProvider.cs b/MediaBrowser.Controller/Providers/IImageProvider.cs
index ccf199844..61f5579f4 100644
--- a/MediaBrowser.Controller/Providers/IImageProvider.cs
+++ b/MediaBrowser.Controller/Providers/IImageProvider.cs
@@ -1,9 +1,4 @@
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Providers;
-using System.Collections.Generic;
-using System.Threading;
-using System.Threading.Tasks;
namespace MediaBrowser.Controller.Providers
{
@@ -26,26 +21,9 @@ namespace MediaBrowser.Controller.Providers
bool Supports(IHasImages item);
/// <summary>
- /// Gets the images.
+ /// Gets the order.
/// </summary>
- /// <param name="item">The item.</param>
- /// <param name="imageType">Type of the image.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
- Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken);
-
- /// <summary>
- /// Gets the images.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
- Task<IEnumerable<RemoteImageInfo>> GetAllImages(IHasImages item, CancellationToken cancellationToken);
-
- /// <summary>
- /// Gets the priority.
- /// </summary>
- /// <value>The priority.</value>
- int Priority { get; }
+ /// <value>The order.</value>
+ int Order { get; }
}
}
diff --git a/MediaBrowser.Controller/Providers/ILocalImageProvider.cs b/MediaBrowser.Controller/Providers/ILocalImageProvider.cs
new file mode 100644
index 000000000..5c3ebd9ac
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/ILocalImageProvider.cs
@@ -0,0 +1,66 @@
+using MediaBrowser.Controller.Drawing;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Model.Entities;
+using System.Collections.Generic;
+using System.IO;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Controller.Providers
+{
+ /// <summary>
+ /// This is just a marker interface
+ /// </summary>
+ public interface ILocalImageProvider : IImageProvider
+ {
+ }
+
+ public interface IImageFileProvider : ILocalImageProvider
+ {
+ List<LocalImageInfo> GetImages(IHasImages item);
+ }
+
+ public class LocalImageInfo
+ {
+ public string Path { get; set; }
+ public ImageType Type { get; set; }
+ }
+
+ public interface IDynamicImageProvider : ILocalImageProvider
+ {
+ /// <summary>
+ /// Gets the supported images.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>IEnumerable{ImageType}.</returns>
+ IEnumerable<ImageType> GetSupportedImages(IHasImages item);
+
+ /// <summary>
+ /// Gets the image.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="type">The type.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task{DynamicImageResponse}.</returns>
+ Task<DynamicImageResponse> GetImage(IHasImages item, ImageType type, CancellationToken cancellationToken);
+ }
+
+ public class DynamicImageInfo
+ {
+ public string ImageId { get; set; }
+ public ImageType Type { get; set; }
+ }
+
+ public class DynamicImageResponse
+ {
+ public string Path { get; set; }
+ public Stream Stream { get; set; }
+ public ImageFormat Format { get; set; }
+ public bool HasImage { get; set; }
+
+ public void SetFormatFromMimeType(string mimeType)
+ {
+
+ }
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/IMetadataProvider.cs b/MediaBrowser.Controller/Providers/IMetadataProvider.cs
new file mode 100644
index 000000000..843ba263b
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/IMetadataProvider.cs
@@ -0,0 +1,68 @@
+using System;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Controller.Providers
+{
+ /// <summary>
+ /// Marker interface
+ /// </summary>
+ public interface IMetadataProvider
+ {
+ /// <summary>
+ /// Gets the name.
+ /// </summary>
+ /// <value>The name.</value>
+ string Name { get; }
+ }
+
+ public interface IMetadataProvider<TItemType> : IMetadataProvider
+ where TItemType : IHasMetadata
+ {
+ }
+
+ public interface ILocalMetadataProvider : IMetadataProvider
+ {
+ /// <summary>
+ /// Determines whether [has local metadata] [the specified item].
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns><c>true</c> if [has local metadata] [the specified item]; otherwise, <c>false</c>.</returns>
+ bool HasLocalMetadata(IHasMetadata item);
+ }
+
+ public interface IRemoteMetadataProvider : IMetadataProvider
+ {
+ }
+
+ public interface IRemoteMetadataProvider<TItemType> : IMetadataProvider<TItemType>, IRemoteMetadataProvider
+ where TItemType : IHasMetadata
+ {
+ Task<MetadataResult<TItemType>> GetMetadata(ItemId id, CancellationToken cancellationToken);
+ }
+
+ public interface ILocalMetadataProvider<TItemType> : IMetadataProvider<TItemType>, ILocalMetadataProvider
+ where TItemType : IHasMetadata
+ {
+ Task<MetadataResult<TItemType>> GetMetadata(string path, CancellationToken cancellationToken);
+ }
+
+ public interface IHasChangeMonitor
+ {
+ /// <summary>
+ /// Determines whether the specified item has changed.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="date">The date.</param>
+ /// <returns><c>true</c> if the specified item has changed; otherwise, <c>false</c>.</returns>
+ bool HasChanged(IHasMetadata item, DateTime date);
+ }
+
+ public class MetadataResult<T>
+ where T : IHasMetadata
+ {
+ public bool HasMetadata { get; set; }
+ public T Item { get; set; }
+ }
+
+}
diff --git a/MediaBrowser.Controller/Providers/IMetadataService.cs b/MediaBrowser.Controller/Providers/IMetadataService.cs
new file mode 100644
index 000000000..c6cc2b716
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/IMetadataService.cs
@@ -0,0 +1,38 @@
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Controller.Providers
+{
+ public interface IMetadataService
+ {
+ /// <summary>
+ /// Adds the parts.
+ /// </summary>
+ /// <param name="providers">The providers.</param>
+ /// <param name="imageProviders">The image providers.</param>
+ void AddParts(IEnumerable<IMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders);
+
+ /// <summary>
+ /// Determines whether this instance can refresh the specified item.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns><c>true</c> if this instance can refresh the specified item; otherwise, <c>false</c>.</returns>
+ bool CanRefresh(IHasMetadata item);
+
+ /// <summary>
+ /// Refreshes the metadata.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="options">The options.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ Task RefreshMetadata(IHasMetadata item, MetadataRefreshOptions options, CancellationToken cancellationToken);
+
+ /// <summary>
+ /// Gets the order.
+ /// </summary>
+ /// <value>The order.</value>
+ int Order { get; }
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/IProviderManager.cs b/MediaBrowser.Controller/Providers/IProviderManager.cs
index 728030ecc..dc57552c4 100644
--- a/MediaBrowser.Controller/Providers/IProviderManager.cs
+++ b/MediaBrowser.Controller/Providers/IProviderManager.cs
@@ -15,14 +15,22 @@ namespace MediaBrowser.Controller.Providers
public interface IProviderManager
{
/// <summary>
+ /// Refreshes the metadata.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="options">The options.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ Task RefreshMetadata(IHasMetadata item, MetadataRefreshOptions options, CancellationToken cancellationToken);
+
+ /// <summary>
/// Executes the metadata providers.
/// </summary>
/// <param name="item">The item.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="force">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
/// <returns>Task{System.Boolean}.</returns>
- Task<ItemUpdateType?> ExecuteMetadataProviders(BaseItem item, CancellationToken cancellationToken, bool force = false, bool allowSlowProviders = true);
+ Task<ItemUpdateType?> ExecuteMetadataProviders(BaseItem item, CancellationToken cancellationToken, bool force = false);
/// <summary>
/// Saves the image.
@@ -54,7 +62,9 @@ namespace MediaBrowser.Controller.Providers
/// </summary>
/// <param name="providers">The providers.</param>
/// <param name="imageProviders">The image providers.</param>
- void AddParts(IEnumerable<BaseMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders);
+ /// <param name="metadataServices">The metadata services.</param>
+ /// <param name="metadataProviders">The metadata providers.</param>
+ void AddParts(IEnumerable<BaseMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders, IEnumerable<IMetadataService> metadataServices, IEnumerable<IMetadataProvider> metadataProviders);
/// <summary>
/// Gets the available remote images.
@@ -70,7 +80,7 @@ namespace MediaBrowser.Controller.Providers
/// Gets the image providers.
/// </summary>
/// <param name="item">The item.</param>
- /// <returns>IEnumerable{IImageProvider}.</returns>
- IEnumerable<IImageProvider> GetImageProviders(BaseItem item);
+ /// <returns>IEnumerable{ImageProviderInfo}.</returns>
+ IEnumerable<ImageProviderInfo> GetImageProviderInfo(BaseItem item);
}
} \ No newline at end of file
diff --git a/MediaBrowser.Controller/Providers/IProviderRepository.cs b/MediaBrowser.Controller/Providers/IProviderRepository.cs
new file mode 100644
index 000000000..1c0ad2cd7
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/IProviderRepository.cs
@@ -0,0 +1,48 @@
+using MediaBrowser.Controller.Persistence;
+using System;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Controller.Providers
+{
+ public interface IProviderRepository : IRepository
+ {
+ /// <summary>
+ /// Gets the provider history.
+ /// </summary>
+ /// <param name="itemId">The item identifier.</param>
+ /// <returns>IEnumerable{BaseProviderInfo}.</returns>
+ IEnumerable<BaseProviderInfo> GetProviderHistory(Guid itemId);
+
+ /// <summary>
+ /// Saves the provider history.
+ /// </summary>
+ /// <param name="id">The identifier.</param>
+ /// <param name="history">The history.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ Task SaveProviderHistory(Guid id, IEnumerable<BaseProviderInfo> history, CancellationToken cancellationToken);
+
+ /// <summary>
+ /// Gets the metadata status.
+ /// </summary>
+ /// <param name="itemId">The item identifier.</param>
+ /// <returns>MetadataStatus.</returns>
+ MetadataStatus GetMetadataStatus(Guid itemId);
+
+ /// <summary>
+ /// Saves the metadata status.
+ /// </summary>
+ /// <param name="status">The status.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ Task SaveMetadataStatus(MetadataStatus status, CancellationToken cancellationToken);
+
+ /// <summary>
+ /// Initializes this instance.
+ /// </summary>
+ /// <returns>Task.</returns>
+ Task Initialize();
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/IRemoteImageProvider.cs b/MediaBrowser.Controller/Providers/IRemoteImageProvider.cs
new file mode 100644
index 000000000..23fda2bfa
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/IRemoteImageProvider.cs
@@ -0,0 +1,48 @@
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Providers;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Controller.Providers
+{
+ /// <summary>
+ /// Interface IImageProvider
+ /// </summary>
+ public interface IRemoteImageProvider : IImageProvider
+ {
+ /// <summary>
+ /// Gets the supported images.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>IEnumerable{ImageType}.</returns>
+ IEnumerable<ImageType> GetSupportedImages(IHasImages item);
+
+ /// <summary>
+ /// Gets the images.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="imageType">Type of the image.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
+ Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken);
+
+ /// <summary>
+ /// Gets the images.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
+ Task<IEnumerable<RemoteImageInfo>> GetAllImages(IHasImages item, CancellationToken cancellationToken);
+
+ /// <summary>
+ /// Gets the image response.
+ /// </summary>
+ /// <param name="url">The URL.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task{HttpResponseInfo}.</returns>
+ Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken);
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/ItemId.cs b/MediaBrowser.Controller/Providers/ItemId.cs
new file mode 100644
index 000000000..1116eb8b5
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/ItemId.cs
@@ -0,0 +1,35 @@
+using MediaBrowser.Model.Entities;
+using System;
+using System.Collections.Generic;
+
+namespace MediaBrowser.Controller.Providers
+{
+ public class ItemId : IHasProviderIds
+ {
+ /// <summary>
+ /// Gets or sets the name.
+ /// </summary>
+ /// <value>The name.</value>
+ public string Name { get; set; }
+ /// <summary>
+ /// Gets or sets the metadata language.
+ /// </summary>
+ /// <value>The metadata language.</value>
+ public string MetadataLanguage { get; set; }
+ /// <summary>
+ /// Gets or sets the metadata country code.
+ /// </summary>
+ /// <value>The metadata country code.</value>
+ public string MetadataCountryCode { get; set; }
+ /// <summary>
+ /// Gets or sets the provider ids.
+ /// </summary>
+ /// <value>The provider ids.</value>
+ public Dictionary<string, string> ProviderIds { get; set; }
+
+ public ItemId()
+ {
+ ProviderIds = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
+ }
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/MetadataRefreshOptions.cs b/MediaBrowser.Controller/Providers/MetadataRefreshOptions.cs
new file mode 100644
index 000000000..d6e8a3afe
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/MetadataRefreshOptions.cs
@@ -0,0 +1,49 @@
+using System;
+
+namespace MediaBrowser.Controller.Providers
+{
+ public class MetadataRefreshOptions : ImageRefreshOptions
+ {
+ /// <summary>
+ /// When paired with MetadataRefreshMode=FullRefresh, all existing data will be overwritten with new data from the providers.
+ /// </summary>
+ public bool ReplaceAllMetadata { get; set; }
+
+ public MetadataRefreshMode MetadataRefreshMode { get; set; }
+
+ /// <summary>
+ /// TODO: deprecate. Keeping this for now, for api compatibility
+ /// </summary>
+ [Obsolete]
+ public bool ForceSave { get; set; }
+
+ /// <summary>
+ /// TODO: deprecate. Keeping this for now, for api compatibility
+ /// </summary>
+ [Obsolete]
+ public bool ResetResolveArgs { get; set; }
+ }
+
+ public class ImageRefreshOptions
+ {
+ public MetadataRefreshMode ImageRefreshMode { get; set; }
+ }
+
+ public enum MetadataRefreshMode
+ {
+ /// <summary>
+ /// Providers will be executed based on default rules
+ /// </summary>
+ EnsureMetadata,
+
+ /// <summary>
+ /// No providers will be executed
+ /// </summary>
+ None,
+
+ /// <summary>
+ /// All providers will be executed to search for new metadata
+ /// </summary>
+ FullRefresh
+ }
+}
diff --git a/MediaBrowser.Controller/Providers/MetadataStatus.cs b/MediaBrowser.Controller/Providers/MetadataStatus.cs
new file mode 100644
index 000000000..834d8ec35
--- /dev/null
+++ b/MediaBrowser.Controller/Providers/MetadataStatus.cs
@@ -0,0 +1,122 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using MediaBrowser.Common.Extensions;
+
+namespace MediaBrowser.Controller.Providers
+{
+ public class MetadataStatus
+ {
+ /// <summary>
+ /// Gets or sets the item identifier.
+ /// </summary>
+ /// <value>The item identifier.</value>
+ public Guid ItemId { get; set; }
+
+ /// <summary>
+ /// Gets or sets the date last metadata refresh.
+ /// </summary>
+ /// <value>The date last metadata refresh.</value>
+ public DateTime? DateLastMetadataRefresh { get; set; }
+
+ /// <summary>
+ /// Gets or sets the date last images refresh.
+ /// </summary>
+ /// <value>The date last images refresh.</value>
+ public DateTime? DateLastImagesRefresh { get; set; }
+
+ /// <summary>
+ /// Gets or sets the last result.
+ /// </summary>
+ /// <value>The last result.</value>
+ public ProviderRefreshStatus LastStatus { get; set; }
+
+ /// <summary>
+ /// Gets or sets the last result error message.
+ /// </summary>
+ /// <value>The last result error message.</value>
+ public string LastErrorMessage { get; set; }
+
+ /// <summary>
+ /// Gets or sets the providers refreshed.
+ /// </summary>
+ /// <value>The providers refreshed.</value>
+ public List<Guid> MetadataProvidersRefreshed { get; set; }
+ public List<Guid> ImageProvidersRefreshed { get; set; }
+
+ public void AddStatus(ProviderRefreshStatus status, string errorMessage)
+ {
+ if (LastStatus != status)
+ {
+ IsDirty = true;
+ }
+
+ if (string.IsNullOrEmpty(LastErrorMessage))
+ {
+ LastErrorMessage = errorMessage;
+ }
+ if (LastStatus == ProviderRefreshStatus.Success)
+ {
+ LastStatus = status;
+ }
+ }
+
+ public MetadataStatus()
+ {
+ LastStatus = ProviderRefreshStatus.Success;
+
+ MetadataProvidersRefreshed = new List<Guid>();
+ ImageProvidersRefreshed = new List<Guid>();
+ }
+
+ public bool IsDirty { get; private set; }
+
+ public void SetDateLastMetadataRefresh(DateTime date)
+ {
+ if (date != (DateLastMetadataRefresh ?? DateTime.MinValue))
+ {
+ IsDirty = true;
+ }
+
+ DateLastMetadataRefresh = date;
+ }
+
+ public void SetDateLastImagesRefresh(DateTime date)
+ {
+ if (date != (DateLastImagesRefresh ?? DateTime.MinValue))
+ {
+ IsDirty = true;
+ }
+
+ DateLastImagesRefresh = date;
+ }
+
+ public void AddImageProvidersRefreshed(List<Guid> providerIds)
+ {
+ var count = ImageProvidersRefreshed.Count;
+
+ providerIds.AddRange(ImageProvidersRefreshed);
+
+ ImageProvidersRefreshed = providerIds.Distinct().ToList();
+
+ if (ImageProvidersRefreshed.Count != count)
+ {
+ IsDirty = true;
+ }
+ }
+
+ public void AddMetadataProvidersRefreshed(List<Guid> providerIds)
+ {
+ var count = MetadataProvidersRefreshed.Count;
+
+ providerIds.AddRange(MetadataProvidersRefreshed);
+
+ MetadataProvidersRefreshed = providerIds.Distinct().ToList();
+
+ if (MetadataProvidersRefreshed.Count != count)
+ {
+ IsDirty = true;
+ }
+ }
+ }
+}
diff --git a/MediaBrowser.Model/Configuration/ServerConfiguration.cs b/MediaBrowser.Model/Configuration/ServerConfiguration.cs
index f017fdf16..923c5ab74 100644
--- a/MediaBrowser.Model/Configuration/ServerConfiguration.cs
+++ b/MediaBrowser.Model/Configuration/ServerConfiguration.cs
@@ -165,7 +165,7 @@ namespace MediaBrowser.Model.Configuration
/// different directories and files.
/// </summary>
/// <value>The file watcher delay.</value>
- public int FileWatcherDelay { get; set; }
+ public int RealtimeWatcherDelay { get; set; }
/// <summary>
/// Gets or sets a value indicating whether [enable dashboard response caching].
@@ -250,7 +250,7 @@ namespace MediaBrowser.Model.Configuration
MaxResumePct = 90;
MinResumeDurationSeconds = Convert.ToInt32(TimeSpan.FromMinutes(5).TotalSeconds);
- FileWatcherDelay = 8;
+ RealtimeWatcherDelay = 20;
RecentItemDays = 10;
diff --git a/MediaBrowser.Model/Dto/BaseItemDto.cs b/MediaBrowser.Model/Dto/BaseItemDto.cs
index a8c486d84..1a90e75d9 100644
--- a/MediaBrowser.Model/Dto/BaseItemDto.cs
+++ b/MediaBrowser.Model/Dto/BaseItemDto.cs
@@ -117,6 +117,12 @@ namespace MediaBrowser.Model.Dto
public string Overview { get; set; }
/// <summary>
+ /// Gets or sets the name of the TMDB collection.
+ /// </summary>
+ /// <value>The name of the TMDB collection.</value>
+ public string TmdbCollectionName { get; set; }
+
+ /// <summary>
/// Gets or sets the taglines.
/// </summary>
/// <value>The taglines.</value>
diff --git a/MediaBrowser.Model/Entities/IHasProviderIds.cs b/MediaBrowser.Model/Entities/IHasProviderIds.cs
index 1c54455da..efb75412f 100644
--- a/MediaBrowser.Model/Entities/IHasProviderIds.cs
+++ b/MediaBrowser.Model/Entities/IHasProviderIds.cs
@@ -21,6 +21,17 @@ namespace MediaBrowser.Model.Entities
public static class ProviderIdsExtensions
{
/// <summary>
+ /// Determines whether [has provider identifier] [the specified instance].
+ /// </summary>
+ /// <param name="instance">The instance.</param>
+ /// <param name="provider">The provider.</param>
+ /// <returns><c>true</c> if [has provider identifier] [the specified instance]; otherwise, <c>false</c>.</returns>
+ public static bool HasProviderId(this IHasProviderIds instance, MetadataProviders provider)
+ {
+ return !string.IsNullOrEmpty(instance.GetProviderId(provider.ToString()));
+ }
+
+ /// <summary>
/// Gets a provider id
/// </summary>
/// <param name="instance">The instance.</param>
diff --git a/MediaBrowser.Model/Providers/ImageProviderInfo.cs b/MediaBrowser.Model/Providers/ImageProviderInfo.cs
index 325aa90cb..1b8a2816a 100644
--- a/MediaBrowser.Model/Providers/ImageProviderInfo.cs
+++ b/MediaBrowser.Model/Providers/ImageProviderInfo.cs
@@ -12,9 +12,9 @@
public string Name { get; set; }
/// <summary>
- /// Gets or sets the priority.
+ /// Gets or sets the order.
/// </summary>
- /// <value>The priority.</value>
- public int Priority { get; set; }
+ /// <value>The order.</value>
+ public int Order { get; set; }
}
}
diff --git a/MediaBrowser.Model/Querying/ItemFields.cs b/MediaBrowser.Model/Querying/ItemFields.cs
index 6f73e73f7..406957daa 100644
--- a/MediaBrowser.Model/Querying/ItemFields.cs
+++ b/MediaBrowser.Model/Querying/ItemFields.cs
@@ -150,6 +150,11 @@ namespace MediaBrowser.Model.Querying
/// The tags
/// </summary>
Tags,
+
+ /// <summary>
+ /// The TMDB collection name
+ /// </summary>
+ TmdbCollectionName,
/// <summary>
/// The trailer url of the item
diff --git a/MediaBrowser.Providers/All/LocalImageProvider.cs b/MediaBrowser.Providers/All/LocalImageProvider.cs
new file mode 100644
index 000000000..88a68bc15
--- /dev/null
+++ b/MediaBrowser.Providers/All/LocalImageProvider.cs
@@ -0,0 +1,327 @@
+using MediaBrowser.Common.IO;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Entities.Movies;
+using MediaBrowser.Controller.Entities.TV;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using System;
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Linq;
+
+namespace MediaBrowser.Providers.All
+{
+ public class LocalImageProvider : IImageFileProvider
+ {
+ private readonly IFileSystem _fileSystem;
+
+ public LocalImageProvider(IFileSystem fileSystem)
+ {
+ _fileSystem = fileSystem;
+ }
+
+ public string Name
+ {
+ get { return "Local Images"; }
+ }
+
+ public int Order
+ {
+ get { return 0; }
+ }
+
+ public bool Supports(IHasImages item)
+ {
+ var locationType = item.LocationType;
+
+ if (locationType == LocationType.FileSystem)
+ {
+ // Episode has it's own provider
+ if (item is Episode)
+ {
+ return false;
+ }
+
+ return true;
+ }
+ if (locationType == LocationType.Virtual)
+ {
+ var season = item as Season;
+
+ if (season != null)
+ {
+ var series = season.Series;
+
+ if (series != null && series.LocationType == LocationType.FileSystem)
+ {
+ return true;
+ }
+ }
+ }
+
+ return false;
+ }
+
+ private IEnumerable<string> GetFiles(IHasImages item, bool includeDirectories)
+ {
+ if (item.LocationType != LocationType.FileSystem)
+ {
+ return new List<string>();
+ }
+
+ var path = item.Path;
+ var fileInfo = _fileSystem.GetFileSystemInfo(path) as DirectoryInfo;
+
+ if (fileInfo == null)
+ {
+ path = Path.GetDirectoryName(path);
+ }
+
+ if (includeDirectories)
+ {
+ return Directory.EnumerateFileSystemEntries(path, "*", SearchOption.TopDirectoryOnly);
+ }
+ return Directory.EnumerateFiles(path, "*", SearchOption.TopDirectoryOnly);
+ }
+
+ public List<LocalImageInfo> GetImages(IHasImages item)
+ {
+ var files = GetFileDictionary(GetFiles(item, true));
+
+ var list = new List<LocalImageInfo>();
+
+ PopulateImages(item, list, files);
+
+ return list;
+ }
+
+ private void PopulateImages(IHasImages item, List<LocalImageInfo> images, Dictionary<string, string> files)
+ {
+ var imagePrefix = string.Empty;
+
+ var baseItem = item as BaseItem;
+ if (baseItem != null && baseItem.IsInMixedFolder)
+ {
+ imagePrefix = Path.GetFileNameWithoutExtension(item.Path) + "-";
+ }
+
+ PopulatePrimaryImages(item, images, files, imagePrefix);
+ PopulateBackdrops(item, images, files, imagePrefix);
+ PopulateScreenshots(images, files, imagePrefix);
+
+ AddImage(files, images, imagePrefix + "logo", ImageType.Logo);
+ AddImage(files, images, imagePrefix + "clearart", ImageType.Art);
+ AddImage(files, images, imagePrefix + "disc", ImageType.Disc);
+ AddImage(files, images, imagePrefix + "cdart", ImageType.Disc);
+ AddImage(files, images, imagePrefix + "box", ImageType.Box);
+ AddImage(files, images, imagePrefix + "back", ImageType.BoxRear);
+ AddImage(files, images, imagePrefix + "boxrear", ImageType.BoxRear);
+ AddImage(files, images, imagePrefix + "menu", ImageType.Menu);
+
+ // Banner
+ AddImage(files, images, imagePrefix + "banner", ImageType.Banner);
+
+ // Thumb
+ AddImage(files, images, imagePrefix + "thumb", ImageType.Thumb);
+ AddImage(files, images, imagePrefix + "landscape", ImageType.Thumb);
+
+ var season = item as Season;
+
+ if (season != null)
+ {
+ PopulateSeasonImagesFromSeriesFolder(season, images);
+ }
+ }
+
+ private void PopulatePrimaryImages(IHasImages item, List<LocalImageInfo> images, Dictionary<string, string> files, string imagePrefix)
+ {
+ AddImage(files, images, imagePrefix + "folder", ImageType.Primary);
+ AddImage(files, images, imagePrefix + "cover", ImageType.Primary);
+ AddImage(files, images, imagePrefix + "poster", ImageType.Primary);
+ AddImage(files, images, imagePrefix + "default", ImageType.Primary);
+
+ // Support plex/xbmc convention
+ if (item is Series)
+ {
+ AddImage(files, images, imagePrefix + "show", ImageType.Primary);
+ }
+
+ // Support plex/xbmc convention
+ if (item is Movie || item is MusicVideo || item is AdultVideo)
+ {
+ AddImage(files, images, imagePrefix + "movie", ImageType.Primary);
+ }
+
+ if (string.IsNullOrEmpty(item.Path))
+ {
+ var name = Path.GetFileNameWithoutExtension(item.Path);
+
+ if (!string.IsNullOrEmpty(name))
+ {
+ AddImage(files, images, name, ImageType.Primary);
+ AddImage(files, images, name + "-poster", ImageType.Primary);
+ }
+ }
+ }
+
+ private void PopulateBackdrops(IHasImages item, List<LocalImageInfo> images, Dictionary<string, string> files, string imagePrefix)
+ {
+ PopulateBackdrops(images, files, imagePrefix, "backdrop", "backdrop", ImageType.Backdrop);
+
+ if (string.IsNullOrEmpty(item.Path))
+ {
+ var name = Path.GetFileNameWithoutExtension(item.Path);
+
+ if (!string.IsNullOrEmpty(name))
+ {
+ AddImage(files, images, imagePrefix + name + "-fanart", ImageType.Backdrop);
+ }
+ }
+
+ PopulateBackdrops(images, files, imagePrefix, "fanart", "fanart-", ImageType.Backdrop);
+ PopulateBackdrops(images, files, imagePrefix, "background", "background-", ImageType.Backdrop);
+ PopulateBackdrops(images, files, imagePrefix, "art", "art-", ImageType.Backdrop);
+
+ string extraFanartFolder;
+ if (files.TryGetValue("extrafanart", out extraFanartFolder))
+ {
+ PopulateBackdropsFromExtraFanart(extraFanartFolder, images);
+ }
+ }
+
+ private void PopulateBackdropsFromExtraFanart(string path, List<LocalImageInfo> images)
+ {
+ var imageFiles = Directory.EnumerateFiles(path, "*", SearchOption.TopDirectoryOnly)
+ .Where(i =>
+ {
+ var extension = Path.GetExtension(i);
+
+ if (string.IsNullOrEmpty(extension))
+ {
+ return false;
+ }
+
+ return BaseItem.SupportedImageExtensions.Contains(extension, StringComparer.OrdinalIgnoreCase);
+ });
+
+ images.AddRange(imageFiles.Select(i => new LocalImageInfo
+ {
+ Path = i,
+ Type = ImageType.Backdrop
+ }));
+ }
+
+ private void PopulateScreenshots(List<LocalImageInfo> images, Dictionary<string, string> files, string imagePrefix)
+ {
+ PopulateBackdrops(images, files, imagePrefix, "screenshot", "screenshot", ImageType.Screenshot);
+ }
+
+ private void PopulateBackdrops(List<LocalImageInfo> images, Dictionary<string, string> files, string imagePrefix, string firstFileName, string subsequentFileNamePrefix, ImageType type)
+ {
+ AddImage(files, images, imagePrefix + firstFileName, type);
+
+ var unfound = 0;
+ for (var i = 1; i <= 20; i++)
+ {
+ // Screenshot Image
+ var found = AddImage(files, images, imagePrefix + subsequentFileNamePrefix + i, type);
+
+ if (!found)
+ {
+ unfound++;
+
+ if (unfound >= 3)
+ {
+ break;
+ }
+ }
+ }
+ }
+
+ private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+ private void PopulateSeasonImagesFromSeriesFolder(Season season, List<LocalImageInfo> images)
+ {
+ var seasonNumber = season.IndexNumber;
+
+ var series = season.Series;
+ if (!seasonNumber.HasValue || series.LocationType != LocationType.FileSystem)
+ {
+ return;
+ }
+
+ var files = GetFileDictionary(GetFiles(series, false));
+
+ // Try using the season name
+ var prefix = season.Name.ToLower().Replace(" ", string.Empty);
+
+ var filenamePrefixes = new List<string> { prefix };
+
+ var seasonMarker = seasonNumber.Value == 0
+ ? "-specials"
+ : seasonNumber.Value.ToString("00", _usCulture);
+
+ // Get this one directly from the file system since we have to go up a level
+ if (!string.Equals(prefix, seasonMarker, StringComparison.OrdinalIgnoreCase))
+ {
+ filenamePrefixes.Add("season" + seasonMarker);
+ }
+
+ foreach (var filename in filenamePrefixes)
+ {
+ AddImage(files, images, filename + "-poster", ImageType.Primary);
+ AddImage(files, images, filename + "-fanart", ImageType.Backdrop);
+ AddImage(files, images, filename + "-banner", ImageType.Banner);
+ AddImage(files, images, filename + "-landscape", ImageType.Thumb);
+ }
+ }
+
+ private Dictionary<string, string> GetFileDictionary(IEnumerable<string> paths)
+ {
+ var dict = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
+
+ foreach (var path in paths)
+ {
+ var filename = Path.GetFileName(path);
+
+ if (!string.IsNullOrEmpty(filename))
+ {
+ dict[filename] = path;
+ }
+ }
+
+ return dict;
+ }
+
+ private bool AddImage(Dictionary<string, string> dict, List<LocalImageInfo> images, string name, ImageType type)
+ {
+ var image = GetImage(dict, name);
+
+ if (image != null)
+ {
+ images.Add(new LocalImageInfo
+ {
+ Path = image,
+ Type = type
+ });
+
+ return true;
+ }
+
+ return false;
+ }
+
+ private string GetImage(Dictionary<string, string> dict, string name)
+ {
+ return BaseItem.SupportedImageExtensions
+ .Select(i =>
+ {
+ var filename = name + i;
+ string path;
+
+ return dict.TryGetValue(filename, out path) ? path : null;
+ })
+ .FirstOrDefault(i => i != null);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/BaseXmlProvider.cs b/MediaBrowser.Providers/BaseXmlProvider.cs
new file mode 100644
index 000000000..eab5bb574
--- /dev/null
+++ b/MediaBrowser.Providers/BaseXmlProvider.cs
@@ -0,0 +1,34 @@
+using MediaBrowser.Common.IO;
+using MediaBrowser.Controller.Providers;
+using System;
+using System.IO;
+using System.Threading;
+
+namespace MediaBrowser.Providers
+{
+ public abstract class BaseXmlProvider: IHasChangeMonitor
+ {
+ protected static readonly SemaphoreSlim XmlParsingResourcePool = new SemaphoreSlim(4, 4);
+
+ protected IFileSystem FileSystem;
+
+ protected BaseXmlProvider(IFileSystem fileSystem)
+ {
+ FileSystem = fileSystem;
+ }
+
+ protected abstract string GetXmlPath(string path);
+
+ public bool HasChanged(IHasMetadata item, DateTime date)
+ {
+ var path = GetXmlPath(item.Path);
+
+ return FileSystem.GetLastWriteTimeUtc(path) > date;
+ }
+
+ public bool HasLocalMetadata(IHasMetadata item)
+ {
+ return File.Exists(GetXmlPath(item.Path));
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/CollectionFolderImageProvider.cs b/MediaBrowser.Providers/CollectionFolderImageProvider.cs
index 6c36dbf7e..12f13262d 100644
--- a/MediaBrowser.Providers/CollectionFolderImageProvider.cs
+++ b/MediaBrowser.Providers/CollectionFolderImageProvider.cs
@@ -1,5 +1,4 @@
-using System.Collections.Generic;
-using MediaBrowser.Common.IO;
+using MediaBrowser.Common.IO;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
@@ -7,6 +6,7 @@ using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
using System;
+using System.Collections.Generic;
using System.IO;
using System.Linq;
diff --git a/MediaBrowser.Providers/ImagesByName/GameGenresManualImageProvider.cs b/MediaBrowser.Providers/GameGenres/GameGenreImageProvider.cs
index 8207bb042..0dbf11450 100644
--- a/MediaBrowser.Providers/ImagesByName/GameGenresManualImageProvider.cs
+++ b/MediaBrowser.Providers/GameGenres/GameGenreImageProvider.cs
@@ -5,15 +5,17 @@ using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
+using MediaBrowser.Providers.Genres;
+using MediaBrowser.Providers.ImagesByName;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Providers.ImagesByName
+namespace MediaBrowser.Providers.GameGenres
{
- public class GameGenresManualImageProvider : IImageProvider
+ public class GameGenreImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly IHttpClient _httpClient;
@@ -21,7 +23,7 @@ namespace MediaBrowser.Providers.ImagesByName
private readonly SemaphoreSlim _listResourcePool = new SemaphoreSlim(1, 1);
- public GameGenresManualImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
+ public GameGenreImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
{
_config = config;
_httpClient = httpClient;
@@ -43,6 +45,15 @@ namespace MediaBrowser.Providers.ImagesByName
return item is GameGenre;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb
+ };
+ }
+
public Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
return GetImages(item, imageType == ImageType.Primary, imageType == ImageType.Thumb, cancellationToken);
@@ -120,9 +131,19 @@ namespace MediaBrowser.Providers.ImagesByName
return ImageUtils.EnsureList(url, file, _httpClient, _fileSystem, _listResourcePool, cancellationToken);
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = GenreImageProvider.ImageDownloadResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/GameGenres/GameGenreMetadataService.cs b/MediaBrowser.Providers/GameGenres/GameGenreMetadataService.cs
new file mode 100644
index 000000000..389e2a275
--- /dev/null
+++ b/MediaBrowser.Providers/GameGenres/GameGenreMetadataService.cs
@@ -0,0 +1,42 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.GameGenres
+{
+ public class GameGenreMetadataService : MetadataService<GameGenre>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public GameGenreMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ /// <param name="mergeMetadataSettings">if set to <c>true</c> [merge metadata settings].</param>
+ protected override void MergeData(GameGenre source, GameGenre target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(GameGenre item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/ImagesByName/GenresManualImageProvider.cs b/MediaBrowser.Providers/Genres/GenreImageProvider.cs
index 469e133d0..189cc8cde 100644
--- a/MediaBrowser.Providers/ImagesByName/GenresManualImageProvider.cs
+++ b/MediaBrowser.Providers/Genres/GenreImageProvider.cs
@@ -5,15 +5,16 @@ using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
+using MediaBrowser.Providers.ImagesByName;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Providers.ImagesByName
+namespace MediaBrowser.Providers.Genres
{
- public class GenresManualImageProvider : IImageProvider
+ public class GenreImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly IHttpClient _httpClient;
@@ -21,7 +22,9 @@ namespace MediaBrowser.Providers.ImagesByName
private readonly SemaphoreSlim _listResourcePool = new SemaphoreSlim(1, 1);
- public GenresManualImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
+ public static SemaphoreSlim ImageDownloadResourcePool = new SemaphoreSlim(5, 5);
+
+ public GenreImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
{
_config = config;
_httpClient = httpClient;
@@ -43,6 +46,15 @@ namespace MediaBrowser.Providers.ImagesByName
return item is Genre;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb
+ };
+ }
+
public Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
return GetImages(item, imageType == ImageType.Primary, imageType == ImageType.Thumb, cancellationToken);
@@ -120,9 +132,19 @@ namespace MediaBrowser.Providers.ImagesByName
return ImageUtils.EnsureList(url, file, _httpClient, _fileSystem, _listResourcePool, cancellationToken);
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = ImageDownloadResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/Genres/GenreMetadataService.cs b/MediaBrowser.Providers/Genres/GenreMetadataService.cs
new file mode 100644
index 000000000..83253a190
--- /dev/null
+++ b/MediaBrowser.Providers/Genres/GenreMetadataService.cs
@@ -0,0 +1,42 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.Genres
+{
+ public class GenreMetadataService : MetadataService<Genre>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public GenreMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ /// <param name="mergeMetadataSettings">if set to <c>true</c> [merge metadata settings].</param>
+ protected override void MergeData(Genre source, Genre target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(Genre item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/ImageFromMediaLocationProvider.cs b/MediaBrowser.Providers/ImageFromMediaLocationProvider.cs
index 2146d927b..bc58f3178 100644
--- a/MediaBrowser.Providers/ImageFromMediaLocationProvider.cs
+++ b/MediaBrowser.Providers/ImageFromMediaLocationProvider.cs
@@ -145,17 +145,6 @@ namespace MediaBrowser.Providers
cancellationToken.ThrowIfCancellationRequested();
- // Make sure current backdrop paths still exist
- item.ValidateBackdrops();
-
- var hasScreenshots = item as IHasScreenshots;
- if (hasScreenshots != null)
- {
- hasScreenshots.ValidateScreenshots();
- }
-
- cancellationToken.ThrowIfCancellationRequested();
-
var args = GetResolveArgsContainingImages(item);
PopulateBaseItemImages(item, args);
diff --git a/MediaBrowser.Providers/ImagesByName/GameGenreImageProvider.cs b/MediaBrowser.Providers/ImagesByName/GameGenreImageProvider.cs
deleted file mode 100644
index a0dbb83dc..000000000
--- a/MediaBrowser.Providers/ImagesByName/GameGenreImageProvider.cs
+++ /dev/null
@@ -1,160 +0,0 @@
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Net;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.ImagesByName
-{
- public class GameGenreImageProvider : BaseMetadataProvider
- {
- private readonly IProviderManager _providerManager;
- private readonly SemaphoreSlim _resourcePool = new SemaphoreSlim(5, 5);
-
- public GameGenreImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- }
-
- public override bool Supports(BaseItem item)
- {
- return item is GameGenre;
- }
-
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- return false;
- }
-
- // Try again periodically in case new images were added
- if ((DateTime.UtcNow - providerInfo.LastRefreshed).TotalDays > 7)
- {
- return true;
- }
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "8";
- }
- }
-
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, GameGenresManualImageProvider.ProviderName).ConfigureAwait(false);
-
- await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- if (!item.LockedFields.Contains(MetadataFields.Images))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Primary))
- {
- await SaveImage(item, images, ImageType.Primary, cancellationToken).ConfigureAwait(false);
- }
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Thumb))
- {
- await SaveImage(item, images, ImageType.Thumb, cancellationToken).ConfigureAwait(false);
- }
- }
-
- if (!item.LockedFields.Contains(MetadataFields.Backdrops))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (item.BackdropImagePaths.Count == 0)
- {
- foreach (var image in images.Where(i => i.Type == ImageType.Backdrop))
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, ImageType.Backdrop, null, cancellationToken)
- .ConfigureAwait(false);
-
- break;
- }
- }
- }
- }
-
-
- private async Task SaveImage(BaseItem item, IEnumerable<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
- {
- foreach (var image in images.Where(i => i.Type == type))
- {
- try
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, type, null, cancellationToken).ConfigureAwait(false);
- break;
- }
- catch (HttpException ex)
- {
- // Sometimes fanart has bad url's in their xml
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
- {
- continue;
- }
- break;
- }
- }
- }
-
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Third; }
- }
- }
-}
diff --git a/MediaBrowser.Providers/ImagesByName/GenreImageProvider.cs b/MediaBrowser.Providers/ImagesByName/GenreImageProvider.cs
deleted file mode 100644
index 5744ef5fa..000000000
--- a/MediaBrowser.Providers/ImagesByName/GenreImageProvider.cs
+++ /dev/null
@@ -1,160 +0,0 @@
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Net;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.ImagesByName
-{
- public class GenreImageProvider : BaseMetadataProvider
- {
- private readonly IProviderManager _providerManager;
- private readonly SemaphoreSlim _resourcePool = new SemaphoreSlim(5, 5);
-
- public GenreImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- }
-
- public override bool Supports(BaseItem item)
- {
- return item is Genre;
- }
-
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- return false;
- }
-
- // Try again periodically in case new images were added
- if ((DateTime.UtcNow - providerInfo.LastRefreshed).TotalDays > 7)
- {
- return true;
- }
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "8";
- }
- }
-
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, GenresManualImageProvider.ProviderName).ConfigureAwait(false);
-
- await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- if (!item.LockedFields.Contains(MetadataFields.Images))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Primary))
- {
- await SaveImage(item, images, ImageType.Primary, cancellationToken).ConfigureAwait(false);
- }
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Thumb))
- {
- await SaveImage(item, images, ImageType.Thumb, cancellationToken).ConfigureAwait(false);
- }
- }
-
- if (!item.LockedFields.Contains(MetadataFields.Backdrops))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (item.BackdropImagePaths.Count == 0)
- {
- foreach (var image in images.Where(i => i.Type == ImageType.Backdrop))
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, ImageType.Backdrop, null, cancellationToken)
- .ConfigureAwait(false);
-
- break;
- }
- }
- }
- }
-
-
- private async Task SaveImage(BaseItem item, IEnumerable<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
- {
- foreach (var image in images.Where(i => i.Type == type))
- {
- try
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, type, null, cancellationToken).ConfigureAwait(false);
- break;
- }
- catch (HttpException ex)
- {
- // Sometimes fanart has bad url's in their xml
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
- {
- continue;
- }
- break;
- }
- }
- }
-
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Third; }
- }
- }
-}
diff --git a/MediaBrowser.Providers/ImagesByName/MusicGenreImageProvider.cs b/MediaBrowser.Providers/ImagesByName/MusicGenreImageProvider.cs
deleted file mode 100644
index 5b05a7b63..000000000
--- a/MediaBrowser.Providers/ImagesByName/MusicGenreImageProvider.cs
+++ /dev/null
@@ -1,161 +0,0 @@
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Entities.Audio;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Net;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.ImagesByName
-{
- public class MusicGenreImageProvider : BaseMetadataProvider
- {
- private readonly IProviderManager _providerManager;
- private readonly SemaphoreSlim _resourcePool = new SemaphoreSlim(5, 5);
-
- public MusicGenreImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- }
-
- public override bool Supports(BaseItem item)
- {
- return item is MusicGenre;
- }
-
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- return false;
- }
-
- // Try again periodically in case new images were added
- if ((DateTime.UtcNow - providerInfo.LastRefreshed).TotalDays > 7)
- {
- return true;
- }
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "8";
- }
- }
-
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, MusicGenresManualImageProvider.ProviderName).ConfigureAwait(false);
-
- await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- if (!item.LockedFields.Contains(MetadataFields.Images))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Primary))
- {
- await SaveImage(item, images, ImageType.Primary, cancellationToken).ConfigureAwait(false);
- }
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Thumb))
- {
- await SaveImage(item, images, ImageType.Thumb, cancellationToken).ConfigureAwait(false);
- }
- }
-
- if (!item.LockedFields.Contains(MetadataFields.Backdrops))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (item.BackdropImagePaths.Count == 0)
- {
- foreach (var image in images.Where(i => i.Type == ImageType.Backdrop))
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, ImageType.Backdrop, null, cancellationToken)
- .ConfigureAwait(false);
-
- break;
- }
- }
- }
- }
-
-
- private async Task SaveImage(BaseItem item, IEnumerable<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
- {
- foreach (var image in images.Where(i => i.Type == type))
- {
- try
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, type, null, cancellationToken).ConfigureAwait(false);
- break;
- }
- catch (HttpException ex)
- {
- // Sometimes fanart has bad url's in their xml
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
- {
- continue;
- }
- break;
- }
- }
- }
-
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Third; }
- }
- }
-}
diff --git a/MediaBrowser.Providers/ImagesByName/StudioImageProvider.cs b/MediaBrowser.Providers/ImagesByName/StudioImageProvider.cs
deleted file mode 100644
index 3035b6014..000000000
--- a/MediaBrowser.Providers/ImagesByName/StudioImageProvider.cs
+++ /dev/null
@@ -1,160 +0,0 @@
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Net;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.ImagesByName
-{
- public class StudioImageProvider : BaseMetadataProvider
- {
- private readonly IProviderManager _providerManager;
- private readonly SemaphoreSlim _resourcePool = new SemaphoreSlim(5, 5);
-
- public StudioImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- }
-
- public override bool Supports(BaseItem item)
- {
- return item is Studio;
- }
-
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- return false;
- }
-
- // Try again periodically in case new images were added
- if ((DateTime.UtcNow - providerInfo.LastRefreshed).TotalDays > 7)
- {
- return true;
- }
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "6";
- }
- }
-
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Thumb))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, StudiosManualImageProvider.ProviderName).ConfigureAwait(false);
-
- await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- if (!item.LockedFields.Contains(MetadataFields.Images))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Primary))
- {
- await SaveImage(item, images, ImageType.Primary, cancellationToken).ConfigureAwait(false);
- }
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!item.HasImage(ImageType.Thumb))
- {
- await SaveImage(item, images, ImageType.Thumb, cancellationToken).ConfigureAwait(false);
- }
- }
-
- if (!item.LockedFields.Contains(MetadataFields.Backdrops))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- if (item.BackdropImagePaths.Count == 0)
- {
- foreach (var image in images.Where(i => i.Type == ImageType.Backdrop))
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, ImageType.Backdrop, null, cancellationToken)
- .ConfigureAwait(false);
-
- break;
- }
- }
- }
- }
-
-
- private async Task SaveImage(BaseItem item, IEnumerable<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
- {
- foreach (var image in images.Where(i => i.Type == type))
- {
- try
- {
- await _providerManager.SaveImage(item, image.Url, _resourcePool, type, null, cancellationToken).ConfigureAwait(false);
- break;
- }
- catch (HttpException ex)
- {
- // Sometimes fanart has bad url's in their xml
- if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
- {
- continue;
- }
- break;
- }
- }
- }
-
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Third; }
- }
- }
-}
diff --git a/MediaBrowser.Providers/LiveTv/ChannelMetadataService.cs b/MediaBrowser.Providers/LiveTv/ChannelMetadataService.cs
new file mode 100644
index 000000000..dd44ba7aa
--- /dev/null
+++ b/MediaBrowser.Providers/LiveTv/ChannelMetadataService.cs
@@ -0,0 +1,37 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.LiveTv;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.LiveTv
+{
+ public class ChannelMetadataService : MetadataService<LiveTvChannel>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public ChannelMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ protected override void MergeData(LiveTvChannel source, LiveTvChannel target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(LiveTvChannel item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/LiveTv/ChannelProviderFromXml.cs b/MediaBrowser.Providers/LiveTv/ChannelProviderFromXml.cs
deleted file mode 100644
index 8ee2553d0..000000000
--- a/MediaBrowser.Providers/LiveTv/ChannelProviderFromXml.cs
+++ /dev/null
@@ -1,91 +0,0 @@
-using MediaBrowser.Common.IO;
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.LiveTv;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Logging;
-using System;
-using System.IO;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.LiveTv
-{
- class ChannelProviderFromXml : BaseMetadataProvider
- {
- private readonly IFileSystem _fileSystem;
-
- public ChannelProviderFromXml(ILogManager logManager, IServerConfigurationManager configurationManager, IFileSystem fileSystem)
- : base(logManager, configurationManager)
- {
- _fileSystem = fileSystem;
- }
-
- /// <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 LiveTvChannel;
- }
-
- /// <summary>
- /// Gets the priority.
- /// </summary>
- /// <value>The priority.</value>
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Second; }
- }
-
- private const string XmlFileName = "channel.xml";
- protected override bool NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
- {
- var xml = item.ResolveArgs.GetMetaFileByPath(Path.Combine(item.MetaLocation, XmlFileName));
-
- if (xml == null)
- {
- return false;
- }
-
- return _fileSystem.GetLastWriteTimeUtc(xml) > item.DateLastSaved;
- }
-
- /// <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 async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- var metadataFile = item.ResolveArgs.GetMetaFileByPath(Path.Combine(item.MetaLocation, XmlFileName));
-
- if (metadataFile != null)
- {
- var path = metadataFile.FullName;
-
- await XmlParsingResourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
-
- try
- {
- new BaseItemXmlParser<LiveTvChannel>(Logger).Fetch((LiveTvChannel)item, path, cancellationToken);
- }
- finally
- {
- XmlParsingResourcePool.Release();
- }
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- return false;
- }
- }
-}
diff --git a/MediaBrowser.Providers/LiveTv/ChannelXmlProvider.cs b/MediaBrowser.Providers/LiveTv/ChannelXmlProvider.cs
new file mode 100644
index 000000000..544685f66
--- /dev/null
+++ b/MediaBrowser.Providers/LiveTv/ChannelXmlProvider.cs
@@ -0,0 +1,59 @@
+using MediaBrowser.Common.IO;
+using MediaBrowser.Controller.LiveTv;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Logging;
+using System.IO;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.LiveTv
+{
+ public class ChannelXmlProvider : BaseXmlProvider, ILocalMetadataProvider<LiveTvChannel>
+ {
+ private readonly ILogger _logger;
+
+ public ChannelXmlProvider(IFileSystem fileSystem, ILogger logger)
+ : base(fileSystem)
+ {
+ _logger = logger;
+ }
+
+ public async Task<MetadataResult<LiveTvChannel>> GetMetadata(string path, CancellationToken cancellationToken)
+ {
+ path = GetXmlPath(path);
+
+ var result = new MetadataResult<LiveTvChannel>();
+
+ await XmlParsingResourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
+
+ try
+ {
+ var item = new LiveTvChannel();
+
+ new BaseItemXmlParser<LiveTvChannel>(_logger).Fetch(item, path, cancellationToken);
+ result.HasMetadata = true;
+ result.Item = item;
+ }
+ catch (FileNotFoundException)
+ {
+ result.HasMetadata = false;
+ }
+ finally
+ {
+ XmlParsingResourcePool.Release();
+ }
+
+ return result;
+ }
+
+ public string Name
+ {
+ get { return "Media Browser Xml"; }
+ }
+
+ protected override string GetXmlPath(string path)
+ {
+ return Path.Combine(path, "channel.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/LiveTv/ProgramMetadataService.cs b/MediaBrowser.Providers/LiveTv/ProgramMetadataService.cs
new file mode 100644
index 000000000..da032eb8f
--- /dev/null
+++ b/MediaBrowser.Providers/LiveTv/ProgramMetadataService.cs
@@ -0,0 +1,41 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.LiveTv;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.LiveTv
+{
+ public class ProgramMetadataService : MetadataService<LiveTvProgram>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public ProgramMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ protected override void MergeData(LiveTvProgram source, LiveTvProgram target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(LiveTvProgram item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Server.Implementations/Providers/ImageSaver.cs b/MediaBrowser.Providers/Manager/ImageSaver.cs
index ec797b688..2decba161 100644
--- a/MediaBrowser.Server.Implementations/Providers/ImageSaver.cs
+++ b/MediaBrowser.Providers/Manager/ImageSaver.cs
@@ -3,7 +3,7 @@ using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Entities.TV;
-using MediaBrowser.Controller.IO;
+using MediaBrowser.Controller.Library;
using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
@@ -15,7 +15,7 @@ using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Server.Implementations.Providers
+namespace MediaBrowser.Providers.Manager
{
/// <summary>
/// Class ImageSaver
@@ -36,7 +36,7 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <summary>
/// The _directory watchers
/// </summary>
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly IFileSystem _fileSystem;
private readonly ILogger _logger;
@@ -44,11 +44,11 @@ namespace MediaBrowser.Server.Implementations.Providers
/// Initializes a new instance of the <see cref="ImageSaver"/> class.
/// </summary>
/// <param name="config">The config.</param>
- /// <param name="directoryWatchers">The directory watchers.</param>
- public ImageSaver(IServerConfigurationManager config, IDirectoryWatchers directoryWatchers, IFileSystem fileSystem, ILogger logger)
+ /// <param name="libraryMonitor">The directory watchers.</param>
+ public ImageSaver(IServerConfigurationManager config, ILibraryMonitor libraryMonitor, IFileSystem fileSystem, ILogger logger)
{
_config = config;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_fileSystem = fileSystem;
_logger = logger;
_remoteImageCache = new FileSystemRepository(config.ApplicationPaths.DownloadedImagesDataPath);
@@ -160,7 +160,7 @@ namespace MediaBrowser.Server.Implementations.Providers
// Delete the current path
if (!string.IsNullOrEmpty(currentPath) && !paths.Contains(currentPath, StringComparer.OrdinalIgnoreCase))
{
- _directoryWatchers.TemporarilyIgnore(currentPath);
+ _libraryMonitor.ReportFileSystemChangeBeginning(currentPath);
try
{
@@ -179,7 +179,7 @@ namespace MediaBrowser.Server.Implementations.Providers
}
finally
{
- _directoryWatchers.RemoveTempIgnore(currentPath);
+ _libraryMonitor.ReportFileSystemChangeComplete(currentPath, false);
}
}
}
@@ -197,8 +197,8 @@ namespace MediaBrowser.Server.Implementations.Providers
var parentFolder = Path.GetDirectoryName(path);
- _directoryWatchers.TemporarilyIgnore(path);
- _directoryWatchers.TemporarilyIgnore(parentFolder);
+ _libraryMonitor.ReportFileSystemChangeBeginning(path);
+ _libraryMonitor.ReportFileSystemChangeBeginning(parentFolder);
try
{
@@ -223,8 +223,8 @@ namespace MediaBrowser.Server.Implementations.Providers
}
finally
{
- _directoryWatchers.RemoveTempIgnore(path);
- _directoryWatchers.RemoveTempIgnore(parentFolder);
+ _libraryMonitor.ReportFileSystemChangeComplete(path, false);
+ _libraryMonitor.ReportFileSystemChangeComplete(parentFolder, false);
}
}
@@ -348,6 +348,9 @@ namespace MediaBrowser.Server.Implementations.Providers
case ImageType.Art:
filename = "clearart";
break;
+ case ImageType.BoxRear:
+ filename = "back";
+ break;
case ImageType.Disc:
filename = item is MusicAlbum ? "cdart" : "disc";
break;
diff --git a/MediaBrowser.Providers/Manager/ItemImageProvider.cs b/MediaBrowser.Providers/Manager/ItemImageProvider.cs
new file mode 100644
index 000000000..e8bae1d2f
--- /dev/null
+++ b/MediaBrowser.Providers/Manager/ItemImageProvider.cs
@@ -0,0 +1,435 @@
+using MediaBrowser.Common.Extensions;
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Model.Net;
+using MediaBrowser.Model.Providers;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Net;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.Manager
+{
+ public class ItemImageProvider
+ {
+ private readonly ILogger _logger;
+ private readonly IProviderManager _providerManager;
+ private readonly IServerConfigurationManager _config;
+
+ public ItemImageProvider(ILogger logger, IProviderManager providerManager, IServerConfigurationManager config)
+ {
+ _logger = logger;
+ _providerManager = providerManager;
+ _config = config;
+ }
+
+ public bool ValidateImages(IHasImages item, IEnumerable<IImageProvider> providers)
+ {
+ var hasChanges = item.ValidateImages();
+
+ foreach (var provider in providers.OfType<IImageFileProvider>())
+ {
+ var images = provider.GetImages(item);
+
+ if (MergeImages(item, images))
+ {
+ hasChanges = true;
+ }
+ }
+
+ return hasChanges;
+ }
+
+ public async Task<RefreshResult> RefreshImages(IHasImages item, IEnumerable<IImageProvider> imageProviders, ImageRefreshOptions options, CancellationToken cancellationToken)
+ {
+ var result = new RefreshResult { UpdateType = ItemUpdateType.Unspecified };
+
+ var providers = GetImageProviders(item, imageProviders).ToList();
+
+ var providerIds = new List<Guid>();
+
+ foreach (var provider in providers.OfType<IRemoteImageProvider>())
+ {
+ await RefreshFromProvider(item, provider, options, result, cancellationToken).ConfigureAwait(false);
+
+ providerIds.Add(provider.GetType().FullName.GetMD5());
+ }
+
+ foreach (var provider in providers.OfType<IDynamicImageProvider>())
+ {
+ await RefreshFromProvider(item, provider, result, cancellationToken).ConfigureAwait(false);
+
+ providerIds.Add(provider.GetType().FullName.GetMD5());
+ }
+
+ result.Providers = providerIds;
+
+ return result;
+ }
+
+ /// <summary>
+ /// Refreshes from provider.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="provider">The provider.</param>
+ /// <param name="result">The result.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ private async Task RefreshFromProvider(IHasImages item, IDynamicImageProvider provider, RefreshResult result, CancellationToken cancellationToken)
+ {
+ _logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name);
+
+ try
+ {
+ var images = provider.GetSupportedImages(item);
+
+ foreach (var imageType in images)
+ {
+ if (!item.HasImage(imageType))
+ {
+ var response = await provider.GetImage(item, imageType, cancellationToken).ConfigureAwait(false);
+
+ if (response.HasImage)
+ {
+ var mimeType = "image/" + response.Format.ToString().ToLower();
+
+ await _providerManager.SaveImage((BaseItem)item, response.Stream, mimeType, imageType, null, Guid.NewGuid().ToString(), cancellationToken).ConfigureAwait(false);
+
+ result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
+ }
+ }
+ }
+ }
+ catch (OperationCanceledException)
+ {
+ throw;
+ }
+ catch (Exception ex)
+ {
+ result.ErrorMessage = ex.Message;
+ result.Status = ProviderRefreshStatus.CompletedWithErrors;
+ _logger.ErrorException("Error in {0}", ex, provider.Name);
+ }
+ }
+
+ /// <summary>
+ /// Image types that are only one per item
+ /// </summary>
+ private readonly ImageType[] _singularImages =
+ {
+ ImageType.Primary,
+ ImageType.Art,
+ ImageType.Banner,
+ ImageType.Box,
+ ImageType.BoxRear,
+ ImageType.Disc,
+ ImageType.Logo,
+ ImageType.Menu,
+ ImageType.Thumb
+ };
+
+ /// <summary>
+ /// Determines if an item already contains the given images
+ /// </summary>
+ /// <param name="item"></param>
+ /// <param name="images"></param>
+ /// <returns></returns>
+ private bool ContainsImages(IHasImages item, List<ImageType> images)
+ {
+ if (_singularImages.Any(i => images.Contains(i) && !item.HasImage(i)))
+ {
+ return false;
+ }
+
+ if (images.Contains(ImageType.Backdrop) && item.BackdropImagePaths.Count < GetMaxBackdropCount(item))
+ {
+ return false;
+ }
+
+ if (images.Contains(ImageType.Screenshot))
+ {
+ var hasScreenshots = item as IHasScreenshots;
+ if (hasScreenshots != null)
+ {
+ if (hasScreenshots.ScreenshotImagePaths.Count < GetMaxBackdropCount(item))
+ {
+ return false;
+ }
+ }
+ }
+
+ return true;
+ }
+
+ /// <summary>
+ /// Refreshes from provider.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="provider">The provider.</param>
+ /// <param name="options">The options.</param>
+ /// <param name="result">The result.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task.</returns>
+ private async Task RefreshFromProvider(IHasImages item, IRemoteImageProvider provider, ImageRefreshOptions options, RefreshResult result, CancellationToken cancellationToken)
+ {
+ try
+ {
+ // TODO: Also factor in IsConfiguredToDownloadImage
+ if (ContainsImages(item, provider.GetSupportedImages(item).ToList()))
+ {
+ return;
+ }
+
+ _logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name);
+
+ var images = await provider.GetAllImages(item, cancellationToken).ConfigureAwait(false);
+ var list = images.ToList();
+
+ foreach (var type in _singularImages)
+ {
+ if (IsConfiguredToDownloadImage(item, type) && !item.HasImage(type))
+ {
+ await DownloadImage(item, provider, result, list, type, cancellationToken).ConfigureAwait(false);
+ }
+ }
+
+ await DownloadBackdrops(item, provider, result, list, cancellationToken).ConfigureAwait(false);
+
+ var hasScreenshots = item as IHasScreenshots;
+ if (hasScreenshots != null)
+ {
+ await DownloadScreenshots(hasScreenshots, provider, result, list, cancellationToken).ConfigureAwait(false);
+ }
+ }
+ catch (OperationCanceledException)
+ {
+ throw;
+ }
+ catch (Exception ex)
+ {
+ result.ErrorMessage = ex.Message;
+ result.Status = ProviderRefreshStatus.CompletedWithErrors;
+ _logger.ErrorException("Error in {0}", ex, provider.Name);
+ }
+ }
+
+ /// <summary>
+ /// Gets the image providers.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="imageProviders">The image providers.</param>
+ /// <returns>IEnumerable{IImageProvider}.</returns>
+ private IEnumerable<IImageProvider> GetImageProviders(IHasImages item, IEnumerable<IImageProvider> imageProviders)
+ {
+ var providers = imageProviders.Where(i =>
+ {
+ try
+ {
+ return i.Supports(item);
+ }
+ catch (Exception ex)
+ {
+ _logger.ErrorException("Error in ImageProvider.Supports", ex, i.Name);
+
+ return false;
+ }
+ });
+
+ if (!_config.Configuration.EnableInternetProviders)
+ {
+ providers = providers.Where(i => !(i is IRemoteImageProvider));
+ }
+
+ return providers.OrderBy(i => i.Order);
+ }
+
+ private bool MergeImages(IHasImages item, List<LocalImageInfo> images)
+ {
+ var changed = false;
+
+ foreach (var type in _singularImages)
+ {
+ var image = images.FirstOrDefault(i => i.Type == type);
+
+ if (image != null)
+ {
+ var oldPath = item.GetImagePath(type);
+
+ item.SetImagePath(type, image.Path);
+
+ if (!string.Equals(oldPath, image.Path, StringComparison.OrdinalIgnoreCase))
+ {
+ changed = true;
+ }
+ }
+ }
+
+ // The change reporting will only be accurate at the count level
+ // Improve this if/when needed
+ var backdrops = images.Where(i => i.Type == ImageType.Backdrop).ToList();
+ if (backdrops.Count > 0)
+ {
+ var oldCount = item.BackdropImagePaths.Count;
+
+ item.BackdropImagePaths = item.BackdropImagePaths
+ .Concat(backdrops.Select(i => i.Path))
+ .Distinct(StringComparer.OrdinalIgnoreCase)
+ .ToList();
+
+ if (oldCount != item.BackdropImagePaths.Count)
+ {
+ changed = true;
+ }
+ }
+
+ var hasScreenshots = item as IHasScreenshots;
+ if (hasScreenshots != null)
+ {
+ var screenshots = images.Where(i => i.Type == ImageType.Screenshot).ToList();
+
+ if (screenshots.Count > 0)
+ {
+ var oldCount = hasScreenshots.ScreenshotImagePaths.Count;
+
+ hasScreenshots.ScreenshotImagePaths = hasScreenshots.ScreenshotImagePaths
+ .Concat(screenshots.Select(i => i.Path))
+ .Distinct(StringComparer.OrdinalIgnoreCase)
+ .ToList();
+
+ if (oldCount != hasScreenshots.ScreenshotImagePaths.Count)
+ {
+ changed = true;
+ }
+ }
+ }
+
+ return changed;
+ }
+
+ private async Task DownloadImage(IHasImages item, IRemoteImageProvider provider, RefreshResult result, IEnumerable<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
+ {
+ foreach (var image in images.Where(i => i.Type == type))
+ {
+ var url = image.Url;
+
+ try
+ {
+ var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);
+
+ await _providerManager.SaveImage((BaseItem)item, response.Content, response.ContentType, type, null, url, cancellationToken).ConfigureAwait(false);
+
+ result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
+ break;
+ }
+ catch (HttpException ex)
+ {
+ // Sometimes providers send back bad url's. Just move onto the next image
+ if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
+ {
+ continue;
+ }
+ break;
+ }
+ }
+ }
+
+ private async Task DownloadBackdrops(IHasImages item, IRemoteImageProvider provider, RefreshResult result, IEnumerable<RemoteImageInfo> images, CancellationToken cancellationToken)
+ {
+ const ImageType imageType = ImageType.Backdrop;
+ var maxCount = GetMaxBackdropCount(item);
+
+ foreach (var image in images.Where(i => i.Type == imageType))
+ {
+ if (item.BackdropImagePaths.Count >= maxCount)
+ {
+ break;
+ }
+
+ var url = image.Url;
+
+ if (item.ContainsImageWithSourceUrl(url))
+ {
+ continue;
+ }
+
+ try
+ {
+ var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);
+
+ await _providerManager.SaveImage((BaseItem)item, response.Content, response.ContentType, imageType, null, url, cancellationToken).ConfigureAwait(false);
+ result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
+ break;
+ }
+ catch (HttpException ex)
+ {
+ // Sometimes providers send back bad url's. Just move onto the next image
+ if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
+ {
+ continue;
+ }
+ break;
+ }
+ }
+ }
+
+ private async Task DownloadScreenshots(IHasScreenshots item, IRemoteImageProvider provider, RefreshResult result, IEnumerable<RemoteImageInfo> images, CancellationToken cancellationToken)
+ {
+ const ImageType imageType = ImageType.Screenshot;
+ var maxCount = GetMaxScreenshotCount(item);
+
+ foreach (var image in images.Where(i => i.Type == imageType))
+ {
+ if (item.ScreenshotImagePaths.Count >= maxCount)
+ {
+ break;
+ }
+
+ var url = image.Url;
+
+ if (item.ContainsImageWithSourceUrl(url))
+ {
+ continue;
+ }
+
+ try
+ {
+ var response = await provider.GetImageResponse(url, cancellationToken).ConfigureAwait(false);
+
+ await _providerManager.SaveImage((BaseItem)item, response.Content, response.ContentType, imageType, null, url, cancellationToken).ConfigureAwait(false);
+ result.UpdateType = result.UpdateType | ItemUpdateType.ImageUpdate;
+ break;
+ }
+ catch (HttpException ex)
+ {
+ // Sometimes providers send back bad url's. Just move onto the next image
+ if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
+ {
+ continue;
+ }
+ break;
+ }
+ }
+ }
+
+ private bool IsConfiguredToDownloadImage(IHasImages item, ImageType type)
+ {
+ return true;
+ }
+
+ private int GetMaxBackdropCount(IHasImages item)
+ {
+ return 1;
+ }
+
+ private int GetMaxScreenshotCount(IHasScreenshots item)
+ {
+ return 1;
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/Manager/MetadataService.cs b/MediaBrowser.Providers/Manager/MetadataService.cs
new file mode 100644
index 000000000..7916d7e86
--- /dev/null
+++ b/MediaBrowser.Providers/Manager/MetadataService.cs
@@ -0,0 +1,358 @@
+using MediaBrowser.Common.Extensions;
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.Manager
+{
+ public abstract class MetadataService<TItemType> : IMetadataService
+ where TItemType : IHasMetadata, new()
+ {
+ protected readonly IServerConfigurationManager ServerConfigurationManager;
+ protected readonly ILogger Logger;
+ protected readonly IProviderManager ProviderManager;
+ private readonly IProviderRepository _providerRepo;
+
+ private IMetadataProvider<TItemType>[] _providers = { };
+
+ private IImageProvider[] _imageProviders = { };
+
+ protected MetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo)
+ {
+ ServerConfigurationManager = serverConfigurationManager;
+ Logger = logger;
+ ProviderManager = providerManager;
+ _providerRepo = providerRepo;
+ }
+
+ /// <summary>
+ /// Adds the parts.
+ /// </summary>
+ /// <param name="providers">The providers.</param>
+ /// <param name="imageProviders">The image providers.</param>
+ public void AddParts(IEnumerable<IMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders)
+ {
+ _providers = providers.OfType<IMetadataProvider<TItemType>>()
+ .ToArray();
+
+ _imageProviders = imageProviders.OrderBy(i => i.Order).ToArray();
+ }
+
+ /// <summary>
+ /// Saves the provider result.
+ /// </summary>
+ /// <param name="result">The result.</param>
+ /// <returns>Task.</returns>
+ protected Task SaveProviderResult(MetadataStatus result)
+ {
+ return _providerRepo.SaveMetadataStatus(result, CancellationToken.None);
+ }
+
+ /// <summary>
+ /// Gets the last result.
+ /// </summary>
+ /// <param name="itemId">The item identifier.</param>
+ /// <returns>ProviderResult.</returns>
+ protected MetadataStatus GetLastResult(Guid itemId)
+ {
+ return _providerRepo.GetMetadataStatus(itemId) ?? new MetadataStatus { ItemId = itemId };
+ }
+
+ public async Task RefreshMetadata(IHasMetadata item, MetadataRefreshOptions options, CancellationToken cancellationToken)
+ {
+ var itemOfType = (TItemType)item;
+
+ var updateType = ItemUpdateType.Unspecified;
+ var lastResult = GetLastResult(item.Id);
+ var refreshResult = lastResult;
+ refreshResult.LastErrorMessage = string.Empty;
+ refreshResult.LastStatus = ProviderRefreshStatus.Success;
+
+ var imageProviders = GetImageProviders(item).ToList();
+ var itemImageProvider = new ItemImageProvider(Logger, ProviderManager, ServerConfigurationManager);
+ var localImagesFailed = false;
+
+ // Start by validating images
+ try
+ {
+ // Always validate images and check for new locally stored ones.
+ if (itemImageProvider.ValidateImages(item, imageProviders))
+ {
+ updateType = updateType | ItemUpdateType.ImageUpdate;
+ }
+ }
+ catch (Exception ex)
+ {
+ localImagesFailed = true;
+ Logger.ErrorException("Error validating images for {0}", ex, item.Path ?? item.Name);
+ refreshResult.AddStatus(ProviderRefreshStatus.Failure, ex.Message);
+ }
+
+ // Next run metadata providers
+ if (options.MetadataRefreshMode != MetadataRefreshMode.None)
+ {
+ var providers = GetProviders(item, lastResult.DateLastMetadataRefresh.HasValue, options).ToList();
+
+ if (providers.Count > 0)
+ {
+ var result = await RefreshWithProviders(itemOfType, options, providers, cancellationToken).ConfigureAwait(false);
+
+ updateType = updateType | result.UpdateType;
+ refreshResult.AddStatus(result.Status, result.ErrorMessage);
+ refreshResult.SetDateLastMetadataRefresh(DateTime.UtcNow);
+ refreshResult.AddImageProvidersRefreshed(result.Providers);
+ }
+ }
+
+ // Next run remote image providers, but only if local image providers didn't throw an exception
+ if (!localImagesFailed)
+ {
+ if ((options.ImageRefreshMode == MetadataRefreshMode.EnsureMetadata && !lastResult.DateLastImagesRefresh.HasValue) ||
+ options.ImageRefreshMode == MetadataRefreshMode.FullRefresh)
+ {
+ var result = await itemImageProvider.RefreshImages(itemOfType, imageProviders, options, cancellationToken).ConfigureAwait(false);
+
+ updateType = updateType | result.UpdateType;
+ refreshResult.AddStatus(result.Status, result.ErrorMessage);
+ refreshResult.SetDateLastImagesRefresh(DateTime.UtcNow);
+ refreshResult.AddImageProvidersRefreshed(result.Providers);
+ }
+ }
+
+ var providersHadChanges = updateType > ItemUpdateType.Unspecified;
+
+ if (options.ForceSave || providersHadChanges)
+ {
+ if (string.IsNullOrEmpty(item.Name))
+ {
+ throw new InvalidOperationException("Item has no name");
+ }
+
+ // Save to database
+ await SaveItem(itemOfType, updateType, cancellationToken);
+ }
+
+ if (providersHadChanges || refreshResult.IsDirty)
+ {
+ await SaveProviderResult(refreshResult).ConfigureAwait(false);
+ }
+ }
+
+ /// <summary>
+ /// Gets the providers.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <param name="hasRefreshedMetadata">if set to <c>true</c> [has refreshed metadata].</param>
+ /// <param name="options">The options.</param>
+ /// <returns>IEnumerable{`0}.</returns>
+ protected virtual IEnumerable<IMetadataProvider> GetProviders(IHasMetadata item, bool hasRefreshedMetadata, MetadataRefreshOptions options)
+ {
+ // Get providers to refresh
+ var providers = _providers.Where(i => CanRefresh(i, item)).ToList();
+
+ // Run all if either of these flags are true
+ var runAllProviders = options.ReplaceAllMetadata || options.MetadataRefreshMode == MetadataRefreshMode.FullRefresh || !hasRefreshedMetadata;
+
+ if (!runAllProviders)
+ {
+ // Avoid implicitly captured closure
+ var currentItem = item;
+
+ var providersWithChanges = providers.OfType<IHasChangeMonitor>()
+ .Where(i => i.HasChanged(currentItem, currentItem.DateLastSaved))
+ .ToList();
+
+ // If local providers are the only ones with changes, then just run those
+ if (providersWithChanges.All(i => i is ILocalMetadataProvider))
+ {
+ providers = providers.Where(i => i is ILocalMetadataProvider).ToList();
+ }
+ }
+
+ return providers;
+ }
+
+ /// <summary>
+ /// Determines whether this instance can refresh the specified provider.
+ /// </summary>
+ /// <param name="provider">The provider.</param>
+ /// <param name="item">The item.</param>
+ /// <returns><c>true</c> if this instance can refresh the specified provider; otherwise, <c>false</c>.</returns>
+ protected bool CanRefresh(IMetadataProvider provider, IHasMetadata item)
+ {
+ if (!ServerConfigurationManager.Configuration.EnableInternetProviders && provider is IRemoteMetadataProvider)
+ {
+ return false;
+ }
+
+ if (item.LocationType != LocationType.FileSystem && provider is ILocalMetadataProvider)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ protected abstract Task SaveItem(TItemType item, ItemUpdateType reason, CancellationToken cancellationToken);
+
+ protected virtual ItemId GetId(IHasMetadata item)
+ {
+ return new ItemId
+ {
+ MetadataCountryCode = item.GetPreferredMetadataCountryCode(),
+ MetadataLanguage = item.GetPreferredMetadataLanguage(),
+ Name = item.Name,
+ ProviderIds = item.ProviderIds
+ };
+ }
+
+ public bool CanRefresh(IHasMetadata item)
+ {
+ return item is TItemType;
+ }
+
+ protected virtual async Task<RefreshResult> RefreshWithProviders(TItemType item, MetadataRefreshOptions options, List<IMetadataProvider> providers, CancellationToken cancellationToken)
+ {
+ var refreshResult = new RefreshResult
+ {
+ UpdateType = ItemUpdateType.Unspecified,
+ Providers = providers.Select(i => i.GetType().FullName.GetMD5()).ToList()
+ };
+
+ var temp = new TItemType();
+
+ // If replacing all metadata, run internet providers first
+ if (options.ReplaceAllMetadata)
+ {
+ await ExecuteRemoteProviders(item, temp, providers.OfType<IRemoteMetadataProvider<TItemType>>(), refreshResult, cancellationToken).ConfigureAwait(false);
+ }
+
+ var hasLocalMetadata = false;
+
+ foreach (var provider in providers.OfType<ILocalMetadataProvider<TItemType>>())
+ {
+ Logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name);
+
+ try
+ {
+ var localItem = await provider.GetMetadata(item.Path, cancellationToken).ConfigureAwait(false);
+
+ if (localItem.HasMetadata)
+ {
+ MergeData(localItem.Item, temp, new List<MetadataFields>(), false, true);
+ refreshResult.UpdateType = refreshResult.UpdateType | ItemUpdateType.MetadataImport;
+
+ // Only one local provider allowed per item
+ hasLocalMetadata = true;
+ break;
+ }
+ }
+ catch (OperationCanceledException)
+ {
+ throw;
+ }
+ catch (Exception ex)
+ {
+ // If a local provider fails, consider that a failure
+ refreshResult.Status = ProviderRefreshStatus.Failure;
+ refreshResult.ErrorMessage = ex.Message;
+ Logger.ErrorException("Error in {0}", ex, provider.Name);
+
+ // If the local provider fails don't continue with remote providers because the user's saved metadata could be lost
+ return refreshResult;
+ }
+ }
+
+ if (!options.ReplaceAllMetadata && !hasLocalMetadata)
+ {
+ await ExecuteRemoteProviders(item, temp, providers.OfType<IRemoteMetadataProvider<TItemType>>(), refreshResult, cancellationToken).ConfigureAwait(false);
+ }
+
+ MergeData(temp, item, item.LockedFields, true, true);
+
+ return refreshResult;
+ }
+
+ private async Task ExecuteRemoteProviders(TItemType item, TItemType temp, IEnumerable<IRemoteMetadataProvider<TItemType>> providers, RefreshResult refreshResult, CancellationToken cancellationToken)
+ {
+ var id = GetId(item);
+
+ foreach (var provider in providers)
+ {
+ Logger.Debug("Running {0} for {1}", provider.GetType().Name, item.Path ?? item.Name);
+
+ try
+ {
+ var result = await provider.GetMetadata(id, cancellationToken).ConfigureAwait(false);
+
+ if (result.HasMetadata)
+ {
+ MergeData(result.Item, temp, new List<MetadataFields>(), false, false);
+
+ refreshResult.UpdateType = refreshResult.UpdateType | ItemUpdateType.MetadataDownload;
+ }
+ }
+ catch (OperationCanceledException)
+ {
+ throw;
+ }
+ catch (Exception ex)
+ {
+ refreshResult.Status = ProviderRefreshStatus.CompletedWithErrors;
+ refreshResult.ErrorMessage = ex.Message;
+ Logger.ErrorException("Error in {0}", ex, provider.Name);
+ }
+ }
+ }
+
+ protected abstract void MergeData(TItemType source, TItemType target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings);
+
+ public virtual int Order
+ {
+ get
+ {
+ return 0;
+ }
+ }
+
+ private IEnumerable<IImageProvider> GetImageProviders(IHasImages item)
+ {
+ var providers = _imageProviders.Where(i =>
+ {
+ try
+ {
+ return i.Supports(item);
+ }
+ catch (Exception ex)
+ {
+ Logger.ErrorException("Error in ImageProvider.Supports", ex, i.Name);
+
+ return false;
+ }
+ });
+
+ if (!ServerConfigurationManager.Configuration.EnableInternetProviders)
+ {
+ providers = providers.Where(i => !(i is IRemoteImageProvider));
+ }
+
+ return providers.OrderBy(i => i.Order);
+ }
+ }
+
+ public class RefreshResult
+ {
+ public ItemUpdateType UpdateType { get; set; }
+ public ProviderRefreshStatus Status { get; set; }
+ public string ErrorMessage { get; set; }
+ public List<Guid> Providers { get; set; }
+ }
+}
diff --git a/MediaBrowser.Server.Implementations/Providers/ProviderManager.cs b/MediaBrowser.Providers/Manager/ProviderManager.cs
index cbfd7d74d..3696bd02f 100644
--- a/MediaBrowser.Server.Implementations/Providers/ProviderManager.cs
+++ b/MediaBrowser.Providers/Manager/ProviderManager.cs
@@ -2,7 +2,6 @@
using MediaBrowser.Common.Net;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Entities.Movies;
using MediaBrowser.Controller.IO;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Persistence;
@@ -17,7 +16,7 @@ using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Server.Implementations.Providers
+namespace MediaBrowser.Providers.Manager
{
/// <summary>
/// Class ProviderManager
@@ -37,7 +36,7 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <summary>
/// The _directory watchers
/// </summary>
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
/// <summary>
/// Gets or sets the configuration manager.
@@ -51,26 +50,32 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <value>The metadata providers enumerable.</value>
private BaseMetadataProvider[] MetadataProviders { get; set; }
+ private IRemoteImageProvider[] RemoteImageProviders { get; set; }
private IImageProvider[] ImageProviders { get; set; }
+
private readonly IFileSystem _fileSystem;
- private readonly IItemRepository _itemRepo;
+ private readonly IProviderRepository _providerRepo;
+
+ private IMetadataService[] _metadataServices = { };
/// <summary>
/// Initializes a new instance of the <see cref="ProviderManager" /> class.
/// </summary>
/// <param name="httpClient">The HTTP client.</param>
/// <param name="configurationManager">The configuration manager.</param>
- /// <param name="directoryWatchers">The directory watchers.</param>
+ /// <param name="libraryMonitor">The directory watchers.</param>
/// <param name="logManager">The log manager.</param>
- public ProviderManager(IHttpClient httpClient, IServerConfigurationManager configurationManager, IDirectoryWatchers directoryWatchers, ILogManager logManager, IFileSystem fileSystem, IItemRepository itemRepo)
+ /// <param name="fileSystem">The file system.</param>
+ /// <param name="providerRepo">The provider repo.</param>
+ public ProviderManager(IHttpClient httpClient, IServerConfigurationManager configurationManager, ILibraryMonitor libraryMonitor, ILogManager logManager, IFileSystem fileSystem, IProviderRepository providerRepo)
{
_logger = logManager.GetLogger("ProviderManager");
_httpClient = httpClient;
ConfigurationManager = configurationManager;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_fileSystem = fileSystem;
- _itemRepo = itemRepo;
+ _providerRepo = providerRepo;
}
/// <summary>
@@ -78,11 +83,34 @@ namespace MediaBrowser.Server.Implementations.Providers
/// </summary>
/// <param name="providers">The providers.</param>
/// <param name="imageProviders">The image providers.</param>
- public void AddParts(IEnumerable<BaseMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders)
+ /// <param name="metadataServices">The metadata services.</param>
+ /// <param name="metadataProviders">The metadata providers.</param>
+ public void AddParts(IEnumerable<BaseMetadataProvider> providers, IEnumerable<IImageProvider> imageProviders, IEnumerable<IMetadataService> metadataServices, IEnumerable<IMetadataProvider> metadataProviders)
{
MetadataProviders = providers.OrderBy(e => e.Priority).ToArray();
- ImageProviders = imageProviders.OrderByDescending(i => i.Priority).ToArray();
+ ImageProviders = imageProviders.OrderBy(i => i.Order).ToArray();
+ RemoteImageProviders = ImageProviders.OfType<IRemoteImageProvider>().ToArray();
+
+ _metadataServices = metadataServices.OrderBy(i => i.Order).ToArray();
+
+ var providerList = metadataProviders.ToList();
+ foreach (var service in _metadataServices)
+ {
+ service.AddParts(providerList, ImageProviders);
+ }
+ }
+
+ public Task RefreshMetadata(IHasMetadata item, MetadataRefreshOptions options, CancellationToken cancellationToken)
+ {
+ var service = _metadataServices.FirstOrDefault(i => i.CanRefresh(item));
+
+ if (service != null)
+ {
+ return service.RefreshMetadata(item, options, cancellationToken);
+ }
+
+ return ((BaseItem)item).RefreshMetadataDirect(cancellationToken, options.ForceSave, options.ReplaceAllMetadata);
}
/// <summary>
@@ -91,9 +119,9 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <param name="item">The item.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="force">if set to <c>true</c> [force].</param>
- /// <param name="allowSlowProviders">if set to <c>true</c> [allow slow providers].</param>
/// <returns>Task{System.Boolean}.</returns>
- public async Task<ItemUpdateType?> ExecuteMetadataProviders(BaseItem item, CancellationToken cancellationToken, bool force = false, bool allowSlowProviders = true)
+ /// <exception cref="System.ArgumentNullException">item</exception>
+ public async Task<ItemUpdateType?> ExecuteMetadataProviders(BaseItem item, CancellationToken cancellationToken, bool force = false)
{
if (item == null)
{
@@ -108,7 +136,7 @@ namespace MediaBrowser.Server.Implementations.Providers
var providerHistories = item.DateLastSaved == default(DateTime) ?
new List<BaseProviderInfo>() :
- _itemRepo.GetProviderHistory(item.Id).ToList();
+ _providerRepo.GetProviderHistory(item.Id).ToList();
// Run the normal providers sequentially in order of priority
foreach (var provider in MetadataProviders)
@@ -126,12 +154,6 @@ namespace MediaBrowser.Server.Implementations.Providers
continue;
}
- // Skip if is slow and we aren't allowing slow ones
- if (provider.IsSlow && !allowSlowProviders)
- {
- continue;
- }
-
// Put this check below the await because the needs refresh of the next tier of providers may depend on the previous ones running
// This is the case for the fan art provider which depends on the movie and tv providers having run before them
if (provider.RequiresInternet && item.DontFetchMeta && provider.EnforceDontFetchMetadata)
@@ -179,7 +201,7 @@ namespace MediaBrowser.Server.Implementations.Providers
if (result.HasValue || force)
{
- await _itemRepo.SaveProviderHistory(item.Id, providerHistories, cancellationToken);
+ await _providerRepo.SaveProviderHistory(item.Id, providerHistories, cancellationToken);
}
return result;
@@ -293,7 +315,7 @@ namespace MediaBrowser.Server.Implementations.Providers
}
//Tell the watchers to ignore
- _directoryWatchers.TemporarilyIgnore(path);
+ _libraryMonitor.ReportFileSystemChangeBeginning(path);
if (dataToSave.CanSeek)
{
@@ -316,7 +338,7 @@ namespace MediaBrowser.Server.Implementations.Providers
finally
{
//Remove the ignore
- _directoryWatchers.RemoveTempIgnore(path);
+ _libraryMonitor.ReportFileSystemChangeComplete(path, false);
}
}
@@ -358,7 +380,7 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <returns>Task.</returns>
public Task SaveImage(BaseItem item, Stream source, string mimeType, ImageType type, int? imageIndex, string sourceUrl, CancellationToken cancellationToken)
{
- return new ImageSaver(ConfigurationManager, _directoryWatchers, _fileSystem, _logger).SaveImage(item, source, mimeType, type, imageIndex, sourceUrl, cancellationToken);
+ return new ImageSaver(ConfigurationManager, _libraryMonitor, _fileSystem, _logger).SaveImage(item, source, mimeType, type, imageIndex, sourceUrl, cancellationToken);
}
/// <summary>
@@ -371,7 +393,7 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
public async Task<IEnumerable<RemoteImageInfo>> GetAvailableRemoteImages(BaseItem item, CancellationToken cancellationToken, string providerName = null, ImageType? type = null)
{
- var providers = GetImageProviders(item);
+ var providers = GetRemoteImageProviders(item);
if (!string.IsNullOrEmpty(providerName))
{
@@ -396,7 +418,7 @@ namespace MediaBrowser.Server.Implementations.Providers
/// <param name="preferredLanguage">The preferred language.</param>
/// <param name="type">The type.</param>
/// <returns>Task{IEnumerable{RemoteImageInfo}}.</returns>
- private async Task<IEnumerable<RemoteImageInfo>> GetImages(BaseItem item, CancellationToken cancellationToken, IImageProvider i, string preferredLanguage, ImageType? type = null)
+ private async Task<IEnumerable<RemoteImageInfo>> GetImages(BaseItem item, CancellationToken cancellationToken, IRemoteImageProvider i, string preferredLanguage, ImageType? type = null)
{
try
{
@@ -414,7 +436,7 @@ namespace MediaBrowser.Server.Implementations.Providers
}
catch (Exception ex)
{
- _logger.ErrorException("{0} failed in GetImages for type {1}", ex, i.GetType().Name, item.GetType().Name);
+ _logger.ErrorException("{0} failed in GetImageInfos for type {1}", ex, i.GetType().Name, item.GetType().Name);
return new List<RemoteImageInfo>();
}
}
@@ -430,14 +452,9 @@ namespace MediaBrowser.Server.Implementations.Providers
return images;
}
- /// <summary>
- /// Gets the supported image providers.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <returns>IEnumerable{IImageProvider}.</returns>
- public IEnumerable<IImageProvider> GetImageProviders(BaseItem item)
+ private IEnumerable<IRemoteImageProvider> GetRemoteImageProviders(BaseItem item)
{
- return ImageProviders.Where(i =>
+ return RemoteImageProviders.Where(i =>
{
try
{
@@ -448,6 +465,22 @@ namespace MediaBrowser.Server.Implementations.Providers
_logger.ErrorException("{0} failed in Supports for type {1}", ex, i.GetType().Name, item.GetType().Name);
return false;
}
+
+ });
+ }
+
+ /// <summary>
+ /// Gets the supported image providers.
+ /// </summary>
+ /// <param name="item">The item.</param>
+ /// <returns>IEnumerable{IImageProvider}.</returns>
+ public IEnumerable<ImageProviderInfo> GetImageProviderInfo(BaseItem item)
+ {
+ return GetRemoteImageProviders(item).Select(i => new ImageProviderInfo
+ {
+ Name = i.Name,
+ Order = i.Order
+
});
}
}
diff --git a/MediaBrowser.Providers/MediaBrowser.Providers.csproj b/MediaBrowser.Providers/MediaBrowser.Providers.csproj
index e686e7eda..b44d3608e 100644
--- a/MediaBrowser.Providers/MediaBrowser.Providers.csproj
+++ b/MediaBrowser.Providers/MediaBrowser.Providers.csproj
@@ -64,6 +64,17 @@
</Reference>
</ItemGroup>
<ItemGroup>
+ <Compile Include="All\LocalImageProvider.cs" />
+ <Compile Include="GameGenres\GameGenreMetadataService.cs" />
+ <Compile Include="Genres\GenreMetadataService.cs" />
+ <Compile Include="LiveTv\ChannelMetadataService.cs" />
+ <Compile Include="LiveTv\ChannelXmlProvider.cs" />
+ <Compile Include="LiveTv\ProgramMetadataService.cs" />
+ <Compile Include="Manager\ImageSaver.cs" />
+ <Compile Include="Manager\ItemImageProvider.cs" />
+ <Compile Include="Manager\ProviderManager.cs" />
+ <Compile Include="Manager\MetadataService.cs" />
+ <Compile Include="BaseXmlProvider.cs" />
<Compile Include="CollectionFolderImageProvider.cs" />
<Compile Include="FanartBaseProvider.cs" />
<Compile Include="FolderProviderFromXml.cs" />
@@ -72,14 +83,10 @@
<Compile Include="Games\GameSystemProviderFromXml.cs" />
<Compile Include="ImageFromMediaLocationProvider.cs" />
<Compile Include="ImagesByNameProvider.cs" />
- <Compile Include="ImagesByName\MusicGenreImageProvider.cs" />
- <Compile Include="ImagesByName\MusicGenresManualImageProvider.cs" />
- <Compile Include="ImagesByName\GameGenreImageProvider.cs" />
- <Compile Include="ImagesByName\GameGenresManualImageProvider.cs" />
- <Compile Include="ImagesByName\GenreImageProvider.cs" />
- <Compile Include="ImagesByName\GenresManualImageProvider.cs" />
+ <Compile Include="MusicGenres\MusicGenreImageProvider.cs" />
+ <Compile Include="GameGenres\GameGenreImageProvider.cs" />
+ <Compile Include="Genres\GenreImageProvider.cs" />
<Compile Include="ImagesByName\ImageUtils.cs" />
- <Compile Include="LiveTv\ChannelProviderFromXml.cs" />
<Compile Include="MediaInfo\AudioImageProvider.cs" />
<Compile Include="MediaInfo\BaseFFProbeProvider.cs" />
<Compile Include="MediaInfo\FFProbeAudioInfoProvider.cs" />
@@ -88,8 +95,8 @@
<Compile Include="Movies\BoxSetProviderFromXml.cs" />
<Compile Include="Movies\ManualMovieDbImageProvider.cs" />
<Compile Include="Movies\ManualFanartMovieImageProvider.cs" />
- <Compile Include="Movies\ManualMovieDbPersonImageProvider.cs" />
- <Compile Include="Movies\MovieDbPersonImageProvider.cs" />
+ <Compile Include="MusicGenres\MusicGenreMetadataService.cs" />
+ <Compile Include="People\MovieDbPersonImageProvider.cs" />
<Compile Include="Movies\MovieUpdatesPrescanTask.cs" />
<Compile Include="Movies\MovieXmlParser.cs" />
<Compile Include="Movies\FanArtMovieProvider.cs" />
@@ -98,8 +105,6 @@
<Compile Include="Movies\MovieDbProvider.cs" />
<Compile Include="Movies\MovieProviderFromXml.cs" />
<Compile Include="Movies\OpenMovieDatabaseProvider.cs" />
- <Compile Include="Movies\PersonProviderFromXml.cs" />
- <Compile Include="Movies\MovieDbPersonProvider.cs" />
<Compile Include="Music\AlbumInfoFromSongProvider.cs" />
<Compile Include="Music\AlbumProviderFromXml.cs" />
<Compile Include="Music\ArtistInfoFromSongProvider.cs" />
@@ -118,7 +123,11 @@
<Compile Include="Music\MusicBrainzAlbumProvider.cs" />
<Compile Include="Music\MusicVideoXmlParser.cs" />
<Compile Include="Music\SoundtrackPostScanTask.cs" />
+ <Compile Include="People\PersonMetadataService.cs" />
+ <Compile Include="People\PersonXmlProvider.cs" />
+ <Compile Include="People\MovieDbPersonProvider.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />
+ <Compile Include="ProviderUtils.cs" />
<Compile Include="RefreshIntrosTask.cs" />
<Compile Include="Savers\AlbumXmlSaver.cs" />
<Compile Include="Savers\ArtistXmlSaver.cs" />
@@ -133,8 +142,8 @@
<Compile Include="Savers\SeasonXmlSaver.cs" />
<Compile Include="Savers\SeriesXmlSaver.cs" />
<Compile Include="Savers\XmlSaverHelpers.cs" />
- <Compile Include="ImagesByName\StudioImageProvider.cs" />
- <Compile Include="ImagesByName\StudiosManualImageProvider.cs" />
+ <Compile Include="Studios\StudiosImageProvider.cs" />
+ <Compile Include="Studios\StudioMetadataService.cs" />
<Compile Include="TV\EpisodeImageFromMediaLocationProvider.cs" />
<Compile Include="TV\EpisodeIndexNumberProvider.cs" />
<Compile Include="TV\EpisodeProviderFromXml.cs" />
@@ -145,7 +154,7 @@
<Compile Include="TV\ManualFanartSeasonProvider.cs" />
<Compile Include="TV\ManualFanartSeriesProvider.cs" />
<Compile Include="TV\ManualTvdbEpisodeImageProvider.cs" />
- <Compile Include="TV\ManualTvdbPersonImageProvider.cs" />
+ <Compile Include="People\TvdbPersonImageProvider.cs" />
<Compile Include="TV\ManualTvdbSeasonImageProvider.cs" />
<Compile Include="TV\ManualTvdbSeriesImageProvider.cs" />
<Compile Include="TV\SeasonIndexNumberProvider.cs" />
@@ -157,7 +166,6 @@
<Compile Include="TV\SeriesPostScanTask.cs" />
<Compile Include="TV\SeriesProviderFromXml.cs" />
<Compile Include="TV\SeriesXmlParser.cs" />
- <Compile Include="TV\TvdbPersonImageProvider.cs" />
<Compile Include="TV\TvdbPrescanTask.cs" />
<Compile Include="TV\TvdbSeriesImageProvider.cs" />
<Compile Include="UserRootFolderNameProvider.cs" />
@@ -180,6 +188,7 @@
<ItemGroup>
<None Include="packages.config" />
</ItemGroup>
+ <ItemGroup />
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<Import Project="$(SolutionDir)\.nuget\NuGet.targets" Condition=" '$(ConfigurationName)' != 'Release Mono' " />
<!-- To modify your build process, add your task inside one of the targets below and uncomment it.
diff --git a/MediaBrowser.Providers/Movies/ManualFanartMovieImageProvider.cs b/MediaBrowser.Providers/Movies/ManualFanartMovieImageProvider.cs
index fae8cd591..c34bd47d7 100644
--- a/MediaBrowser.Providers/Movies/ManualFanartMovieImageProvider.cs
+++ b/MediaBrowser.Providers/Movies/ManualFanartMovieImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Dto;
@@ -16,14 +17,16 @@ using System.Xml;
namespace MediaBrowser.Providers.Movies
{
- public class ManualFanartMovieImageProvider : IImageProvider
+ public class ManualFanartMovieImageProvider : IRemoteImageProvider
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualFanartMovieImageProvider(IServerConfigurationManager config)
+ public ManualFanartMovieImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -41,6 +44,20 @@ namespace MediaBrowser.Providers.Movies
return FanArtMovieProvider.SupportsItem(item);
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb,
+ ImageType.Art,
+ ImageType.Logo,
+ ImageType.Disc,
+ ImageType.Banner,
+ ImageType.Backdrop
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -294,9 +311,19 @@ namespace MediaBrowser.Providers.Movies
}
}
- public int Priority
+ public int Order
{
get { return 1; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = FanartBaseProvider.FanArtResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/Movies/ManualMovieDbImageProvider.cs b/MediaBrowser.Providers/Movies/ManualMovieDbImageProvider.cs
index 21e50f400..ee1f14e09 100644
--- a/MediaBrowser.Providers/Movies/ManualMovieDbImageProvider.cs
+++ b/MediaBrowser.Providers/Movies/ManualMovieDbImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Dto;
@@ -14,15 +15,17 @@ using System.Threading.Tasks;
namespace MediaBrowser.Providers.Movies
{
- class ManualMovieDbImageProvider : IImageProvider
+ class ManualMovieDbImageProvider : IRemoteImageProvider
{
private readonly IJsonSerializer _jsonSerializer;
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualMovieDbImageProvider(IJsonSerializer jsonSerializer, IServerConfigurationManager config)
+ public ManualMovieDbImageProvider(IJsonSerializer jsonSerializer, IServerConfigurationManager config, IHttpClient httpClient)
{
_jsonSerializer = jsonSerializer;
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -40,6 +43,15 @@ namespace MediaBrowser.Providers.Movies
return MovieDbImagesProvider.SupportsItem(item);
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Backdrop
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -167,9 +179,19 @@ namespace MediaBrowser.Providers.Movies
return null;
}
- public int Priority
+ public int Order
{
- get { return 2; }
+ get { return 0; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = MovieDbProvider.Current.MovieDbResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/Movies/MovieDbPersonImageProvider.cs b/MediaBrowser.Providers/Movies/MovieDbPersonImageProvider.cs
deleted file mode 100644
index f6c908a7c..000000000
--- a/MediaBrowser.Providers/Movies/MovieDbPersonImageProvider.cs
+++ /dev/null
@@ -1,207 +0,0 @@
-using MediaBrowser.Common.IO;
-using MediaBrowser.Common.Net;
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.IO;
-using System.Linq;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.Movies
-{
- /// <summary>
- /// Class MovieDbPersonImageProvider.
- /// </summary>
- public class MovieDbPersonImageProvider : BaseMetadataProvider
- {
- /// <summary>
- /// The _provider manager
- /// </summary>
- private readonly IProviderManager _providerManager;
-
- private readonly IFileSystem _fileSystem;
-
- /// <summary>
- /// Initializes a new instance of the <see cref="MediaBrowser.Providers.Movies.MovieDbImagesProvider"/> class.
- /// </summary>
- /// <param name="logManager">The log manager.</param>
- /// <param name="configurationManager">The configuration manager.</param>
- /// <param name="providerManager">The provider manager.</param>
- public MovieDbPersonImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager, IFileSystem fileSystem)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- _fileSystem = fileSystem;
- }
-
- /// <summary>
- /// Gets the priority.
- /// </summary>
- /// <value>The priority.</value>
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Third; }
- }
-
- /// <summary>
- /// Supports 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 Person;
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
- }
-
- /// <summary>
- /// Gets a value indicating whether [requires internet].
- /// </summary>
- /// <value><c>true</c> if [requires internet]; otherwise, <c>false</c>.</value>
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- /// <summary>
- /// Gets a value indicating whether [refresh on version change].
- /// </summary>
- /// <value><c>true</c> if [refresh on version change]; otherwise, <c>false</c>.</value>
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- /// <summary>
- /// Gets the provider version.
- /// </summary>
- /// <value>The provider version.</value>
- protected override string ProviderVersion
- {
- get
- {
- return "3";
- }
- }
-
- /// <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)
- {
- if (string.IsNullOrEmpty(item.GetProviderId(MetadataProviders.Tmdb)))
- {
- return false;
- }
-
- // Don't refresh if we already have both poster and backdrop and we're not refreshing images
- if (item.HasImage(ImageType.Primary))
- {
- return false;
- }
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- /// <summary>
- /// Needses the refresh based on compare date.
- /// </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 NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
- {
- var provderId = item.GetProviderId(MetadataProviders.Tmdb);
-
- if (!string.IsNullOrEmpty(provderId))
- {
- // Process images
- var path = MovieDbPersonProvider.GetPersonDataFilePath(ConfigurationManager.ApplicationPaths, provderId);
-
- var fileInfo = new FileInfo(path);
-
- if (fileInfo.Exists)
- {
- return _fileSystem.GetLastWriteTimeUtc(fileInfo) > providerInfo.LastRefreshed;
- }
-
- return false;
- }
-
- return false;
- }
-
- /// <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 async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, ManualMovieDbPersonImageProvider.ProviderName).ConfigureAwait(false);
- await ProcessImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- /// <summary>
- /// Processes the images.
- /// </summary>
- /// <param name="item">The item.</param>
- /// <param name="images">The images.</param>
- /// <param name="cancellationToken">The cancellation token</param>
- /// <returns>Task.</returns>
- private async Task ProcessImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- var eligiblePosters = images
- .Where(i => i.Type == ImageType.Primary)
- .ToList();
-
- // poster
- if (eligiblePosters.Count > 0 && !item.HasImage(ImageType.Primary) && !item.LockedFields.Contains(MetadataFields.Images))
- {
- var poster = eligiblePosters[0];
-
- var url = poster.Url;
-
- var img = await MovieDbProvider.Current.GetMovieDbResponse(new HttpRequestOptions
- {
- Url = url,
- CancellationToken = cancellationToken
-
- }).ConfigureAwait(false);
-
- await _providerManager.SaveImage(item, img, MimeTypes.GetMimeType(url), ImageType.Primary, null, url, cancellationToken)
- .ConfigureAwait(false);
- }
- }
- }
-}
diff --git a/MediaBrowser.Providers/Movies/MovieDbPersonProvider.cs b/MediaBrowser.Providers/Movies/MovieDbPersonProvider.cs
deleted file mode 100644
index 70ad6611a..000000000
--- a/MediaBrowser.Providers/Movies/MovieDbPersonProvider.cs
+++ /dev/null
@@ -1,440 +0,0 @@
-using MediaBrowser.Common.Configuration;
-using MediaBrowser.Common.Extensions;
-using MediaBrowser.Common.IO;
-using MediaBrowser.Common.Net;
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Serialization;
-using System;
-using System.Collections.Generic;
-using System.Globalization;
-using System.IO;
-using System.Net;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.Movies
-{
- /// <summary>
- /// Class TmdbPersonProvider
- /// </summary>
- public class MovieDbPersonProvider : BaseMetadataProvider
- {
- protected readonly IProviderManager ProviderManager;
-
- internal static MovieDbPersonProvider Current { get; private set; }
-
- const string DataFileName = "info.json";
- private readonly IFileSystem _fileSystem;
-
- public MovieDbPersonProvider(IJsonSerializer jsonSerializer, ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager, IFileSystem fileSystem)
- : base(logManager, configurationManager)
- {
- if (jsonSerializer == null)
- {
- throw new ArgumentNullException("jsonSerializer");
- }
- JsonSerializer = jsonSerializer;
- ProviderManager = providerManager;
- _fileSystem = fileSystem;
- Current = this;
- }
-
- /// <summary>
- /// Gets the json serializer.
- /// </summary>
- /// <value>The json serializer.</value>
- protected IJsonSerializer JsonSerializer { get; private set; }
-
- /// <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 Person;
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "3";
- }
- }
-
- public override ItemUpdateType ItemUpdateType
- {
- get
- {
- return ItemUpdateType.MetadataDownload;
- }
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- if (HasAltMeta(item))
- return false;
-
- return base.NeedsRefreshInternal(item, providerInfo);
- }
-
- protected override bool NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
- {
- var provderId = item.GetProviderId(MetadataProviders.Tmdb);
-
- if (!string.IsNullOrEmpty(provderId))
- {
- // Process images
- var path = GetPersonDataPath(ConfigurationManager.ApplicationPaths, provderId);
-
- var file = Path.Combine(path, DataFileName);
- var fileInfo = new FileInfo(file);
-
- if (fileInfo.Exists)
- {
- return _fileSystem.GetLastWriteTimeUtc(fileInfo) > providerInfo.LastRefreshed;
- }
-
- return true;
- }
-
- return base.NeedsRefreshBasedOnCompareDate(item, providerInfo);
- }
-
- internal static string GetPersonDataPath(IApplicationPaths appPaths, string tmdbId)
- {
- var letter = tmdbId.GetMD5().ToString().Substring(0, 1);
-
- var seriesDataPath = Path.Combine(GetPersonsDataPath(appPaths), letter, tmdbId);
-
- return seriesDataPath;
- }
-
- internal static string GetPersonDataFilePath(IApplicationPaths appPaths, string tmdbId)
- {
- var letter = tmdbId.GetMD5().ToString().Substring(0, 1);
-
- var seriesDataPath = Path.Combine(GetPersonsDataPath(appPaths), letter, tmdbId);
-
- return Path.Combine(seriesDataPath, DataFileName);
- }
-
- internal static string GetPersonsDataPath(IApplicationPaths appPaths)
- {
- var dataPath = Path.Combine(appPaths.DataPath, "tmdb-people");
-
- return dataPath;
- }
-
- private bool HasAltMeta(BaseItem item)
- {
- return item.LocationType == LocationType.FileSystem && item.ResolveArgs.ContainsMetaFileByName("person.xml");
- }
-
- /// <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 async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- var person = (Person)item;
-
- var id = person.GetProviderId(MetadataProviders.Tmdb);
-
- // We don't already have an Id, need to fetch it
- if (string.IsNullOrEmpty(id))
- {
- id = await GetTmdbId(item, cancellationToken).ConfigureAwait(false);
- }
-
- cancellationToken.ThrowIfCancellationRequested();
-
- if (!string.IsNullOrEmpty(id))
- {
- await FetchInfo(person, id, force, cancellationToken).ConfigureAwait(false);
- }
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- /// <summary>
- /// Gets the priority.
- /// </summary>
- /// <value>The priority.</value>
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Second; }
- }
-
- /// <summary>
- /// Gets a value indicating whether [requires internet].
- /// </summary>
- /// <value><c>true</c> if [requires internet]; otherwise, <c>false</c>.</value>
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- private readonly CultureInfo _usCulture = new CultureInfo("en-US");
-
- /// <summary>
- /// Gets the TMDB id.
- /// </summary>
- /// <param name="person">The person.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task{System.String}.</returns>
- private async Task<string> GetTmdbId(BaseItem person, CancellationToken cancellationToken)
- {
- string url = string.Format(@"http://api.themoviedb.org/3/search/person?api_key={1}&query={0}", WebUtility.UrlEncode(person.Name), MovieDbProvider.ApiKey);
- PersonSearchResults searchResult = null;
-
- using (var json = await MovieDbProvider.Current.GetMovieDbResponse(new HttpRequestOptions
- {
- Url = url,
- CancellationToken = cancellationToken,
- AcceptHeader = MovieDbProvider.AcceptHeader
-
- }).ConfigureAwait(false))
- {
- searchResult = JsonSerializer.DeserializeFromStream<PersonSearchResults>(json);
- }
-
- return searchResult != null && searchResult.Total_Results > 0 ? searchResult.Results[0].Id.ToString(_usCulture) : null;
- }
-
- /// <summary>
- /// Fetches the info.
- /// </summary>
- /// <param name="person">The person.</param>
- /// <param name="id">The id.</param>
- /// <param name="isForcedRefresh">if set to <c>true</c> [is forced refresh].</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task.</returns>
- private async Task FetchInfo(Person person, string id, bool isForcedRefresh, CancellationToken cancellationToken)
- {
- await EnsurePersonInfo(id, cancellationToken).ConfigureAwait(false);
-
- if (isForcedRefresh || !HasAltMeta(person))
- {
- var dataFilePath = GetPersonDataFilePath(ConfigurationManager.ApplicationPaths, id);
-
- var info = JsonSerializer.DeserializeFromFile<PersonResult>(dataFilePath);
-
- cancellationToken.ThrowIfCancellationRequested();
-
- ProcessInfo(person, info);
- }
- }
-
- internal async Task EnsurePersonInfo(string id, CancellationToken cancellationToken)
- {
- var personDataPath = GetPersonDataPath(ConfigurationManager.ApplicationPaths, id);
-
- var fileInfo = _fileSystem.GetFileSystemInfo(personDataPath);
-
- if (fileInfo.Exists && (DateTime.UtcNow - _fileSystem.GetLastWriteTimeUtc(fileInfo)).TotalDays <= 7)
- {
- return;
- }
-
- var url = string.Format(@"http://api.themoviedb.org/3/person/{1}?api_key={0}&append_to_response=credits,images", MovieDbProvider.ApiKey, id);
-
- using (var json = await MovieDbProvider.Current.GetMovieDbResponse(new HttpRequestOptions
- {
- Url = url,
- CancellationToken = cancellationToken,
- AcceptHeader = MovieDbProvider.AcceptHeader
-
- }).ConfigureAwait(false))
- {
- Directory.CreateDirectory(personDataPath);
-
- using (var fs = _fileSystem.GetFileStream(Path.Combine(personDataPath, DataFileName), FileMode.Create, FileAccess.Write, FileShare.Read, true))
- {
- await json.CopyToAsync(fs).ConfigureAwait(false);
- }
- }
- }
-
- /// <summary>
- /// Processes the info.
- /// </summary>
- /// <param name="person">The person.</param>
- /// <param name="searchResult">The search result.</param>
- protected void ProcessInfo(Person person, PersonResult searchResult)
- {
- if (!person.LockedFields.Contains(MetadataFields.Overview))
- {
- person.Overview = searchResult.biography;
- }
-
- DateTime date;
-
- if (DateTime.TryParseExact(searchResult.birthday, "yyyy-MM-dd", new CultureInfo("en-US"), DateTimeStyles.None, out date))
- {
- person.PremiereDate = date.ToUniversalTime();
- }
-
- if (DateTime.TryParseExact(searchResult.deathday, "yyyy-MM-dd", new CultureInfo("en-US"), DateTimeStyles.None, out date))
- {
- person.EndDate = date.ToUniversalTime();
- }
-
- if (!string.IsNullOrEmpty(searchResult.homepage))
- {
- person.HomePageUrl = searchResult.homepage;
- }
-
- if (!person.LockedFields.Contains(MetadataFields.ProductionLocations))
- {
- if (!string.IsNullOrEmpty(searchResult.place_of_birth))
- {
- person.PlaceOfBirth = searchResult.place_of_birth;
- }
- }
-
- person.SetProviderId(MetadataProviders.Tmdb, searchResult.id.ToString(_usCulture));
- }
-
- #region Result Objects
- /// <summary>
- /// Class PersonSearchResult
- /// </summary>
- public class PersonSearchResult
- {
- /// <summary>
- /// Gets or sets a value indicating whether this <see cref="PersonSearchResult" /> is adult.
- /// </summary>
- /// <value><c>true</c> if adult; otherwise, <c>false</c>.</value>
- public bool Adult { get; set; }
- /// <summary>
- /// Gets or sets the id.
- /// </summary>
- /// <value>The id.</value>
- public int Id { get; set; }
- /// <summary>
- /// Gets or sets the name.
- /// </summary>
- /// <value>The name.</value>
- public string Name { get; set; }
- /// <summary>
- /// Gets or sets the profile_ path.
- /// </summary>
- /// <value>The profile_ path.</value>
- public string Profile_Path { get; set; }
- }
-
- /// <summary>
- /// Class PersonSearchResults
- /// </summary>
- public class PersonSearchResults
- {
- /// <summary>
- /// Gets or sets the page.
- /// </summary>
- /// <value>The page.</value>
- public int Page { get; set; }
- /// <summary>
- /// Gets or sets the results.
- /// </summary>
- /// <value>The results.</value>
- public List<PersonSearchResult> Results { get; set; }
- /// <summary>
- /// Gets or sets the total_ pages.
- /// </summary>
- /// <value>The total_ pages.</value>
- public int Total_Pages { get; set; }
- /// <summary>
- /// Gets or sets the total_ results.
- /// </summary>
- /// <value>The total_ results.</value>
- public int Total_Results { get; set; }
- }
-
- public class Cast
- {
- public int id { get; set; }
- public string title { get; set; }
- public string character { get; set; }
- public string original_title { get; set; }
- public string poster_path { get; set; }
- public string release_date { get; set; }
- public bool adult { get; set; }
- }
-
- public class Crew
- {
- public int id { get; set; }
- public string title { get; set; }
- public string original_title { get; set; }
- public string department { get; set; }
- public string job { get; set; }
- public string poster_path { get; set; }
- public string release_date { get; set; }
- public bool adult { get; set; }
- }
-
- public class Credits
- {
- public List<Cast> cast { get; set; }
- public List<Crew> crew { get; set; }
- }
-
- public class Profile
- {
- public string file_path { get; set; }
- public int width { get; set; }
- public int height { get; set; }
- public object iso_639_1 { get; set; }
- public double aspect_ratio { get; set; }
- }
-
- public class Images
- {
- public List<Profile> profiles { get; set; }
- }
-
- public class PersonResult
- {
- public bool adult { get; set; }
- public List<object> also_known_as { get; set; }
- public string biography { get; set; }
- public string birthday { get; set; }
- public string deathday { get; set; }
- public string homepage { get; set; }
- public int id { get; set; }
- public string imdb_id { get; set; }
- public string name { get; set; }
- public string place_of_birth { get; set; }
- public double popularity { get; set; }
- public string profile_path { get; set; }
- public Credits credits { get; set; }
- public Images images { get; set; }
- }
-
- #endregion
- }
-}
diff --git a/MediaBrowser.Providers/Movies/MovieDbProvider.cs b/MediaBrowser.Providers/Movies/MovieDbProvider.cs
index d6937293c..6f32aa135 100644
--- a/MediaBrowser.Providers/Movies/MovieDbProvider.cs
+++ b/MediaBrowser.Providers/Movies/MovieDbProvider.cs
@@ -697,7 +697,8 @@ namespace MediaBrowser.Providers.Movies
}
if (!movie.LockedFields.Contains(MetadataFields.Overview))
{
- movie.Overview = WebUtility.HtmlDecode(movieData.overview);
+ // Bug in Mono: WebUtility.HtmlDecode should return null if the string is null but in Mono it generate an System.ArgumentNullException.
+ movie.Overview = movieData.overview != null ? WebUtility.HtmlDecode(movieData.overview) : null;
movie.Overview = movie.Overview != null ? movie.Overview.Replace("\n\n", "\n") : null;
}
movie.HomePageUrl = movieData.homepage;
diff --git a/MediaBrowser.Providers/Movies/PersonProviderFromXml.cs b/MediaBrowser.Providers/Movies/PersonProviderFromXml.cs
deleted file mode 100644
index 014580025..000000000
--- a/MediaBrowser.Providers/Movies/PersonProviderFromXml.cs
+++ /dev/null
@@ -1,89 +0,0 @@
-using MediaBrowser.Common.IO;
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Logging;
-using System;
-using System.IO;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.Movies
-{
- class PersonProviderFromXml : BaseMetadataProvider
- {
- private readonly IFileSystem _fileSystem;
-
- public PersonProviderFromXml(ILogManager logManager, IServerConfigurationManager configurationManager, IFileSystem fileSystem)
- : base(logManager, configurationManager)
- {
- _fileSystem = fileSystem;
- }
-
- /// <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 Person;
- }
-
- /// <summary>
- /// Gets the priority.
- /// </summary>
- /// <value>The priority.</value>
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Second; }
- }
-
- private const string XmlFileName = "person.xml";
- protected override bool NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
- {
- var xml = item.ResolveArgs.GetMetaFileByPath(Path.Combine(item.MetaLocation, XmlFileName));
-
- if (xml == null)
- {
- return false;
- }
-
- return _fileSystem.GetLastWriteTimeUtc(xml) > item.DateLastSaved;
- }
-
- /// <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="providerInfo">The provider information.</param>
- /// <param name="cancellationToken">The cancellation token.</param>
- /// <returns>Task{System.Boolean}.</returns>
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- var metadataFile = item.ResolveArgs.GetMetaFileByPath(Path.Combine(item.MetaLocation, XmlFileName));
-
- if (metadataFile != null)
- {
- var path = metadataFile.FullName;
-
- await XmlParsingResourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
-
- try
- {
- new BaseItemXmlParser<Person>(Logger).Fetch((Person)item, path, cancellationToken);
- }
- finally
- {
- XmlParsingResourcePool.Release();
- }
- }
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
- }
-}
diff --git a/MediaBrowser.Providers/Music/ManualFanartAlbumProvider.cs b/MediaBrowser.Providers/Music/ManualFanartAlbumProvider.cs
index 5c923869f..d5108f309 100644
--- a/MediaBrowser.Providers/Music/ManualFanartAlbumProvider.cs
+++ b/MediaBrowser.Providers/Music/ManualFanartAlbumProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Providers;
@@ -17,14 +18,16 @@ using System.Xml;
namespace MediaBrowser.Providers.Music
{
- public class ManualFanartAlbumProvider : IImageProvider
+ public class ManualFanartAlbumProvider : IRemoteImageProvider
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualFanartAlbumProvider(IServerConfigurationManager config)
+ public ManualFanartAlbumProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -42,6 +45,15 @@ namespace MediaBrowser.Providers.Music
return item is MusicAlbum;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Disc
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -325,9 +337,19 @@ namespace MediaBrowser.Providers.Music
list.Add(info);
}
- public int Priority
+ public int Order
{
- get { return 1; }
+ get { return 0; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = FanartBaseProvider.FanArtResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/Music/ManualFanartArtistProvider.cs b/MediaBrowser.Providers/Music/ManualFanartArtistProvider.cs
index ddf5064aa..f100a3e31 100644
--- a/MediaBrowser.Providers/Music/ManualFanartArtistProvider.cs
+++ b/MediaBrowser.Providers/Music/ManualFanartArtistProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Providers;
@@ -17,14 +18,16 @@ using System.Xml;
namespace MediaBrowser.Providers.Music
{
- public class ManualFanartArtistProvider : IImageProvider
+ public class ManualFanartArtistProvider : IRemoteImageProvider
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualFanartArtistProvider(IServerConfigurationManager config)
+ public ManualFanartArtistProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -42,6 +45,18 @@ namespace MediaBrowser.Providers.Music
return item is MusicArtist;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Logo,
+ ImageType.Art,
+ ImageType.Banner,
+ ImageType.Backdrop
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -334,9 +349,19 @@ namespace MediaBrowser.Providers.Music
list.Add(info);
}
- public int Priority
+ public int Order
{
- get { return 1; }
+ get { return 0; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = FanartBaseProvider.FanArtResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/Music/ManualLastFmImageProvider.cs b/MediaBrowser.Providers/Music/ManualLastFmImageProvider.cs
index 6d6f1ec7b..aa7d00fb6 100644
--- a/MediaBrowser.Providers/Music/ManualLastFmImageProvider.cs
+++ b/MediaBrowser.Providers/Music/ManualLastFmImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Entities;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
@@ -11,8 +12,15 @@ using System.Threading.Tasks;
namespace MediaBrowser.Providers.Music
{
- public class ManualLastFmImageProvider : IImageProvider
+ public class ManualLastFmImageProvider : IRemoteImageProvider
{
+ private readonly IHttpClient _httpClient;
+
+ public ManualLastFmImageProvider(IHttpClient httpClient)
+ {
+ _httpClient = httpClient;
+ }
+
public string Name
{
get { return ProviderName; }
@@ -28,6 +36,14 @@ namespace MediaBrowser.Providers.Music
return item is MusicAlbum || item is MusicArtist;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -72,7 +88,8 @@ namespace MediaBrowser.Providers.Music
var info = new RemoteImageInfo
{
ProviderName = Name,
- Url = url
+ Url = url,
+ Type = ImageType.Primary
};
if (string.Equals(size, "mega", StringComparison.OrdinalIgnoreCase))
@@ -95,9 +112,19 @@ namespace MediaBrowser.Providers.Music
return info;
}
- public int Priority
+ public int Order
+ {
+ get { return 1; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
{
- get { return 0; }
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = LastfmBaseProvider.LastfmResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/Music/MusicBrainzAlbumProvider.cs b/MediaBrowser.Providers/Music/MusicBrainzAlbumProvider.cs
index e678271f2..29edac6b0 100644
--- a/MediaBrowser.Providers/Music/MusicBrainzAlbumProvider.cs
+++ b/MediaBrowser.Providers/Music/MusicBrainzAlbumProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Common.Net;
+using MediaBrowser.Common;
+using MediaBrowser.Common.Net;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.Audio;
@@ -20,11 +21,13 @@ namespace MediaBrowser.Providers.Music
internal static MusicBrainzAlbumProvider Current;
private readonly IHttpClient _httpClient;
+ private readonly IApplicationHost _appHost;
- public MusicBrainzAlbumProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IHttpClient httpClient)
+ public MusicBrainzAlbumProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IHttpClient httpClient, IApplicationHost appHost)
: base(logManager, configurationManager)
{
_httpClient = httpClient;
+ _appHost = appHost;
Current = this;
}
@@ -83,7 +86,7 @@ namespace MediaBrowser.Providers.Music
private async Task<ReleaseResult> GetReleaseResult(string albumName, string artistId, CancellationToken cancellationToken)
{
- var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" and arid:{1}",
+ var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" AND arid:{1}",
WebUtility.UrlEncode(albumName),
artistId);
@@ -94,7 +97,7 @@ namespace MediaBrowser.Providers.Music
private async Task<ReleaseResult> GetReleaseResultByArtistName(string albumName, string artistName, CancellationToken cancellationToken)
{
- var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" and artist:\"{1}\"",
+ var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" AND artist:\"{1}\"",
WebUtility.UrlEncode(albumName),
WebUtility.UrlEncode(artistName));
@@ -189,11 +192,13 @@ namespace MediaBrowser.Providers.Music
var doc = new XmlDocument();
+ var userAgent = _appHost.Name + "/" + _appHost.ApplicationVersion;
+
using (var xml = await _httpClient.Get(new HttpRequestOptions
{
Url = url,
CancellationToken = cancellationToken,
- UserAgent = Environment.MachineName
+ UserAgent = userAgent
}).ConfigureAwait(false))
{
diff --git a/MediaBrowser.Providers/ImagesByName/MusicGenresManualImageProvider.cs b/MediaBrowser.Providers/MusicGenres/MusicGenreImageProvider.cs
index f21e867d1..8ae217e0b 100644
--- a/MediaBrowser.Providers/ImagesByName/MusicGenresManualImageProvider.cs
+++ b/MediaBrowser.Providers/MusicGenres/MusicGenreImageProvider.cs
@@ -6,15 +6,17 @@ using MediaBrowser.Controller.Entities.Audio;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
+using MediaBrowser.Providers.Genres;
+using MediaBrowser.Providers.ImagesByName;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Providers.ImagesByName
+namespace MediaBrowser.Providers.MusicGenres
{
- public class MusicGenresManualImageProvider : IImageProvider
+ public class MusicGenreImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly IHttpClient _httpClient;
@@ -22,7 +24,7 @@ namespace MediaBrowser.Providers.ImagesByName
private readonly SemaphoreSlim _listResourcePool = new SemaphoreSlim(1, 1);
- public MusicGenresManualImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
+ public MusicGenreImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
{
_config = config;
_httpClient = httpClient;
@@ -44,6 +46,15 @@ namespace MediaBrowser.Providers.ImagesByName
return item is MusicGenre;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb
+ };
+ }
+
public Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
return GetImages(item, imageType == ImageType.Primary, imageType == ImageType.Thumb, cancellationToken);
@@ -121,9 +132,19 @@ namespace MediaBrowser.Providers.ImagesByName
return ImageUtils.EnsureList(url, file, _httpClient, _fileSystem, _listResourcePool, cancellationToken);
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = GenreImageProvider.ImageDownloadResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/MusicGenres/MusicGenreMetadataService.cs b/MediaBrowser.Providers/MusicGenres/MusicGenreMetadataService.cs
new file mode 100644
index 000000000..b88ca92bc
--- /dev/null
+++ b/MediaBrowser.Providers/MusicGenres/MusicGenreMetadataService.cs
@@ -0,0 +1,42 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities.Audio;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.MusicGenres
+{
+ public class MusicGenreMetadataService : MetadataService<MusicGenre>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public MusicGenreMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ /// <param name="mergeMetadataSettings">if set to <c>true</c> [merge metadata settings].</param>
+ protected override void MergeData(MusicGenre source, MusicGenre target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(MusicGenre item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/Movies/ManualMovieDbPersonImageProvider.cs b/MediaBrowser.Providers/People/MovieDbPersonImageProvider.cs
index 4d3a5baac..2b4e43f3c 100644
--- a/MediaBrowser.Providers/Movies/ManualMovieDbPersonImageProvider.cs
+++ b/MediaBrowser.Providers/People/MovieDbPersonImageProvider.cs
@@ -1,26 +1,30 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
using MediaBrowser.Model.Serialization;
+using MediaBrowser.Providers.Movies;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Providers.Movies
+namespace MediaBrowser.Providers.People
{
- public class ManualMovieDbPersonImageProvider : IImageProvider
+ public class MovieDbPersonImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly IJsonSerializer _jsonSerializer;
+ private readonly IHttpClient _httpClient;
- public ManualMovieDbPersonImageProvider(IServerConfigurationManager config, IJsonSerializer jsonSerializer)
+ public MovieDbPersonImageProvider(IServerConfigurationManager config, IJsonSerializer jsonSerializer, IHttpClient httpClient)
{
_config = config;
_jsonSerializer = jsonSerializer;
+ _httpClient = httpClient;
}
public string Name
@@ -38,6 +42,14 @@ namespace MediaBrowser.Providers.Movies
return item is Person;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -120,9 +132,19 @@ namespace MediaBrowser.Providers.Movies
return profile.iso_639_1 == null ? null : profile.iso_639_1.ToString();
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = MovieDbProvider.Current.MovieDbResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/People/MovieDbPersonProvider.cs b/MediaBrowser.Providers/People/MovieDbPersonProvider.cs
new file mode 100644
index 000000000..088ba0322
--- /dev/null
+++ b/MediaBrowser.Providers/People/MovieDbPersonProvider.cs
@@ -0,0 +1,289 @@
+using MediaBrowser.Common.Configuration;
+using MediaBrowser.Common.Extensions;
+using MediaBrowser.Common.IO;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Serialization;
+using MediaBrowser.Providers.Movies;
+using System;
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Net;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.People
+{
+ public class MovieDbPersonProvider : IRemoteMetadataProvider<Person>
+ {
+ const string DataFileName = "info.json";
+
+ internal static MovieDbPersonProvider Current { get; private set; }
+
+ private readonly IJsonSerializer _jsonSerializer;
+ private readonly IFileSystem _fileSystem;
+ private readonly IServerConfigurationManager _configurationManager;
+
+ public MovieDbPersonProvider(IFileSystem fileSystem, IServerConfigurationManager configurationManager, IJsonSerializer jsonSerializer)
+ {
+ _fileSystem = fileSystem;
+ _configurationManager = configurationManager;
+ _jsonSerializer = jsonSerializer;
+ Current = this;
+ }
+
+ public string Name
+ {
+ get { return "TheMovieDb"; }
+ }
+
+ public async Task<MetadataResult<Person>> GetMetadata(ItemId id, CancellationToken cancellationToken)
+ {
+ var tmdbId = id.GetProviderId(MetadataProviders.Tmdb);
+
+ // We don't already have an Id, need to fetch it
+ if (string.IsNullOrEmpty(tmdbId))
+ {
+ tmdbId = await GetTmdbId(id.Name, cancellationToken).ConfigureAwait(false);
+ }
+
+ var result = new MetadataResult<Person>();
+
+ if (!string.IsNullOrEmpty(tmdbId))
+ {
+ await EnsurePersonInfo(tmdbId, cancellationToken).ConfigureAwait(false);
+
+ var dataFilePath = GetPersonDataFilePath(_configurationManager.ApplicationPaths, tmdbId);
+
+ var info = _jsonSerializer.DeserializeFromFile<PersonResult>(dataFilePath);
+
+ var item = new Person();
+ result.HasMetadata = true;
+
+ item.Name = info.name;
+ item.HomePageUrl = info.homepage;
+ item.PlaceOfBirth = info.place_of_birth;
+ item.Overview = info.biography;
+
+ DateTime date;
+
+ if (DateTime.TryParseExact(info.birthday, "yyyy-MM-dd", new CultureInfo("en-US"), DateTimeStyles.None, out date))
+ {
+ item.PremiereDate = date.ToUniversalTime();
+ }
+
+ if (DateTime.TryParseExact(info.deathday, "yyyy-MM-dd", new CultureInfo("en-US"), DateTimeStyles.None, out date))
+ {
+ item.EndDate = date.ToUniversalTime();
+ }
+
+ item.SetProviderId(MetadataProviders.Tmdb, info.id.ToString(_usCulture));
+
+ if (!string.IsNullOrEmpty(info.imdb_id))
+ {
+ item.SetProviderId(MetadataProviders.Imdb, info.imdb_id);
+ }
+
+ result.Item = item;
+ }
+
+ return result;
+ }
+
+ private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+
+ /// <summary>
+ /// Gets the TMDB id.
+ /// </summary>
+ /// <param name="name">The name.</param>
+ /// <param name="cancellationToken">The cancellation token.</param>
+ /// <returns>Task{System.String}.</returns>
+ private async Task<string> GetTmdbId(string name, CancellationToken cancellationToken)
+ {
+ string url = string.Format(@"http://api.themoviedb.org/3/search/person?api_key={1}&query={0}", WebUtility.UrlEncode(name), MovieDbProvider.ApiKey);
+ PersonSearchResults searchResult = null;
+
+ using (var json = await MovieDbProvider.Current.GetMovieDbResponse(new HttpRequestOptions
+ {
+ Url = url,
+ CancellationToken = cancellationToken,
+ AcceptHeader = MovieDbProvider.AcceptHeader
+
+ }).ConfigureAwait(false))
+ {
+ searchResult = _jsonSerializer.DeserializeFromStream<PersonSearchResults>(json);
+ }
+
+ return searchResult != null && searchResult.Total_Results > 0 ? searchResult.Results[0].Id.ToString(_usCulture) : null;
+ }
+
+ internal async Task EnsurePersonInfo(string id, CancellationToken cancellationToken)
+ {
+ var dataFilePath = GetPersonDataFilePath(_configurationManager.ApplicationPaths, id);
+
+ var fileInfo = _fileSystem.GetFileSystemInfo(dataFilePath);
+
+ if (fileInfo.Exists && (DateTime.UtcNow - _fileSystem.GetLastWriteTimeUtc(fileInfo)).TotalDays <= 7)
+ {
+ return;
+ }
+
+ var url = string.Format(@"http://api.themoviedb.org/3/person/{1}?api_key={0}&append_to_response=credits,images", MovieDbProvider.ApiKey, id);
+
+ using (var json = await MovieDbProvider.Current.GetMovieDbResponse(new HttpRequestOptions
+ {
+ Url = url,
+ CancellationToken = cancellationToken,
+ AcceptHeader = MovieDbProvider.AcceptHeader
+
+ }).ConfigureAwait(false))
+ {
+ Directory.CreateDirectory(Path.GetDirectoryName(dataFilePath));
+
+ using (var fs = _fileSystem.GetFileStream(dataFilePath, FileMode.Create, FileAccess.Write, FileShare.Read, true))
+ {
+ await json.CopyToAsync(fs).ConfigureAwait(false);
+ }
+ }
+ }
+
+ private static string GetPersonDataPath(IApplicationPaths appPaths, string tmdbId)
+ {
+ var letter = tmdbId.GetMD5().ToString().Substring(0, 1);
+
+ return Path.Combine(GetPersonsDataPath(appPaths), letter, tmdbId);
+ }
+
+ internal static string GetPersonDataFilePath(IApplicationPaths appPaths, string tmdbId)
+ {
+ return Path.Combine(GetPersonDataPath(appPaths, tmdbId), DataFileName);
+ }
+
+ private static string GetPersonsDataPath(IApplicationPaths appPaths)
+ {
+ return Path.Combine(appPaths.DataPath, "tmdb-people");
+ }
+
+ #region Result Objects
+ /// <summary>
+ /// Class PersonSearchResult
+ /// </summary>
+ public class PersonSearchResult
+ {
+ /// <summary>
+ /// Gets or sets a value indicating whether this <see cref="MovieDbPersonProvider.PersonSearchResult" /> is adult.
+ /// </summary>
+ /// <value><c>true</c> if adult; otherwise, <c>false</c>.</value>
+ public bool Adult { get; set; }
+ /// <summary>
+ /// Gets or sets the id.
+ /// </summary>
+ /// <value>The id.</value>
+ public int Id { get; set; }
+ /// <summary>
+ /// Gets or sets the name.
+ /// </summary>
+ /// <value>The name.</value>
+ public string Name { get; set; }
+ /// <summary>
+ /// Gets or sets the profile_ path.
+ /// </summary>
+ /// <value>The profile_ path.</value>
+ public string Profile_Path { get; set; }
+ }
+
+ /// <summary>
+ /// Class PersonSearchResults
+ /// </summary>
+ public class PersonSearchResults
+ {
+ /// <summary>
+ /// Gets or sets the page.
+ /// </summary>
+ /// <value>The page.</value>
+ public int Page { get; set; }
+ /// <summary>
+ /// Gets or sets the results.
+ /// </summary>
+ /// <value>The results.</value>
+ public List<MovieDbPersonProvider.PersonSearchResult> Results { get; set; }
+ /// <summary>
+ /// Gets or sets the total_ pages.
+ /// </summary>
+ /// <value>The total_ pages.</value>
+ public int Total_Pages { get; set; }
+ /// <summary>
+ /// Gets or sets the total_ results.
+ /// </summary>
+ /// <value>The total_ results.</value>
+ public int Total_Results { get; set; }
+ }
+
+ public class Cast
+ {
+ public int id { get; set; }
+ public string title { get; set; }
+ public string character { get; set; }
+ public string original_title { get; set; }
+ public string poster_path { get; set; }
+ public string release_date { get; set; }
+ public bool adult { get; set; }
+ }
+
+ public class Crew
+ {
+ public int id { get; set; }
+ public string title { get; set; }
+ public string original_title { get; set; }
+ public string department { get; set; }
+ public string job { get; set; }
+ public string poster_path { get; set; }
+ public string release_date { get; set; }
+ public bool adult { get; set; }
+ }
+
+ public class Credits
+ {
+ public List<Cast> cast { get; set; }
+ public List<Crew> crew { get; set; }
+ }
+
+ public class Profile
+ {
+ public string file_path { get; set; }
+ public int width { get; set; }
+ public int height { get; set; }
+ public object iso_639_1 { get; set; }
+ public double aspect_ratio { get; set; }
+ }
+
+ public class Images
+ {
+ public List<Profile> profiles { get; set; }
+ }
+
+ public class PersonResult
+ {
+ public bool adult { get; set; }
+ public List<object> also_known_as { get; set; }
+ public string biography { get; set; }
+ public string birthday { get; set; }
+ public string deathday { get; set; }
+ public string homepage { get; set; }
+ public int id { get; set; }
+ public string imdb_id { get; set; }
+ public string name { get; set; }
+ public string place_of_birth { get; set; }
+ public double popularity { get; set; }
+ public string profile_path { get; set; }
+ public Credits credits { get; set; }
+ public Images images { get; set; }
+ }
+
+ #endregion
+ }
+}
diff --git a/MediaBrowser.Providers/People/PersonMetadataService.cs b/MediaBrowser.Providers/People/PersonMetadataService.cs
new file mode 100644
index 000000000..e04013934
--- /dev/null
+++ b/MediaBrowser.Providers/People/PersonMetadataService.cs
@@ -0,0 +1,47 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.People
+{
+ public class PersonMetadataService : MetadataService<Person>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public PersonMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ /// <param name="mergeMetadataSettings">if set to <c>true</c> [merge metadata settings].</param>
+ protected override void MergeData(Person source, Person target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+
+ if (replaceData || string.IsNullOrEmpty(target.PlaceOfBirth))
+ {
+ target.PlaceOfBirth = source.PlaceOfBirth;
+ }
+ }
+
+ protected override Task SaveItem(Person item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/People/PersonXmlProvider.cs b/MediaBrowser.Providers/People/PersonXmlProvider.cs
new file mode 100644
index 000000000..900b7956c
--- /dev/null
+++ b/MediaBrowser.Providers/People/PersonXmlProvider.cs
@@ -0,0 +1,59 @@
+using MediaBrowser.Common.IO;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Logging;
+using System.IO;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.People
+{
+ public class PersonXmlProvider : BaseXmlProvider, ILocalMetadataProvider<Person>
+ {
+ private readonly ILogger _logger;
+
+ public PersonXmlProvider(IFileSystem fileSystem, ILogger logger)
+ : base(fileSystem)
+ {
+ _logger = logger;
+ }
+
+ public async Task<MetadataResult<Person>> GetMetadata(string path, CancellationToken cancellationToken)
+ {
+ path = GetXmlPath(path);
+
+ var result = new MetadataResult<Person>();
+
+ await XmlParsingResourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
+
+ try
+ {
+ var person = new Person();
+
+ new BaseItemXmlParser<Person>(_logger).Fetch(person, path, cancellationToken);
+ result.HasMetadata = true;
+ result.Item = person;
+ }
+ catch (FileNotFoundException)
+ {
+ result.HasMetadata = false;
+ }
+ finally
+ {
+ XmlParsingResourcePool.Release();
+ }
+
+ return result;
+ }
+
+ public string Name
+ {
+ get { return "Media Browser Xml"; }
+ }
+
+ protected override string GetXmlPath(string path)
+ {
+ return Path.Combine(path, "person.xml");
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/TV/ManualTvdbPersonImageProvider.cs b/MediaBrowser.Providers/People/TvdbPersonImageProvider.cs
index 456db1048..aa4b9e0d8 100644
--- a/MediaBrowser.Providers/TV/ManualTvdbPersonImageProvider.cs
+++ b/MediaBrowser.Providers/People/TvdbPersonImageProvider.cs
@@ -1,10 +1,12 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
+using MediaBrowser.Providers.TV;
using System;
using System.Collections.Generic;
using System.IO;
@@ -14,17 +16,19 @@ using System.Threading;
using System.Threading.Tasks;
using System.Xml;
-namespace MediaBrowser.Providers.TV
+namespace MediaBrowser.Providers.People
{
- public class ManualTvdbPersonImageProvider : IImageProvider
+ public class TvdbPersonImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly ILibraryManager _library;
+ private readonly IHttpClient _httpClient;
- public ManualTvdbPersonImageProvider(IServerConfigurationManager config, ILibraryManager library)
+ public TvdbPersonImageProvider(IServerConfigurationManager config, ILibraryManager library, IHttpClient httpClient)
{
_config = config;
_library = library;
+ _httpClient = httpClient;
}
public string Name
@@ -42,6 +46,14 @@ namespace MediaBrowser.Providers.TV
return item is Person;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -184,9 +196,19 @@ namespace MediaBrowser.Providers.TV
return null;
}
- public int Priority
+ public int Order
{
- get { return 0; }
+ get { return 1; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = TvdbSeriesProvider.Current.TvDbResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/ProviderUtils.cs b/MediaBrowser.Providers/ProviderUtils.cs
new file mode 100644
index 000000000..416ada42c
--- /dev/null
+++ b/MediaBrowser.Providers/ProviderUtils.cs
@@ -0,0 +1,126 @@
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Model.Entities;
+using System.Collections.Generic;
+
+namespace MediaBrowser.Providers
+{
+ public static class ProviderUtils
+ {
+ public static void MergeBaseItemData(BaseItem source, BaseItem target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ if (!lockedFields.Contains(MetadataFields.Name))
+ {
+ if (replaceData || string.IsNullOrEmpty(target.Name))
+ {
+ target.Name = source.Name;
+ }
+ }
+
+ if (replaceData || !target.CommunityRating.HasValue)
+ {
+ target.CommunityRating = source.CommunityRating;
+ }
+
+ if (replaceData || !target.EndDate.HasValue)
+ {
+ target.EndDate = source.EndDate;
+ }
+
+ if (!lockedFields.Contains(MetadataFields.Genres))
+ {
+ if (replaceData || target.Genres.Count == 0)
+ {
+ target.Genres = source.Genres;
+ }
+ }
+
+ if (replaceData || string.IsNullOrEmpty(target.HomePageUrl))
+ {
+ target.HomePageUrl = source.HomePageUrl;
+ }
+
+ if (replaceData || !target.IndexNumber.HasValue)
+ {
+ target.IndexNumber = source.IndexNumber;
+ }
+
+ if (!lockedFields.Contains(MetadataFields.OfficialRating))
+ {
+ if (replaceData || string.IsNullOrEmpty(target.OfficialRating))
+ {
+ target.OfficialRating = source.OfficialRating;
+ }
+ }
+
+ if (replaceData || string.IsNullOrEmpty(target.OfficialRatingDescription))
+ {
+ target.OfficialRatingDescription = source.OfficialRatingDescription;
+ }
+
+ if (!lockedFields.Contains(MetadataFields.Overview))
+ {
+ if (replaceData || string.IsNullOrEmpty(target.Overview))
+ {
+ target.Overview = source.Overview;
+ }
+ }
+
+ if (replaceData || !target.ParentIndexNumber.HasValue)
+ {
+ target.ParentIndexNumber = source.ParentIndexNumber;
+ }
+
+ if (!lockedFields.Contains(MetadataFields.Cast))
+ {
+ if (replaceData || target.People.Count == 0)
+ {
+ target.People = source.People;
+ }
+ }
+
+ if (replaceData || !target.PremiereDate.HasValue)
+ {
+ target.PremiereDate = source.PremiereDate;
+ }
+
+ if (replaceData || !target.ProductionYear.HasValue)
+ {
+ target.ProductionYear = source.ProductionYear;
+ }
+
+ if (!lockedFields.Contains(MetadataFields.Runtime))
+ {
+ if (replaceData || !target.RunTimeTicks.HasValue)
+ {
+ target.RunTimeTicks = source.RunTimeTicks;
+ }
+ }
+
+ if (!lockedFields.Contains(MetadataFields.Studios))
+ {
+ if (replaceData || target.Studios.Count == 0)
+ {
+ target.Studios = source.Studios;
+ }
+ }
+
+ if (replaceData || !target.VoteCount.HasValue)
+ {
+ target.VoteCount = source.VoteCount;
+ }
+
+ foreach (var id in source.ProviderIds)
+ {
+ target.ProviderIds[id.Key] = id.Value;
+ }
+
+ if (mergeMetadataSettings)
+ {
+ target.ForcedSortName = source.ForcedSortName;
+ target.LockedFields = source.LockedFields;
+ target.DontFetchMeta = source.DontFetchMeta;
+ target.DisplayMediaType = source.DisplayMediaType;
+ }
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/Studios/StudioMetadataService.cs b/MediaBrowser.Providers/Studios/StudioMetadataService.cs
new file mode 100644
index 000000000..1a35b94b3
--- /dev/null
+++ b/MediaBrowser.Providers/Studios/StudioMetadataService.cs
@@ -0,0 +1,41 @@
+using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Controller.Entities;
+using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
+using MediaBrowser.Model.Entities;
+using MediaBrowser.Model.Logging;
+using MediaBrowser.Providers.Manager;
+using System.Collections.Generic;
+using System.Threading;
+using System.Threading.Tasks;
+
+namespace MediaBrowser.Providers.Studios
+{
+ public class StudioMetadataService : MetadataService<Studio>
+ {
+ private readonly ILibraryManager _libraryManager;
+
+ public StudioMetadataService(IServerConfigurationManager serverConfigurationManager, ILogger logger, IProviderManager providerManager, IProviderRepository providerRepo, ILibraryManager libraryManager)
+ : base(serverConfigurationManager, logger, providerManager, providerRepo)
+ {
+ _libraryManager = libraryManager;
+ }
+
+ /// <summary>
+ /// Merges the specified source.
+ /// </summary>
+ /// <param name="source">The source.</param>
+ /// <param name="target">The target.</param>
+ /// <param name="lockedFields">The locked fields.</param>
+ /// <param name="replaceData">if set to <c>true</c> [replace data].</param>
+ protected override void MergeData(Studio source, Studio target, List<MetadataFields> lockedFields, bool replaceData, bool mergeMetadataSettings)
+ {
+ ProviderUtils.MergeBaseItemData(source, target, lockedFields, replaceData, mergeMetadataSettings);
+ }
+
+ protected override Task SaveItem(Studio item, ItemUpdateType reason, CancellationToken cancellationToken)
+ {
+ return _libraryManager.UpdateItem(item, reason, cancellationToken);
+ }
+ }
+}
diff --git a/MediaBrowser.Providers/ImagesByName/StudiosManualImageProvider.cs b/MediaBrowser.Providers/Studios/StudiosImageProvider.cs
index 062f08020..eacec5b69 100644
--- a/MediaBrowser.Providers/ImagesByName/StudiosManualImageProvider.cs
+++ b/MediaBrowser.Providers/Studios/StudiosImageProvider.cs
@@ -5,15 +5,17 @@ using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Providers;
+using MediaBrowser.Providers.Genres;
+using MediaBrowser.Providers.ImagesByName;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
-namespace MediaBrowser.Providers.ImagesByName
+namespace MediaBrowser.Providers.Studios
{
- public class StudiosManualImageProvider : IImageProvider
+ public class StudiosImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly IHttpClient _httpClient;
@@ -21,7 +23,7 @@ namespace MediaBrowser.Providers.ImagesByName
private readonly SemaphoreSlim _listResourcePool = new SemaphoreSlim(1, 1);
- public StudiosManualImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
+ public StudiosImageProvider(IServerConfigurationManager config, IHttpClient httpClient, IFileSystem fileSystem)
{
_config = config;
_httpClient = httpClient;
@@ -43,6 +45,15 @@ namespace MediaBrowser.Providers.ImagesByName
return item is Studio;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb
+ };
+ }
+
public Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
return GetImages(item, imageType == ImageType.Primary, imageType == ImageType.Thumb, cancellationToken);
@@ -120,9 +131,19 @@ namespace MediaBrowser.Providers.ImagesByName
return ImageUtils.EnsureList(url, file, _httpClient, _fileSystem, _listResourcePool, cancellationToken);
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = GenreImageProvider.ImageDownloadResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/TV/ManualFanartSeasonProvider.cs b/MediaBrowser.Providers/TV/ManualFanartSeasonProvider.cs
index 503c56d0d..1d6f82a3a 100644
--- a/MediaBrowser.Providers/TV/ManualFanartSeasonProvider.cs
+++ b/MediaBrowser.Providers/TV/ManualFanartSeasonProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Providers;
@@ -17,14 +18,16 @@ using System.Xml;
namespace MediaBrowser.Providers.TV
{
- public class ManualFanartSeasonImageProvider : IImageProvider
+ public class ManualFanartSeasonImageProvider : IRemoteImageProvider
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualFanartSeasonImageProvider(IServerConfigurationManager config)
+ public ManualFanartSeasonImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -42,6 +45,15 @@ namespace MediaBrowser.Providers.TV
return item is Season;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Backdrop,
+ ImageType.Thumb
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -245,9 +257,19 @@ namespace MediaBrowser.Providers.TV
}
}
- public int Priority
+ public int Order
{
- get { return 0; }
+ get { return 1; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = FanartBaseProvider.FanArtResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/TV/ManualFanartSeriesProvider.cs b/MediaBrowser.Providers/TV/ManualFanartSeriesProvider.cs
index 6a80f720a..b88a94e4e 100644
--- a/MediaBrowser.Providers/TV/ManualFanartSeriesProvider.cs
+++ b/MediaBrowser.Providers/TV/ManualFanartSeriesProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Providers;
@@ -17,14 +18,16 @@ using System.Xml;
namespace MediaBrowser.Providers.TV
{
- public class ManualFanartSeriesImageProvider : IImageProvider
+ public class ManualFanartSeriesImageProvider : IRemoteImageProvider
{
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
- public ManualFanartSeriesImageProvider(IServerConfigurationManager config)
+ public ManualFanartSeriesImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -42,6 +45,19 @@ namespace MediaBrowser.Providers.TV
return item is Series;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Thumb,
+ ImageType.Art,
+ ImageType.Logo,
+ ImageType.Backdrop,
+ ImageType.Banner
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -302,9 +318,19 @@ namespace MediaBrowser.Providers.TV
}
}
- public int Priority
+ public int Order
{
- get { return 0; }
+ get { return 1; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = FanartBaseProvider.FanArtResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/TV/ManualTvdbEpisodeImageProvider.cs b/MediaBrowser.Providers/TV/ManualTvdbEpisodeImageProvider.cs
index 6d38dee2e..abccc2947 100644
--- a/MediaBrowser.Providers/TV/ManualTvdbEpisodeImageProvider.cs
+++ b/MediaBrowser.Providers/TV/ManualTvdbEpisodeImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
@@ -16,14 +17,16 @@ using System.Xml;
namespace MediaBrowser.Providers.TV
{
- public class ManualTvdbEpisodeImageProvider : IImageProvider
+ public class ManualTvdbEpisodeImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+ private readonly IHttpClient _httpClient;
- public ManualTvdbEpisodeImageProvider(IServerConfigurationManager config)
+ public ManualTvdbEpisodeImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -36,6 +39,14 @@ namespace MediaBrowser.Providers.TV
return item is Episode;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -161,9 +172,19 @@ namespace MediaBrowser.Providers.TV
};
}
- public int Priority
+ public int Order
{
get { return 0; }
}
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = TvdbSeriesProvider.Current.TvDbResourcePool
+ });
+ }
}
}
diff --git a/MediaBrowser.Providers/TV/ManualTvdbSeasonImageProvider.cs b/MediaBrowser.Providers/TV/ManualTvdbSeasonImageProvider.cs
index d9a6f6507..f672942db 100644
--- a/MediaBrowser.Providers/TV/ManualTvdbSeasonImageProvider.cs
+++ b/MediaBrowser.Providers/TV/ManualTvdbSeasonImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
@@ -18,14 +19,16 @@ using System.Xml;
namespace MediaBrowser.Providers.TV
{
- public class ManualTvdbSeasonImageProvider : IImageProvider
+ public class ManualTvdbSeasonImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
+ private readonly IHttpClient _httpClient;
- public ManualTvdbSeasonImageProvider(IServerConfigurationManager config)
+ public ManualTvdbSeasonImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -43,6 +46,16 @@ namespace MediaBrowser.Providers.TV
return item is Season;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Banner,
+ ImageType.Backdrop
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -308,9 +321,19 @@ namespace MediaBrowser.Providers.TV
}
- public int Priority
+ public int Order
{
- get { return 1; }
+ get { return 0; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = TvdbSeriesProvider.Current.TvDbResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/TV/ManualTvdbSeriesImageProvider.cs b/MediaBrowser.Providers/TV/ManualTvdbSeriesImageProvider.cs
index 644cad93b..a1c7114fc 100644
--- a/MediaBrowser.Providers/TV/ManualTvdbSeriesImageProvider.cs
+++ b/MediaBrowser.Providers/TV/ManualTvdbSeriesImageProvider.cs
@@ -1,4 +1,5 @@
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
+using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
@@ -18,14 +19,16 @@ using System.Xml;
namespace MediaBrowser.Providers.TV
{
- public class ManualTvdbSeriesImageProvider : IImageProvider
+ public class ManualTvdbSeriesImageProvider : IRemoteImageProvider
{
private readonly IServerConfigurationManager _config;
+ private readonly IHttpClient _httpClient;
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
- public ManualTvdbSeriesImageProvider(IServerConfigurationManager config)
+ public ManualTvdbSeriesImageProvider(IServerConfigurationManager config, IHttpClient httpClient)
{
_config = config;
+ _httpClient = httpClient;
}
public string Name
@@ -43,6 +46,16 @@ namespace MediaBrowser.Providers.TV
return item is Series;
}
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
+ {
+ return new List<ImageType>
+ {
+ ImageType.Primary,
+ ImageType.Banner,
+ ImageType.Backdrop
+ };
+ }
+
public async Task<IEnumerable<RemoteImageInfo>> GetImages(IHasImages item, ImageType imageType, CancellationToken cancellationToken)
{
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
@@ -304,9 +317,19 @@ namespace MediaBrowser.Providers.TV
}
- public int Priority
+ public int Order
{
- get { return 1; }
+ get { return 0; }
+ }
+
+ public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
+ {
+ return _httpClient.GetResponse(new HttpRequestOptions
+ {
+ CancellationToken = cancellationToken,
+ Url = url,
+ ResourcePool = TvdbSeriesProvider.Current.TvDbResourcePool
+ });
}
}
}
diff --git a/MediaBrowser.Providers/TV/TvdbPersonImageProvider.cs b/MediaBrowser.Providers/TV/TvdbPersonImageProvider.cs
deleted file mode 100644
index f2ce92efd..000000000
--- a/MediaBrowser.Providers/TV/TvdbPersonImageProvider.cs
+++ /dev/null
@@ -1,98 +0,0 @@
-using MediaBrowser.Controller.Configuration;
-using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Providers;
-using MediaBrowser.Model.Entities;
-using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Providers;
-using System;
-using System.Collections.Generic;
-using System.Linq;
-using System.Threading;
-using System.Threading.Tasks;
-
-namespace MediaBrowser.Providers.TV
-{
- public class TvdbPersonImageProvider : BaseMetadataProvider
- {
- private readonly IProviderManager _providerManager;
-
- public TvdbPersonImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
- : base(logManager, configurationManager)
- {
- _providerManager = providerManager;
- }
-
- protected override bool RefreshOnVersionChange
- {
- get
- {
- return true;
- }
- }
-
- protected override string ProviderVersion
- {
- get
- {
- return "2";
- }
- }
-
- public override bool RequiresInternet
- {
- get
- {
- return true;
- }
- }
-
- public override bool Supports(BaseItem item)
- {
- return item is Person;
- }
-
- /// <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 async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
- {
- if (string.IsNullOrEmpty(item.PrimaryImagePath))
- {
- cancellationToken.ThrowIfCancellationRequested();
-
- var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, ManualTvdbPersonImageProvider.ProviderName).ConfigureAwait(false);
-
- await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
- {
- if (!item.HasImage(ImageType.Primary) && !item.LockedFields.Contains(MetadataFields.Images))
- {
- var image = images.FirstOrDefault(i => i.Type == ImageType.Primary);
-
- if (image != null)
- {
- await _providerManager.SaveImage(item, image.Url, TvdbSeriesProvider.Current.TvDbResourcePool, ImageType.Primary, null, cancellationToken)
- .ConfigureAwait(false);
- }
- }
- }
-
- public override MetadataProviderPriority Priority
- {
- get { return MetadataProviderPriority.Fourth; }
- }
- }
-}
diff --git a/MediaBrowser.Providers/VirtualItemImageValidator.cs b/MediaBrowser.Providers/VirtualItemImageValidator.cs
index f3f32b2b3..892275d38 100644
--- a/MediaBrowser.Providers/VirtualItemImageValidator.cs
+++ b/MediaBrowser.Providers/VirtualItemImageValidator.cs
@@ -44,14 +44,6 @@ namespace MediaBrowser.Providers
public override Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
{
item.ValidateImages();
- item.ValidateBackdrops();
-
- var hasScreenshots = item as IHasScreenshots;
-
- if (hasScreenshots != null)
- {
- hasScreenshots.ValidateScreenshots();
- }
SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
return TrueTaskResult;
diff --git a/MediaBrowser.Server.Implementations/Drawing/ImageProcessor.cs b/MediaBrowser.Server.Implementations/Drawing/ImageProcessor.cs
index 6378cef52..06a03ba1c 100644
--- a/MediaBrowser.Server.Implementations/Drawing/ImageProcessor.cs
+++ b/MediaBrowser.Server.Implementations/Drawing/ImageProcessor.cs
@@ -388,18 +388,18 @@ namespace MediaBrowser.Server.Implementations.Drawing
/// <param name="image">The image.</param>
/// <param name="outputFormat">The output format.</param>
/// <returns>ImageFormat.</returns>
- private ImageFormat GetOutputFormat(Image image, ImageOutputFormat outputFormat)
+ private System.Drawing.Imaging.ImageFormat GetOutputFormat(Image image, ImageOutputFormat outputFormat)
{
switch (outputFormat)
{
case ImageOutputFormat.Bmp:
- return ImageFormat.Bmp;
+ return System.Drawing.Imaging.ImageFormat.Bmp;
case ImageOutputFormat.Gif:
- return ImageFormat.Gif;
+ return System.Drawing.Imaging.ImageFormat.Gif;
case ImageOutputFormat.Jpg:
- return ImageFormat.Jpeg;
+ return System.Drawing.Imaging.ImageFormat.Jpeg;
case ImageOutputFormat.Png:
- return ImageFormat.Png;
+ return System.Drawing.Imaging.ImageFormat.Png;
default:
return image.RawFormat;
}
@@ -787,7 +787,7 @@ namespace MediaBrowser.Server.Implementations.Drawing
//And then save it in the cache
using (var outputStream = _fileSystem.GetFileStream(enhancedImagePath, FileMode.Create, FileAccess.Write, FileShare.Read, false))
{
- newImage.Save(ImageFormat.Png, outputStream, 100);
+ newImage.Save(System.Drawing.Imaging.ImageFormat.Png, outputStream, 100);
}
}
}
diff --git a/MediaBrowser.Server.Implementations/Dto/DtoService.cs b/MediaBrowser.Server.Implementations/Dto/DtoService.cs
index 7a9735e0e..b427b0c45 100644
--- a/MediaBrowser.Server.Implementations/Dto/DtoService.cs
+++ b/MediaBrowser.Server.Implementations/Dto/DtoService.cs
@@ -1024,6 +1024,11 @@ namespace MediaBrowser.Server.Implementations.Dto
{
dto.SpecialFeatureCount = specialFeatureCount;
}
+
+ if (fields.Contains(ItemFields.TmdbCollectionName))
+ {
+ dto.TmdbCollectionName = movie.TmdbCollectionName;
+ }
}
// Add EpisodeInfo
diff --git a/MediaBrowser.Server.Implementations/FileOrganization/EpisodeFileOrganizer.cs b/MediaBrowser.Server.Implementations/FileOrganization/EpisodeFileOrganizer.cs
index ece21df7a..a2e094e9a 100644
--- a/MediaBrowser.Server.Implementations/FileOrganization/EpisodeFileOrganizer.cs
+++ b/MediaBrowser.Server.Implementations/FileOrganization/EpisodeFileOrganizer.cs
@@ -22,7 +22,7 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
{
public class EpisodeFileOrganizer
{
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly ILibraryManager _libraryManager;
private readonly ILogger _logger;
private readonly IFileSystem _fileSystem;
@@ -31,14 +31,14 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
private readonly CultureInfo _usCulture = new CultureInfo("en-US");
- public EpisodeFileOrganizer(IFileOrganizationService organizationService, IServerConfigurationManager config, IFileSystem fileSystem, ILogger logger, ILibraryManager libraryManager, IDirectoryWatchers directoryWatchers)
+ public EpisodeFileOrganizer(IFileOrganizationService organizationService, IServerConfigurationManager config, IFileSystem fileSystem, ILogger logger, ILibraryManager libraryManager, ILibraryMonitor libraryMonitor)
{
_organizationService = organizationService;
_config = config;
_fileSystem = fileSystem;
_logger = logger;
_libraryManager = libraryManager;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
}
public async Task<FileOrganizationResult> OrganizeEpisodeFile(string path, TvFileOrganizationOptions options, bool overwriteExisting)
@@ -174,6 +174,8 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
{
_logger.Debug("Removing duplicate episode {0}", path);
+ _libraryMonitor.ReportFileSystemChangeBeginning(path);
+
try
{
File.Delete(path);
@@ -182,6 +184,10 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
{
_logger.ErrorException("Error removing duplicate episode", ex, path);
}
+ finally
+ {
+ _libraryMonitor.ReportFileSystemChangeComplete(path, true);
+ }
}
}
}
@@ -232,7 +238,7 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
private void PerformFileSorting(TvFileOrganizationOptions options, FileOrganizationResult result)
{
- _directoryWatchers.TemporarilyIgnore(result.TargetPath);
+ _libraryMonitor.ReportFileSystemChangeBeginning(result.TargetPath);
Directory.CreateDirectory(Path.GetDirectoryName(result.TargetPath));
@@ -264,7 +270,7 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
}
finally
{
- _directoryWatchers.RemoveTempIgnore(result.TargetPath);
+ _libraryMonitor.ReportFileSystemChangeComplete(result.TargetPath, true);
}
if (copy)
@@ -376,8 +382,8 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
private string GetEpisodeFileName(string sourcePath, string seriesName, int seasonNumber, int episodeNumber, int? endingEpisodeNumber, string episodeTitle, TvFileOrganizationOptions options)
{
- seriesName = _fileSystem.GetValidFilename(seriesName);
- episodeTitle = _fileSystem.GetValidFilename(episodeTitle);
+ seriesName = _fileSystem.GetValidFilename(seriesName).Trim();
+ episodeTitle = _fileSystem.GetValidFilename(episodeTitle).Trim();
var sourceExtension = (Path.GetExtension(sourcePath) ?? string.Empty).TrimStart('.');
diff --git a/MediaBrowser.Server.Implementations/FileOrganization/FileOrganizationService.cs b/MediaBrowser.Server.Implementations/FileOrganization/FileOrganizationService.cs
index bbd0f74e5..518a7bb48 100644
--- a/MediaBrowser.Server.Implementations/FileOrganization/FileOrganizationService.cs
+++ b/MediaBrowser.Server.Implementations/FileOrganization/FileOrganizationService.cs
@@ -21,17 +21,17 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
private readonly ITaskManager _taskManager;
private readonly IFileOrganizationRepository _repo;
private readonly ILogger _logger;
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly ILibraryManager _libraryManager;
private readonly IServerConfigurationManager _config;
private readonly IFileSystem _fileSystem;
- public FileOrganizationService(ITaskManager taskManager, IFileOrganizationRepository repo, ILogger logger, IDirectoryWatchers directoryWatchers, ILibraryManager libraryManager, IServerConfigurationManager config, IFileSystem fileSystem)
+ public FileOrganizationService(ITaskManager taskManager, IFileOrganizationRepository repo, ILogger logger, ILibraryMonitor libraryMonitor, ILibraryManager libraryManager, IServerConfigurationManager config, IFileSystem fileSystem)
{
_taskManager = taskManager;
_repo = repo;
_logger = logger;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_libraryManager = libraryManager;
_config = config;
_fileSystem = fileSystem;
@@ -91,13 +91,10 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
}
var organizer = new EpisodeFileOrganizer(this, _config, _fileSystem, _logger, _libraryManager,
- _directoryWatchers);
+ _libraryMonitor);
await organizer.OrganizeEpisodeFile(result.OriginalPath, _config.Configuration.TvFileOrganizationOptions, true)
.ConfigureAwait(false);
-
- await _libraryManager.ValidateMediaLibrary(new Progress<double>(), CancellationToken.None)
- .ConfigureAwait(false);
}
public Task ClearLog()
@@ -108,12 +105,9 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
public async Task PerformEpisodeOrganization(EpisodeFileOrganizationRequest request)
{
var organizer = new EpisodeFileOrganizer(this, _config, _fileSystem, _logger, _libraryManager,
- _directoryWatchers);
+ _libraryMonitor);
await organizer.OrganizeWithCorrection(request, _config.Configuration.TvFileOrganizationOptions).ConfigureAwait(false);
-
- await _libraryManager.ValidateMediaLibrary(new Progress<double>(), CancellationToken.None)
- .ConfigureAwait(false);
}
}
}
diff --git a/MediaBrowser.Server.Implementations/FileOrganization/OrganizerScheduledTask.cs b/MediaBrowser.Server.Implementations/FileOrganization/OrganizerScheduledTask.cs
index 340038e4b..3c5e1ed0e 100644
--- a/MediaBrowser.Server.Implementations/FileOrganization/OrganizerScheduledTask.cs
+++ b/MediaBrowser.Server.Implementations/FileOrganization/OrganizerScheduledTask.cs
@@ -14,16 +14,16 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
{
public class OrganizerScheduledTask : IScheduledTask, IConfigurableScheduledTask
{
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly ILibraryManager _libraryManager;
private readonly ILogger _logger;
private readonly IFileSystem _fileSystem;
private readonly IServerConfigurationManager _config;
private readonly IFileOrganizationService _organizationService;
- public OrganizerScheduledTask(IDirectoryWatchers directoryWatchers, ILibraryManager libraryManager, ILogger logger, IFileSystem fileSystem, IServerConfigurationManager config, IFileOrganizationService organizationService)
+ public OrganizerScheduledTask(ILibraryMonitor libraryMonitor, ILibraryManager libraryManager, ILogger logger, IFileSystem fileSystem, IServerConfigurationManager config, IFileOrganizationService organizationService)
{
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_libraryManager = libraryManager;
_logger = logger;
_fileSystem = fileSystem;
@@ -48,7 +48,7 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
public Task Execute(CancellationToken cancellationToken, IProgress<double> progress)
{
- return new TvFolderOrganizer(_libraryManager, _logger, _fileSystem, _directoryWatchers, _organizationService, _config)
+ return new TvFolderOrganizer(_libraryManager, _logger, _fileSystem, _libraryMonitor, _organizationService, _config)
.Organize(_config.Configuration.TvFileOrganizationOptions, cancellationToken, progress);
}
diff --git a/MediaBrowser.Server.Implementations/FileOrganization/TvFolderOrganizer.cs b/MediaBrowser.Server.Implementations/FileOrganization/TvFolderOrganizer.cs
index 6a413f2f0..24f21e339 100644
--- a/MediaBrowser.Server.Implementations/FileOrganization/TvFolderOrganizer.cs
+++ b/MediaBrowser.Server.Implementations/FileOrganization/TvFolderOrganizer.cs
@@ -18,19 +18,19 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
{
public class TvFolderOrganizer
{
- private readonly IDirectoryWatchers _directoryWatchers;
+ private readonly ILibraryMonitor _libraryMonitor;
private readonly ILibraryManager _libraryManager;
private readonly ILogger _logger;
private readonly IFileSystem _fileSystem;
private readonly IFileOrganizationService _organizationService;
private readonly IServerConfigurationManager _config;
- public TvFolderOrganizer(ILibraryManager libraryManager, ILogger logger, IFileSystem fileSystem, IDirectoryWatchers directoryWatchers, IFileOrganizationService organizationService, IServerConfigurationManager config)
+ public TvFolderOrganizer(ILibraryManager libraryManager, ILogger logger, IFileSystem fileSystem, ILibraryMonitor libraryMonitor, IFileOrganizationService organizationService, IServerConfigurationManager config)
{
_libraryManager = libraryManager;
_logger = logger;
_fileSystem = fileSystem;
- _directoryWatchers = directoryWatchers;
+ _libraryMonitor = libraryMonitor;
_organizationService = organizationService;
_config = config;
}
@@ -57,7 +57,7 @@ namespace MediaBrowser.Server.Implementations.FileOrganization
foreach (var file in eligibleFiles)
{
var organizer = new EpisodeFileOrganizer(_organizationService, _config, _fileSystem, _logger, _libraryManager,
- _directoryWatchers);
+ _libraryMonitor);
var result = await organizer.OrganizeEpisodeFile(file.FullName, options, false).ConfigureAwait(false);
diff --git a/MediaBrowser.Server.Implementations/IO/DirectoryWatchers.cs b/MediaBrowser.Server.Implementations/IO/LibraryMonitor.cs
index 1efc3bc70..0716a3d83 100644
--- a/MediaBrowser.Server.Implementations/IO/DirectoryWatchers.cs
+++ b/MediaBrowser.Server.Implementations/IO/LibraryMonitor.cs
@@ -2,7 +2,6 @@
using MediaBrowser.Common.ScheduledTasks;
using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.IO;
using MediaBrowser.Controller.Library;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
@@ -18,10 +17,7 @@ using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.IO
{
- /// <summary>
- /// Class DirectoryWatchers
- /// </summary>
- public class DirectoryWatchers : IDirectoryWatchers
+ public class LibraryMonitor : ILibraryMonitor
{
/// <summary>
/// The file system watchers
@@ -55,17 +51,28 @@ namespace MediaBrowser.Server.Implementations.IO
/// Add the path to our temporary ignore list. Use when writing to a path within our listening scope.
/// </summary>
/// <param name="path">The path.</param>
- public void TemporarilyIgnore(string path)
+ private void TemporarilyIgnore(string path)
{
_tempIgnoredPaths[path] = path;
}
- /// <summary>
- /// Removes the temp ignore.
- /// </summary>
- /// <param name="path">The path.</param>
- public async void RemoveTempIgnore(string path)
+ public void ReportFileSystemChangeBeginning(string path)
+ {
+ if (string.IsNullOrEmpty(path))
+ {
+ throw new ArgumentNullException("path");
+ }
+
+ TemporarilyIgnore(path);
+ }
+
+ public async void ReportFileSystemChangeComplete(string path, bool refreshPath)
{
+ if (string.IsNullOrEmpty(path))
+ {
+ throw new ArgumentNullException("path");
+ }
+
// This is an arbitraty amount of time, but delay it because file system writes often trigger events after RemoveTempIgnore has been called.
// Seeing long delays in some situations, especially over the network.
// Seeing delays up to 40 seconds, but not going to ignore changes for that long.
@@ -73,6 +80,11 @@ namespace MediaBrowser.Server.Implementations.IO
string val;
_tempIgnoredPaths.TryRemove(path, out val);
+
+ if (refreshPath)
+ {
+ ReportFileSystemChanged(path);
+ }
}
/// <summary>
@@ -91,11 +103,11 @@ namespace MediaBrowser.Server.Implementations.IO
private IServerConfigurationManager ConfigurationManager { get; set; }
private readonly IFileSystem _fileSystem;
-
+
/// <summary>
- /// Initializes a new instance of the <see cref="DirectoryWatchers" /> class.
+ /// Initializes a new instance of the <see cref="LibraryMonitor" /> class.
/// </summary>
- public DirectoryWatchers(ILogManager logManager, ITaskManager taskManager, ILibraryManager libraryManager, IServerConfigurationManager configurationManager, IFileSystem fileSystem)
+ public LibraryMonitor(ILogManager logManager, ITaskManager taskManager, ILibraryManager libraryManager, IServerConfigurationManager configurationManager, IFileSystem fileSystem)
{
if (taskManager == null)
{
@@ -104,7 +116,7 @@ namespace MediaBrowser.Server.Implementations.IO
LibraryManager = libraryManager;
TaskManager = taskManager;
- Logger = logManager.GetLogger("DirectoryWatchers");
+ Logger = logManager.GetLogger(GetType().Name);
ConfigurationManager = configurationManager;
_fileSystem = fileSystem;
@@ -328,31 +340,30 @@ namespace MediaBrowser.Server.Implementations.IO
{
OnWatcherChanged(e);
}
- catch (IOException ex)
+ catch (Exception ex)
{
- Logger.ErrorException("IOException in watcher changed. Path: {0}", ex, e.FullPath);
+ Logger.ErrorException("Exception in watcher changed. Path: {0}", ex, e.FullPath);
}
}
private void OnWatcherChanged(FileSystemEventArgs e)
{
- var name = e.Name;
+ Logger.Debug("Watcher sees change of type " + e.ChangeType + " to " + e.FullPath);
- // Ignore certain files
- if (_alwaysIgnoreFiles.Contains(name, StringComparer.OrdinalIgnoreCase))
- {
- return;
- }
+ ReportFileSystemChanged(e.FullPath);
+ }
- var nameFromFullPath = Path.GetFileName(e.FullPath);
- // Ignore certain files
- if (!string.IsNullOrEmpty(nameFromFullPath) && _alwaysIgnoreFiles.Contains(nameFromFullPath, StringComparer.OrdinalIgnoreCase))
+ public void ReportFileSystemChanged(string path)
+ {
+ if (string.IsNullOrEmpty(path))
{
- return;
+ throw new ArgumentNullException("path");
}
+
+ var filename = Path.GetFileName(path);
- // Ignore when someone manually creates a new folder
- if (e.ChangeType == WatcherChangeTypes.Created && name == "New folder")
+ // Ignore certain files
+ if (!string.IsNullOrEmpty(filename) && _alwaysIgnoreFiles.Contains(filename, StringComparer.OrdinalIgnoreCase))
{
return;
}
@@ -362,36 +373,35 @@ namespace MediaBrowser.Server.Implementations.IO
// If the parent of an ignored path has a change event, ignore that too
if (tempIgnorePaths.Any(i =>
{
- if (string.Equals(i, e.FullPath, StringComparison.OrdinalIgnoreCase))
+ if (string.Equals(i, path, StringComparison.OrdinalIgnoreCase))
{
- Logger.Debug("Watcher ignoring change to {0}", e.FullPath);
+ Logger.Debug("Ignoring change to {0}", path);
return true;
}
- // Go up a level
- var parent = Path.GetDirectoryName(i);
- if (string.Equals(parent, e.FullPath, StringComparison.OrdinalIgnoreCase))
+ if (_fileSystem.ContainsSubPath(i, path))
{
- Logger.Debug("Watcher ignoring change to {0}", e.FullPath);
+ Logger.Debug("Ignoring change to {0}", path);
return true;
}
- // Go up another level
+ // Go up a level
+ var parent = Path.GetDirectoryName(i);
if (!string.IsNullOrEmpty(parent))
{
- parent = Path.GetDirectoryName(i);
- if (string.Equals(parent, e.FullPath, StringComparison.OrdinalIgnoreCase))
+ if (string.Equals(parent, path, StringComparison.OrdinalIgnoreCase))
{
- Logger.Debug("Watcher ignoring change to {0}", e.FullPath);
+ Logger.Debug("Ignoring change to {0}", path);
return true;
}
- }
- if (i.StartsWith(e.FullPath, StringComparison.OrdinalIgnoreCase) ||
- e.FullPath.StartsWith(i, StringComparison.OrdinalIgnoreCase))
- {
- Logger.Debug("Watcher ignoring change to {0}", e.FullPath);
- return true;
+ // Go up another level
+ parent = Path.GetDirectoryName(i);
+ if (string.Equals(parent, path, StringComparison.OrdinalIgnoreCase))
+ {
+ Logger.Debug("Ignoring change to {0}", path);
+ return true;
+ }
}
return false;
@@ -401,22 +411,19 @@ namespace MediaBrowser.Server.Implementations.IO
return;
}
- Logger.Info("Watcher sees change of type " + e.ChangeType + " to " + e.FullPath);
-
- //Since we're watching created, deleted and renamed we always want the parent of the item to be the affected path
- var affectedPath = e.FullPath;
-
- _affectedPaths.AddOrUpdate(affectedPath, affectedPath, (key, oldValue) => affectedPath);
+ // Avoid implicitly captured closure
+ var affectedPath = path;
+ _affectedPaths.AddOrUpdate(path, path, (key, oldValue) => affectedPath);
lock (_timerLock)
{
if (_updateTimer == null)
{
- _updateTimer = new Timer(TimerStopped, null, TimeSpan.FromSeconds(ConfigurationManager.Configuration.FileWatcherDelay), TimeSpan.FromMilliseconds(-1));
+ _updateTimer = new Timer(TimerStopped, null, TimeSpan.FromSeconds(ConfigurationManager.Configuration.RealtimeWatcherDelay), TimeSpan.FromMilliseconds(-1));
}
else
{
- _updateTimer.Change(TimeSpan.FromSeconds(ConfigurationManager.Configuration.FileWatcherDelay), TimeSpan.FromMilliseconds(-1));
+ _updateTimer.Change(TimeSpan.FromSeconds(ConfigurationManager.Configuration.RealtimeWatcherDelay), TimeSpan.FromMilliseconds(-1));
}
}
}
@@ -427,24 +434,9 @@ namespace MediaBrowser.Server.Implementations.IO
/// <param name="stateInfo">The state info.</param>
private async void TimerStopped(object stateInfo)
{
- lock (_timerLock)
- {
- // Extend the timer as long as any of the paths are still being written to.
- if (_affectedPaths.Any(p => IsFileLocked(p.Key)))
- {
- Logger.Info("Timer extended.");
- _updateTimer.Change(TimeSpan.FromSeconds(ConfigurationManager.Configuration.FileWatcherDelay), TimeSpan.FromMilliseconds(-1));
- return;
- }
-
- Logger.Info("Timer stopped.");
+ Logger.Debug("Timer stopped.");
- if (_updateTimer != null)
- {
- _updateTimer.Dispose();
- _updateTimer = null;
- }
- }
+ DisposeTimer();
var paths = _affectedPaths.Keys.ToList();
_affectedPaths.Clear();
@@ -452,59 +444,16 @@ namespace MediaBrowser.Server.Implementations.IO
await ProcessPathChanges(paths).ConfigureAwait(false);
}
- /// <summary>
- /// Try and determine if a file is locked
- /// This is not perfect, and is subject to race conditions, so I'd rather not make this a re-usable library method.
- /// </summary>
- /// <param name="path">The path.</param>
- /// <returns><c>true</c> if [is file locked] [the specified path]; otherwise, <c>false</c>.</returns>
- private bool IsFileLocked(string path)
+ private void DisposeTimer()
{
- try
- {
- var data = _fileSystem.GetFileSystemInfo(path);
-
- if (!data.Exists
- || data.Attributes.HasFlag(FileAttributes.Directory)
- || data.Attributes.HasFlag(FileAttributes.ReadOnly))
- {
- return false;
- }
- }
- catch (IOException)
- {
- return false;
- }
-
- try
+ lock (_timerLock)
{
- using (_fileSystem.GetFileStream(path, FileMode.Open, FileAccess.ReadWrite, FileShare.ReadWrite))
+ if (_updateTimer != null)
{
- //file is not locked
- return false;
+ _updateTimer.Dispose();
+ _updateTimer = null;
}
}
- catch (DirectoryNotFoundException)
- {
- return false;
- }
- catch (FileNotFoundException)
- {
- return false;
- }
- catch (IOException)
- {
- //the file is unavailable because it is:
- //still being written to
- //or being processed by another thread
- //or does not exist (has already been processed)
- Logger.Debug("{0} is locked.", path);
- return true;
- }
- catch
- {
- return false;
- }
}
/// <summary>
@@ -599,14 +548,7 @@ namespace MediaBrowser.Server.Implementations.IO
watcher.Dispose();
}
- lock (_timerLock)
- {
- if (_updateTimer != null)
- {
- _updateTimer.Dispose();
- _updateTimer = null;
- }
- }
+ DisposeTimer();
_fileSystemWatchers.Clear();
_affectedPaths.Clear();
diff --git a/MediaBrowser.Server.Implementations/Library/LibraryManager.cs b/MediaBrowser.Server.Implementations/Library/LibraryManager.cs
index 736c70ad5..17b5ea424 100644
--- a/MediaBrowser.Server.Implementations/Library/LibraryManager.cs
+++ b/MediaBrowser.Server.Implementations/Library/LibraryManager.cs
@@ -137,7 +137,7 @@ namespace MediaBrowser.Server.Implementations.Library
private IEnumerable<IMetadataSaver> _savers;
- private readonly Func<IDirectoryWatchers> _directoryWatchersFactory;
+ private readonly Func<ILibraryMonitor> _libraryMonitorFactory;
/// <summary>
/// The _library items cache
@@ -180,14 +180,14 @@ namespace MediaBrowser.Server.Implementations.Library
/// <param name="userManager">The user manager.</param>
/// <param name="configurationManager">The configuration manager.</param>
/// <param name="userDataRepository">The user data repository.</param>
- public LibraryManager(ILogger logger, ITaskManager taskManager, IUserManager userManager, IServerConfigurationManager configurationManager, IUserDataManager userDataRepository, Func<IDirectoryWatchers> directoryWatchersFactory, IFileSystem fileSystem)
+ public LibraryManager(ILogger logger, ITaskManager taskManager, IUserManager userManager, IServerConfigurationManager configurationManager, IUserDataManager userDataRepository, Func<ILibraryMonitor> libraryMonitorFactory, IFileSystem fileSystem)
{
_logger = logger;
_taskManager = taskManager;
_userManager = userManager;
ConfigurationManager = configurationManager;
_userDataRepository = userDataRepository;
- _directoryWatchersFactory = directoryWatchersFactory;
+ _libraryMonitorFactory = libraryMonitorFactory;
_fileSystem = fileSystem;
ByReferenceItems = new ConcurrentDictionary<Guid, BaseItem>();
@@ -934,7 +934,7 @@ namespace MediaBrowser.Server.Implementations.Library
/// <returns>Task.</returns>
public async Task ValidateMediaLibraryInternal(IProgress<double> progress, CancellationToken cancellationToken)
{
- _directoryWatchersFactory().Stop();
+ _libraryMonitorFactory().Stop();
try
{
@@ -942,7 +942,7 @@ namespace MediaBrowser.Server.Implementations.Library
}
finally
{
- _directoryWatchersFactory().Start();
+ _libraryMonitorFactory().Start();
}
}
@@ -1462,13 +1462,13 @@ namespace MediaBrowser.Server.Implementations.Library
var semaphore = _fileLocks.GetOrAdd(path, key => new SemaphoreSlim(1, 1));
- var directoryWatchers = _directoryWatchersFactory();
+ var libraryMonitor = _libraryMonitorFactory();
await semaphore.WaitAsync().ConfigureAwait(false);
try
{
- directoryWatchers.TemporarilyIgnore(path);
+ libraryMonitor.ReportFileSystemChangeBeginning(path);
saver.Save(item, CancellationToken.None);
}
catch (Exception ex)
@@ -1477,7 +1477,7 @@ namespace MediaBrowser.Server.Implementations.Library
}
finally
{
- directoryWatchers.RemoveTempIgnore(path);
+ libraryMonitor.ReportFileSystemChangeComplete(path, false);
semaphore.Release();
}
}
diff --git a/MediaBrowser.Server.Implementations/Library/ResolverHelper.cs b/MediaBrowser.Server.Implementations/Library/ResolverHelper.cs
index e32fcd627..4ce5f11d4 100644
--- a/MediaBrowser.Server.Implementations/Library/ResolverHelper.cs
+++ b/MediaBrowser.Server.Implementations/Library/ResolverHelper.cs
@@ -46,7 +46,7 @@ namespace MediaBrowser.Server.Implementations.Library
}
// Make sure the item has a name
- EnsureName(item);
+ EnsureName(item, args);
item.DontFetchMeta = item.Path.IndexOf("[dontfetchmeta]", StringComparison.OrdinalIgnoreCase) != -1 ||
item.Parents.Any(i => i.DontFetchMeta);
@@ -59,13 +59,13 @@ namespace MediaBrowser.Server.Implementations.Library
/// Ensures the name.
/// </summary>
/// <param name="item">The item.</param>
- private static void EnsureName(BaseItem item)
+ private static void EnsureName(BaseItem item, ItemResolveArgs args)
{
// If the subclass didn't supply a name, add it here
if (string.IsNullOrEmpty(item.Name) && !string.IsNullOrEmpty(item.Path))
{
//we use our resolve args name here to get the name of the containg folder, not actual video file
- item.Name = GetMBName(item.ResolveArgs.FileInfo.Name, (item.ResolveArgs.FileInfo.Attributes & FileAttributes.Directory) == FileAttributes.Directory);
+ item.Name = GetMBName(args.FileInfo.Name, (args.FileInfo.Attributes & FileAttributes.Directory) == FileAttributes.Directory);
}
}
diff --git a/MediaBrowser.Server.Implementations/Library/UserManager.cs b/MediaBrowser.Server.Implementations/Library/UserManager.cs
index d4a74f2b6..ce76dd21b 100644
--- a/MediaBrowser.Server.Implementations/Library/UserManager.cs
+++ b/MediaBrowser.Server.Implementations/Library/UserManager.cs
@@ -4,6 +4,7 @@ using MediaBrowser.Controller.Configuration;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Persistence;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Logging;
using System;
using System.Collections.Generic;
@@ -192,7 +193,11 @@ namespace MediaBrowser.Server.Implementations.Library
/// <returns>Task.</returns>
public Task RefreshUsersMetadata(CancellationToken cancellationToken, bool force = false)
{
- var tasks = Users.Select(user => user.RefreshMetadata(cancellationToken, forceRefresh: force)).ToList();
+ var tasks = Users.Select(user => user.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ReplaceAllMetadata = force
+
+ }, cancellationToken)).ToList();
return Task.WhenAll(tasks);
}
diff --git a/MediaBrowser.Server.Implementations/Library/Validators/GenresPostScanTask.cs b/MediaBrowser.Server.Implementations/Library/Validators/GenresPostScanTask.cs
index b4907a70c..d7add8574 100644
--- a/MediaBrowser.Server.Implementations/Library/Validators/GenresPostScanTask.cs
+++ b/MediaBrowser.Server.Implementations/Library/Validators/GenresPostScanTask.cs
@@ -16,7 +16,6 @@ namespace MediaBrowser.Server.Implementations.Library.Validators
/// 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)
{
_libraryManager = libraryManager;
diff --git a/MediaBrowser.Server.Implementations/Library/Validators/PeoplePostScanTask.cs b/MediaBrowser.Server.Implementations/Library/Validators/PeoplePostScanTask.cs
index 0104b2b7e..c8094302c 100644
--- a/MediaBrowser.Server.Implementations/Library/Validators/PeoplePostScanTask.cs
+++ b/MediaBrowser.Server.Implementations/Library/Validators/PeoplePostScanTask.cs
@@ -1,5 +1,6 @@
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.Library;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Logging;
using System;
using System.Collections.Generic;
@@ -88,7 +89,14 @@ namespace MediaBrowser.Server.Implementations.Library.Validators
var itemByName = _libraryManager.GetPerson(name);
- await itemByName.RefreshMetadata(cancellationToken, allowSlowProviders: false).ConfigureAwait(false);
+ // The only purpose here is to be able to react to image changes without running the people task.
+ // All other metadata can wait for that.
+ await itemByName.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ImageRefreshMode = MetadataRefreshMode.None,
+ MetadataRefreshMode = MetadataRefreshMode.None
+
+ }, cancellationToken).ConfigureAwait(false);
foreach (var libraryId in counts.Keys)
{
diff --git a/MediaBrowser.Server.Implementations/LiveTv/ChannelImageProvider.cs b/MediaBrowser.Server.Implementations/LiveTv/ChannelImageProvider.cs
index f1e10e175..9fdca568e 100644
--- a/MediaBrowser.Server.Implementations/LiveTv/ChannelImageProvider.cs
+++ b/MediaBrowser.Server.Implementations/LiveTv/ChannelImageProvider.cs
@@ -1,154 +1,111 @@
-using MediaBrowser.Common.IO;
-using MediaBrowser.Common.Net;
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
using System;
-using System.IO;
+using System.Collections.Generic;
using System.Linq;
-using System.Net;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.LiveTv
{
- public class ChannelImageProvider : BaseMetadataProvider
+ public class ChannelImageProvider : IDynamicImageProvider, IHasChangeMonitor
{
private readonly ILiveTvManager _liveTvManager;
- private readonly IProviderManager _providerManager;
- private readonly IFileSystem _fileSystem;
private readonly IHttpClient _httpClient;
+ private readonly ILogger _logger;
- public ChannelImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, ILiveTvManager liveTvManager, IProviderManager providerManager, IFileSystem fileSystem, IHttpClient httpClient)
- : base(logManager, configurationManager)
+ public ChannelImageProvider(ILiveTvManager liveTvManager, IHttpClient httpClient, ILogger logger)
{
_liveTvManager = liveTvManager;
- _providerManager = providerManager;
- _fileSystem = fileSystem;
_httpClient = httpClient;
+ _logger = logger;
}
- public override bool Supports(BaseItem item)
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
{
- return item is LiveTvChannel;
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- return !item.HasImage(ImageType.Primary);
+ return new[] { ImageType.Primary };
}
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
+ public async Task<DynamicImageResponse> GetImage(IHasImages item, ImageType type, CancellationToken cancellationToken)
{
- if (item.HasImage(ImageType.Primary))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var changed = true;
-
- try
- {
- changed = await DownloadImage((LiveTvChannel)item, cancellationToken).ConfigureAwait(false);
- }
- catch (HttpException ex)
- {
- // Don't fail the provider on a 404
- if (!ex.StatusCode.HasValue || ex.StatusCode.Value != HttpStatusCode.NotFound)
- {
- throw;
- }
- }
-
- if (changed)
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- }
-
- return changed;
- }
+ var liveTvItem = (LiveTvChannel)item;
- private async Task<bool> DownloadImage(LiveTvChannel item, CancellationToken cancellationToken)
- {
- Stream imageStream = null;
- string contentType = null;
+ var imageResponse = new DynamicImageResponse();
- if (!string.IsNullOrEmpty(item.ProviderImagePath))
+ if (!string.IsNullOrEmpty(liveTvItem.ProviderImagePath))
{
- contentType = "image/" + Path.GetExtension(item.ProviderImagePath).ToLower();
- imageStream = _fileSystem.GetFileStream(item.ProviderImagePath, FileMode.Open, FileAccess.Read, FileShare.Read, true);
+ imageResponse.Path = liveTvItem.ProviderImagePath;
+ imageResponse.HasImage = true;
}
- else if (!string.IsNullOrEmpty(item.ProviderImageUrl))
+ else if (!string.IsNullOrEmpty(liveTvItem.ProviderImageUrl))
{
var options = new HttpRequestOptions
{
CancellationToken = cancellationToken,
- Url = item.ProviderImageUrl
+ Url = liveTvItem.ProviderImageUrl
};
var response = await _httpClient.GetResponse(options).ConfigureAwait(false);
- if (!response.ContentType.StartsWith("image/", StringComparison.OrdinalIgnoreCase))
+ if (response.ContentType.StartsWith("image/", StringComparison.OrdinalIgnoreCase))
{
- Logger.Error("Provider did not return an image content type.");
- return false;
+ imageResponse.HasImage = true;
+ imageResponse.Stream = response.Content;
+ imageResponse.SetFormatFromMimeType(response.ContentType);
+ }
+ else
+ {
+ _logger.Error("Provider did not return an image content type.");
}
-
- imageStream = response.Content;
- contentType = response.ContentType;
}
- else if (item.HasProviderImage ?? true)
+ else if (liveTvItem.HasProviderImage ?? true)
{
- var service = _liveTvManager.Services.FirstOrDefault(i => string.Equals(i.Name, item.ServiceName, StringComparison.OrdinalIgnoreCase));
+ var service = _liveTvManager.Services.FirstOrDefault(i => string.Equals(i.Name, liveTvItem.ServiceName, StringComparison.OrdinalIgnoreCase));
if (service != null)
{
try
{
- var response = await service.GetChannelImageAsync(item.ExternalId, cancellationToken).ConfigureAwait(false);
+ var response = await service.GetChannelImageAsync(liveTvItem.ExternalId, cancellationToken).ConfigureAwait(false);
if (response != null)
{
- imageStream = response.Stream;
- contentType = response.MimeType;
+ imageResponse.HasImage = true;
+ imageResponse.Stream = response.Stream;
+ imageResponse.Format = response.Format;
}
}
catch (NotImplementedException)
{
- return false;
}
}
}
- if (imageStream != null)
- {
- // Dummy up the original url
- var url = item.ServiceName + item.ExternalId;
+ return imageResponse;
+ }
- await _providerManager.SaveImage(item, imageStream, contentType, ImageType.Primary, null, url, cancellationToken).ConfigureAwait(false);
- return true;
- }
+ public string Name
+ {
+ get { return "Live TV Service Provider"; }
+ }
- return false;
+ public bool Supports(IHasImages item)
+ {
+ return item is LiveTvChannel;
}
- public override MetadataProviderPriority Priority
+ public int Order
{
- get { return MetadataProviderPriority.Second; }
+ get { return 0; }
}
- public override ItemUpdateType ItemUpdateType
+ public bool HasChanged(IHasMetadata item, DateTime date)
{
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
+ return !item.HasImage(ImageType.Primary) && (DateTime.UtcNow - date).TotalDays >= 1;
}
}
}
diff --git a/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs b/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
index e256d7da5..9501d2d12 100644
--- a/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
+++ b/MediaBrowser.Server.Implementations/LiveTv/LiveTvManager.cs
@@ -9,6 +9,7 @@ using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Controller.MediaInfo;
using MediaBrowser.Controller.Persistence;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Controller.Sorting;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.LiveTv;
@@ -328,7 +329,12 @@ namespace MediaBrowser.Server.Implementations.LiveTv
// Set this now so we don't cause additional file system access during provider executions
item.ResetResolveArgs(fileInfo);
- await item.RefreshMetadata(cancellationToken, forceSave: isNew, resetResolveArgs: false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = isNew,
+ ResetResolveArgs = false
+
+ }, cancellationToken);
return item;
}
@@ -383,7 +389,12 @@ namespace MediaBrowser.Server.Implementations.LiveTv
item.RunTimeTicks = (info.EndDate - info.StartDate).Ticks;
item.StartDate = info.StartDate;
- await item.RefreshMetadata(cancellationToken, forceSave: isNew, resetResolveArgs: false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = isNew,
+ ResetResolveArgs = false
+
+ }, cancellationToken);
return item;
}
@@ -435,7 +446,12 @@ namespace MediaBrowser.Server.Implementations.LiveTv
item.RecordingInfo = info;
item.ServiceName = serviceName;
- await item.RefreshMetadata(cancellationToken, forceSave: isNew, resetResolveArgs: false);
+ await item.RefreshMetadata(new MetadataRefreshOptions
+ {
+ ForceSave = isNew,
+ ResetResolveArgs = false
+
+ }, cancellationToken);
_libraryManager.RegisterItem((BaseItem)item);
diff --git a/MediaBrowser.Server.Implementations/LiveTv/ProgramImageProvider.cs b/MediaBrowser.Server.Implementations/LiveTv/ProgramImageProvider.cs
index 041925cdd..117cb1da7 100644
--- a/MediaBrowser.Server.Implementations/LiveTv/ProgramImageProvider.cs
+++ b/MediaBrowser.Server.Implementations/LiveTv/ProgramImageProvider.cs
@@ -1,154 +1,111 @@
-using MediaBrowser.Common.IO;
-using MediaBrowser.Common.Net;
-using MediaBrowser.Controller.Configuration;
+using MediaBrowser.Common.Net;
using MediaBrowser.Controller.Entities;
-using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
-using MediaBrowser.Model.Net;
using System;
-using System.IO;
+using System.Collections.Generic;
using System.Linq;
-using System.Net;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.LiveTv
{
- public class ProgramImageProvider : BaseMetadataProvider
+ public class ProgramImageProvider : IDynamicImageProvider, IHasChangeMonitor
{
private readonly ILiveTvManager _liveTvManager;
- private readonly IProviderManager _providerManager;
- private readonly IFileSystem _fileSystem;
private readonly IHttpClient _httpClient;
+ private readonly ILogger _logger;
- public ProgramImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, ILiveTvManager liveTvManager, IProviderManager providerManager, IFileSystem fileSystem, IHttpClient httpClient)
- : base(logManager, configurationManager)
+ public ProgramImageProvider(ILiveTvManager liveTvManager, IHttpClient httpClient, ILogger logger)
{
_liveTvManager = liveTvManager;
- _providerManager = providerManager;
- _fileSystem = fileSystem;
_httpClient = httpClient;
+ _logger = logger;
}
- public override bool Supports(BaseItem item)
+ public IEnumerable<ImageType> GetSupportedImages(IHasImages item)
{
- return item is LiveTvProgram;
- }
-
- protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
- {
- return !item.HasImage(ImageType.Primary);
+ return new[] { ImageType.Primary };
}
- public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
+ public async Task<DynamicImageResponse> GetImage(IHasImages item, ImageType type, CancellationToken cancellationToken)
{
- if (item.HasImage(ImageType.Primary))
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- return true;
- }
-
- var changed = true;
-
- try
- {
- changed = await DownloadImage((LiveTvProgram)item, cancellationToken).ConfigureAwait(false);
- }
- catch (HttpException ex)
- {
- // Don't fail the provider on a 404
- if (!ex.StatusCode.HasValue || ex.StatusCode.Value != HttpStatusCode.NotFound)
- {
- throw;
- }
- }
-
- if (changed)
- {
- SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
- }
-
- return changed;
- }
+ var liveTvItem = (LiveTvProgram)item;
- private async Task<bool> DownloadImage(LiveTvProgram item, CancellationToken cancellationToken)
- {
- Stream imageStream = null;
- string contentType = null;
+ var imageResponse = new DynamicImageResponse();
- if (!string.IsNullOrEmpty(item.ProviderImagePath))
+ if (!string.IsNullOrEmpty(liveTvItem.ProviderImagePath))
{
- contentType = "image/" + Path.GetExtension(item.ProviderImagePath).ToLower();
- imageStream = _fileSystem.GetFileStream(item.ProviderImagePath, FileMode.Open, FileAccess.Read, FileShare.Read, true);
+ imageResponse.Path = liveTvItem.ProviderImagePath;
+ imageResponse.HasImage = true;
}
- else if (!string.IsNullOrEmpty(item.ProviderImageUrl))
+ else if (!string.IsNullOrEmpty(liveTvItem.ProviderImageUrl))
{
var options = new HttpRequestOptions
{
CancellationToken = cancellationToken,
- Url = item.ProviderImageUrl
+ Url = liveTvItem.ProviderImageUrl
};
var response = await _httpClient.GetResponse(options).ConfigureAwait(false);
- if (!response.ContentType.StartsWith("image/", StringComparison.OrdinalIgnoreCase))
+ if (response.ContentType.StartsWith("image/", StringComparison.OrdinalIgnoreCase))
{
- Logger.Error("Provider did not return an image content type.");
- return false;
+ imageResponse.HasImage = true;
+ imageResponse.Stream = response.Content;
+ imageResponse.SetFormatFromMimeType(response.ContentType);
+ }
+ else
+ {
+ _logger.Error("Provider did not return an image content type.");
}
-
- imageStream = response.Content;
- contentType = response.ContentType;
}
- else if (item.HasProviderImage ?? true)
+ else if (liveTvItem.HasProviderImage ?? true)
{
- var service = _liveTvManager.Services.FirstOrDefault(i => string.Equals(i.Name, item.ServiceName, StringComparison.OrdinalIgnoreCase));
+ var service = _liveTvManager.Services.FirstOrDefault(i => string.Equals(i.Name, liveTvItem.ServiceName, StringComparison.OrdinalIgnoreCase));
if (service != null)
{
try
{
- var response = await service.GetProgramImageAsync(item.ExternalId, item.ExternalChannelId, cancellationToken).ConfigureAwait(false);
+ var response = await service.GetProgramImageAsync(liveTvItem.ExternalId, liveTvItem.ExternalChannelId, cancellationToken).ConfigureAwait(false);
if (response != null)
{
- imageStream = response.Stream;
- contentType = response.MimeType;
+ imageResponse.HasImage = true;
+ imageResponse.Stream = response.Stream;
+ imageResponse.Format = response.Format;
}
}
catch (NotImplementedException)
{
- return false;
}
}
}
- if (imageStream != null)
- {
- // Dummy up the original url
- var url = item.ServiceName + item.ExternalId;
+ return imageResponse;
+ }
- await _providerManager.SaveImage(item, imageStream, contentType, ImageType.Primary, null, url, cancellationToken).ConfigureAwait(false);
- return true;
- }
+ public string Name
+ {
+ get { return "Live TV Service Provider"; }
+ }
- return false;
+ public bool Supports(IHasImages item)
+ {
+ return item is LiveTvProgram;
}
- public override MetadataProviderPriority Priority
+ public int Order
{
- get { return MetadataProviderPriority.Second; }
+ get { return 0; }
}
- public override ItemUpdateType ItemUpdateType
+ public bool HasChanged(IHasMetadata item, DateTime date)
{
- get
- {
- return ItemUpdateType.ImageUpdate;
- }
+ return !item.HasImage(ImageType.Primary) && (DateTime.UtcNow - date).TotalHours >= 12;
}
}
}
diff --git a/MediaBrowser.Server.Implementations/LiveTv/RecordingImageProvider.cs b/MediaBrowser.Server.Implementations/LiveTv/RecordingImageProvider.cs
index 9f6ab85a4..ce7c1286b 100644
--- a/MediaBrowser.Server.Implementations/LiveTv/RecordingImageProvider.cs
+++ b/MediaBrowser.Server.Implementations/LiveTv/RecordingImageProvider.cs
@@ -118,7 +118,7 @@ namespace MediaBrowser.Server.Implementations.LiveTv
if (response != null)
{
imageStream = response.Stream;
- contentType = response.MimeType;
+ contentType = "image/" + response.Format.ToString().ToLower();
}
}
catch (NotImplementedException)
diff --git a/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj b/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
index 314e7a458..fe4283368 100644
--- a/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
+++ b/MediaBrowser.Server.Implementations/MediaBrowser.Server.Implementations.csproj
@@ -137,7 +137,7 @@
<Compile Include="HttpServer\StreamWriter.cs" />
<Compile Include="HttpServer\SwaggerService.cs" />
<Compile Include="Drawing\ImageProcessor.cs" />
- <Compile Include="IO\DirectoryWatchers.cs" />
+ <Compile Include="IO\LibraryMonitor.cs" />
<Compile Include="Library\CoreResolutionIgnoreRule.cs" />
<Compile Include="Library\LibraryManager.cs" />
<Compile Include="Library\SearchEngine.cs" />
@@ -189,8 +189,6 @@
<Compile Include="Persistence\SqliteShrinkMemoryTimer.cs" />
<Compile Include="Persistence\TypeMapper.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />
- <Compile Include="Providers\ImageSaver.cs" />
- <Compile Include="Providers\ProviderManager.cs" />
<Compile Include="Roku\RokuControllerFactory.cs" />
<Compile Include="ScheduledTasks\PeopleValidationTask.cs" />
<Compile Include="ScheduledTasks\ChapterImagesTask.cs" />
diff --git a/MediaBrowser.Server.Implementations/Persistence/SqliteItemRepository.cs b/MediaBrowser.Server.Implementations/Persistence/SqliteItemRepository.cs
index 200898a62..6b463bbdf 100644
--- a/MediaBrowser.Server.Implementations/Persistence/SqliteItemRepository.cs
+++ b/MediaBrowser.Server.Implementations/Persistence/SqliteItemRepository.cs
@@ -58,7 +58,6 @@ namespace MediaBrowser.Server.Implementations.Persistence
private SqliteChapterRepository _chapterRepository;
private SqliteMediaStreamsRepository _mediaStreamsRepository;
- private SqliteProviderInfoRepository _providerInfoRepository;
private IDbCommand _deleteChildrenCommand;
private IDbCommand _saveChildrenCommand;
@@ -99,10 +98,6 @@ namespace MediaBrowser.Server.Implementations.Persistence
var mediaStreamsDbFile = Path.Combine(_appPaths.DataPath, "mediainfo.db");
var mediaStreamsConnection = SqliteExtensions.ConnectToDb(mediaStreamsDbFile, _logger).Result;
_mediaStreamsRepository = new SqliteMediaStreamsRepository(mediaStreamsConnection, logManager);
-
- var providerInfosDbFile = Path.Combine(_appPaths.DataPath, "providerinfo.db");
- var providerInfoConnection = SqliteExtensions.ConnectToDb(providerInfosDbFile, _logger).Result;
- _providerInfoRepository = new SqliteProviderInfoRepository(providerInfoConnection, logManager);
}
/// <summary>
@@ -134,7 +129,6 @@ namespace MediaBrowser.Server.Implementations.Persistence
PrepareStatements();
_mediaStreamsRepository.Initialize();
- _providerInfoRepository.Initialize();
_chapterRepository.Initialize();
_shrinkMemoryTimer = new SqliteShrinkMemoryTimer(_connection, _writeLock, _logger);
@@ -436,12 +430,6 @@ namespace MediaBrowser.Server.Implementations.Persistence
_mediaStreamsRepository.Dispose();
_mediaStreamsRepository = null;
}
-
- if (_providerInfoRepository != null)
- {
- _providerInfoRepository.Dispose();
- _providerInfoRepository = null;
- }
}
}
catch (Exception ex)
@@ -556,15 +544,5 @@ namespace MediaBrowser.Server.Implementations.Persistence
{
return _mediaStreamsRepository.SaveMediaStreams(id, streams, cancellationToken);
}
-
- public IEnumerable<BaseProviderInfo> GetProviderHistory(Guid itemId)
- {
- return _providerInfoRepository.GetBaseProviderInfos(itemId);
- }
-
- public Task SaveProviderHistory(Guid id, IEnumerable<BaseProviderInfo> history, CancellationToken cancellationToken)
- {
- return _providerInfoRepository.SaveProviderInfos(id, history, cancellationToken);
- }
}
} \ No newline at end of file
diff --git a/MediaBrowser.Server.Implementations/Persistence/SqliteProviderInfoRepository.cs b/MediaBrowser.Server.Implementations/Persistence/SqliteProviderInfoRepository.cs
index 9971c7460..8a82c062d 100644
--- a/MediaBrowser.Server.Implementations/Persistence/SqliteProviderInfoRepository.cs
+++ b/MediaBrowser.Server.Implementations/Persistence/SqliteProviderInfoRepository.cs
@@ -1,4 +1,6 @@
-using MediaBrowser.Controller.Providers;
+using System.IO;
+using MediaBrowser.Common.Configuration;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Logging;
using System;
using System.Collections.Generic;
@@ -9,7 +11,7 @@ using System.Threading.Tasks;
namespace MediaBrowser.Server.Implementations.Persistence
{
- class SqliteProviderInfoRepository
+ public class SqliteProviderInfoRepository : IProviderRepository
{
private IDbConnection _connection;
@@ -17,32 +19,47 @@ namespace MediaBrowser.Server.Implementations.Persistence
private IDbCommand _deleteInfosCommand;
private IDbCommand _saveInfoCommand;
+ private IDbCommand _saveStatusCommand;
+ private readonly IApplicationPaths _appPaths;
- public SqliteProviderInfoRepository(IDbConnection connection, ILogManager logManager)
+ public SqliteProviderInfoRepository(IApplicationPaths appPaths, ILogManager logManager)
{
- _connection = connection;
-
+ _appPaths = appPaths;
_logger = logManager.GetLogger(GetType().Name);
}
private SqliteShrinkMemoryTimer _shrinkMemoryTimer;
-
+
+ /// <summary>
+ /// Gets the name of the repository
+ /// </summary>
+ /// <value>The name.</value>
+ public string Name
+ {
+ get
+ {
+ return "SQLite";
+ }
+ }
+
/// <summary>
/// Opens the connection to the database
/// </summary>
/// <returns>Task.</returns>
- public void Initialize()
+ public async Task Initialize()
{
- var createTableCommand
- = "create table if not exists providerinfos ";
+ var dbFile = Path.Combine(_appPaths.DataPath, "providerinfo.db");
- createTableCommand += "(ItemId GUID, ProviderId GUID, ProviderVersion TEXT, FileStamp GUID, LastRefreshStatus TEXT, LastRefreshed datetime, PRIMARY KEY (ItemId, ProviderId))";
+ _connection = await SqliteExtensions.ConnectToDb(dbFile, _logger).ConfigureAwait(false);
string[] queries = {
- createTableCommand,
+ "create table if not exists providerinfos (ItemId GUID, ProviderId GUID, ProviderVersion TEXT, FileStamp GUID, LastRefreshStatus TEXT, LastRefreshed datetime, PRIMARY KEY (ItemId, ProviderId))",
"create index if not exists idx_providerinfos on providerinfos(ItemId, ProviderId)",
+ "create table if not exists MetadataStatus (ItemId GUID PRIMARY KEY, DateLastMetadataRefresh datetime, DateLastImagesRefresh datetime, LastStatus TEXT, LastErrorMessage TEXT, MetadataProvidersRefreshed TEXT, ImageProvidersRefreshed TEXT)",
+ "create index if not exists idx_MetadataStatus on MetadataStatus(ItemId)",
+
//pragmas
"pragma temp_store = memory",
@@ -56,7 +73,7 @@ namespace MediaBrowser.Server.Implementations.Persistence
_shrinkMemoryTimer = new SqliteShrinkMemoryTimer(_connection, _writeLock, _logger);
}
- private static readonly string[] SaveColumns =
+ private static readonly string[] SaveHistoryColumns =
{
"ItemId",
"ProviderId",
@@ -66,7 +83,18 @@ namespace MediaBrowser.Server.Implementations.Persistence
"LastRefreshed"
};
- private readonly string[] _selectColumns = SaveColumns.Skip(1).ToArray();
+ private readonly string[] _historySelectColumns = SaveHistoryColumns.Skip(1).ToArray();
+
+ private static readonly string[] StatusColumns =
+ {
+ "ItemId",
+ "DateLastMetadataRefresh",
+ "DateLastImagesRefresh",
+ "LastStatus",
+ "LastErrorMessage",
+ "MetadataProvidersRefreshed",
+ "ImageProvidersRefreshed"
+ };
/// <summary>
/// The _write lock
@@ -85,16 +113,27 @@ namespace MediaBrowser.Server.Implementations.Persistence
_saveInfoCommand = _connection.CreateCommand();
_saveInfoCommand.CommandText = string.Format("replace into providerinfos ({0}) values ({1})",
- string.Join(",", SaveColumns),
- string.Join(",", SaveColumns.Select(i => "@" + i).ToArray()));
+ string.Join(",", SaveHistoryColumns),
+ string.Join(",", SaveHistoryColumns.Select(i => "@" + i).ToArray()));
- foreach (var col in SaveColumns)
+ foreach (var col in SaveHistoryColumns)
{
_saveInfoCommand.Parameters.Add(_saveInfoCommand, "@" + col);
}
+
+ _saveStatusCommand = _connection.CreateCommand();
+
+ _saveStatusCommand.CommandText = string.Format("replace into MetadataStatus ({0}) values ({1})",
+ string.Join(",", StatusColumns),
+ string.Join(",", StatusColumns.Select(i => "@" + i).ToArray()));
+
+ foreach (var col in StatusColumns)
+ {
+ _saveStatusCommand.Parameters.Add(_saveStatusCommand, "@" + col);
+ }
}
- public IEnumerable<BaseProviderInfo> GetBaseProviderInfos(Guid itemId)
+ public IEnumerable<BaseProviderInfo> GetProviderHistory(Guid itemId)
{
if (itemId == Guid.Empty)
{
@@ -103,7 +142,7 @@ namespace MediaBrowser.Server.Implementations.Persistence
using (var cmd = _connection.CreateCommand())
{
- var cmdText = "select " + string.Join(",", _selectColumns) + " from providerinfos where";
+ var cmdText = "select " + string.Join(",", _historySelectColumns) + " from providerinfos where";
cmdText += " ItemId=@ItemId";
cmd.Parameters.Add(cmd, "@ItemId", DbType.Guid).Value = itemId;
@@ -121,10 +160,10 @@ namespace MediaBrowser.Server.Implementations.Persistence
}
/// <summary>
- /// Gets the chapter.
+ /// Gets the base provider information.
/// </summary>
/// <param name="reader">The reader.</param>
- /// <returns>ChapterInfo.</returns>
+ /// <returns>BaseProviderInfo.</returns>
private BaseProviderInfo GetBaseProviderInfo(IDataReader reader)
{
var item = new BaseProviderInfo
@@ -144,7 +183,7 @@ namespace MediaBrowser.Server.Implementations.Persistence
return item;
}
- public async Task SaveProviderInfos(Guid id, IEnumerable<BaseProviderInfo> infos, CancellationToken cancellationToken)
+ public async Task SaveProviderHistory(Guid id, IEnumerable<BaseProviderInfo> infos, CancellationToken cancellationToken)
{
if (id == Guid.Empty)
{
@@ -166,7 +205,6 @@ namespace MediaBrowser.Server.Implementations.Persistence
{
transaction = _connection.BeginTransaction();
- // First delete chapters
_deleteInfosCommand.GetParameter(0).Value = id;
_deleteInfosCommand.Transaction = transaction;
@@ -221,6 +259,136 @@ namespace MediaBrowser.Server.Implementations.Persistence
}
}
+ public MetadataStatus GetMetadataStatus(Guid itemId)
+ {
+ if (itemId == Guid.Empty)
+ {
+ throw new ArgumentNullException("itemId");
+ }
+
+ using (var cmd = _connection.CreateCommand())
+ {
+ var cmdText = "select " + string.Join(",", StatusColumns) + " from MetadataStatus where";
+
+ cmdText += " ItemId=@ItemId";
+ cmd.Parameters.Add(cmd, "@ItemId", DbType.Guid).Value = itemId;
+
+ cmd.CommandText = cmdText;
+
+ using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult | CommandBehavior.SingleRow))
+ {
+ while (reader.Read())
+ {
+ return GetStatus(reader);
+ }
+
+ return null;
+ }
+ }
+ }
+
+ private MetadataStatus GetStatus(IDataReader reader)
+ {
+ var result = new MetadataStatus
+ {
+ ItemId = reader.GetGuid(0)
+ };
+
+ if (!reader.IsDBNull(1))
+ {
+ result.DateLastMetadataRefresh = reader.GetDateTime(1).ToUniversalTime();
+ }
+
+ if (!reader.IsDBNull(2))
+ {
+ result.DateLastImagesRefresh = reader.GetDateTime(2).ToUniversalTime();
+ }
+
+ if (!reader.IsDBNull(3))
+ {
+ result.LastStatus = (ProviderRefreshStatus)Enum.Parse(typeof(ProviderRefreshStatus), reader.GetString(3), true);
+ }
+
+ if (!reader.IsDBNull(4))
+ {
+ result.LastErrorMessage = reader.GetString(4);
+ }
+
+ if (!reader.IsDBNull(5))
+ {
+ result.MetadataProvidersRefreshed = reader.GetString(5).Split('|').Where(i => !string.IsNullOrEmpty(i)).Select(i => new Guid(i)).ToList();
+ }
+
+ if (!reader.IsDBNull(6))
+ {
+ result.ImageProvidersRefreshed = reader.GetString(6).Split('|').Where(i => !string.IsNullOrEmpty(i)).Select(i => new Guid(i)).ToList();
+ }
+
+ return result;
+ }
+
+ public async Task SaveMetadataStatus(MetadataStatus status, CancellationToken cancellationToken)
+ {
+ if (status == null)
+ {
+ throw new ArgumentNullException("status");
+ }
+
+ cancellationToken.ThrowIfCancellationRequested();
+
+ await _writeLock.WaitAsync(cancellationToken).ConfigureAwait(false);
+
+ IDbTransaction transaction = null;
+
+ try
+ {
+ transaction = _connection.BeginTransaction();
+
+ _saveStatusCommand.GetParameter(0).Value = status.ItemId;
+ _saveStatusCommand.GetParameter(1).Value = status.DateLastMetadataRefresh;
+ _saveStatusCommand.GetParameter(2).Value = status.DateLastImagesRefresh;
+ _saveStatusCommand.GetParameter(3).Value = status.LastStatus.ToString();
+ _saveStatusCommand.GetParameter(4).Value = status.LastErrorMessage;
+ _saveStatusCommand.GetParameter(5).Value = string.Join("|", status.MetadataProvidersRefreshed.ToArray());
+ _saveStatusCommand.GetParameter(6).Value = string.Join("|", status.ImageProvidersRefreshed.ToArray());
+
+ _saveStatusCommand.Transaction = transaction;
+
+ _saveStatusCommand.ExecuteNonQuery();
+
+ transaction.Commit();
+ }
+ catch (OperationCanceledException)
+ {
+ if (transaction != null)
+ {
+ transaction.Rollback();
+ }
+
+ throw;
+ }
+ catch (Exception e)
+ {
+ _logger.ErrorException("Failed to save provider info:", e);
+
+ if (transaction != null)
+ {
+ transaction.Rollback();
+ }
+
+ throw;
+ }
+ finally
+ {
+ if (transaction != null)
+ {
+ transaction.Dispose();
+ }
+
+ _writeLock.Release();
+ }
+ }
+
/// <summary>
/// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.
/// </summary>
diff --git a/MediaBrowser.ServerApplication/ApplicationHost.cs b/MediaBrowser.ServerApplication/ApplicationHost.cs
index b78a0f36d..4b15ca8d0 100644
--- a/MediaBrowser.ServerApplication/ApplicationHost.cs
+++ b/MediaBrowser.ServerApplication/ApplicationHost.cs
@@ -14,7 +14,6 @@ using MediaBrowser.Controller.Drawing;
using MediaBrowser.Controller.Dto;
using MediaBrowser.Controller.Entities;
using MediaBrowser.Controller.FileOrganization;
-using MediaBrowser.Controller.IO;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.LiveTv;
using MediaBrowser.Controller.Localization;
@@ -33,6 +32,7 @@ using MediaBrowser.Model.MediaInfo;
using MediaBrowser.Model.System;
using MediaBrowser.Model.Updates;
using MediaBrowser.Providers;
+using MediaBrowser.Providers.Manager;
using MediaBrowser.Server.Implementations;
using MediaBrowser.Server.Implementations.BdInfo;
using MediaBrowser.Server.Implementations.Configuration;
@@ -47,7 +47,6 @@ using MediaBrowser.Server.Implementations.LiveTv;
using MediaBrowser.Server.Implementations.Localization;
using MediaBrowser.Server.Implementations.MediaEncoder;
using MediaBrowser.Server.Implementations.Persistence;
-using MediaBrowser.Server.Implementations.Providers;
using MediaBrowser.Server.Implementations.ServerManager;
using MediaBrowser.Server.Implementations.Session;
using MediaBrowser.Server.Implementations.WebSocket;
@@ -137,7 +136,7 @@ namespace MediaBrowser.ServerApplication
/// Gets or sets the directory watchers.
/// </summary>
/// <value>The directory watchers.</value>
- private IDirectoryWatchers DirectoryWatchers { get; set; }
+ private ILibraryMonitor LibraryMonitor { get; set; }
/// <summary>
/// Gets or sets the provider manager.
/// </summary>
@@ -173,6 +172,7 @@ namespace MediaBrowser.ServerApplication
internal IItemRepository ItemRepository { get; set; }
private INotificationsRepository NotificationsRepository { get; set; }
private IFileOrganizationRepository FileOrganizationRepository { get; set; }
+ private IProviderRepository ProviderRepository { get; set; }
/// <summary>
/// Initializes a new instance of the <see cref="ApplicationHost"/> class.
@@ -267,19 +267,22 @@ namespace MediaBrowser.ServerApplication
ItemRepository = new SqliteItemRepository(ApplicationPaths, JsonSerializer, LogManager);
RegisterSingleInstance(ItemRepository);
+ ProviderRepository = new SqliteProviderInfoRepository(ApplicationPaths, LogManager);
+ RegisterSingleInstance(ProviderRepository);
+
FileOrganizationRepository = await GetFileOrganizationRepository().ConfigureAwait(false);
RegisterSingleInstance(FileOrganizationRepository);
UserManager = new UserManager(Logger, ServerConfigurationManager, UserRepository);
RegisterSingleInstance(UserManager);
- LibraryManager = new LibraryManager(Logger, TaskManager, UserManager, ServerConfigurationManager, UserDataManager, () => DirectoryWatchers, FileSystemManager);
+ LibraryManager = new LibraryManager(Logger, TaskManager, UserManager, ServerConfigurationManager, UserDataManager, () => LibraryMonitor, FileSystemManager);
RegisterSingleInstance(LibraryManager);
- DirectoryWatchers = new DirectoryWatchers(LogManager, TaskManager, LibraryManager, ServerConfigurationManager, FileSystemManager);
- RegisterSingleInstance(DirectoryWatchers);
+ LibraryMonitor = new LibraryMonitor(LogManager, TaskManager, LibraryManager, ServerConfigurationManager, FileSystemManager);
+ RegisterSingleInstance(LibraryMonitor);
- ProviderManager = new ProviderManager(HttpClient, ServerConfigurationManager, DirectoryWatchers, LogManager, FileSystemManager, ItemRepository);
+ ProviderManager = new ProviderManager(HttpClient, ServerConfigurationManager, LibraryMonitor, LogManager, FileSystemManager, ProviderRepository);
RegisterSingleInstance(ProviderManager);
RegisterSingleInstance<ISearchEngine>(() => new SearchEngine(LogManager, LibraryManager, UserManager));
@@ -306,7 +309,7 @@ namespace MediaBrowser.ServerApplication
var newsService = new Server.Implementations.News.NewsService(ApplicationPaths, JsonSerializer);
RegisterSingleInstance<INewsService>(newsService);
- var fileOrganizationService = new FileOrganizationService(TaskManager, FileOrganizationRepository, Logger, DirectoryWatchers, LibraryManager, ServerConfigurationManager, FileSystemManager);
+ var fileOrganizationService = new FileOrganizationService(TaskManager, FileOrganizationRepository, Logger, LibraryMonitor, LibraryManager, ServerConfigurationManager, FileSystemManager);
RegisterSingleInstance<IFileOrganizationService>(fileOrganizationService);
progress.Report(15);
@@ -427,6 +430,8 @@ namespace MediaBrowser.ServerApplication
{
await ItemRepository.Initialize().ConfigureAwait(false);
+ await ProviderRepository.Initialize().ConfigureAwait(false);
+
((LibraryManager)LibraryManager).ItemRepository = ItemRepository;
}
@@ -491,7 +496,7 @@ namespace MediaBrowser.ServerApplication
GetExports<IPeoplePrescanTask>(),
GetExports<IMetadataSaver>());
- ProviderManager.AddParts(GetExports<BaseMetadataProvider>(), GetExports<IImageProvider>());
+ ProviderManager.AddParts(GetExports<BaseMetadataProvider>(), GetExports<IImageProvider>(), GetExports<IMetadataService>(), GetExports<IMetadataProvider>());
ImageProcessor.AddParts(GetExports<IImageEnhancer>());
diff --git a/MediaBrowser.ServerApplication/LibraryExplorer.xaml.cs b/MediaBrowser.ServerApplication/LibraryExplorer.xaml.cs
index 1a5d73e6b..51d661518 100644
--- a/MediaBrowser.ServerApplication/LibraryExplorer.xaml.cs
+++ b/MediaBrowser.ServerApplication/LibraryExplorer.xaml.cs
@@ -5,6 +5,7 @@ using MediaBrowser.Controller.Entities.TV;
using MediaBrowser.Controller.Library;
using MediaBrowser.Controller.Localization;
using MediaBrowser.Controller.Persistence;
+using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Entities;
using MediaBrowser.Model.Logging;
using MediaBrowser.Model.Querying;
@@ -220,14 +221,14 @@ namespace MediaBrowser.ServerApplication
if (item is IHasMediaStreams)
{
var mediaStreams = _itemRepository.GetMediaStreams(new MediaStreamQuery
- {
+ {
ItemId = item.Id
}).ToList();
if (mediaStreams.Count > 0)
{
- json += "\n\nMedia Streams:\n\n"+FormatJson(_jsonSerializer.SerializeToString(mediaStreams));
+ json += "\n\nMedia Streams:\n\n" + FormatJson(_jsonSerializer.SerializeToString(mediaStreams));
}
}
@@ -356,7 +357,7 @@ namespace MediaBrowser.ServerApplication
var item = ((TreeViewItem)tvwLibrary.SelectedItem).Tag as BaseItem;
if (item != null)
{
- item.RefreshMetadata(CancellationToken.None, forceRefresh: cbxForce.IsChecked.Value);
+ item.RefreshMetadata(new MetadataRefreshOptions { ReplaceAllMetadata = cbxForce.IsChecked.Value }, CancellationToken.None);
tvwLibrary_SelectedItemChanged(this, null);
}
}