FanArtTVProvider.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. using MediaBrowser.Common.Configuration;
  2. using MediaBrowser.Common.IO;
  3. using MediaBrowser.Common.Net;
  4. using MediaBrowser.Controller.Configuration;
  5. using MediaBrowser.Controller.Entities;
  6. using MediaBrowser.Controller.Entities.TV;
  7. using MediaBrowser.Controller.Library;
  8. using MediaBrowser.Controller.Providers;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Logging;
  11. using MediaBrowser.Model.Providers;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Globalization;
  15. using System.IO;
  16. using System.Linq;
  17. using System.Threading;
  18. using System.Threading.Tasks;
  19. namespace MediaBrowser.Providers.TV
  20. {
  21. class FanArtTvProvider : FanartBaseProvider
  22. {
  23. protected string FanArtBaseUrl = "http://api.fanart.tv/webservice/series/{0}/{1}/xml/all/1/1";
  24. internal static FanArtTvProvider Current { get; private set; }
  25. /// <summary>
  26. /// Gets the HTTP client.
  27. /// </summary>
  28. /// <value>The HTTP client.</value>
  29. protected IHttpClient HttpClient { get; private set; }
  30. private readonly IProviderManager _providerManager;
  31. private readonly IFileSystem _fileSystem;
  32. public FanArtTvProvider(IHttpClient httpClient, ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager, IFileSystem fileSystem)
  33. : base(logManager, configurationManager)
  34. {
  35. if (httpClient == null)
  36. {
  37. throw new ArgumentNullException("httpClient");
  38. }
  39. HttpClient = httpClient;
  40. _providerManager = providerManager;
  41. _fileSystem = fileSystem;
  42. Current = this;
  43. }
  44. public override bool Supports(BaseItem item)
  45. {
  46. return item is Series;
  47. }
  48. /// <summary>
  49. /// Gets the priority.
  50. /// </summary>
  51. /// <value>The priority.</value>
  52. public override MetadataProviderPriority Priority
  53. {
  54. get { return MetadataProviderPriority.Third; }
  55. }
  56. public override ItemUpdateType ItemUpdateType
  57. {
  58. get
  59. {
  60. return ItemUpdateType.ImageUpdate;
  61. }
  62. }
  63. /// <summary>
  64. /// Needses the refresh internal.
  65. /// </summary>
  66. /// <param name="item">The item.</param>
  67. /// <param name="providerInfo">The provider info.</param>
  68. /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
  69. protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
  70. {
  71. if (string.IsNullOrEmpty(item.GetProviderId(MetadataProviders.Tvdb)))
  72. {
  73. return false;
  74. }
  75. return base.NeedsRefreshInternal(item, providerInfo);
  76. }
  77. protected override bool NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
  78. {
  79. var id = item.GetProviderId(MetadataProviders.Tvdb);
  80. if (!string.IsNullOrEmpty(id))
  81. {
  82. // Process images
  83. var xmlPath = GetFanartXmlPath(id);
  84. var fileInfo = new FileInfo(xmlPath);
  85. return !fileInfo.Exists || _fileSystem.GetLastWriteTimeUtc(fileInfo) > providerInfo.LastRefreshed;
  86. }
  87. return base.NeedsRefreshBasedOnCompareDate(item, providerInfo);
  88. }
  89. /// <summary>
  90. /// Gets a value indicating whether [refresh on version change].
  91. /// </summary>
  92. /// <value><c>true</c> if [refresh on version change]; otherwise, <c>false</c>.</value>
  93. protected override bool RefreshOnVersionChange
  94. {
  95. get
  96. {
  97. return true;
  98. }
  99. }
  100. /// <summary>
  101. /// Gets the provider version.
  102. /// </summary>
  103. /// <value>The provider version.</value>
  104. protected override string ProviderVersion
  105. {
  106. get
  107. {
  108. return "1";
  109. }
  110. }
  111. /// <summary>
  112. /// Gets the series data path.
  113. /// </summary>
  114. /// <param name="appPaths">The app paths.</param>
  115. /// <param name="seriesId">The series id.</param>
  116. /// <returns>System.String.</returns>
  117. internal static string GetSeriesDataPath(IApplicationPaths appPaths, string seriesId)
  118. {
  119. var seriesDataPath = Path.Combine(GetSeriesDataPath(appPaths), seriesId);
  120. return seriesDataPath;
  121. }
  122. /// <summary>
  123. /// Gets the series data path.
  124. /// </summary>
  125. /// <param name="appPaths">The app paths.</param>
  126. /// <returns>System.String.</returns>
  127. internal static string GetSeriesDataPath(IApplicationPaths appPaths)
  128. {
  129. var dataPath = Path.Combine(appPaths.DataPath, "fanart-tv");
  130. return dataPath;
  131. }
  132. public string GetFanartXmlPath(string tvdbId)
  133. {
  134. var dataPath = GetSeriesDataPath(ConfigurationManager.ApplicationPaths, tvdbId);
  135. return Path.Combine(dataPath, "fanart.xml");
  136. }
  137. protected readonly CultureInfo UsCulture = new CultureInfo("en-US");
  138. public override async Task<bool> FetchAsync(BaseItem item, bool force, CancellationToken cancellationToken)
  139. {
  140. cancellationToken.ThrowIfCancellationRequested();
  141. var seriesId = item.GetProviderId(MetadataProviders.Tvdb);
  142. if (!string.IsNullOrEmpty(seriesId))
  143. {
  144. var xmlPath = GetFanartXmlPath(seriesId);
  145. // Only download the xml if it doesn't already exist. The prescan task will take care of getting updates
  146. if (!File.Exists(xmlPath))
  147. {
  148. await DownloadSeriesXml(seriesId, cancellationToken).ConfigureAwait(false);
  149. }
  150. var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, ManualFanartSeriesImageProvider.ProviderName).ConfigureAwait(false);
  151. await FetchFromXml(item, images.ToList(), cancellationToken).ConfigureAwait(false);
  152. }
  153. SetLastRefreshed(item, DateTime.UtcNow);
  154. return true;
  155. }
  156. /// <summary>
  157. /// Fetches from XML.
  158. /// </summary>
  159. /// <param name="item">The item.</param>
  160. /// <param name="images">The images.</param>
  161. /// <param name="cancellationToken">The cancellation token.</param>
  162. /// <returns>Task.</returns>
  163. private async Task FetchFromXml(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
  164. {
  165. cancellationToken.ThrowIfCancellationRequested();
  166. if (ConfigurationManager.Configuration.DownloadSeriesImages.Primary && !item.HasImage(ImageType.Primary))
  167. {
  168. var image = images.FirstOrDefault(i => i.Type == ImageType.Primary);
  169. if (image != null)
  170. {
  171. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Primary, null, cancellationToken).ConfigureAwait(false);
  172. }
  173. }
  174. cancellationToken.ThrowIfCancellationRequested();
  175. if (ConfigurationManager.Configuration.DownloadSeriesImages.Logo && !item.HasImage(ImageType.Logo))
  176. {
  177. var image = images.FirstOrDefault(i => i.Type == ImageType.Logo);
  178. if (image != null)
  179. {
  180. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Logo, null, cancellationToken).ConfigureAwait(false);
  181. }
  182. }
  183. cancellationToken.ThrowIfCancellationRequested();
  184. if (ConfigurationManager.Configuration.DownloadSeriesImages.Art && !item.HasImage(ImageType.Art))
  185. {
  186. var image = images.FirstOrDefault(i => i.Type == ImageType.Art);
  187. if (image != null)
  188. {
  189. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Art, null, cancellationToken).ConfigureAwait(false);
  190. }
  191. }
  192. cancellationToken.ThrowIfCancellationRequested();
  193. if (ConfigurationManager.Configuration.DownloadSeriesImages.Thumb && !item.HasImage(ImageType.Thumb))
  194. {
  195. var image = images.FirstOrDefault(i => i.Type == ImageType.Thumb);
  196. if (image != null)
  197. {
  198. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Thumb, null, cancellationToken).ConfigureAwait(false);
  199. }
  200. }
  201. cancellationToken.ThrowIfCancellationRequested();
  202. if (ConfigurationManager.Configuration.DownloadSeriesImages.Banner && !item.HasImage(ImageType.Banner))
  203. {
  204. var image = images.FirstOrDefault(i => i.Type == ImageType.Banner);
  205. if (image != null)
  206. {
  207. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Banner, null, cancellationToken).ConfigureAwait(false);
  208. }
  209. }
  210. cancellationToken.ThrowIfCancellationRequested();
  211. var backdropLimit = ConfigurationManager.Configuration.MaxBackdrops;
  212. if (ConfigurationManager.Configuration.DownloadSeriesImages.Backdrops &&
  213. item.BackdropImagePaths.Count < backdropLimit)
  214. {
  215. var numBackdrops = item.BackdropImagePaths.Count;
  216. foreach (var image in images.Where(i => i.Type == ImageType.Backdrop))
  217. {
  218. await _providerManager.SaveImage(item, image.Url, FanArtResourcePool, ImageType.Backdrop, numBackdrops, cancellationToken)
  219. .ConfigureAwait(false);
  220. numBackdrops++;
  221. if (item.BackdropImagePaths.Count >= backdropLimit) break;
  222. }
  223. }
  224. }
  225. /// <summary>
  226. /// Downloads the series XML.
  227. /// </summary>
  228. /// <param name="tvdbId">The TVDB id.</param>
  229. /// <param name="cancellationToken">The cancellation token.</param>
  230. /// <returns>Task.</returns>
  231. internal async Task DownloadSeriesXml(string tvdbId, CancellationToken cancellationToken)
  232. {
  233. cancellationToken.ThrowIfCancellationRequested();
  234. var url = string.Format(FanArtBaseUrl, ApiKey, tvdbId);
  235. var xmlPath = GetFanartXmlPath(tvdbId);
  236. Directory.CreateDirectory(Path.GetDirectoryName(xmlPath));
  237. using (var response = await HttpClient.Get(new HttpRequestOptions
  238. {
  239. Url = url,
  240. ResourcePool = FanArtResourcePool,
  241. CancellationToken = cancellationToken
  242. }).ConfigureAwait(false))
  243. {
  244. using (var xmlFileStream = _fileSystem.GetFileStream(xmlPath, FileMode.Create, FileAccess.Write, FileShare.Read, true))
  245. {
  246. await response.CopyToAsync(xmlFileStream).ConfigureAwait(false);
  247. }
  248. }
  249. }
  250. }
  251. }