TmdbSeriesProvider.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Net.Http;
  6. using System.Threading;
  7. using System.Threading.Tasks;
  8. using Jellyfin.Data.Enums;
  9. using Jellyfin.Extensions;
  10. using MediaBrowser.Common.Net;
  11. using MediaBrowser.Controller.Entities;
  12. using MediaBrowser.Controller.Entities.TV;
  13. using MediaBrowser.Controller.Library;
  14. using MediaBrowser.Controller.Providers;
  15. using MediaBrowser.Model.Entities;
  16. using MediaBrowser.Model.Providers;
  17. using TMDbLib.Objects.Find;
  18. using TMDbLib.Objects.Search;
  19. using TMDbLib.Objects.TvShows;
  20. namespace MediaBrowser.Providers.Plugins.Tmdb.TV
  21. {
  22. /// <summary>
  23. /// TV series provider powered by TheMovieDb.
  24. /// </summary>
  25. public class TmdbSeriesProvider : IRemoteMetadataProvider<Series, SeriesInfo>, IHasOrder
  26. {
  27. private readonly IHttpClientFactory _httpClientFactory;
  28. private readonly ILibraryManager _libraryManager;
  29. private readonly TmdbClientManager _tmdbClientManager;
  30. /// <summary>
  31. /// Initializes a new instance of the <see cref="TmdbSeriesProvider"/> class.
  32. /// </summary>
  33. /// <param name="libraryManager">The <see cref="ILibraryManager"/>.</param>
  34. /// <param name="httpClientFactory">The <see cref="IHttpClientFactory"/>.</param>
  35. /// <param name="tmdbClientManager">The <see cref="TmdbClientManager"/>.</param>
  36. public TmdbSeriesProvider(
  37. ILibraryManager libraryManager,
  38. IHttpClientFactory httpClientFactory,
  39. TmdbClientManager tmdbClientManager)
  40. {
  41. _libraryManager = libraryManager;
  42. _httpClientFactory = httpClientFactory;
  43. _tmdbClientManager = tmdbClientManager;
  44. }
  45. /// <inheritdoc />
  46. public string Name => TmdbUtils.ProviderName;
  47. /// <inheritdoc />
  48. public int Order => 1;
  49. /// <inheritdoc />
  50. public async Task<IEnumerable<RemoteSearchResult>> GetSearchResults(SeriesInfo searchInfo, CancellationToken cancellationToken)
  51. {
  52. if (searchInfo.TryGetProviderId(MetadataProvider.Tmdb, out var tmdbId))
  53. {
  54. var series = await _tmdbClientManager
  55. .GetSeriesAsync(Convert.ToInt32(tmdbId, CultureInfo.InvariantCulture), searchInfo.MetadataLanguage, searchInfo.MetadataLanguage, cancellationToken)
  56. .ConfigureAwait(false);
  57. if (series is not null)
  58. {
  59. var remoteResult = MapTvShowToRemoteSearchResult(series);
  60. return new[] { remoteResult };
  61. }
  62. }
  63. if (searchInfo.TryGetProviderId(MetadataProvider.Imdb, out var imdbId))
  64. {
  65. var findResult = await _tmdbClientManager
  66. .FindByExternalIdAsync(imdbId, FindExternalSource.Imdb, searchInfo.MetadataLanguage, cancellationToken)
  67. .ConfigureAwait(false);
  68. var tvResults = findResult?.TvResults;
  69. if (tvResults is not null)
  70. {
  71. var imdbIdResults = new RemoteSearchResult[tvResults.Count];
  72. for (var i = 0; i < tvResults.Count; i++)
  73. {
  74. var remoteResult = MapSearchTvToRemoteSearchResult(tvResults[i]);
  75. remoteResult.SetProviderId(MetadataProvider.Imdb, imdbId);
  76. imdbIdResults[i] = remoteResult;
  77. }
  78. return imdbIdResults;
  79. }
  80. }
  81. if (searchInfo.TryGetProviderId(MetadataProvider.Tvdb, out var tvdbId))
  82. {
  83. var findResult = await _tmdbClientManager
  84. .FindByExternalIdAsync(tvdbId, FindExternalSource.TvDb, searchInfo.MetadataLanguage, cancellationToken)
  85. .ConfigureAwait(false);
  86. var tvResults = findResult?.TvResults;
  87. if (tvResults is not null)
  88. {
  89. var tvIdResults = new RemoteSearchResult[tvResults.Count];
  90. for (var i = 0; i < tvResults.Count; i++)
  91. {
  92. var remoteResult = MapSearchTvToRemoteSearchResult(tvResults[i]);
  93. remoteResult.SetProviderId(MetadataProvider.Tvdb, tvdbId);
  94. tvIdResults[i] = remoteResult;
  95. }
  96. return tvIdResults;
  97. }
  98. }
  99. var tvSearchResults = await _tmdbClientManager.SearchSeriesAsync(searchInfo.Name, searchInfo.MetadataLanguage, cancellationToken: cancellationToken)
  100. .ConfigureAwait(false);
  101. var remoteResults = new RemoteSearchResult[tvSearchResults.Count];
  102. for (var i = 0; i < tvSearchResults.Count; i++)
  103. {
  104. remoteResults[i] = MapSearchTvToRemoteSearchResult(tvSearchResults[i]);
  105. }
  106. return remoteResults;
  107. }
  108. private RemoteSearchResult MapTvShowToRemoteSearchResult(TvShow series)
  109. {
  110. var remoteResult = new RemoteSearchResult
  111. {
  112. Name = series.Name ?? series.OriginalName,
  113. SearchProviderName = Name,
  114. ImageUrl = _tmdbClientManager.GetPosterUrl(series.PosterPath),
  115. Overview = series.Overview
  116. };
  117. remoteResult.SetProviderId(MetadataProvider.Tmdb, series.Id.ToString(CultureInfo.InvariantCulture));
  118. if (series.ExternalIds is not null)
  119. {
  120. if (!string.IsNullOrEmpty(series.ExternalIds.ImdbId))
  121. {
  122. remoteResult.SetProviderId(MetadataProvider.Imdb, series.ExternalIds.ImdbId);
  123. }
  124. if (!string.IsNullOrEmpty(series.ExternalIds.TvdbId))
  125. {
  126. remoteResult.SetProviderId(MetadataProvider.Tvdb, series.ExternalIds.TvdbId);
  127. }
  128. }
  129. remoteResult.PremiereDate = series.FirstAirDate?.ToUniversalTime();
  130. return remoteResult;
  131. }
  132. private RemoteSearchResult MapSearchTvToRemoteSearchResult(SearchTv series)
  133. {
  134. var remoteResult = new RemoteSearchResult
  135. {
  136. Name = series.Name ?? series.OriginalName,
  137. SearchProviderName = Name,
  138. ImageUrl = _tmdbClientManager.GetPosterUrl(series.PosterPath),
  139. Overview = series.Overview
  140. };
  141. remoteResult.SetProviderId(MetadataProvider.Tmdb, series.Id.ToString(CultureInfo.InvariantCulture));
  142. remoteResult.PremiereDate = series.FirstAirDate?.ToUniversalTime();
  143. return remoteResult;
  144. }
  145. /// <inheritdoc />
  146. public async Task<MetadataResult<Series>> GetMetadata(SeriesInfo info, CancellationToken cancellationToken)
  147. {
  148. var result = new MetadataResult<Series>
  149. {
  150. QueriedById = true
  151. };
  152. var tmdbId = info.GetProviderId(MetadataProvider.Tmdb);
  153. if (string.IsNullOrEmpty(tmdbId) && info.TryGetProviderId(MetadataProvider.Imdb, out var imdbId))
  154. {
  155. var searchResult = await _tmdbClientManager.FindByExternalIdAsync(imdbId, FindExternalSource.Imdb, info.MetadataLanguage, cancellationToken).ConfigureAwait(false);
  156. if (searchResult?.TvResults.Count > 0)
  157. {
  158. tmdbId = searchResult.TvResults[0].Id.ToString(CultureInfo.InvariantCulture);
  159. }
  160. }
  161. if (string.IsNullOrEmpty(tmdbId) && info.TryGetProviderId(MetadataProvider.Tvdb, out var tvdbId))
  162. {
  163. var searchResult = await _tmdbClientManager.FindByExternalIdAsync(tvdbId, FindExternalSource.TvDb, info.MetadataLanguage, cancellationToken).ConfigureAwait(false);
  164. if (searchResult?.TvResults.Count > 0)
  165. {
  166. tmdbId = searchResult.TvResults[0].Id.ToString(CultureInfo.InvariantCulture);
  167. }
  168. }
  169. if (string.IsNullOrEmpty(tmdbId))
  170. {
  171. result.QueriedById = false;
  172. // ParseName is required here.
  173. // Caller provides the filename with extension stripped and NOT the parsed filename
  174. var parsedName = _libraryManager.ParseName(info.Name);
  175. var cleanedName = TmdbUtils.CleanName(parsedName.Name);
  176. var searchResults = await _tmdbClientManager.SearchSeriesAsync(cleanedName, info.MetadataLanguage, info.Year ?? parsedName.Year ?? 0, cancellationToken).ConfigureAwait(false);
  177. if (searchResults.Count > 0)
  178. {
  179. tmdbId = searchResults[0].Id.ToString(CultureInfo.InvariantCulture);
  180. }
  181. }
  182. if (!int.TryParse(tmdbId, CultureInfo.InvariantCulture, out int tmdbIdInt))
  183. {
  184. return result;
  185. }
  186. cancellationToken.ThrowIfCancellationRequested();
  187. var tvShow = await _tmdbClientManager
  188. .GetSeriesAsync(tmdbIdInt, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage), cancellationToken)
  189. .ConfigureAwait(false);
  190. if (tvShow is null)
  191. {
  192. return result;
  193. }
  194. result = new MetadataResult<Series>
  195. {
  196. Item = MapTvShowToSeries(tvShow, info.MetadataCountryCode),
  197. ResultLanguage = info.MetadataLanguage ?? tvShow.OriginalLanguage
  198. };
  199. foreach (var person in GetPersons(tvShow))
  200. {
  201. result.AddPerson(person);
  202. }
  203. result.HasMetadata = result.Item is not null;
  204. return result;
  205. }
  206. private static Series MapTvShowToSeries(TvShow seriesResult, string preferredCountryCode)
  207. {
  208. var series = new Series
  209. {
  210. Name = seriesResult.Name,
  211. OriginalTitle = seriesResult.OriginalName
  212. };
  213. series.SetProviderId(MetadataProvider.Tmdb, seriesResult.Id.ToString(CultureInfo.InvariantCulture));
  214. series.CommunityRating = Convert.ToSingle(seriesResult.VoteAverage);
  215. series.Overview = seriesResult.Overview;
  216. if (seriesResult.Networks is not null)
  217. {
  218. series.Studios = seriesResult.Networks.Select(i => i.Name).ToArray();
  219. }
  220. if (seriesResult.Genres is not null)
  221. {
  222. series.Genres = seriesResult.Genres.Select(i => i.Name).ToArray();
  223. }
  224. if (seriesResult.Keywords?.Results is not null)
  225. {
  226. for (var i = 0; i < seriesResult.Keywords.Results.Count; i++)
  227. {
  228. series.AddTag(seriesResult.Keywords.Results[i].Name);
  229. }
  230. }
  231. series.HomePageUrl = seriesResult.Homepage;
  232. series.RunTimeTicks = seriesResult.EpisodeRunTime.Select(i => TimeSpan.FromMinutes(i).Ticks).FirstOrDefault();
  233. if (string.Equals(seriesResult.Status, "Ended", StringComparison.OrdinalIgnoreCase)
  234. || string.Equals(seriesResult.Status, "Canceled", StringComparison.OrdinalIgnoreCase))
  235. {
  236. series.Status = SeriesStatus.Ended;
  237. series.EndDate = seriesResult.LastAirDate;
  238. }
  239. else
  240. {
  241. series.Status = SeriesStatus.Continuing;
  242. }
  243. series.PremiereDate = seriesResult.FirstAirDate;
  244. var ids = seriesResult.ExternalIds;
  245. if (ids is not null)
  246. {
  247. if (!string.IsNullOrWhiteSpace(ids.ImdbId))
  248. {
  249. series.SetProviderId(MetadataProvider.Imdb, ids.ImdbId);
  250. }
  251. if (!string.IsNullOrEmpty(ids.TvrageId))
  252. {
  253. series.SetProviderId(MetadataProvider.TvRage, ids.TvrageId);
  254. }
  255. if (!string.IsNullOrEmpty(ids.TvdbId))
  256. {
  257. series.SetProviderId(MetadataProvider.Tvdb, ids.TvdbId);
  258. }
  259. }
  260. var contentRatings = seriesResult.ContentRatings.Results ?? new List<ContentRating>();
  261. var ourRelease = contentRatings.FirstOrDefault(c => string.Equals(c.Iso_3166_1, preferredCountryCode, StringComparison.OrdinalIgnoreCase));
  262. var usRelease = contentRatings.FirstOrDefault(c => string.Equals(c.Iso_3166_1, "US", StringComparison.OrdinalIgnoreCase));
  263. var minimumRelease = contentRatings.FirstOrDefault();
  264. if (ourRelease is not null)
  265. {
  266. series.OfficialRating = TmdbUtils.BuildParentalRating(ourRelease.Iso_3166_1, ourRelease.Rating);
  267. }
  268. else if (usRelease is not null)
  269. {
  270. series.OfficialRating = usRelease.Rating;
  271. }
  272. else if (minimumRelease is not null)
  273. {
  274. series.OfficialRating = minimumRelease.Rating;
  275. }
  276. if (seriesResult.Videos?.Results is not null)
  277. {
  278. foreach (var video in seriesResult.Videos.Results)
  279. {
  280. if (TmdbUtils.IsTrailerType(video))
  281. {
  282. series.AddTrailerUrl("https://www.youtube.com/watch?v=" + video.Key);
  283. }
  284. }
  285. }
  286. return series;
  287. }
  288. private IEnumerable<PersonInfo> GetPersons(TvShow seriesResult)
  289. {
  290. if (seriesResult.Credits?.Cast is not null)
  291. {
  292. foreach (var actor in seriesResult.Credits.Cast.OrderBy(a => a.Order).Take(Plugin.Instance.Configuration.MaxCastMembers))
  293. {
  294. var personInfo = new PersonInfo
  295. {
  296. Name = actor.Name.Trim(),
  297. Role = actor.Character,
  298. Type = PersonKind.Actor,
  299. SortOrder = actor.Order,
  300. ImageUrl = _tmdbClientManager.GetPosterUrl(actor.ProfilePath)
  301. };
  302. if (actor.Id > 0)
  303. {
  304. personInfo.SetProviderId(MetadataProvider.Tmdb, actor.Id.ToString(CultureInfo.InvariantCulture));
  305. }
  306. yield return personInfo;
  307. }
  308. }
  309. if (seriesResult.Credits?.Crew is not null)
  310. {
  311. var keepTypes = new[]
  312. {
  313. PersonType.Director,
  314. PersonType.Writer,
  315. PersonType.Producer
  316. };
  317. foreach (var person in seriesResult.Credits.Crew)
  318. {
  319. // Normalize this
  320. var type = TmdbUtils.MapCrewToPersonType(person);
  321. if (!TmdbUtils.WantedCrewKinds.Contains(type)
  322. && !TmdbUtils.WantedCrewTypes.Contains(person.Job ?? string.Empty, StringComparison.OrdinalIgnoreCase))
  323. {
  324. continue;
  325. }
  326. yield return new PersonInfo
  327. {
  328. Name = person.Name.Trim(),
  329. Role = person.Job,
  330. Type = type
  331. };
  332. }
  333. }
  334. }
  335. /// <inheritdoc />
  336. public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
  337. {
  338. return _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(url, cancellationToken);
  339. }
  340. }
  341. }