mirror of
https://github.com/jellyfin/jellyfin.git
synced 2025-12-16 05:53:03 +03:00
325 lines
12 KiB
C#
325 lines
12 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Net.Http;
|
|
using System.Text;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using Jellyfin.Data.Enums;
|
|
using Jellyfin.Extensions;
|
|
using MediaBrowser.Common.Net;
|
|
using MediaBrowser.Controller.Entities;
|
|
using MediaBrowser.Controller.Entities.TV;
|
|
using MediaBrowser.Controller.Providers;
|
|
using MediaBrowser.Model.Entities;
|
|
using MediaBrowser.Model.Providers;
|
|
using TMDbLib.Objects.TvShows;
|
|
|
|
namespace MediaBrowser.Providers.Plugins.Tmdb.TV
|
|
{
|
|
/// <summary>
|
|
/// TV episode provider powered by TheMovieDb.
|
|
/// </summary>
|
|
public class TmdbEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>, IHasOrder
|
|
{
|
|
private readonly IHttpClientFactory _httpClientFactory;
|
|
private readonly TmdbClientManager _tmdbClientManager;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="TmdbEpisodeProvider"/> class.
|
|
/// </summary>
|
|
/// <param name="httpClientFactory">The <see cref="IHttpClientFactory"/>.</param>
|
|
/// <param name="tmdbClientManager">The <see cref="TmdbClientManager"/>.</param>
|
|
public TmdbEpisodeProvider(IHttpClientFactory httpClientFactory, TmdbClientManager tmdbClientManager)
|
|
{
|
|
_httpClientFactory = httpClientFactory;
|
|
_tmdbClientManager = tmdbClientManager;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public int Order => 1;
|
|
|
|
/// <inheritdoc />
|
|
public string Name => TmdbUtils.ProviderName;
|
|
|
|
/// <inheritdoc />
|
|
public async Task<IEnumerable<RemoteSearchResult>> GetSearchResults(EpisodeInfo searchInfo, CancellationToken cancellationToken)
|
|
{
|
|
// The search query must either provide an episode number or date
|
|
if (!searchInfo.IndexNumber.HasValue)
|
|
{
|
|
return Enumerable.Empty<RemoteSearchResult>();
|
|
}
|
|
|
|
var metadataResult = await GetMetadata(searchInfo, cancellationToken).ConfigureAwait(false);
|
|
|
|
if (!metadataResult.HasMetadata)
|
|
{
|
|
return Enumerable.Empty<RemoteSearchResult>();
|
|
}
|
|
|
|
var item = metadataResult.Item;
|
|
|
|
return new[]
|
|
{
|
|
new RemoteSearchResult
|
|
{
|
|
IndexNumber = item.IndexNumber,
|
|
Name = item.Name,
|
|
ParentIndexNumber = item.ParentIndexNumber,
|
|
PremiereDate = item.PremiereDate,
|
|
ProductionYear = item.ProductionYear,
|
|
ProviderIds = item.ProviderIds,
|
|
SearchProviderName = Name,
|
|
IndexNumberEnd = item.IndexNumberEnd
|
|
}
|
|
};
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public async Task<MetadataResult<Episode>> GetMetadata(EpisodeInfo info, CancellationToken cancellationToken)
|
|
{
|
|
var metadataResult = new MetadataResult<Episode>();
|
|
var config = Plugin.Instance.Configuration;
|
|
|
|
// Allowing this will dramatically increase scan times
|
|
if (info.IsMissingEpisode)
|
|
{
|
|
return metadataResult;
|
|
}
|
|
|
|
info.SeriesProviderIds.TryGetValue(MetadataProvider.Tmdb.ToString(), out string? tmdbId);
|
|
|
|
var seriesTmdbId = Convert.ToInt32(tmdbId, CultureInfo.InvariantCulture);
|
|
if (seriesTmdbId <= 0)
|
|
{
|
|
return metadataResult;
|
|
}
|
|
|
|
var seasonNumber = info.ParentIndexNumber ?? 1;
|
|
var episodeNumber = info.IndexNumber;
|
|
|
|
if (!episodeNumber.HasValue)
|
|
{
|
|
return metadataResult;
|
|
}
|
|
|
|
TvEpisode? episodeResult = null;
|
|
if (info.IndexNumberEnd.HasValue)
|
|
{
|
|
var startindex = episodeNumber;
|
|
var endindex = info.IndexNumberEnd;
|
|
List<TvEpisode>? result = null;
|
|
for (int? episode = startindex; episode <= endindex; episode++)
|
|
{
|
|
var episodeInfo = await _tmdbClientManager.GetEpisodeAsync(seriesTmdbId, seasonNumber, episode.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage, info.MetadataCountryCode), info.MetadataCountryCode, cancellationToken).ConfigureAwait(false);
|
|
if (episodeInfo is not null)
|
|
{
|
|
(result ??= new List<TvEpisode>()).Add(episodeInfo);
|
|
}
|
|
}
|
|
|
|
if (result is not null)
|
|
{
|
|
// Forces a deep copy of the first TvEpisode, so we don't modify the original because it's cached
|
|
episodeResult = new TvEpisode()
|
|
{
|
|
Name = result[0].Name,
|
|
Overview = result[0].Overview,
|
|
AirDate = result[0].AirDate,
|
|
VoteAverage = result[0].VoteAverage,
|
|
ExternalIds = result[0].ExternalIds,
|
|
Videos = result[0].Videos,
|
|
Credits = result[0].Credits
|
|
};
|
|
|
|
if (result.Count > 1)
|
|
{
|
|
var name = new StringBuilder(episodeResult.Name);
|
|
var overview = new StringBuilder(episodeResult.Overview);
|
|
|
|
for (int i = 1; i < result.Count; i++)
|
|
{
|
|
name.Append(" / ").Append(result[i].Name);
|
|
overview.Append(" / ").Append(result[i].Overview);
|
|
}
|
|
|
|
episodeResult.Name = name.ToString();
|
|
episodeResult.Overview = overview.ToString();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return metadataResult;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
episodeResult = await _tmdbClientManager
|
|
.GetEpisodeAsync(seriesTmdbId, seasonNumber, episodeNumber.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage, info.MetadataCountryCode), info.MetadataCountryCode, cancellationToken)
|
|
.ConfigureAwait(false);
|
|
}
|
|
|
|
if (episodeResult is null)
|
|
{
|
|
return metadataResult;
|
|
}
|
|
|
|
metadataResult.HasMetadata = true;
|
|
metadataResult.QueriedById = true;
|
|
|
|
if (!string.IsNullOrEmpty(episodeResult.Overview))
|
|
{
|
|
// if overview is non-empty, we can assume that localized data was returned
|
|
metadataResult.ResultLanguage = info.MetadataLanguage;
|
|
}
|
|
|
|
var item = new Episode
|
|
{
|
|
IndexNumber = episodeNumber,
|
|
ParentIndexNumber = seasonNumber,
|
|
IndexNumberEnd = info.IndexNumberEnd,
|
|
Name = episodeResult.Name,
|
|
PremiereDate = episodeResult.AirDate,
|
|
ProductionYear = episodeResult.AirDate?.Year,
|
|
Overview = episodeResult.Overview,
|
|
CommunityRating = Convert.ToSingle(episodeResult.VoteAverage)
|
|
};
|
|
|
|
var externalIds = episodeResult.ExternalIds;
|
|
item.TrySetProviderId(MetadataProvider.Tvdb, externalIds?.TvdbId);
|
|
item.TrySetProviderId(MetadataProvider.Imdb, externalIds?.ImdbId);
|
|
item.TrySetProviderId(MetadataProvider.TvRage, externalIds?.TvrageId);
|
|
|
|
if (episodeResult.Videos?.Results is not null)
|
|
{
|
|
foreach (var video in episodeResult.Videos.Results)
|
|
{
|
|
if (TmdbUtils.IsTrailerType(video))
|
|
{
|
|
item.AddTrailerUrl("https://www.youtube.com/watch?v=" + video.Key);
|
|
}
|
|
}
|
|
}
|
|
|
|
var credits = episodeResult.Credits;
|
|
|
|
if (credits?.Cast is not null)
|
|
{
|
|
var castQuery = config.HideMissingCastMembers
|
|
? credits.Cast.Where(a => !string.IsNullOrEmpty(a.ProfilePath)).OrderBy(a => a.Order)
|
|
: credits.Cast.OrderBy(a => a.Order);
|
|
|
|
foreach (var actor in castQuery.Take(config.MaxCastMembers))
|
|
{
|
|
if (string.IsNullOrWhiteSpace(actor.Name))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
var personInfo = new PersonInfo
|
|
{
|
|
Name = actor.Name.Trim(),
|
|
Role = actor.Character?.Trim() ?? string.Empty,
|
|
Type = PersonKind.Actor,
|
|
SortOrder = actor.Order,
|
|
ImageUrl = _tmdbClientManager.GetProfileUrl(actor.ProfilePath)
|
|
};
|
|
|
|
if (actor.Id > 0)
|
|
{
|
|
personInfo.SetProviderId(MetadataProvider.Tmdb, actor.Id.ToString(CultureInfo.InvariantCulture));
|
|
}
|
|
|
|
metadataResult.AddPerson(personInfo);
|
|
}
|
|
}
|
|
|
|
if (credits?.GuestStars is not null)
|
|
{
|
|
var guestQuery = config.HideMissingCastMembers
|
|
? credits.GuestStars.Where(a => !string.IsNullOrEmpty(a.ProfilePath)).OrderBy(a => a.Order)
|
|
: credits.GuestStars.OrderBy(a => a.Order);
|
|
|
|
foreach (var guest in guestQuery.Take(config.MaxCastMembers))
|
|
{
|
|
if (string.IsNullOrWhiteSpace(guest.Name))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
var personInfo = new PersonInfo
|
|
{
|
|
Name = guest.Name.Trim(),
|
|
Role = guest.Character?.Trim() ?? string.Empty,
|
|
Type = PersonKind.GuestStar,
|
|
SortOrder = guest.Order,
|
|
ImageUrl = _tmdbClientManager.GetProfileUrl(guest.ProfilePath)
|
|
};
|
|
|
|
if (guest.Id > 0)
|
|
{
|
|
personInfo.SetProviderId(MetadataProvider.Tmdb, guest.Id.ToString(CultureInfo.InvariantCulture));
|
|
}
|
|
|
|
metadataResult.AddPerson(personInfo);
|
|
}
|
|
}
|
|
|
|
if (credits?.Crew is not null)
|
|
{
|
|
var crewQuery = credits.Crew
|
|
.Select(crewMember => new
|
|
{
|
|
CrewMember = crewMember,
|
|
PersonType = TmdbUtils.MapCrewToPersonType(crewMember)
|
|
})
|
|
.Where(entry =>
|
|
TmdbUtils.WantedCrewKinds.Contains(entry.PersonType) ||
|
|
TmdbUtils.WantedCrewTypes.Contains(entry.CrewMember.Job ?? string.Empty, StringComparison.OrdinalIgnoreCase));
|
|
|
|
if (config.HideMissingCrewMembers)
|
|
{
|
|
crewQuery = crewQuery.Where(entry => !string.IsNullOrEmpty(entry.CrewMember.ProfilePath));
|
|
}
|
|
|
|
foreach (var entry in crewQuery.Take(config.MaxCrewMembers))
|
|
{
|
|
var crewMember = entry.CrewMember;
|
|
|
|
if (string.IsNullOrWhiteSpace(crewMember.Name))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
var personInfo = new PersonInfo
|
|
{
|
|
Name = crewMember.Name.Trim(),
|
|
Role = crewMember.Job?.Trim() ?? string.Empty,
|
|
Type = entry.PersonType,
|
|
ImageUrl = _tmdbClientManager.GetProfileUrl(crewMember.ProfilePath)
|
|
};
|
|
|
|
if (crewMember.Id > 0)
|
|
{
|
|
personInfo.SetProviderId(MetadataProvider.Tmdb, crewMember.Id.ToString(CultureInfo.InvariantCulture));
|
|
}
|
|
|
|
metadataResult.AddPerson(personInfo);
|
|
}
|
|
}
|
|
|
|
metadataResult.Item = item;
|
|
|
|
return metadataResult;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
|
|
{
|
|
return _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(url, cancellationToken);
|
|
}
|
|
}
|
|
}
|