TvdbEpisodeProvider.cs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Threading;
  4. using System.Threading.Tasks;
  5. using MediaBrowser.Common.Net;
  6. using MediaBrowser.Controller.Entities;
  7. using MediaBrowser.Controller.Entities.TV;
  8. using MediaBrowser.Controller.Providers;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Providers;
  11. using Microsoft.Extensions.Logging;
  12. using TvDbSharper;
  13. using TvDbSharper.Dto;
  14. namespace MediaBrowser.Providers.TV
  15. {
  16. /// <summary>
  17. /// Class RemoteEpisodeProvider
  18. /// </summary>
  19. class TvdbEpisodeProvider : IRemoteMetadataProvider<Episode, EpisodeInfo>, IHasOrder
  20. {
  21. private readonly IHttpClient _httpClient;
  22. private readonly ILogger _logger;
  23. private readonly TvDbClientManager _tvDbClientManager;
  24. public TvdbEpisodeProvider(IHttpClient httpClient, ILogger logger)
  25. {
  26. _httpClient = httpClient;
  27. _logger = logger;
  28. _tvDbClientManager = TvDbClientManager.Instance;
  29. }
  30. public async Task<IEnumerable<RemoteSearchResult>> GetSearchResults(EpisodeInfo searchInfo, CancellationToken cancellationToken)
  31. {
  32. var list = new List<RemoteSearchResult>();
  33. // The search query must either provide an episode number or date
  34. // TODO premieredate functionality is ded, could grab all episodes and search for it
  35. if (!searchInfo.IndexNumber.HasValue || !searchInfo.PremiereDate.HasValue)
  36. {
  37. return list;
  38. }
  39. if (TvdbSeriesProvider.IsValidSeries(searchInfo.SeriesProviderIds))
  40. {
  41. try
  42. {
  43. var episodeTvdbId = searchInfo.GetProviderId(MetadataProviders.Tvdb);
  44. if (string.IsNullOrEmpty(episodeTvdbId))
  45. {
  46. episodeTvdbId = await _tvDbClientManager.GetEpisodeTvdbId(searchInfo, cancellationToken);
  47. if (string.IsNullOrEmpty(episodeTvdbId))
  48. {
  49. _logger.LogError("Episode {SeasonNumber}x{EpisodeNumber} found for series {SeriesTvdbId}",
  50. searchInfo.ParentIndexNumber, searchInfo.IndexNumber);
  51. return list;
  52. }
  53. }
  54. var episodeResult = await _tvDbClientManager.GetEpisodesAsync(Convert.ToInt32(episodeTvdbId), cancellationToken);
  55. var metadataResult = MapEpisodeToResult(searchInfo, episodeResult.Data);
  56. if (metadataResult.HasMetadata)
  57. {
  58. var item = metadataResult.Item;
  59. list.Add(new RemoteSearchResult
  60. {
  61. IndexNumber = item.IndexNumber,
  62. Name = item.Name,
  63. ParentIndexNumber = item.ParentIndexNumber,
  64. PremiereDate = item.PremiereDate,
  65. ProductionYear = item.ProductionYear,
  66. ProviderIds = item.ProviderIds,
  67. SearchProviderName = Name,
  68. IndexNumberEnd = item.IndexNumberEnd
  69. });
  70. }
  71. }
  72. catch (TvDbServerException e)
  73. {
  74. _logger.LogError(e, "Failed to retrieve episode with id {TvDbId}", searchInfo.IndexNumber);
  75. }
  76. }
  77. return list;
  78. }
  79. public string Name => "TheTVDB";
  80. public async Task<MetadataResult<Episode>> GetMetadata(EpisodeInfo searchInfo, CancellationToken cancellationToken)
  81. {
  82. var result = new MetadataResult<Episode>
  83. {
  84. QueriedById = true
  85. };
  86. if (TvdbSeriesProvider.IsValidSeries(searchInfo.SeriesProviderIds) &&
  87. (searchInfo.IndexNumber.HasValue || searchInfo.PremiereDate.HasValue))
  88. {
  89. var tvdbId = searchInfo.GetProviderId(MetadataProviders.Tvdb);
  90. try
  91. {
  92. if (string.IsNullOrEmpty(tvdbId))
  93. {
  94. tvdbId = await _tvDbClientManager.GetEpisodeTvdbId(searchInfo, cancellationToken);
  95. if (string.IsNullOrEmpty(tvdbId))
  96. {
  97. _logger.LogError("Episode {SeasonNumber}x{EpisodeNumber}found for series {SeriesTvdbId}",
  98. searchInfo.ParentIndexNumber, searchInfo.IndexNumber, tvdbId);
  99. return result;
  100. }
  101. }
  102. var episodeResult = await _tvDbClientManager.GetEpisodesAsync(
  103. Convert.ToInt32(tvdbId),
  104. cancellationToken);
  105. result = MapEpisodeToResult(searchInfo, episodeResult.Data);
  106. }
  107. catch (TvDbServerException e)
  108. {
  109. _logger.LogError(e, "Failed to retrieve episode with id {TvDbId}", tvdbId);
  110. }
  111. }
  112. else
  113. {
  114. _logger.LogDebug("No series identity found for {0}", searchInfo.Name);
  115. }
  116. return result;
  117. }
  118. private static MetadataResult<Episode> MapEpisodeToResult(EpisodeInfo id, EpisodeRecord episode)
  119. {
  120. var result = new MetadataResult<Episode>
  121. {
  122. HasMetadata = true,
  123. Item = new Episode
  124. {
  125. IndexNumber = id.IndexNumber,
  126. ParentIndexNumber = id.ParentIndexNumber,
  127. IndexNumberEnd = id.IndexNumberEnd,
  128. AirsBeforeEpisodeNumber = episode.AirsBeforeEpisode,
  129. AirsAfterSeasonNumber = episode.AirsAfterSeason,
  130. AirsBeforeSeasonNumber = episode.AirsBeforeSeason,
  131. Name = episode.EpisodeName,
  132. Overview = episode.Overview,
  133. CommunityRating = (float?)episode.SiteRating,
  134. }
  135. };
  136. result.ResetPeople();
  137. var item = result.Item;
  138. item.SetProviderId(MetadataProviders.Tvdb, episode.Id.ToString());
  139. item.SetProviderId(MetadataProviders.Imdb, episode.ImdbId);
  140. if (string.Equals(id.SeriesDisplayOrder, "dvd", StringComparison.OrdinalIgnoreCase))
  141. {
  142. item.IndexNumber = Convert.ToInt32(episode.DvdEpisodeNumber ?? episode.AiredEpisodeNumber);
  143. item.ParentIndexNumber = episode.DvdSeason ?? episode.AiredSeason;
  144. }
  145. else if (episode.AiredEpisodeNumber.HasValue)
  146. {
  147. item.IndexNumber = episode.AiredEpisodeNumber;
  148. }
  149. else if (episode.AiredSeason.HasValue)
  150. {
  151. item.ParentIndexNumber = episode.AiredSeason;
  152. }
  153. if (DateTime.TryParse(episode.FirstAired, out var date))
  154. {
  155. date = date.ToUniversalTime();
  156. item.PremiereDate = date;
  157. item.ProductionYear = date.Year;
  158. }
  159. foreach (var director in episode.Directors)
  160. {
  161. result.AddPerson(new PersonInfo
  162. {
  163. Name = director,
  164. Type = PersonType.Director
  165. });
  166. }
  167. foreach (var person in episode.GuestStars)
  168. {
  169. var index = person.IndexOf('(');
  170. string role = null;
  171. var name = person;
  172. if (index != -1)
  173. {
  174. role = person.Substring(index + 1).Trim().TrimEnd(')');
  175. name = person.Substring(0, index).Trim();
  176. }
  177. result.AddPerson(new PersonInfo
  178. {
  179. Type = PersonType.GuestStar,
  180. Name = name,
  181. Role = role
  182. });
  183. }
  184. foreach (var writer in episode.Writers)
  185. {
  186. result.AddPerson(new PersonInfo
  187. {
  188. Name = writer,
  189. Type = PersonType.Writer
  190. });
  191. }
  192. result.ResultLanguage = episode.Language.EpisodeName;
  193. return result;
  194. }
  195. public Task<HttpResponseInfo> GetImageResponse(string url, CancellationToken cancellationToken)
  196. {
  197. return _httpClient.GetResponse(new HttpRequestOptions
  198. {
  199. CancellationToken = cancellationToken,
  200. Url = url
  201. });
  202. }
  203. public int Order => 0;
  204. }
  205. }