MusicBrainzAlbumProvider.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. using MediaBrowser.Common.Net;
  2. using MediaBrowser.Controller.Configuration;
  3. using MediaBrowser.Controller.Entities;
  4. using MediaBrowser.Controller.Entities.Audio;
  5. using MediaBrowser.Controller.Providers;
  6. using MediaBrowser.Model.Entities;
  7. using MediaBrowser.Model.Logging;
  8. using System;
  9. using System.IO;
  10. using System.Net;
  11. using System.Text;
  12. using System.Threading;
  13. using System.Threading.Tasks;
  14. using System.Xml;
  15. namespace MediaBrowser.Providers.Music
  16. {
  17. public class MusicBrainzAlbumProvider : BaseMetadataProvider
  18. {
  19. internal static MusicBrainzAlbumProvider Current;
  20. private readonly IHttpClient _httpClient;
  21. public MusicBrainzAlbumProvider(ILogManager logManager, IServerConfigurationManager configurationManager, IHttpClient httpClient)
  22. : base(logManager, configurationManager)
  23. {
  24. _httpClient = httpClient;
  25. Current = this;
  26. }
  27. public override bool Supports(BaseItem item)
  28. {
  29. return item is MusicAlbum;
  30. }
  31. public override async Task<bool> FetchAsync(BaseItem item, bool force, CancellationToken cancellationToken)
  32. {
  33. var releaseId = item.GetProviderId(MetadataProviders.Musicbrainz);
  34. var releaseGroupId = item.GetProviderId(MetadataProviders.MusicBrainzReleaseGroup);
  35. if (string.IsNullOrEmpty(releaseId))
  36. {
  37. var result = await GetReleaseResult((MusicAlbum)item, cancellationToken).ConfigureAwait(false);
  38. if (!string.IsNullOrEmpty(result.ReleaseId))
  39. {
  40. releaseId = result.ReleaseId;
  41. item.SetProviderId(MetadataProviders.Musicbrainz, releaseId);
  42. }
  43. if (!string.IsNullOrEmpty(result.ReleaseGroupId))
  44. {
  45. releaseGroupId = result.ReleaseGroupId;
  46. item.SetProviderId(MetadataProviders.MusicBrainzReleaseGroup, releaseGroupId);
  47. }
  48. }
  49. // If we have a release Id but not a release group Id...
  50. if (!string.IsNullOrEmpty(releaseId) && string.IsNullOrEmpty(releaseGroupId))
  51. {
  52. releaseGroupId = await GetReleaseGroupId(releaseId, cancellationToken).ConfigureAwait(false);
  53. item.SetProviderId(MetadataProviders.MusicBrainzReleaseGroup, releaseGroupId);
  54. }
  55. SetLastRefreshed(item, DateTime.UtcNow);
  56. return true;
  57. }
  58. private Task<ReleaseResult> GetReleaseResult(MusicAlbum album, CancellationToken cancellationToken)
  59. {
  60. var artist = album.Parent;
  61. var artistId = artist.GetProviderId(MetadataProviders.Musicbrainz);
  62. if (!string.IsNullOrEmpty(artistId))
  63. {
  64. return GetReleaseResult(album.Name, artistId, cancellationToken);
  65. }
  66. return GetReleaseResultByArtistName(album.Name, artist.Name, cancellationToken);
  67. }
  68. private async Task<ReleaseResult> GetReleaseResult(string albumName, string artistId, CancellationToken cancellationToken)
  69. {
  70. var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" and arid:{1}",
  71. WebUtility.UrlEncode(albumName),
  72. artistId);
  73. var doc = await GetMusicBrainzResponse(url, cancellationToken).ConfigureAwait(false);
  74. return GetReleaseResult(doc);
  75. }
  76. private async Task<ReleaseResult> GetReleaseResultByArtistName(string albumName, string artistName, CancellationToken cancellationToken)
  77. {
  78. var url = string.Format("http://www.musicbrainz.org/ws/2/release/?query=\"{0}\" and artist:\"{1}\"",
  79. WebUtility.UrlEncode(albumName),
  80. WebUtility.UrlEncode(artistName));
  81. var doc = await GetMusicBrainzResponse(url, cancellationToken).ConfigureAwait(false);
  82. return GetReleaseResult(doc);
  83. }
  84. private ReleaseResult GetReleaseResult(XmlDocument doc)
  85. {
  86. var ns = new XmlNamespaceManager(doc.NameTable);
  87. ns.AddNamespace("mb", "http://musicbrainz.org/ns/mmd-2.0#");
  88. var result = new ReleaseResult
  89. {
  90. };
  91. var releaseIdNode = doc.SelectSingleNode("//mb:release-list/mb:release/@id", ns);
  92. if (releaseIdNode != null)
  93. {
  94. result.ReleaseId = releaseIdNode.Value;
  95. }
  96. var releaseGroupIdNode = doc.SelectSingleNode("//mb:release-list/mb:release/mb:release-group/@id", ns);
  97. if (releaseGroupIdNode != null)
  98. {
  99. result.ReleaseGroupId = releaseGroupIdNode.Value;
  100. }
  101. return result;
  102. }
  103. private class ReleaseResult
  104. {
  105. public string ReleaseId;
  106. public string ReleaseGroupId;
  107. }
  108. /// <summary>
  109. /// Gets the release group id internal.
  110. /// </summary>
  111. /// <param name="releaseEntryId">The release entry id.</param>
  112. /// <param name="cancellationToken">The cancellation token.</param>
  113. /// <returns>Task{System.String}.</returns>
  114. private async Task<string> GetReleaseGroupId(string releaseEntryId, CancellationToken cancellationToken)
  115. {
  116. var url = string.Format("http://www.musicbrainz.org/ws/2/release-group/?query=reid:{0}", releaseEntryId);
  117. var doc = await GetMusicBrainzResponse(url, cancellationToken).ConfigureAwait(false);
  118. var ns = new XmlNamespaceManager(doc.NameTable);
  119. ns.AddNamespace("mb", "http://musicbrainz.org/ns/mmd-2.0#");
  120. var node = doc.SelectSingleNode("//mb:release-group-list/mb:release-group/@id", ns);
  121. return node != null ? node.Value : null;
  122. }
  123. /// <summary>
  124. /// The _last music brainz request
  125. /// </summary>
  126. private DateTime _lastRequestDate = DateTime.MinValue;
  127. /// <summary>
  128. /// The _music brainz resource pool
  129. /// </summary>
  130. private readonly SemaphoreSlim _musicBrainzResourcePool = new SemaphoreSlim(1, 1);
  131. /// <summary>
  132. /// Gets the music brainz response.
  133. /// </summary>
  134. /// <param name="url">The URL.</param>
  135. /// <param name="cancellationToken">The cancellation token.</param>
  136. /// <returns>Task{XmlDocument}.</returns>
  137. internal async Task<XmlDocument> GetMusicBrainzResponse(string url, CancellationToken cancellationToken)
  138. {
  139. await _musicBrainzResourcePool.WaitAsync(cancellationToken).ConfigureAwait(false);
  140. try
  141. {
  142. var diff = 1500 - (DateTime.Now - _lastRequestDate).TotalMilliseconds;
  143. // MusicBrainz is extremely adamant about limiting to one request per second
  144. if (diff > 0)
  145. {
  146. await Task.Delay(Convert.ToInt32(diff), cancellationToken).ConfigureAwait(false);
  147. }
  148. _lastRequestDate = DateTime.Now;
  149. var doc = new XmlDocument();
  150. using (var xml = await _httpClient.Get(new HttpRequestOptions
  151. {
  152. Url = url,
  153. CancellationToken = cancellationToken,
  154. UserAgent = Environment.MachineName
  155. }).ConfigureAwait(false))
  156. {
  157. using (var oReader = new StreamReader(xml, Encoding.UTF8))
  158. {
  159. doc.Load(oReader);
  160. }
  161. }
  162. return doc;
  163. }
  164. finally
  165. {
  166. _lastRequestDate = DateTime.Now;
  167. _musicBrainzResourcePool.Release();
  168. }
  169. }
  170. public override MetadataProviderPriority Priority
  171. {
  172. get { return MetadataProviderPriority.Third; }
  173. }
  174. }
  175. }