TmdbEpisodeProvider.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Net.Http;
  6. using System.Text;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using Jellyfin.Data.Enums;
  10. using Jellyfin.Extensions;
  11. using MediaBrowser.Common.Net;
  12. using MediaBrowser.Controller.Entities;
  13. using MediaBrowser.Controller.Entities.TV;
  14. using MediaBrowser.Controller.Providers;
  15. using MediaBrowser.Model.Entities;
  16. using MediaBrowser.Model.Providers;
  17. using TMDbLib.Objects.TvShows;
  18. namespace MediaBrowser.Providers.Plugins.Tmdb.TV
  19. {
  20. /// <summary>
  21. /// TV episode provider powered by TheMovieDb.
  22. /// </summary>
  23. public class TmdbEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>, IHasOrder
  24. {
  25. private readonly IHttpClientFactory _httpClientFactory;
  26. private readonly TmdbClientManager _tmdbClientManager;
  27. /// <summary>
  28. /// Initializes a new instance of the <see cref="TmdbEpisodeProvider"/> class.
  29. /// </summary>
  30. /// <param name="httpClientFactory">The <see cref="IHttpClientFactory"/>.</param>
  31. /// <param name="tmdbClientManager">The <see cref="TmdbClientManager"/>.</param>
  32. public TmdbEpisodeProvider(IHttpClientFactory httpClientFactory, TmdbClientManager tmdbClientManager)
  33. {
  34. _httpClientFactory = httpClientFactory;
  35. _tmdbClientManager = tmdbClientManager;
  36. }
  37. /// <inheritdoc />
  38. public int Order => 1;
  39. /// <inheritdoc />
  40. public string Name => TmdbUtils.ProviderName;
  41. /// <inheritdoc />
  42. public async Task<IEnumerable<RemoteSearchResult>> GetSearchResults(EpisodeInfo searchInfo, CancellationToken cancellationToken)
  43. {
  44. // The search query must either provide an episode number or date
  45. if (!searchInfo.IndexNumber.HasValue || !searchInfo.ParentIndexNumber.HasValue)
  46. {
  47. return Enumerable.Empty<RemoteSearchResult>();
  48. }
  49. var metadataResult = await GetMetadata(searchInfo, cancellationToken).ConfigureAwait(false);
  50. if (!metadataResult.HasMetadata)
  51. {
  52. return Enumerable.Empty<RemoteSearchResult>();
  53. }
  54. var item = metadataResult.Item;
  55. return new[]
  56. {
  57. new RemoteSearchResult
  58. {
  59. IndexNumber = item.IndexNumber,
  60. Name = item.Name,
  61. ParentIndexNumber = item.ParentIndexNumber,
  62. PremiereDate = item.PremiereDate,
  63. ProductionYear = item.ProductionYear,
  64. ProviderIds = item.ProviderIds,
  65. SearchProviderName = Name,
  66. IndexNumberEnd = item.IndexNumberEnd
  67. }
  68. };
  69. }
  70. /// <inheritdoc />
  71. public async Task<MetadataResult<Episode>> GetMetadata(EpisodeInfo info, CancellationToken cancellationToken)
  72. {
  73. var metadataResult = new MetadataResult<Episode>();
  74. // Allowing this will dramatically increase scan times
  75. if (info.IsMissingEpisode)
  76. {
  77. return metadataResult;
  78. }
  79. info.SeriesProviderIds.TryGetValue(MetadataProvider.Tmdb.ToString(), out string? tmdbId);
  80. var seriesTmdbId = Convert.ToInt32(tmdbId, CultureInfo.InvariantCulture);
  81. if (seriesTmdbId <= 0)
  82. {
  83. return metadataResult;
  84. }
  85. var seasonNumber = info.ParentIndexNumber;
  86. var episodeNumber = info.IndexNumber;
  87. if (!seasonNumber.HasValue || !episodeNumber.HasValue)
  88. {
  89. return metadataResult;
  90. }
  91. TvEpisode? episodeResult = null;
  92. if (info.IndexNumberEnd.HasValue)
  93. {
  94. var startindex = episodeNumber;
  95. var endindex = info.IndexNumberEnd;
  96. List<TvEpisode>? result = null;
  97. for (int? episode = startindex; episode <= endindex; episode++)
  98. {
  99. var episodeInfo = await _tmdbClientManager.GetEpisodeAsync(seriesTmdbId, seasonNumber.Value, episode.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage), cancellationToken).ConfigureAwait(false);
  100. if (episodeInfo is not null)
  101. {
  102. (result ??= new List<TvEpisode>()).Add(episodeInfo);
  103. }
  104. }
  105. if (result is not null)
  106. {
  107. // Forces a deep copy of the first TvEpisode, so we don't modify the original because it's cached
  108. episodeResult = new TvEpisode()
  109. {
  110. Name = result[0].Name,
  111. Overview = result[0].Overview,
  112. AirDate = result[0].AirDate,
  113. VoteAverage = result[0].VoteAverage,
  114. ExternalIds = result[0].ExternalIds,
  115. Videos = result[0].Videos,
  116. Credits = result[0].Credits
  117. };
  118. if (result.Count > 1)
  119. {
  120. var name = new StringBuilder(episodeResult.Name);
  121. var overview = new StringBuilder(episodeResult.Overview);
  122. for (int i = 1; i < result.Count; i++)
  123. {
  124. name.Append(" / ").Append(result[i].Name);
  125. overview.Append(" / ").Append(result[i].Overview);
  126. }
  127. episodeResult.Name = name.ToString();
  128. episodeResult.Overview = overview.ToString();
  129. }
  130. }
  131. else
  132. {
  133. return metadataResult;
  134. }
  135. }
  136. else
  137. {
  138. episodeResult = await _tmdbClientManager
  139. .GetEpisodeAsync(seriesTmdbId, seasonNumber.Value, episodeNumber.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage), cancellationToken)
  140. .ConfigureAwait(false);
  141. }
  142. if (episodeResult is null)
  143. {
  144. return metadataResult;
  145. }
  146. metadataResult.HasMetadata = true;
  147. metadataResult.QueriedById = true;
  148. if (!string.IsNullOrEmpty(episodeResult.Overview))
  149. {
  150. // if overview is non-empty, we can assume that localized data was returned
  151. metadataResult.ResultLanguage = info.MetadataLanguage;
  152. }
  153. var item = new Episode
  154. {
  155. IndexNumber = info.IndexNumber,
  156. ParentIndexNumber = info.ParentIndexNumber,
  157. IndexNumberEnd = info.IndexNumberEnd,
  158. Name = episodeResult.Name,
  159. PremiereDate = episodeResult.AirDate,
  160. ProductionYear = episodeResult.AirDate?.Year,
  161. Overview = episodeResult.Overview,
  162. CommunityRating = Convert.ToSingle(episodeResult.VoteAverage)
  163. };
  164. var externalIds = episodeResult.ExternalIds;
  165. item.TrySetProviderId(MetadataProvider.Tvdb, externalIds?.TvdbId);
  166. item.TrySetProviderId(MetadataProvider.Imdb, externalIds?.ImdbId);
  167. item.TrySetProviderId(MetadataProvider.TvRage, externalIds?.TvrageId);
  168. if (episodeResult.Videos?.Results is not null)
  169. {
  170. foreach (var video in episodeResult.Videos.Results)
  171. {
  172. if (TmdbUtils.IsTrailerType(video))
  173. {
  174. item.AddTrailerUrl("https://www.youtube.com/watch?v=" + video.Key);
  175. }
  176. }
  177. }
  178. var credits = episodeResult.Credits;
  179. if (credits?.Cast is not null)
  180. {
  181. foreach (var actor in credits.Cast.OrderBy(a => a.Order).Take(Plugin.Instance.Configuration.MaxCastMembers))
  182. {
  183. metadataResult.AddPerson(new PersonInfo
  184. {
  185. Name = actor.Name.Trim(),
  186. Role = actor.Character,
  187. Type = PersonKind.Actor,
  188. SortOrder = actor.Order
  189. });
  190. }
  191. }
  192. if (credits?.GuestStars is not null)
  193. {
  194. foreach (var guest in credits.GuestStars.OrderBy(a => a.Order).Take(Plugin.Instance.Configuration.MaxCastMembers))
  195. {
  196. metadataResult.AddPerson(new PersonInfo
  197. {
  198. Name = guest.Name.Trim(),
  199. Role = guest.Character,
  200. Type = PersonKind.GuestStar,
  201. SortOrder = guest.Order
  202. });
  203. }
  204. }
  205. // and the rest from crew
  206. if (credits?.Crew is not null)
  207. {
  208. foreach (var person in credits.Crew)
  209. {
  210. // Normalize this
  211. var type = TmdbUtils.MapCrewToPersonType(person);
  212. if (!TmdbUtils.WantedCrewKinds.Contains(type)
  213. && !TmdbUtils.WantedCrewTypes.Contains(person.Job ?? string.Empty, StringComparison.OrdinalIgnoreCase))
  214. {
  215. continue;
  216. }
  217. metadataResult.AddPerson(new PersonInfo
  218. {
  219. Name = person.Name.Trim(),
  220. Role = person.Job,
  221. Type = type
  222. });
  223. }
  224. }
  225. metadataResult.Item = item;
  226. return metadataResult;
  227. }
  228. /// <inheritdoc />
  229. public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
  230. {
  231. return _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(url, cancellationToken);
  232. }
  233. }
  234. }