TmdbSeriesProvider.cs 15 KB

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