LastfmBaseProvider.cs 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. using System.Net;
  2. using MediaBrowser.Common.Net;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Model.Entities;
  5. using MediaBrowser.Model.Logging;
  6. using System;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using MediaBrowser.Model.Serialization;
  10. namespace MediaBrowser.Controller.Providers.Music
  11. {
  12. class LastfmProviderException : ApplicationException
  13. {
  14. public LastfmProviderException(string msg)
  15. : base(msg)
  16. {
  17. }
  18. }
  19. /// <summary>
  20. /// Class MovieDbProvider
  21. /// </summary>
  22. public abstract class LastfmBaseProvider : BaseMetadataProvider
  23. {
  24. /// <summary>
  25. /// Gets the json serializer.
  26. /// </summary>
  27. /// <value>The json serializer.</value>
  28. protected IJsonSerializer JsonSerializer { get; private set; }
  29. /// <summary>
  30. /// Gets the HTTP client.
  31. /// </summary>
  32. /// <value>The HTTP client.</value>
  33. protected IHttpClient HttpClient { get; private set; }
  34. /// <summary>
  35. /// The name of the local json meta file for this item type
  36. /// </summary>
  37. protected string LocalMetaFileName { get; set; }
  38. /// <summary>
  39. /// Initializes a new instance of the <see cref="LastfmBaseProvider" /> class.
  40. /// </summary>
  41. /// <param name="jsonSerializer">The json serializer.</param>
  42. /// <param name="httpClient">The HTTP client.</param>
  43. /// <param name="logManager">The Log manager</param>
  44. /// <exception cref="System.ArgumentNullException">jsonSerializer</exception>
  45. public LastfmBaseProvider(IJsonSerializer jsonSerializer, IHttpClient httpClient, ILogManager logManager)
  46. : base(logManager)
  47. {
  48. if (jsonSerializer == null)
  49. {
  50. throw new ArgumentNullException("jsonSerializer");
  51. }
  52. if (httpClient == null)
  53. {
  54. throw new ArgumentNullException("httpClient");
  55. }
  56. JsonSerializer = jsonSerializer;
  57. HttpClient = httpClient;
  58. }
  59. /// <summary>
  60. /// Gets the priority.
  61. /// </summary>
  62. /// <value>The priority.</value>
  63. public override MetadataProviderPriority Priority
  64. {
  65. get { return MetadataProviderPriority.Second; }
  66. }
  67. /// <summary>
  68. /// Gets a value indicating whether [requires internet].
  69. /// </summary>
  70. /// <value><c>true</c> if [requires internet]; otherwise, <c>false</c>.</value>
  71. public override bool RequiresInternet
  72. {
  73. get
  74. {
  75. return true;
  76. }
  77. }
  78. /// <summary>
  79. /// If we save locally, refresh if they delete something
  80. /// </summary>
  81. protected override bool RefreshOnFileSystemStampChange
  82. {
  83. get
  84. {
  85. return Kernel.Instance.Configuration.SaveLocalMeta;
  86. }
  87. }
  88. protected const string RootUrl = @"http://ws.audioscrobbler.com/2.0/?";
  89. protected static string ApiKey = "7b76553c3eb1d341d642755aecc40a33";
  90. protected override bool NeedsRefreshInternal(BaseItem item, BaseProviderInfo providerInfo)
  91. {
  92. if (item.DontFetchMeta) return false;
  93. if (Kernel.Instance.Configuration.SaveLocalMeta && HasFileSystemStampChanged(item, providerInfo))
  94. {
  95. //If they deleted something from file system, chances are, this item was mis-identified the first time
  96. item.SetProviderId(MetadataProviders.Musicbrainz, null);
  97. Logger.Debug("LastfmProvider reports file system stamp change...");
  98. return true;
  99. }
  100. if (providerInfo.LastRefreshStatus == ProviderRefreshStatus.CompletedWithErrors)
  101. {
  102. Logger.Debug("LastfmProvider for {0} - last attempt had errors. Will try again.", item.Path);
  103. return true;
  104. }
  105. var downloadDate = providerInfo.LastRefreshed;
  106. if (Kernel.Instance.Configuration.MetadataRefreshDays == -1 && downloadDate != DateTime.MinValue)
  107. {
  108. return false;
  109. }
  110. if (DateTime.Today.Subtract(item.DateCreated).TotalDays > 180 && downloadDate != DateTime.MinValue)
  111. return false; // don't trigger a refresh data for item that are more than 6 months old and have been refreshed before
  112. if (DateTime.Today.Subtract(downloadDate).TotalDays < Kernel.Instance.Configuration.MetadataRefreshDays) // only refresh every n days
  113. return false;
  114. Logger.Debug("LastfmProvider - " + item.Name + " needs refresh. Download date: " + downloadDate + " item created date: " + item.DateCreated + " Check for Update age: " + Kernel.Instance.Configuration.MetadataRefreshDays);
  115. return true;
  116. }
  117. /// <summary>
  118. /// Fetches metadata and returns true or false indicating if any work that requires persistence was done
  119. /// </summary>
  120. /// <param name="item">The item.</param>
  121. /// <param name="force">if set to <c>true</c> [force].</param>
  122. /// <param name="cancellationToken">The cancellation token</param>
  123. /// <returns>Task{System.Boolean}.</returns>
  124. protected override async Task<bool> FetchAsyncInternal(BaseItem item, bool force, CancellationToken cancellationToken)
  125. {
  126. if (item.DontFetchMeta)
  127. {
  128. Logger.Info("LastfmProvider - Not fetching because requested to ignore " + item.Name);
  129. return false;
  130. }
  131. cancellationToken.ThrowIfCancellationRequested();
  132. if (!Kernel.Instance.Configuration.SaveLocalMeta || !HasLocalMeta(item) || (force && !HasLocalMeta(item)))
  133. {
  134. try
  135. {
  136. await FetchData(item, cancellationToken).ConfigureAwait(false);
  137. SetLastRefreshed(item, DateTime.UtcNow);
  138. }
  139. catch (LastfmProviderException)
  140. {
  141. SetLastRefreshed(item, DateTime.UtcNow, ProviderRefreshStatus.CompletedWithErrors);
  142. }
  143. return true;
  144. }
  145. Logger.Debug("LastfmProvider not fetching because local meta exists for " + item.Name);
  146. SetLastRefreshed(item, DateTime.UtcNow);
  147. return true;
  148. }
  149. /// <summary>
  150. /// Determines whether [has local meta] [the specified item].
  151. /// </summary>
  152. /// <param name="item">The item.</param>
  153. /// <returns><c>true</c> if [has local meta] [the specified item]; otherwise, <c>false</c>.</returns>
  154. private bool HasLocalMeta(BaseItem item)
  155. {
  156. return item.ResolveArgs.ContainsMetaFileByName(LocalMetaFileName);
  157. }
  158. /// <summary>
  159. /// Fetches the items data.
  160. /// </summary>
  161. /// <param name="item">The item.</param>
  162. /// <param name="cancellationToken"></param>
  163. /// <returns>Task.</returns>
  164. protected async Task FetchData(BaseItem item, CancellationToken cancellationToken)
  165. {
  166. var id = item.GetProviderId(MetadataProviders.Musicbrainz) ?? await FindId(item, cancellationToken).ConfigureAwait(false);
  167. if (id != null)
  168. {
  169. Logger.Debug("LastfmProvider - getting info with id: " + id);
  170. cancellationToken.ThrowIfCancellationRequested();
  171. item.SetProviderId(MetadataProviders.Musicbrainz, id);
  172. await FetchLastfmData(item, id, cancellationToken).ConfigureAwait(false);
  173. }
  174. else
  175. {
  176. Logger.Info("LastfmProvider could not find " + item.Name + ". Check name on Last.fm.");
  177. }
  178. }
  179. protected abstract Task<string> FindId(BaseItem item, CancellationToken cancellationToken);
  180. protected abstract Task FetchLastfmData(BaseItem item, string id, CancellationToken cancellationToken);
  181. /// <summary>
  182. /// Encodes an URL.
  183. /// </summary>
  184. /// <param name="name">The name.</param>
  185. /// <returns>System.String.</returns>
  186. protected static string UrlEncode(string name)
  187. {
  188. return WebUtility.UrlEncode(name);
  189. }
  190. }
  191. }