MusicBrainzAlbumProvider.cs 11 KB

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