#pragma warning disable CS1591 using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Runtime.CompilerServices; using System.Threading; using System.Threading.Tasks; using MediaBrowser.Controller.Providers; using MediaBrowser.Model.Entities; using Microsoft.Extensions.Caching.Memory; using TvDbSharper; using TvDbSharper.Dto; namespace MediaBrowser.Providers.Plugins.TheTvdb { public class TvdbClientManager { private const string DefaultLanguage = "en"; private readonly SemaphoreSlim _cacheWriteLock = new SemaphoreSlim(1, 1); private readonly IMemoryCache _cache; private readonly TvDbClient _tvDbClient; private DateTime _tokenCreatedAt; public TvdbClientManager(IMemoryCache memoryCache) { _cache = memoryCache; _tvDbClient = new TvDbClient(); } private TvDbClient TvDbClient { get { if (string.IsNullOrEmpty(_tvDbClient.Authentication.Token)) { _tvDbClient.Authentication.AuthenticateAsync(TvdbUtils.TvdbApiKey).GetAwaiter().GetResult(); _tokenCreatedAt = DateTime.Now; } // Refresh if necessary if (_tokenCreatedAt < DateTime.Now.Subtract(TimeSpan.FromHours(20))) { try { _tvDbClient.Authentication.RefreshTokenAsync().GetAwaiter().GetResult(); } catch { _tvDbClient.Authentication.AuthenticateAsync(TvdbUtils.TvdbApiKey).GetAwaiter().GetResult(); } _tokenCreatedAt = DateTime.Now; } return _tvDbClient; } } public Task> GetSeriesByNameAsync(string name, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("series", name, language); return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByNameAsync(name, cancellationToken)); } public Task> GetSeriesByIdAsync(int tvdbId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("series", tvdbId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetAsync(tvdbId, cancellationToken)); } public Task> GetEpisodesAsync(int episodeTvdbId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("episode", episodeTvdbId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Episodes.GetAsync(episodeTvdbId, cancellationToken)); } public async Task> GetAllEpisodesAsync(int tvdbId, string language, CancellationToken cancellationToken) { // Traverse all episode pages and join them together var episodes = new List(); var episodePage = await GetEpisodesPageAsync(tvdbId, new EpisodeQuery(), language, cancellationToken) .ConfigureAwait(false); episodes.AddRange(episodePage.Data); if (!episodePage.Links.Next.HasValue || !episodePage.Links.Last.HasValue) { return episodes; } int next = episodePage.Links.Next.Value; int last = episodePage.Links.Last.Value; for (var page = next; page <= last; ++page) { episodePage = await GetEpisodesPageAsync(tvdbId, page, new EpisodeQuery(), language, cancellationToken) .ConfigureAwait(false); episodes.AddRange(episodePage.Data); } return episodes; } public Task> GetSeriesByImdbIdAsync( string imdbId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("series", imdbId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByImdbIdAsync(imdbId, cancellationToken)); } public Task> GetSeriesByZap2ItIdAsync( string zap2ItId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("series", zap2ItId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Search.SearchSeriesByZap2ItIdAsync(zap2ItId, cancellationToken)); } public Task> GetActorsAsync( int tvdbId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("actors", tvdbId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetActorsAsync(tvdbId, cancellationToken)); } public Task> GetImagesAsync( int tvdbId, ImagesQuery imageQuery, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("images", tvdbId, language, imageQuery); return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetImagesAsync(tvdbId, imageQuery, cancellationToken)); } public Task> GetLanguagesAsync(CancellationToken cancellationToken) { return TryGetValue("languages", null, () => TvDbClient.Languages.GetAllAsync(cancellationToken)); } public Task> GetSeriesEpisodeSummaryAsync( int tvdbId, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey("seriesepisodesummary", tvdbId, language); return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetEpisodesSummaryAsync(tvdbId, cancellationToken)); } public Task> GetEpisodesPageAsync( int tvdbId, int page, EpisodeQuery episodeQuery, string language, CancellationToken cancellationToken) { var cacheKey = GenerateKey(language, tvdbId, episodeQuery); return TryGetValue(cacheKey, language, () => TvDbClient.Series.GetEpisodesAsync(tvdbId, page, episodeQuery, cancellationToken)); } public Task GetEpisodeTvdbId( EpisodeInfo searchInfo, string language, CancellationToken cancellationToken) { searchInfo.SeriesProviderIds.TryGetValue(MetadataProvider.Tvdb.ToString(), out var seriesTvdbId); var episodeQuery = new EpisodeQuery(); // Prefer SxE over premiere date as it is more robust if (searchInfo.IndexNumber.HasValue && searchInfo.ParentIndexNumber.HasValue) { switch (searchInfo.SeriesDisplayOrder) { case "dvd": episodeQuery.DvdEpisode = searchInfo.IndexNumber.Value; episodeQuery.DvdSeason = searchInfo.ParentIndexNumber.Value; break; case "absolute": episodeQuery.AbsoluteNumber = searchInfo.IndexNumber.Value; break; default: // aired order episodeQuery.AiredEpisode = searchInfo.IndexNumber.Value; episodeQuery.AiredSeason = searchInfo.ParentIndexNumber.Value; break; } } else if (searchInfo.PremiereDate.HasValue) { // tvdb expects yyyy-mm-dd format episodeQuery.FirstAired = searchInfo.PremiereDate.Value.ToString("yyyy-MM-dd"); } return GetEpisodeTvdbId(Convert.ToInt32(seriesTvdbId), episodeQuery, language, cancellationToken); } public async Task GetEpisodeTvdbId( int seriesTvdbId, EpisodeQuery episodeQuery, string language, CancellationToken cancellationToken) { var episodePage = await GetEpisodesPageAsync(Convert.ToInt32(seriesTvdbId), episodeQuery, language, cancellationToken) .ConfigureAwait(false); return episodePage.Data.FirstOrDefault()?.Id.ToString(); } public Task> GetEpisodesPageAsync( int tvdbId, EpisodeQuery episodeQuery, string language, CancellationToken cancellationToken) { return GetEpisodesPageAsync(tvdbId, 1, episodeQuery, language, cancellationToken); } public async IAsyncEnumerable GetImageKeyTypesForSeriesAsync(int tvdbId, string language, [EnumeratorCancellation] CancellationToken cancellationToken) { var cacheKey = GenerateKey(nameof(TvDbClient.Series.GetImagesSummaryAsync), tvdbId); var imagesSummary = await TryGetValue(cacheKey, language, () => TvDbClient.Series.GetImagesSummaryAsync(tvdbId, cancellationToken)).ConfigureAwait(false); if (imagesSummary.Data.Fanart > 0) { yield return KeyType.Fanart; } if (imagesSummary.Data.Series > 0) { yield return KeyType.Series; } if (imagesSummary.Data.Poster > 0) { yield return KeyType.Poster; } } public async IAsyncEnumerable GetImageKeyTypesForSeasonAsync(int tvdbId, string language, [EnumeratorCancellation] CancellationToken cancellationToken) { var cacheKey = GenerateKey(nameof(TvDbClient.Series.GetImagesSummaryAsync), tvdbId); var imagesSummary = await TryGetValue(cacheKey, language, () => TvDbClient.Series.GetImagesSummaryAsync(tvdbId, cancellationToken)).ConfigureAwait(false); if (imagesSummary.Data.Season > 0) { yield return KeyType.Season; } if (imagesSummary.Data.Fanart > 0) { yield return KeyType.Fanart; } // TODO seasonwide is not supported in TvDbSharper } private async Task TryGetValue(string key, string language, Func> resultFactory) { if (_cache.TryGetValue(key, out T cachedValue)) { return cachedValue; } await _cacheWriteLock.WaitAsync().ConfigureAwait(false); try { if (_cache.TryGetValue(key, out cachedValue)) { return cachedValue; } _tvDbClient.AcceptedLanguage = TvdbUtils.NormalizeLanguage(language) ?? DefaultLanguage; var result = await resultFactory.Invoke().ConfigureAwait(false); _cache.Set(key, result, TimeSpan.FromHours(1)); return result; } finally { _cacheWriteLock.Release(); } } private static string GenerateKey(params object[] objects) { var key = string.Empty; foreach (var obj in objects) { var objType = obj.GetType(); if (objType.IsPrimitive || objType == typeof(string)) { key += obj + ";"; } else { foreach (PropertyInfo propertyInfo in objType.GetProperties()) { var currentValue = propertyInfo.GetValue(obj, null); if (currentValue == null) { continue; } key += propertyInfo.Name + "=" + currentValue + ";"; } } } return key; } } }