MusicBrainzAlbumProvider.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Net.Http;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. using Jellyfin.Extensions;
  8. using MediaBrowser.Controller.Entities.Audio;
  9. using MediaBrowser.Controller.Providers;
  10. using MediaBrowser.Model.Entities;
  11. using MediaBrowser.Model.Providers;
  12. using MediaBrowser.Providers.Music;
  13. using MetaBrainz.MusicBrainz;
  14. using MetaBrainz.MusicBrainz.Interfaces.Entities;
  15. using MetaBrainz.MusicBrainz.Interfaces.Searches;
  16. using Microsoft.Extensions.Logging;
  17. namespace MediaBrowser.Providers.Plugins.MusicBrainz;
  18. /// <summary>
  19. /// Music album metadata provider for MusicBrainz.
  20. /// </summary>
  21. public class MusicBrainzAlbumProvider : IRemoteMetadataProvider<MusicAlbum, AlbumInfo>, IHasOrder, IDisposable
  22. {
  23. private readonly ILogger<MusicBrainzAlbumProvider> _logger;
  24. private readonly Query _musicBrainzQuery;
  25. private readonly string _musicBrainzDefaultUri = "https://musicbrainz.org";
  26. /// <summary>
  27. /// Initializes a new instance of the <see cref="MusicBrainzAlbumProvider"/> class.
  28. /// </summary>
  29. /// <param name="logger">The logger.</param>
  30. public MusicBrainzAlbumProvider(ILogger<MusicBrainzAlbumProvider> logger)
  31. {
  32. _logger = logger;
  33. MusicBrainz.Plugin.Instance!.ConfigurationChanged += (_, _) =>
  34. {
  35. if (Uri.TryCreate(MusicBrainz.Plugin.Instance.Configuration.Server, UriKind.Absolute, out var server))
  36. {
  37. Query.DefaultServer = server.Host;
  38. Query.DefaultPort = server.Port;
  39. Query.DefaultUrlScheme = server.Scheme;
  40. }
  41. else
  42. {
  43. // Fallback to official server
  44. _logger.LogWarning("Invalid MusicBrainz server specified, falling back to official server");
  45. var defaultServer = new Uri(_musicBrainzDefaultUri);
  46. Query.DefaultServer = defaultServer.Host;
  47. Query.DefaultPort = defaultServer.Port;
  48. Query.DefaultUrlScheme = defaultServer.Scheme;
  49. }
  50. Query.DelayBetweenRequests = MusicBrainz.Plugin.Instance.Configuration.RateLimit;
  51. };
  52. _musicBrainzQuery = new Query();
  53. }
  54. /// <inheritdoc />
  55. public string Name => "MusicBrainz";
  56. /// <inheritdoc />
  57. public int Order => 0;
  58. /// <inheritdoc />
  59. public async Task<IEnumerable<RemoteSearchResult>> GetSearchResults(AlbumInfo searchInfo, CancellationToken cancellationToken)
  60. {
  61. var releaseId = searchInfo.GetReleaseId();
  62. var releaseGroupId = searchInfo.GetReleaseGroupId();
  63. if (!string.IsNullOrEmpty(releaseId))
  64. {
  65. var releaseResult = await _musicBrainzQuery.LookupReleaseAsync(new Guid(releaseId), Include.ReleaseGroups, cancellationToken).ConfigureAwait(false);
  66. return GetReleaseResult(releaseResult).SingleItemAsEnumerable();
  67. }
  68. if (!string.IsNullOrEmpty(releaseGroupId))
  69. {
  70. var releaseGroupResult = await _musicBrainzQuery.LookupReleaseGroupAsync(new Guid(releaseGroupId), Include.None, null, cancellationToken).ConfigureAwait(false);
  71. return GetReleaseGroupResult(releaseGroupResult.Releases);
  72. }
  73. var artistMusicBrainzId = searchInfo.GetMusicBrainzArtistId();
  74. if (!string.IsNullOrWhiteSpace(artistMusicBrainzId))
  75. {
  76. var releaseSearchResults = await _musicBrainzQuery.FindReleasesAsync($"\"{searchInfo.Name}\" AND arid:{artistMusicBrainzId}", null, null, false, cancellationToken)
  77. .ConfigureAwait(false);
  78. if (releaseSearchResults.Results.Count > 0)
  79. {
  80. return GetReleaseSearchResult(releaseSearchResults.Results);
  81. }
  82. }
  83. else
  84. {
  85. // I'm sure there is a better way but for now it resolves search for 12" Mixes
  86. var queryName = searchInfo.Name.Replace("\"", string.Empty, StringComparison.Ordinal);
  87. var releaseSearchResults = await _musicBrainzQuery.FindReleasesAsync($"\"{queryName}\" AND artist:\"{searchInfo.GetAlbumArtist()}\"c", null, null, false, cancellationToken)
  88. .ConfigureAwait(false);
  89. if (releaseSearchResults.Results.Count > 0)
  90. {
  91. return GetReleaseSearchResult(releaseSearchResults.Results);
  92. }
  93. }
  94. return Enumerable.Empty<RemoteSearchResult>();
  95. }
  96. private IEnumerable<RemoteSearchResult> GetReleaseSearchResult(IEnumerable<ISearchResult<IRelease>>? releaseSearchResults)
  97. {
  98. if (releaseSearchResults is null)
  99. {
  100. yield break;
  101. }
  102. foreach (var result in releaseSearchResults)
  103. {
  104. yield return GetReleaseResult(result.Item);
  105. }
  106. }
  107. private IEnumerable<RemoteSearchResult> GetReleaseGroupResult(IEnumerable<IRelease>? releaseSearchResults)
  108. {
  109. if (releaseSearchResults is null)
  110. {
  111. yield break;
  112. }
  113. foreach (var result in releaseSearchResults)
  114. {
  115. yield return GetReleaseResult(result);
  116. }
  117. }
  118. private RemoteSearchResult GetReleaseResult(IRelease releaseSearchResult)
  119. {
  120. var searchResult = new RemoteSearchResult
  121. {
  122. Name = releaseSearchResult.Title,
  123. ProductionYear = releaseSearchResult.Date?.Year,
  124. PremiereDate = releaseSearchResult.Date?.NearestDate
  125. };
  126. if (releaseSearchResult.ArtistCredit?.Count > 0)
  127. {
  128. searchResult.AlbumArtist = new RemoteSearchResult
  129. {
  130. SearchProviderName = Name,
  131. Name = releaseSearchResult.ArtistCredit[0].Name
  132. };
  133. if (releaseSearchResult.ArtistCredit[0].Artist?.Id is not null)
  134. {
  135. searchResult.AlbumArtist.SetProviderId(MetadataProvider.MusicBrainzArtist, releaseSearchResult.ArtistCredit[0].Artist!.Id.ToString());
  136. }
  137. }
  138. searchResult.SetProviderId(MetadataProvider.MusicBrainzAlbum, releaseSearchResult.Id.ToString());
  139. if (releaseSearchResult.ReleaseGroup?.Id is not null)
  140. {
  141. searchResult.SetProviderId(MetadataProvider.MusicBrainzReleaseGroup, releaseSearchResult.ReleaseGroup.Id.ToString());
  142. }
  143. return searchResult;
  144. }
  145. /// <inheritdoc />
  146. public async Task<MetadataResult<MusicAlbum>> GetMetadata(AlbumInfo info, CancellationToken cancellationToken)
  147. {
  148. // TODO: This sets essentially nothing. As-is, it's mostly useless. Make it actually pull metadata and use it.
  149. var releaseId = info.GetReleaseId();
  150. var releaseGroupId = info.GetReleaseGroupId();
  151. var result = new MetadataResult<MusicAlbum>
  152. {
  153. Item = new MusicAlbum()
  154. };
  155. // If there is a release group, but no release ID, try to match the release
  156. if (string.IsNullOrWhiteSpace(releaseId) && !string.IsNullOrWhiteSpace(releaseGroupId))
  157. {
  158. // TODO: Actually try to match the release. Simply taking the first result is stupid.
  159. var releaseGroup = await _musicBrainzQuery.LookupReleaseGroupAsync(new Guid(releaseGroupId), Include.None, null, cancellationToken).ConfigureAwait(false);
  160. var release = releaseGroup.Releases?.Count > 0 ? releaseGroup.Releases[0] : null;
  161. if (release is not null)
  162. {
  163. releaseId = release.Id.ToString();
  164. result.HasMetadata = true;
  165. }
  166. }
  167. // If there is no release ID, lookup a release with the info we have
  168. if (string.IsNullOrWhiteSpace(releaseId))
  169. {
  170. var artistMusicBrainzId = info.GetMusicBrainzArtistId();
  171. IRelease? releaseResult = null;
  172. if (!string.IsNullOrEmpty(artistMusicBrainzId))
  173. {
  174. var releaseSearchResults = await _musicBrainzQuery.FindReleasesAsync($"\"{info.Name}\" AND arid:{artistMusicBrainzId}", null, null, false, cancellationToken)
  175. .ConfigureAwait(false);
  176. releaseResult = releaseSearchResults.Results.Count > 0 ? releaseSearchResults.Results[0].Item : null;
  177. }
  178. else if (!string.IsNullOrEmpty(info.GetAlbumArtist()))
  179. {
  180. var releaseSearchResults = await _musicBrainzQuery.FindReleasesAsync($"\"{info.Name}\" AND artist:{info.GetAlbumArtist()}", null, null, false, cancellationToken)
  181. .ConfigureAwait(false);
  182. releaseResult = releaseSearchResults.Results.Count > 0 ? releaseSearchResults.Results[0].Item : null;
  183. }
  184. if (releaseResult is not null)
  185. {
  186. releaseId = releaseResult.Id.ToString();
  187. if (releaseResult.ReleaseGroup?.Id is not null)
  188. {
  189. releaseGroupId = releaseResult.ReleaseGroup.Id.ToString();
  190. }
  191. result.HasMetadata = true;
  192. result.Item.ProductionYear = releaseResult.Date?.Year;
  193. result.Item.Overview = releaseResult.Annotation;
  194. }
  195. }
  196. // If we have a release ID but not a release group ID, lookup the release group
  197. if (!string.IsNullOrWhiteSpace(releaseId) && string.IsNullOrWhiteSpace(releaseGroupId))
  198. {
  199. var release = await _musicBrainzQuery.LookupReleaseAsync(new Guid(releaseId), Include.Releases, cancellationToken).ConfigureAwait(false);
  200. releaseGroupId = release.ReleaseGroup?.Id.ToString();
  201. result.HasMetadata = true;
  202. }
  203. // If we have a release ID and a release group ID
  204. if (!string.IsNullOrWhiteSpace(releaseId) || !string.IsNullOrWhiteSpace(releaseGroupId))
  205. {
  206. result.HasMetadata = true;
  207. }
  208. if (result.HasMetadata)
  209. {
  210. if (!string.IsNullOrEmpty(releaseId))
  211. {
  212. result.Item.SetProviderId(MetadataProvider.MusicBrainzAlbum, releaseId);
  213. }
  214. if (!string.IsNullOrEmpty(releaseGroupId))
  215. {
  216. result.Item.SetProviderId(MetadataProvider.MusicBrainzReleaseGroup, releaseGroupId);
  217. }
  218. }
  219. return result;
  220. }
  221. /// <inheritdoc />
  222. public Task<HttpResponseMessage> GetImageResponse(string url, CancellationToken cancellationToken)
  223. {
  224. throw new NotImplementedException();
  225. }
  226. /// <inheritdoc />
  227. public void Dispose()
  228. {
  229. Dispose(true);
  230. GC.SuppressFinalize(this);
  231. }
  232. /// <summary>
  233. /// Dispose all resources.
  234. /// </summary>
  235. /// <param name="disposing">Whether to dispose.</param>
  236. protected virtual void Dispose(bool disposing)
  237. {
  238. if (disposing)
  239. {
  240. _musicBrainzQuery.Dispose();
  241. }
  242. }
  243. }