mirror of
https://github.com/jellyfin/jellyfin.git
synced 2024-11-15 18:08:53 -07:00
#608 - Add manual image selection for People
This commit is contained in:
parent
36dc3a5318
commit
0f311f48c4
@ -113,6 +113,7 @@
|
|||||||
<Compile Include="TV\ManualFanartSeasonProvider.cs" />
|
<Compile Include="TV\ManualFanartSeasonProvider.cs" />
|
||||||
<Compile Include="TV\ManualFanartSeriesProvider.cs" />
|
<Compile Include="TV\ManualFanartSeriesProvider.cs" />
|
||||||
<Compile Include="TV\ManualTvdbEpisodeImageProvider.cs" />
|
<Compile Include="TV\ManualTvdbEpisodeImageProvider.cs" />
|
||||||
|
<Compile Include="TV\ManualTvdbPersonImageProvider.cs" />
|
||||||
<Compile Include="TV\ManualTvdbSeasonImageProvider.cs" />
|
<Compile Include="TV\ManualTvdbSeasonImageProvider.cs" />
|
||||||
<Compile Include="TV\ManualTvdbSeriesImageProvider.cs" />
|
<Compile Include="TV\ManualTvdbSeriesImageProvider.cs" />
|
||||||
<Compile Include="TV\TvdbEpisodeProvider.cs" />
|
<Compile Include="TV\TvdbEpisodeProvider.cs" />
|
||||||
|
192
MediaBrowser.Providers/TV/ManualTvdbPersonImageProvider.cs
Normal file
192
MediaBrowser.Providers/TV/ManualTvdbPersonImageProvider.cs
Normal file
@ -0,0 +1,192 @@
|
|||||||
|
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 System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Xml;
|
||||||
|
|
||||||
|
namespace MediaBrowser.Providers.TV
|
||||||
|
{
|
||||||
|
public class ManualTvdbPersonImageProvider : IImageProvider
|
||||||
|
{
|
||||||
|
private readonly IServerConfigurationManager _config;
|
||||||
|
private readonly ILibraryManager _library;
|
||||||
|
|
||||||
|
public ManualTvdbPersonImageProvider(IServerConfigurationManager config, ILibraryManager library)
|
||||||
|
{
|
||||||
|
_config = config;
|
||||||
|
_library = library;
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Name
|
||||||
|
{
|
||||||
|
get { return ProviderName; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string ProviderName
|
||||||
|
{
|
||||||
|
get { return "TheTVDB"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool Supports(BaseItem item)
|
||||||
|
{
|
||||||
|
return item is Person;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<IEnumerable<RemoteImageInfo>> GetImages(BaseItem item, ImageType imageType, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var images = await GetAllImages(item, cancellationToken).ConfigureAwait(false);
|
||||||
|
|
||||||
|
return images.Where(i => i.Type == imageType);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task<IEnumerable<RemoteImageInfo>> GetAllImages(BaseItem item, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var seriesWithPerson = _library.RootFolder
|
||||||
|
.RecursiveChildren
|
||||||
|
.OfType<Series>()
|
||||||
|
.Where(i => !string.IsNullOrEmpty(i.GetProviderId(MetadataProviders.Tvdb)) && i.People.Any(p => string.Equals(p.Name, item.Name, StringComparison.OrdinalIgnoreCase)))
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
var infos = seriesWithPerson.Select(i => GetImageFromSeriesData(i, item.Name, cancellationToken))
|
||||||
|
.Where(i => i != null)
|
||||||
|
.Take(1);
|
||||||
|
|
||||||
|
return Task.FromResult(infos);
|
||||||
|
}
|
||||||
|
|
||||||
|
private RemoteImageInfo GetImageFromSeriesData(Series series, string personName, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var tvdbPath = TvdbSeriesProvider.GetSeriesDataPath(_config.ApplicationPaths, series.GetProviderId(MetadataProviders.Tvdb));
|
||||||
|
|
||||||
|
var actorXmlPath = Path.Combine(tvdbPath, "actors.xml");
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return GetImageInfo(actorXmlPath, personName, cancellationToken);
|
||||||
|
}
|
||||||
|
catch (FileNotFoundException)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private RemoteImageInfo GetImageInfo(string xmlFile, string personName, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var settings = new XmlReaderSettings
|
||||||
|
{
|
||||||
|
CheckCharacters = false,
|
||||||
|
IgnoreProcessingInstructions = true,
|
||||||
|
IgnoreComments = true,
|
||||||
|
ValidationType = ValidationType.None
|
||||||
|
};
|
||||||
|
|
||||||
|
using (var streamReader = new StreamReader(xmlFile, Encoding.UTF8))
|
||||||
|
{
|
||||||
|
// Use XmlReader for best performance
|
||||||
|
using (var reader = XmlReader.Create(streamReader, settings))
|
||||||
|
{
|
||||||
|
reader.MoveToContent();
|
||||||
|
|
||||||
|
// Loop through each element
|
||||||
|
while (reader.Read())
|
||||||
|
{
|
||||||
|
cancellationToken.ThrowIfCancellationRequested();
|
||||||
|
|
||||||
|
if (reader.NodeType == XmlNodeType.Element)
|
||||||
|
{
|
||||||
|
switch (reader.Name)
|
||||||
|
{
|
||||||
|
case "Actor":
|
||||||
|
{
|
||||||
|
using (var subtree = reader.ReadSubtree())
|
||||||
|
{
|
||||||
|
var info = FetchImageInfoFromActorNode(personName, subtree);
|
||||||
|
|
||||||
|
if (info != null)
|
||||||
|
{
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
reader.Skip();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fetches the data from actor node.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="personName">Name of the person.</param>
|
||||||
|
/// <param name="reader">The reader.</param>
|
||||||
|
/// <returns>System.String.</returns>
|
||||||
|
private RemoteImageInfo FetchImageInfoFromActorNode(string personName, XmlReader reader)
|
||||||
|
{
|
||||||
|
reader.MoveToContent();
|
||||||
|
|
||||||
|
string name = null;
|
||||||
|
string image = null;
|
||||||
|
|
||||||
|
while (reader.Read())
|
||||||
|
{
|
||||||
|
if (reader.NodeType == XmlNodeType.Element)
|
||||||
|
{
|
||||||
|
switch (reader.Name)
|
||||||
|
{
|
||||||
|
case "Name":
|
||||||
|
{
|
||||||
|
name = (reader.ReadElementContentAsString() ?? string.Empty).Trim();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case "Image":
|
||||||
|
{
|
||||||
|
image = (reader.ReadElementContentAsString() ?? string.Empty).Trim();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
reader.Skip();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(name) && !string.IsNullOrEmpty(image) &&
|
||||||
|
string.Equals(name, personName, StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
return new RemoteImageInfo
|
||||||
|
{
|
||||||
|
Url = TVUtils.BannerUrl + image,
|
||||||
|
Type = ImageType.Primary,
|
||||||
|
ProviderName = Name
|
||||||
|
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Priority
|
||||||
|
{
|
||||||
|
get { return 1; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,29 +1,24 @@
|
|||||||
using MediaBrowser.Controller.Configuration;
|
using MediaBrowser.Controller.Configuration;
|
||||||
using MediaBrowser.Controller.Entities;
|
using MediaBrowser.Controller.Entities;
|
||||||
using MediaBrowser.Controller.Entities.TV;
|
|
||||||
using MediaBrowser.Controller.Library;
|
|
||||||
using MediaBrowser.Controller.Providers;
|
using MediaBrowser.Controller.Providers;
|
||||||
using MediaBrowser.Model.Entities;
|
using MediaBrowser.Model.Entities;
|
||||||
using MediaBrowser.Model.Logging;
|
using MediaBrowser.Model.Logging;
|
||||||
|
using MediaBrowser.Model.Providers;
|
||||||
using System;
|
using System;
|
||||||
using System.IO;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Text;
|
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using System.Xml;
|
|
||||||
|
|
||||||
namespace MediaBrowser.Providers.TV
|
namespace MediaBrowser.Providers.TV
|
||||||
{
|
{
|
||||||
public class TvdbPersonImageProvider : BaseMetadataProvider
|
public class TvdbPersonImageProvider : BaseMetadataProvider
|
||||||
{
|
{
|
||||||
private readonly ILibraryManager _library;
|
|
||||||
private readonly IProviderManager _providerManager;
|
private readonly IProviderManager _providerManager;
|
||||||
|
|
||||||
public TvdbPersonImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, ILibraryManager library, IProviderManager providerManager)
|
public TvdbPersonImageProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager)
|
||||||
: base(logManager, configurationManager)
|
: base(logManager, configurationManager)
|
||||||
{
|
{
|
||||||
_library = library;
|
|
||||||
_providerManager = providerManager;
|
_providerManager = providerManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,158 +62,33 @@ namespace MediaBrowser.Providers.TV
|
|||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(item.PrimaryImagePath))
|
if (string.IsNullOrEmpty(item.PrimaryImagePath))
|
||||||
{
|
{
|
||||||
var seriesWithPerson = _library.RootFolder
|
cancellationToken.ThrowIfCancellationRequested();
|
||||||
.RecursiveChildren
|
|
||||||
.OfType<Series>()
|
|
||||||
.Where(i => !string.IsNullOrEmpty(i.GetProviderId(MetadataProviders.Tvdb)) && i.People.Any(p => string.Equals(p.Name, item.Name, StringComparison.OrdinalIgnoreCase)))
|
|
||||||
.ToList();
|
|
||||||
|
|
||||||
foreach (var series in seriesWithPerson)
|
var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, ManualTvdbPersonImageProvider.ProviderName).ConfigureAwait(false);
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
await DownloadImageFromSeries(item, series, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
catch (FileNotFoundException)
|
|
||||||
{
|
|
||||||
// No biggie
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// break once we have an image
|
await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
|
||||||
if (!string.IsNullOrEmpty(item.PrimaryImagePath))
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
SetLastRefreshed(item, DateTime.UtcNow);
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
SetLastRefreshed(item, DateTime.UtcNow);
|
SetLastRefreshed(item, DateTime.UtcNow);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
|
||||||
/// <summary>
|
|
||||||
/// Downloads the image from series.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="item">The item.</param>
|
|
||||||
/// <param name="series">The series.</param>
|
|
||||||
/// <param name="cancellationToken">The cancellation token.</param>
|
|
||||||
/// <returns>Task.</returns>
|
|
||||||
private async Task DownloadImageFromSeries(BaseItem item, Series series, CancellationToken cancellationToken)
|
|
||||||
{
|
{
|
||||||
var tvdbPath = TvdbSeriesProvider.GetSeriesDataPath(ConfigurationManager.ApplicationPaths, series.GetProviderId(MetadataProviders.Tvdb));
|
if (!item.HasImage(ImageType.Primary))
|
||||||
|
|
||||||
var actorXmlPath = Path.Combine(tvdbPath, "actors.xml");
|
|
||||||
|
|
||||||
var url = FetchImageUrl(item, actorXmlPath, cancellationToken);
|
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(url))
|
|
||||||
{
|
{
|
||||||
url = TVUtils.BannerUrl + url;
|
var image = images.FirstOrDefault(i => i.Type == ImageType.Primary);
|
||||||
|
|
||||||
await _providerManager.SaveImage(item, url, TvdbSeriesProvider.Current.TvDbResourcePool,
|
if (image != null)
|
||||||
ImageType.Primary, null, cancellationToken).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
private string FetchImageUrl(BaseItem item, string actorsXmlPath, CancellationToken cancellationToken)
|
|
||||||
{
|
|
||||||
var settings = new XmlReaderSettings
|
|
||||||
{
|
|
||||||
CheckCharacters = false,
|
|
||||||
IgnoreProcessingInstructions = true,
|
|
||||||
IgnoreComments = true,
|
|
||||||
ValidationType = ValidationType.None
|
|
||||||
};
|
|
||||||
|
|
||||||
using (var streamReader = new StreamReader(actorsXmlPath, Encoding.UTF8))
|
|
||||||
{
|
|
||||||
// Use XmlReader for best performance
|
|
||||||
using (var reader = XmlReader.Create(streamReader, settings))
|
|
||||||
{
|
{
|
||||||
reader.MoveToContent();
|
await _providerManager.SaveImage(item, image.Url, TvdbSeriesProvider.Current.TvDbResourcePool, ImageType.Primary, null, cancellationToken)
|
||||||
|
.ConfigureAwait(false);
|
||||||
// Loop through each element
|
|
||||||
while (reader.Read())
|
|
||||||
{
|
|
||||||
cancellationToken.ThrowIfCancellationRequested();
|
|
||||||
|
|
||||||
if (reader.NodeType == XmlNodeType.Element)
|
|
||||||
{
|
|
||||||
switch (reader.Name)
|
|
||||||
{
|
|
||||||
case "Actor":
|
|
||||||
{
|
|
||||||
using (var subtree = reader.ReadSubtree())
|
|
||||||
{
|
|
||||||
var url = FetchImageUrlFromActorNode(item, subtree);
|
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(url))
|
|
||||||
{
|
|
||||||
return url;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
reader.Skip();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Fetches the data from actor node.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="item">The item.</param>
|
|
||||||
/// <param name="reader">The reader.</param>
|
|
||||||
private string FetchImageUrlFromActorNode(BaseItem item, XmlReader reader)
|
|
||||||
{
|
|
||||||
reader.MoveToContent();
|
|
||||||
|
|
||||||
string name = null;
|
|
||||||
string image = null;
|
|
||||||
|
|
||||||
while (reader.Read())
|
|
||||||
{
|
|
||||||
if (reader.NodeType == XmlNodeType.Element)
|
|
||||||
{
|
|
||||||
switch (reader.Name)
|
|
||||||
{
|
|
||||||
case "Name":
|
|
||||||
{
|
|
||||||
name = (reader.ReadElementContentAsString() ?? string.Empty).Trim();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case "Image":
|
|
||||||
{
|
|
||||||
image = (reader.ReadElementContentAsString() ?? string.Empty).Trim();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
reader.Skip();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(name) && !string.IsNullOrEmpty(image) &&
|
|
||||||
string.Equals(name, item.Name, StringComparison.OrdinalIgnoreCase))
|
|
||||||
{
|
|
||||||
return image;
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public override MetadataProviderPriority Priority
|
public override MetadataProviderPriority Priority
|
||||||
{
|
{
|
||||||
get { return MetadataProviderPriority.Third; }
|
get { return MetadataProviderPriority.Third; }
|
||||||
|
@ -36,7 +36,7 @@ namespace MediaBrowser.Server.Implementations.ScheduledTasks
|
|||||||
{
|
{
|
||||||
new DailyTrigger { TimeOfDay = TimeSpan.FromHours(2) },
|
new DailyTrigger { TimeOfDay = TimeSpan.FromHours(2) },
|
||||||
|
|
||||||
new IntervalTrigger{ Interval = TimeSpan.FromHours(12)}
|
new IntervalTrigger{ Interval = TimeSpan.FromHours(24)}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -40,7 +40,7 @@ namespace MediaBrowser.Server.Implementations.ScheduledTasks
|
|||||||
|
|
||||||
new SystemEventTrigger{ SystemEvent = SystemEvent.WakeFromSleep},
|
new SystemEventTrigger{ SystemEvent = SystemEvent.WakeFromSleep},
|
||||||
|
|
||||||
new IntervalTrigger{ Interval = TimeSpan.FromHours(2)}
|
new IntervalTrigger{ Interval = TimeSpan.FromHours(4)}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user