TvdbClientManager.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Reflection;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. using MediaBrowser.Controller.Providers;
  8. using MediaBrowser.Model.Entities;
  9. using Microsoft.Extensions.Caching.Memory;
  10. using TvDbSharper;
  11. using TvDbSharper.Dto;
  12. namespace MediaBrowser.Providers.Plugins.TheTvdb
  13. {
  14. public class TvdbClientManager
  15. {
  16. private const string DefaultLanguage = "en";
  17. private readonly SemaphoreSlim _cacheWriteLock = new SemaphoreSlim(1, 1);
  18. private readonly IMemoryCache _cache;
  19. private readonly TvDbClient _tvDbClient;
  20. private DateTime _tokenCreatedAt;
  21. public TvdbClientManager(IMemoryCache memoryCache)
  22. {
  23. _cache = memoryCache;
  24. _tvDbClient = new TvDbClient();
  25. }
  26. private TvDbClient TvDbClient
  27. {
  28. get
  29. {
  30. if (string.IsNullOrEmpty(_tvDbClient.Authentication.Token))
  31. {
  32. _tvDbClient.Authentication.AuthenticateAsync(TvdbUtils.TvdbApiKey).GetAwaiter().GetResult();
  33. _tokenCreatedAt = DateTime.Now;
  34. }
  35. // Refresh if necessary
  36. if (_tokenCreatedAt < DateTime.Now.Subtract(TimeSpan.FromHours(20)))
  37. {
  38. try
  39. {
  40. _tvDbClient.Authentication.RefreshTokenAsync().GetAwaiter().GetResult();
  41. }
  42. catch
  43. {
  44. _tvDbClient.Authentication.AuthenticateAsync(TvdbUtils.TvdbApiKey).GetAwaiter().GetResult();
  45. }
  46. _tokenCreatedAt = DateTime.Now;
  47. }
  48. return _tvDbClient;
  49. }
  50. }
  51. public Task<TvDbResponse<SeriesSearchResult[]>> GetSeriesByNameAsync(string name, string language,
  52. CancellationToken cancellationToken)
  53. {
  54. var cacheKey = GenerateKey("series", name, language);
  55. return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByNameAsync(name, cancellationToken));
  56. }
  57. public Task<TvDbResponse<Series>> GetSeriesByIdAsync(int tvdbId, string language,
  58. CancellationToken cancellationToken)
  59. {
  60. var cacheKey = GenerateKey("series", tvdbId, language);
  61. return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetAsync(tvdbId, cancellationToken));
  62. }
  63. public Task<TvDbResponse<EpisodeRecord>> GetEpisodesAsync(int episodeTvdbId, string language,
  64. CancellationToken cancellationToken)
  65. {
  66. var cacheKey = GenerateKey("episode", episodeTvdbId, language);
  67. return TryGetValue(cacheKey, language, () => TvDbClient.Episodes.GetAsync(episodeTvdbId, cancellationToken));
  68. }
  69. public async Task<List<EpisodeRecord>> GetAllEpisodesAsync(int tvdbId, string language,
  70. CancellationToken cancellationToken)
  71. {
  72. // Traverse all episode pages and join them together
  73. var episodes = new List<EpisodeRecord>();
  74. var episodePage = await GetEpisodesPageAsync(tvdbId, new EpisodeQuery(), language, cancellationToken)
  75. .ConfigureAwait(false);
  76. episodes.AddRange(episodePage.Data);
  77. if (!episodePage.Links.Next.HasValue || !episodePage.Links.Last.HasValue)
  78. {
  79. return episodes;
  80. }
  81. int next = episodePage.Links.Next.Value;
  82. int last = episodePage.Links.Last.Value;
  83. for (var page = next; page <= last; ++page)
  84. {
  85. episodePage = await GetEpisodesPageAsync(tvdbId, page, new EpisodeQuery(), language, cancellationToken)
  86. .ConfigureAwait(false);
  87. episodes.AddRange(episodePage.Data);
  88. }
  89. return episodes;
  90. }
  91. public Task<TvDbResponse<SeriesSearchResult[]>> GetSeriesByImdbIdAsync(
  92. string imdbId,
  93. string language,
  94. CancellationToken cancellationToken)
  95. {
  96. var cacheKey = GenerateKey("series", imdbId, language);
  97. return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByImdbIdAsync(imdbId, cancellationToken));
  98. }
  99. public Task<TvDbResponse<SeriesSearchResult[]>> GetSeriesByZap2ItIdAsync(
  100. string zap2ItId,
  101. string language,
  102. CancellationToken cancellationToken)
  103. {
  104. var cacheKey = GenerateKey("series", zap2ItId, language);
  105. return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByZap2ItIdAsync(zap2ItId, cancellationToken));
  106. }
  107. public Task<TvDbResponse<Actor[]>> GetActorsAsync(
  108. int tvdbId,
  109. string language,
  110. CancellationToken cancellationToken)
  111. {
  112. var cacheKey = GenerateKey("actors", tvdbId, language);
  113. return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetActorsAsync(tvdbId, cancellationToken));
  114. }
  115. public Task<TvDbResponse<Image[]>> GetImagesAsync(
  116. int tvdbId,
  117. ImagesQuery imageQuery,
  118. string language,
  119. CancellationToken cancellationToken)
  120. {
  121. var cacheKey = GenerateKey("images", tvdbId, language, imageQuery);
  122. return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetImagesAsync(tvdbId, imageQuery, cancellationToken));
  123. }
  124. public Task<TvDbResponse<Language[]>> GetLanguagesAsync(CancellationToken cancellationToken)
  125. {
  126. return TryGetValue("languages", null, () => TvDbClient.Languages.GetAllAsync(cancellationToken));
  127. }
  128. public Task<TvDbResponse<EpisodesSummary>> GetSeriesEpisodeSummaryAsync(
  129. int tvdbId,
  130. string language,
  131. CancellationToken cancellationToken)
  132. {
  133. var cacheKey = GenerateKey("seriesepisodesummary", tvdbId, language);
  134. return TryGetValue(cacheKey, language,
  135. () => TvDbClient.Series.GetEpisodesSummaryAsync(tvdbId, cancellationToken));
  136. }
  137. public Task<TvDbResponse<EpisodeRecord[]>> GetEpisodesPageAsync(
  138. int tvdbId,
  139. int page,
  140. EpisodeQuery episodeQuery,
  141. string language,
  142. CancellationToken cancellationToken)
  143. {
  144. var cacheKey = GenerateKey(language, tvdbId, episodeQuery);
  145. return TryGetValue(cacheKey, language,
  146. () => TvDbClient.Series.GetEpisodesAsync(tvdbId, page, episodeQuery, cancellationToken));
  147. }
  148. public Task<string> GetEpisodeTvdbId(
  149. EpisodeInfo searchInfo,
  150. string language,
  151. CancellationToken cancellationToken)
  152. {
  153. searchInfo.SeriesProviderIds.TryGetValue(MetadataProviders.Tvdb.ToString(),
  154. out var seriesTvdbId);
  155. var episodeQuery = new EpisodeQuery();
  156. // Prefer SxE over premiere date as it is more robust
  157. if (searchInfo.IndexNumber.HasValue && searchInfo.ParentIndexNumber.HasValue)
  158. {
  159. switch (searchInfo.SeriesDisplayOrder)
  160. {
  161. case "dvd":
  162. episodeQuery.DvdEpisode = searchInfo.IndexNumber.Value;
  163. episodeQuery.DvdSeason = searchInfo.ParentIndexNumber.Value;
  164. break;
  165. case "absolute":
  166. episodeQuery.AbsoluteNumber = searchInfo.IndexNumber.Value;
  167. break;
  168. default:
  169. // aired order
  170. episodeQuery.AiredEpisode = searchInfo.IndexNumber.Value;
  171. episodeQuery.AiredSeason = searchInfo.ParentIndexNumber.Value;
  172. break;
  173. }
  174. }
  175. else if (searchInfo.PremiereDate.HasValue)
  176. {
  177. // tvdb expects yyyy-mm-dd format
  178. episodeQuery.FirstAired = searchInfo.PremiereDate.Value.ToString("yyyy-MM-dd");
  179. }
  180. return GetEpisodeTvdbId(Convert.ToInt32(seriesTvdbId), episodeQuery, language, cancellationToken);
  181. }
  182. public async Task<string> GetEpisodeTvdbId(
  183. int seriesTvdbId,
  184. EpisodeQuery episodeQuery,
  185. string language,
  186. CancellationToken cancellationToken)
  187. {
  188. var episodePage =
  189. await GetEpisodesPageAsync(Convert.ToInt32(seriesTvdbId), episodeQuery, language, cancellationToken)
  190. .ConfigureAwait(false);
  191. return episodePage.Data.FirstOrDefault()?.Id.ToString();
  192. }
  193. public Task<TvDbResponse<EpisodeRecord[]>> GetEpisodesPageAsync(
  194. int tvdbId,
  195. EpisodeQuery episodeQuery,
  196. string language,
  197. CancellationToken cancellationToken)
  198. {
  199. return GetEpisodesPageAsync(tvdbId, 1, episodeQuery, language, cancellationToken);
  200. }
  201. private async Task<T> TryGetValue<T>(string key, string language, Func<Task<T>> resultFactory)
  202. {
  203. if (_cache.TryGetValue(key, out T cachedValue))
  204. {
  205. return cachedValue;
  206. }
  207. await _cacheWriteLock.WaitAsync().ConfigureAwait(false);
  208. try
  209. {
  210. if (_cache.TryGetValue(key, out cachedValue))
  211. {
  212. return cachedValue;
  213. }
  214. _tvDbClient.AcceptedLanguage = TvdbUtils.NormalizeLanguage(language) ?? DefaultLanguage;
  215. var result = await resultFactory.Invoke().ConfigureAwait(false);
  216. _cache.Set(key, result, TimeSpan.FromHours(1));
  217. return result;
  218. }
  219. finally
  220. {
  221. _cacheWriteLock.Release();
  222. }
  223. }
  224. private static string GenerateKey(params object[] objects)
  225. {
  226. var key = string.Empty;
  227. foreach (var obj in objects)
  228. {
  229. var objType = obj.GetType();
  230. if (objType.IsPrimitive || objType == typeof(string))
  231. {
  232. key += obj + ";";
  233. }
  234. else
  235. {
  236. foreach (PropertyInfo propertyInfo in objType.GetProperties())
  237. {
  238. var currentValue = propertyInfo.GetValue(obj, null);
  239. if (currentValue == null)
  240. {
  241. continue;
  242. }
  243. key += propertyInfo.Name + "=" + currentValue + ";";
  244. }
  245. }
  246. }
  247. return key;
  248. }
  249. }
  250. }