TmdbSeriesProvider.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  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. remoteResult.TrySetProviderId(MetadataProvider.Imdb, series.ExternalIds.ImdbId);
  121. remoteResult.TrySetProviderId(MetadataProvider.Tvdb, series.ExternalIds.TvdbId);
  122. }
  123. remoteResult.PremiereDate = series.FirstAirDate?.ToUniversalTime();
  124. return remoteResult;
  125. }
  126. private RemoteSearchResult MapSearchTvToRemoteSearchResult(SearchTv series)
  127. {
  128. var remoteResult = new RemoteSearchResult
  129. {
  130. Name = series.Name ?? series.OriginalName,
  131. SearchProviderName = Name,
  132. ImageUrl = _tmdbClientManager.GetPosterUrl(series.PosterPath),
  133. Overview = series.Overview
  134. };
  135. remoteResult.SetProviderId(MetadataProvider.Tmdb, series.Id.ToString(CultureInfo.InvariantCulture));
  136. remoteResult.PremiereDate = series.FirstAirDate?.ToUniversalTime();
  137. return remoteResult;
  138. }
  139. /// <inheritdoc />
  140. public async Task<MetadataResult<Series>> GetMetadata(SeriesInfo info, CancellationToken cancellationToken)
  141. {
  142. var result = new MetadataResult<Series>
  143. {
  144. QueriedById = true
  145. };
  146. var tmdbId = info.GetProviderId(MetadataProvider.Tmdb);
  147. if (string.IsNullOrEmpty(tmdbId) && info.TryGetProviderId(MetadataProvider.Imdb, out var imdbId))
  148. {
  149. var searchResult = await _tmdbClientManager.FindByExternalIdAsync(imdbId, FindExternalSource.Imdb, info.MetadataLanguage, cancellationToken).ConfigureAwait(false);
  150. if (searchResult?.TvResults.Count > 0)
  151. {
  152. tmdbId = searchResult.TvResults[0].Id.ToString(CultureInfo.InvariantCulture);
  153. }
  154. }
  155. if (string.IsNullOrEmpty(tmdbId) && info.TryGetProviderId(MetadataProvider.Tvdb, out var tvdbId))
  156. {
  157. var searchResult = await _tmdbClientManager.FindByExternalIdAsync(tvdbId, FindExternalSource.TvDb, info.MetadataLanguage, cancellationToken).ConfigureAwait(false);
  158. if (searchResult?.TvResults.Count > 0)
  159. {
  160. tmdbId = searchResult.TvResults[0].Id.ToString(CultureInfo.InvariantCulture);
  161. }
  162. }
  163. if (string.IsNullOrEmpty(tmdbId))
  164. {
  165. result.QueriedById = false;
  166. // ParseName is required here.
  167. // Caller provides the filename with extension stripped and NOT the parsed filename
  168. var parsedName = _libraryManager.ParseName(info.Name);
  169. var cleanedName = TmdbUtils.CleanName(parsedName.Name);
  170. var searchResults = await _tmdbClientManager.SearchSeriesAsync(cleanedName, info.MetadataLanguage, info.Year ?? parsedName.Year ?? 0, cancellationToken).ConfigureAwait(false);
  171. if (searchResults.Count > 0)
  172. {
  173. tmdbId = searchResults[0].Id.ToString(CultureInfo.InvariantCulture);
  174. }
  175. }
  176. if (!int.TryParse(tmdbId, CultureInfo.InvariantCulture, out int tmdbIdInt))
  177. {
  178. return result;
  179. }
  180. cancellationToken.ThrowIfCancellationRequested();
  181. var tvShow = await _tmdbClientManager
  182. .GetSeriesAsync(tmdbIdInt, info.MetadataLanguage, TmdbUtils.GetImageLanguagesParam(info.MetadataLanguage), cancellationToken)
  183. .ConfigureAwait(false);
  184. if (tvShow is null)
  185. {
  186. return result;
  187. }
  188. result = new MetadataResult<Series>
  189. {
  190. Item = MapTvShowToSeries(tvShow, info.MetadataCountryCode),
  191. ResultLanguage = info.MetadataLanguage ?? tvShow.OriginalLanguage
  192. };
  193. foreach (var person in GetPersons(tvShow))
  194. {
  195. result.AddPerson(person);
  196. }
  197. result.HasMetadata = result.Item is not null;
  198. return result;
  199. }
  200. private static Series MapTvShowToSeries(TvShow seriesResult, string preferredCountryCode)
  201. {
  202. var series = new Series
  203. {
  204. Name = seriesResult.Name,
  205. OriginalTitle = seriesResult.OriginalName
  206. };
  207. series.SetProviderId(MetadataProvider.Tmdb, seriesResult.Id.ToString(CultureInfo.InvariantCulture));
  208. series.CommunityRating = Convert.ToSingle(seriesResult.VoteAverage);
  209. series.Overview = seriesResult.Overview;
  210. if (seriesResult.Networks is not null)
  211. {
  212. series.Studios = seriesResult.Networks.Select(i => i.Name).ToArray();
  213. }
  214. if (seriesResult.Genres is not null)
  215. {
  216. series.Genres = seriesResult.Genres.Select(i => i.Name).ToArray();
  217. }
  218. if (seriesResult.Keywords?.Results is not null)
  219. {
  220. for (var i = 0; i < seriesResult.Keywords.Results.Count; i++)
  221. {
  222. series.AddTag(seriesResult.Keywords.Results[i].Name);
  223. }
  224. }
  225. series.HomePageUrl = seriesResult.Homepage;
  226. series.RunTimeTicks = seriesResult.EpisodeRunTime.Select(i => TimeSpan.FromMinutes(i).Ticks).FirstOrDefault();
  227. if (Emby.Naming.TV.TvParserHelpers.TryParseSeriesStatus(seriesResult.Status, out var seriesStatus))
  228. {
  229. series.Status = seriesStatus;
  230. }
  231. series.EndDate = seriesResult.LastAirDate;
  232. series.PremiereDate = seriesResult.FirstAirDate;
  233. var ids = seriesResult.ExternalIds;
  234. if (ids is not null)
  235. {
  236. series.TrySetProviderId(MetadataProvider.Imdb, ids.ImdbId);
  237. series.TrySetProviderId(MetadataProvider.TvRage, ids.TvrageId);
  238. series.TrySetProviderId(MetadataProvider.Tvdb, ids.TvdbId);
  239. }
  240. var contentRatings = seriesResult.ContentRatings.Results ?? new List<ContentRating>();
  241. var ourRelease = contentRatings.FirstOrDefault(c => string.Equals(c.Iso_3166_1, preferredCountryCode, StringComparison.OrdinalIgnoreCase));
  242. var usRelease = contentRatings.FirstOrDefault(c => string.Equals(c.Iso_3166_1, "US", StringComparison.OrdinalIgnoreCase));
  243. var minimumRelease = contentRatings.FirstOrDefault();
  244. if (ourRelease is not null)
  245. {
  246. series.OfficialRating = TmdbUtils.BuildParentalRating(ourRelease.Iso_3166_1, ourRelease.Rating);
  247. }
  248. else if (usRelease is not null)
  249. {
  250. series.OfficialRating = usRelease.Rating;
  251. }
  252. else if (minimumRelease is not null)
  253. {
  254. series.OfficialRating = minimumRelease.Rating;
  255. }
  256. if (seriesResult.Videos?.Results is not null)
  257. {
  258. foreach (var video in seriesResult.Videos.Results)
  259. {
  260. if (TmdbUtils.IsTrailerType(video))
  261. {
  262. series.AddTrailerUrl("https://www.youtube.com/watch?v=" + video.Key);
  263. }
  264. }
  265. }
  266. return series;
  267. }
  268. private IEnumerable<PersonInfo> GetPersons(TvShow seriesResult)
  269. {
  270. if (seriesResult.Credits?.Cast is not null)
  271. {
  272. foreach (var actor in seriesResult.Credits.Cast.OrderBy(a => a.Order).Take(Plugin.Instance.Configuration.MaxCastMembers))
  273. {
  274. var personInfo = new PersonInfo
  275. {
  276. Name = actor.Name.Trim(),
  277. Role = actor.Character,
  278. Type = PersonKind.Actor,
  279. SortOrder = actor.Order,
  280. ImageUrl = _tmdbClientManager.GetPosterUrl(actor.ProfilePath)
  281. };
  282. if (actor.Id > 0)
  283. {
  284. personInfo.SetProviderId(MetadataProvider.Tmdb, actor.Id.ToString(CultureInfo.InvariantCulture));
  285. }
  286. yield return personInfo;
  287. }
  288. }
  289. if (seriesResult.Credits?.Crew is not null)
  290. {
  291. var keepTypes = new[]
  292. {
  293. PersonType.Director,
  294. PersonType.Writer,
  295. PersonType.Producer
  296. };
  297. foreach (var person in seriesResult.Credits.Crew)
  298. {
  299. // Normalize this
  300. var type = TmdbUtils.MapCrewToPersonType(person);
  301. if (!TmdbUtils.WantedCrewKinds.Contains(type)
  302. && !TmdbUtils.WantedCrewTypes.Contains(person.Job ?? string.Empty, StringComparison.OrdinalIgnoreCase))
  303. {
  304. continue;
  305. }
  306. yield return new PersonInfo
  307. {
  308. Name = person.Name.Trim(),
  309. Role = person.Job,
  310. Type = type
  311. };
  312. }
  313. }
  314. }
  315. /// <inheritdoc />
  316. public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
  317. {
  318. return _httpClientFactory.CreateClient(NamedClient.Default).GetAsync(url, cancellationToken);
  319. }
  320. }
  321. }