You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Lidarr/src/NzbDrone.Core/Music/Services/AlbumService.cs

300 lines
11 KiB

using System;
using System.Collections.Generic;
using System.Linq;
using NLog;
using NzbDrone.Common.Extensions;
using NzbDrone.Core.Datastore;
using NzbDrone.Core.Messaging.Events;
using NzbDrone.Core.Music.Events;
using NzbDrone.Core.Parser;
namespace NzbDrone.Core.Music
{
public interface IAlbumService
{
Album GetAlbum(int albumId);
List<Album> GetAlbums(IEnumerable<int> albumIds);
List<Album> GetAlbumsByArtist(int artistId);
List<Album> GetNextAlbumsByArtistMetadataId(IEnumerable<int> artistMetadataIds);
List<Album> GetLastAlbumsByArtistMetadataId(IEnumerable<int> artistMetadataIds);
List<Album> GetAlbumsByArtistMetadataId(int artistMetadataId);
List<Album> GetAlbumsForRefresh(int artistMetadataId, List<string> foreignIds);
Album AddAlbum(Album newAlbum, bool doRefresh);
Album FindById(string foreignId);
Album FindByTitle(int artistMetadataId, string title);
Album FindByTitleInexact(int artistMetadataId, string title);
List<Album> GetCandidates(int artistMetadataId, string title);
void DeleteAlbum(int albumId, bool deleteFiles, bool addImportListExclusion = false);
List<Album> GetAllAlbums();
Album UpdateAlbum(Album album);
void SetAlbumMonitored(int albumId, bool monitored);
void SetMonitored(IEnumerable<int> ids, bool monitored);
PagingSpec<Album> AlbumsWithoutFiles(PagingSpec<Album> pagingSpec);
List<Album> AlbumsBetweenDates(DateTime start, DateTime end, bool includeUnmonitored);
List<Album> ArtistAlbumsBetweenDates(Artist artist, DateTime start, DateTime end, bool includeUnmonitored);
void InsertMany(List<Album> albums);
void UpdateMany(List<Album> albums);
void DeleteMany(List<Album> albums);
void SetAddOptions(IEnumerable<Album> albums);
Album FindAlbumByRelease(string albumReleaseId);
Album FindAlbumByTrackId(int trackId);
List<Album> GetArtistAlbumsWithFiles(Artist artist);
}
public class AlbumService : IAlbumService,
IHandle<ArtistsDeletedEvent>
{
private readonly IAlbumRepository _albumRepository;
private readonly IEventAggregator _eventAggregator;
private readonly Logger _logger;
public AlbumService(IAlbumRepository albumRepository,
IEventAggregator eventAggregator,
Logger logger)
{
_albumRepository = albumRepository;
_eventAggregator = eventAggregator;
_logger = logger;
}
public Album AddAlbum(Album newAlbum, bool doRefresh)
{
_albumRepository.Insert(newAlbum);
_eventAggregator.PublishEvent(new AlbumAddedEvent(GetAlbum(newAlbum.Id), doRefresh));
return newAlbum;
}
public void DeleteAlbum(int albumId, bool deleteFiles, bool addImportListExclusion = false)
{
var album = _albumRepository.Get(albumId);
album.Artist.LazyLoad();
_albumRepository.Delete(albumId);
_eventAggregator.PublishEvent(new AlbumDeletedEvent(album, deleteFiles, addImportListExclusion));
}
public Album FindById(string foreignId)
{
return _albumRepository.FindById(foreignId);
}
public Album FindByTitle(int artistMetadataId, string title)
{
return _albumRepository.FindByTitle(artistMetadataId, title);
}
private List<Tuple<Func<Album, string, double>, string>> AlbumScoringFunctions(string title, string cleanTitle)
{
Func<Func<Album, string, double>, string, Tuple<Func<Album, string, double>, string>> tc = Tuple.Create;
var scoringFunctions = new List<Tuple<Func<Album, string, double>, string>>
{
tc((a, t) => a.CleanTitle.FuzzyMatch(t), cleanTitle),
tc((a, t) => a.Title.FuzzyMatch(t), title),
tc((a, t) => a.CleanTitle.FuzzyMatch(t), title.RemoveBracketsAndContents().CleanArtistName()),
tc((a, t) => a.CleanTitle.FuzzyMatch(t), title.RemoveAfterDash().CleanArtistName()),
tc((a, t) => a.CleanTitle.FuzzyMatch(t), title.RemoveBracketsAndContents().RemoveAfterDash().CleanArtistName()),
tc((a, t) => t.FuzzyContains(a.CleanTitle), cleanTitle),
tc((a, t) => t.FuzzyContains(a.Title), title)
};
return scoringFunctions;
}
public Album FindByTitleInexact(int artistMetadataId, string title)
{
var albums = GetAlbumsByArtistMetadataId(artistMetadataId);
foreach (var func in AlbumScoringFunctions(title, title.CleanArtistName()))
{
var results = FindByStringInexact(albums, func.Item1, func.Item2);
if (results.Count == 1)
{
return results[0];
}
}
return null;
}
public List<Album> GetCandidates(int artistMetadataId, string title)
{
var albums = GetAlbumsByArtistMetadataId(artistMetadataId);
var output = new List<Album>();
foreach (var func in AlbumScoringFunctions(title, title.CleanArtistName()))
{
output.AddRange(FindByStringInexact(albums, func.Item1, func.Item2));
}
return output.DistinctBy(x => x.Id).ToList();
}
private List<Album> FindByStringInexact(List<Album> albums, Func<Album, string, double> scoreFunction, string title)
{
const double fuzzThreshold = 0.7;
const double fuzzGap = 0.4;
var sortedAlbums = albums.Select(s => new
{
MatchProb = scoreFunction(s, title),
Album = s
})
.ToList()
.OrderByDescending(s => s.MatchProb)
.ToList();
return sortedAlbums.TakeWhile((x, i) => i == 0 || sortedAlbums[i - 1].MatchProb - x.MatchProb < fuzzGap)
.TakeWhile((x, i) => x.MatchProb > fuzzThreshold || (i > 0 && sortedAlbums[i - 1].MatchProb > fuzzThreshold))
.Select(x => x.Album)
.ToList();
}
public List<Album> GetAllAlbums()
{
return _albumRepository.All().ToList();
}
public Album GetAlbum(int albumId)
{
return _albumRepository.Get(albumId);
}
public List<Album> GetAlbums(IEnumerable<int> albumIds)
{
return _albumRepository.Get(albumIds).ToList();
}
public List<Album> GetAlbumsByArtist(int artistId)
{
return _albumRepository.GetAlbums(artistId).ToList();
}
public List<Album> GetNextAlbumsByArtistMetadataId(IEnumerable<int> artistMetadataIds)
{
return _albumRepository.GetNextAlbums(artistMetadataIds).ToList();
}
public List<Album> GetLastAlbumsByArtistMetadataId(IEnumerable<int> artistMetadataIds)
{
return _albumRepository.GetLastAlbums(artistMetadataIds).ToList();
}
public List<Album> GetAlbumsByArtistMetadataId(int artistMetadataId)
{
return _albumRepository.GetAlbumsByArtistMetadataId(artistMetadataId).ToList();
}
public List<Album> GetAlbumsForRefresh(int artistMetadataId, List<string> foreignIds)
{
return _albumRepository.GetAlbumsForRefresh(artistMetadataId, foreignIds);
}
public Album FindAlbumByRelease(string albumReleaseId)
{
return _albumRepository.FindAlbumByRelease(albumReleaseId);
}
public Album FindAlbumByTrackId(int trackId)
{
return _albumRepository.FindAlbumByTrack(trackId);
}
public void SetAddOptions(IEnumerable<Album> albums)
{
_albumRepository.SetFields(albums.ToList(), s => s.AddOptions);
}
public PagingSpec<Album> AlbumsWithoutFiles(PagingSpec<Album> pagingSpec)
{
var albumResult = _albumRepository.AlbumsWithoutFiles(pagingSpec);
return albumResult;
}
public List<Album> AlbumsBetweenDates(DateTime start, DateTime end, bool includeUnmonitored)
{
var albums = _albumRepository.AlbumsBetweenDates(start.ToUniversalTime(), end.ToUniversalTime(), includeUnmonitored);
return albums;
}
public List<Album> ArtistAlbumsBetweenDates(Artist artist, DateTime start, DateTime end, bool includeUnmonitored)
{
var albums = _albumRepository.ArtistAlbumsBetweenDates(artist, start.ToUniversalTime(), end.ToUniversalTime(), includeUnmonitored);
return albums;
}
public List<Album> GetArtistAlbumsWithFiles(Artist artist)
{
return _albumRepository.GetArtistAlbumsWithFiles(artist);
}
public void InsertMany(List<Album> albums)
{
_albumRepository.InsertMany(albums);
}
public void UpdateMany(List<Album> albums)
{
_albumRepository.UpdateMany(albums);
}
public void DeleteMany(List<Album> albums)
{
_albumRepository.DeleteMany(albums);
foreach (var album in albums)
{
_eventAggregator.PublishEvent(new AlbumDeletedEvent(album, false, false));
}
}
public Album UpdateAlbum(Album album)
{
var storedAlbum = GetAlbum(album.Id);
var updatedAlbum = _albumRepository.Update(album);
// If updatedAlbum has populated the Releases, populate in the storedAlbum too
if (updatedAlbum.AlbumReleases.IsLoaded)
{
storedAlbum.AlbumReleases.LazyLoad();
}
_eventAggregator.PublishEvent(new AlbumEditedEvent(updatedAlbum, storedAlbum));
return updatedAlbum;
}
public void SetAlbumMonitored(int albumId, bool monitored)
{
var album = _albumRepository.Get(albumId);
_albumRepository.SetMonitoredFlat(album, monitored);
// publish album edited event so artist stats update
_eventAggregator.PublishEvent(new AlbumEditedEvent(album, album));
_logger.Debug("Monitored flag for Album:{0} was set to {1}", albumId, monitored);
}
public void SetMonitored(IEnumerable<int> ids, bool monitored)
{
_albumRepository.SetMonitored(ids, monitored);
// publish album edited event so artist stats update
foreach (var album in _albumRepository.Get(ids))
{
_eventAggregator.PublishEvent(new AlbumEditedEvent(album, album));
}
}
public void Handle(ArtistsDeletedEvent message)
{
// TODO Do this in one call instead of one for each artist?
var albums = message.Artists.SelectMany(x => GetAlbumsByArtistMetadataId(x.ArtistMetadataId)).ToList();
DeleteMany(albums);
}
}
}