TvdbSeasonProvider.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. using System.Net;
  2. using MediaBrowser.Common.IO;
  3. using MediaBrowser.Controller.Configuration;
  4. using MediaBrowser.Controller.Entities;
  5. using MediaBrowser.Controller.Entities.TV;
  6. using MediaBrowser.Controller.Library;
  7. using MediaBrowser.Controller.Providers;
  8. using MediaBrowser.Model.Configuration;
  9. using MediaBrowser.Model.Entities;
  10. using MediaBrowser.Model.Logging;
  11. using MediaBrowser.Model.Net;
  12. using MediaBrowser.Model.Providers;
  13. using System;
  14. using System.Collections.Generic;
  15. using System.IO;
  16. using System.Linq;
  17. using System.Threading;
  18. using System.Threading.Tasks;
  19. namespace MediaBrowser.Providers.TV
  20. {
  21. /// <summary>
  22. /// Class RemoteSeasonProvider
  23. /// </summary>
  24. class TvdbSeasonProvider : BaseMetadataProvider
  25. {
  26. /// <summary>
  27. /// The _provider manager
  28. /// </summary>
  29. private readonly IProviderManager _providerManager;
  30. private readonly IFileSystem _fileSystem;
  31. /// <summary>
  32. /// Initializes a new instance of the <see cref="TvdbSeasonProvider"/> class.
  33. /// </summary>
  34. /// <param name="logManager">The log manager.</param>
  35. /// <param name="configurationManager">The configuration manager.</param>
  36. /// <param name="providerManager">The provider manager.</param>
  37. /// <exception cref="System.ArgumentNullException">httpClient</exception>
  38. public TvdbSeasonProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IProviderManager providerManager, IFileSystem fileSystem)
  39. : base(logManager, configurationManager)
  40. {
  41. _providerManager = providerManager;
  42. _fileSystem = fileSystem;
  43. }
  44. /// <summary>
  45. /// Supportses the specified item.
  46. /// </summary>
  47. /// <param name="item">The item.</param>
  48. /// <returns><c>true</c> if XXXX, <c>false</c> otherwise</returns>
  49. public override bool Supports(BaseItem item)
  50. {
  51. return item is Season;
  52. }
  53. /// <summary>
  54. /// Gets the priority.
  55. /// </summary>
  56. /// <value>The priority.</value>
  57. public override MetadataProviderPriority Priority
  58. {
  59. // Run after fanart
  60. get { return MetadataProviderPriority.Fourth; }
  61. }
  62. /// <summary>
  63. /// Gets a value indicating whether [requires internet].
  64. /// </summary>
  65. /// <value><c>true</c> if [requires internet]; otherwise, <c>false</c>.</value>
  66. public override bool RequiresInternet
  67. {
  68. get
  69. {
  70. return true;
  71. }
  72. }
  73. public override ItemUpdateType ItemUpdateType
  74. {
  75. get
  76. {
  77. return ItemUpdateType.ImageUpdate;
  78. }
  79. }
  80. /// <summary>
  81. /// Gets a value indicating whether [refresh on version change].
  82. /// </summary>
  83. /// <value><c>true</c> if [refresh on version change]; otherwise, <c>false</c>.</value>
  84. protected override bool RefreshOnVersionChange
  85. {
  86. get
  87. {
  88. return true;
  89. }
  90. }
  91. /// <summary>
  92. /// Gets the provider version.
  93. /// </summary>
  94. /// <value>The provider version.</value>
  95. protected override string ProviderVersion
  96. {
  97. get
  98. {
  99. return "2";
  100. }
  101. }
  102. protected override bool NeedsRefreshBasedOnCompareDate(BaseItem item, BaseProviderInfo providerInfo)
  103. {
  104. var season = (Season)item;
  105. var seriesId = season.Series != null ? season.Series.GetProviderId(MetadataProviders.Tvdb) : null;
  106. if (!string.IsNullOrEmpty(seriesId))
  107. {
  108. // Process images
  109. var imagesXmlPath = Path.Combine(TvdbSeriesProvider.GetSeriesDataPath(ConfigurationManager.ApplicationPaths, seriesId), "banners.xml");
  110. var imagesFileInfo = new FileInfo(imagesXmlPath);
  111. if (imagesFileInfo.Exists)
  112. {
  113. return _fileSystem.GetLastWriteTimeUtc(imagesFileInfo) > providerInfo.LastRefreshed;
  114. }
  115. }
  116. return false;
  117. }
  118. protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
  119. {
  120. if (item.HasImage(ImageType.Primary) && item.HasImage(ImageType.Banner) && item.BackdropImagePaths.Count > 0)
  121. {
  122. return false;
  123. }
  124. return base.NeedsRefreshInternal(item, providerInfo);
  125. }
  126. /// <summary>
  127. /// Fetches metadata and returns true or false indicating if any work that requires persistence was done
  128. /// </summary>
  129. /// <param name="item">The item.</param>
  130. /// <param name="force">if set to <c>true</c> [force].</param>
  131. /// <param name="cancellationToken">The cancellation token.</param>
  132. /// <returns>Task{System.Boolean}.</returns>
  133. public override async Task<bool> FetchAsync(BaseItem item, bool force, BaseProviderInfo providerInfo, CancellationToken cancellationToken)
  134. {
  135. cancellationToken.ThrowIfCancellationRequested();
  136. var images = await _providerManager.GetAvailableRemoteImages(item, cancellationToken, ManualTvdbSeasonImageProvider.ProviderName).ConfigureAwait(false);
  137. await DownloadImages(item, images.ToList(), cancellationToken).ConfigureAwait(false);
  138. SetLastRefreshed(item, DateTime.UtcNow, providerInfo);
  139. return true;
  140. }
  141. private async Task DownloadImages(BaseItem item, List<RemoteImageInfo> images, CancellationToken cancellationToken)
  142. {
  143. var options = ConfigurationManager.Configuration.GetMetadataOptions("Season") ?? new MetadataOptions();
  144. var backdropLimit = options.GetLimit(ImageType.Backdrop);
  145. if (!item.LockedFields.Contains(MetadataFields.Images))
  146. {
  147. if (!item.HasImage(ImageType.Primary))
  148. {
  149. await SaveImage(item, images, ImageType.Primary, cancellationToken).ConfigureAwait(false);
  150. }
  151. if (options.IsEnabled(ImageType.Banner) && !item.HasImage(ImageType.Banner))
  152. {
  153. await SaveImage(item, images, ImageType.Banner, cancellationToken).ConfigureAwait(false);
  154. }
  155. }
  156. if (options.IsEnabled(ImageType.Backdrop) && item.BackdropImagePaths.Count < backdropLimit && !item.LockedFields.Contains(MetadataFields.Backdrops))
  157. {
  158. foreach (var backdrop in images.Where(i => i.Type == ImageType.Backdrop))
  159. {
  160. var url = backdrop.Url;
  161. await _providerManager.SaveImage(item, url, TvdbSeriesProvider.Current.TvDbResourcePool, ImageType.Backdrop, null, cancellationToken).ConfigureAwait(false);
  162. if (item.BackdropImagePaths.Count >= backdropLimit) break;
  163. }
  164. }
  165. }
  166. private async Task SaveImage(BaseItem item, List<RemoteImageInfo> images, ImageType type, CancellationToken cancellationToken)
  167. {
  168. foreach (var image in images.Where(i => i.Type == type))
  169. {
  170. try
  171. {
  172. await _providerManager.SaveImage(item, image.Url, TvdbSeriesProvider.Current.TvDbResourcePool, type, null, cancellationToken).ConfigureAwait(false);
  173. break;
  174. }
  175. catch (HttpException ex)
  176. {
  177. // Sometimes fanart has bad url's in their xml
  178. if (ex.StatusCode.HasValue && ex.StatusCode.Value == HttpStatusCode.NotFound)
  179. {
  180. continue;
  181. }
  182. break;
  183. }
  184. }
  185. }
  186. }
  187. }