TmdbEpisodeProvider.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  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)
  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. var config = Plugin.Instance.Configuration;
  75. // Allowing this will dramatically increase scan times
  76. if (info.IsMissingEpisode)
  77. {
  78. return metadataResult;
  79. }
  80. info.SeriesProviderIds.TryGetValue(MetadataProvider.Tmdb.ToString(), out string? tmdbId);
  81. var seriesTmdbId = Convert.ToInt32(tmdbId, CultureInfo.InvariantCulture);
  82. if (seriesTmdbId <= 0)
  83. {
  84. return metadataResult;
  85. }
  86. var seasonNumber = info.ParentIndexNumber ?? 1;
  87. var episodeNumber = info.IndexNumber;
  88. if (!episodeNumber.HasValue)
  89. {
  90. return metadataResult;
  91. }
  92. TvEpisode? episodeResult = null;
  93. if (info.IndexNumberEnd.HasValue)
  94. {
  95. var startindex = episodeNumber;
  96. var endindex = info.IndexNumberEnd;
  97. List<TvEpisode>? result = null;
  98. for (int? episode = startindex; episode <= endindex; episode++)
  99. {
  100. var episodeInfo = await _tmdbClientManager.GetEpisodeAsync(seriesTmdbId, seasonNumber, episode.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage, info.MetadataCountryCode), info.MetadataCountryCode, cancellationToken).ConfigureAwait(false);
  101. if (episodeInfo is not null)
  102. {
  103. (result ??= new List<TvEpisode>()).Add(episodeInfo);
  104. }
  105. }
  106. if (result is not null)
  107. {
  108. // Forces a deep copy of the first TvEpisode, so we don't modify the original because it's cached
  109. episodeResult = new TvEpisode()
  110. {
  111. Name = result[0].Name,
  112. Overview = result[0].Overview,
  113. AirDate = result[0].AirDate,
  114. VoteAverage = result[0].VoteAverage,
  115. ExternalIds = result[0].ExternalIds,
  116. Videos = result[0].Videos,
  117. Credits = result[0].Credits
  118. };
  119. if (result.Count > 1)
  120. {
  121. var name = new StringBuilder(episodeResult.Name);
  122. var overview = new StringBuilder(episodeResult.Overview);
  123. for (int i = 1; i < result.Count; i++)
  124. {
  125. name.Append(" / ").Append(result[i].Name);
  126. overview.Append(" / ").Append(result[i].Overview);
  127. }
  128. episodeResult.Name = name.ToString();
  129. episodeResult.Overview = overview.ToString();
  130. }
  131. }
  132. else
  133. {
  134. return metadataResult;
  135. }
  136. }
  137. else
  138. {
  139. episodeResult = await _tmdbClientManager
  140. .GetEpisodeAsync(seriesTmdbId, seasonNumber, episodeNumber.Value, info.SeriesDisplayOrder, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage, info.MetadataCountryCode), info.MetadataCountryCode, cancellationToken)
  141. .ConfigureAwait(false);
  142. }
  143. if (episodeResult is null)
  144. {
  145. return metadataResult;
  146. }
  147. metadataResult.HasMetadata = true;
  148. metadataResult.QueriedById = true;
  149. if (!string.IsNullOrEmpty(episodeResult.Overview))
  150. {
  151. // if overview is non-empty, we can assume that localized data was returned
  152. metadataResult.ResultLanguage = info.MetadataLanguage;
  153. }
  154. var item = new Episode
  155. {
  156. IndexNumber = episodeNumber,
  157. ParentIndexNumber = seasonNumber,
  158. IndexNumberEnd = info.IndexNumberEnd,
  159. Name = episodeResult.Name,
  160. PremiereDate = episodeResult.AirDate,
  161. ProductionYear = episodeResult.AirDate?.Year,
  162. Overview = episodeResult.Overview,
  163. CommunityRating = Convert.ToSingle(episodeResult.VoteAverage)
  164. };
  165. var externalIds = episodeResult.ExternalIds;
  166. item.TrySetProviderId(MetadataProvider.Tvdb, externalIds?.TvdbId);
  167. item.TrySetProviderId(MetadataProvider.Imdb, externalIds?.ImdbId);
  168. item.TrySetProviderId(MetadataProvider.TvRage, externalIds?.TvrageId);
  169. if (episodeResult.Videos?.Results is not null)
  170. {
  171. foreach (var video in episodeResult.Videos.Results)
  172. {
  173. if (TmdbUtils.IsTrailerType(video))
  174. {
  175. item.AddTrailerUrl("https://www.youtube.com/watch?v=" + video.Key);
  176. }
  177. }
  178. }
  179. var credits = episodeResult.Credits;
  180. if (credits?.Cast is not null)
  181. {
  182. var castQuery = config.HideMissingCastMembers
  183. ? credits.Cast.Where(a => !string.IsNullOrEmpty(a.ProfilePath)).OrderBy(a => a.Order)
  184. : credits.Cast.OrderBy(a => a.Order);
  185. foreach (var actor in castQuery.Take(config.MaxCastMembers))
  186. {
  187. if (string.IsNullOrWhiteSpace(actor.Name))
  188. {
  189. continue;
  190. }
  191. var personInfo = new PersonInfo
  192. {
  193. Name = actor.Name.Trim(),
  194. Role = actor.Character?.Trim() ?? string.Empty,
  195. Type = PersonKind.Actor,
  196. SortOrder = actor.Order,
  197. ImageUrl = _tmdbClientManager.GetProfileUrl(actor.ProfilePath)
  198. };
  199. if (actor.Id > 0)
  200. {
  201. personInfo.SetProviderId(MetadataProvider.Tmdb, actor.Id.ToString(CultureInfo.InvariantCulture));
  202. }
  203. metadataResult.AddPerson(personInfo);
  204. }
  205. }
  206. if (credits?.GuestStars is not null)
  207. {
  208. var guestQuery = config.HideMissingCastMembers
  209. ? credits.GuestStars.Where(a => !string.IsNullOrEmpty(a.ProfilePath)).OrderBy(a => a.Order)
  210. : credits.GuestStars.OrderBy(a => a.Order);
  211. foreach (var guest in guestQuery.Take(config.MaxCastMembers))
  212. {
  213. if (string.IsNullOrWhiteSpace(guest.Name))
  214. {
  215. continue;
  216. }
  217. var personInfo = new PersonInfo
  218. {
  219. Name = guest.Name.Trim(),
  220. Role = guest.Character?.Trim() ?? string.Empty,
  221. Type = PersonKind.GuestStar,
  222. SortOrder = guest.Order,
  223. ImageUrl = _tmdbClientManager.GetProfileUrl(guest.ProfilePath)
  224. };
  225. if (guest.Id > 0)
  226. {
  227. personInfo.SetProviderId(MetadataProvider.Tmdb, guest.Id.ToString(CultureInfo.InvariantCulture));
  228. }
  229. metadataResult.AddPerson(personInfo);
  230. }
  231. }
  232. if (credits?.Crew is not null)
  233. {
  234. var crewQuery = credits.Crew
  235. .Select(crewMember => new
  236. {
  237. CrewMember = crewMember,
  238. PersonType = TmdbUtils.MapCrewToPersonType(crewMember)
  239. })
  240. .Where(entry =>
  241. TmdbUtils.WantedCrewKinds.Contains(entry.PersonType) ||
  242. TmdbUtils.WantedCrewTypes.Contains(entry.CrewMember.Job ?? string.Empty, StringComparison.OrdinalIgnoreCase));
  243. if (config.HideMissingCrewMembers)
  244. {
  245. crewQuery = crewQuery.Where(entry => !string.IsNullOrEmpty(entry.CrewMember.ProfilePath));
  246. }
  247. foreach (var entry in crewQuery.Take(config.MaxCrewMembers))
  248. {
  249. var crewMember = entry.CrewMember;
  250. if (string.IsNullOrWhiteSpace(crewMember.Name))
  251. {
  252. continue;
  253. }
  254. var personInfo = new PersonInfo
  255. {
  256. Name = crewMember.Name.Trim(),
  257. Role = crewMember.Job?.Trim() ?? string.Empty,
  258. Type = entry.PersonType,
  259. ImageUrl = _tmdbClientManager.GetProfileUrl(crewMember.ProfilePath)
  260. };
  261. if (crewMember.Id > 0)
  262. {
  263. personInfo.SetProviderId(MetadataProvider.Tmdb, crewMember.Id.ToString(CultureInfo.InvariantCulture));
  264. }
  265. metadataResult.AddPerson(personInfo);
  266. }
  267. }
  268. metadataResult.Item = item;
  269. return metadataResult;
  270. }
  271. /// <inheritdoc />
  272. public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
  273. {
  274. return _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(url, cancellationToken);
  275. }
  276. }
  277. }