|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Linq.Expressions;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using NLog;
|
|
|
|
using NzbDrone.Common.Instrumentation.Extensions;
|
|
|
|
using NzbDrone.Core.Datastore;
|
|
|
|
using NzbDrone.Core.DecisionEngine;
|
|
|
|
using NzbDrone.Core.Download;
|
|
|
|
using NzbDrone.Core.Messaging.Commands;
|
|
|
|
using NzbDrone.Core.Music;
|
|
|
|
using NzbDrone.Core.Queue;
|
|
|
|
|
|
|
|
namespace NzbDrone.Core.IndexerSearch
|
|
|
|
{
|
|
|
|
internal class AlbumSearchService : IExecute<AlbumSearchCommand>,
|
|
|
|
IExecute<MissingAlbumSearchCommand>,
|
|
|
|
IExecute<CutoffUnmetAlbumSearchCommand>
|
|
|
|
{
|
|
|
|
private readonly ISearchForReleases _releaseSearchService;
|
|
|
|
private readonly IAlbumService _albumService;
|
|
|
|
private readonly IAlbumCutoffService _albumCutoffService;
|
|
|
|
private readonly IQueueService _queueService;
|
|
|
|
private readonly IProcessDownloadDecisions _processDownloadDecisions;
|
|
|
|
private readonly Logger _logger;
|
|
|
|
|
|
|
|
public AlbumSearchService(ISearchForReleases nzbSearchService,
|
|
|
|
IAlbumService albumService,
|
|
|
|
IAlbumCutoffService albumCutoffService,
|
|
|
|
IQueueService queueService,
|
|
|
|
IProcessDownloadDecisions processDownloadDecisions,
|
|
|
|
Logger logger)
|
|
|
|
{
|
|
|
|
_releaseSearchService = nzbSearchService;
|
|
|
|
_albumService = albumService;
|
|
|
|
_albumCutoffService = albumCutoffService;
|
|
|
|
_queueService = queueService;
|
|
|
|
_processDownloadDecisions = processDownloadDecisions;
|
|
|
|
_logger = logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task SearchForMissingAlbums(List<Album> albums, bool userInvokedSearch)
|
|
|
|
{
|
|
|
|
_logger.ProgressInfo("Performing missing search for {0} albums", albums.Count);
|
|
|
|
var downloadedCount = 0;
|
|
|
|
|
|
|
|
foreach (var album in albums)
|
|
|
|
{
|
|
|
|
List<DownloadDecision> decisions;
|
|
|
|
decisions = await _releaseSearchService.AlbumSearch(album.Id, false, userInvokedSearch, false);
|
|
|
|
var processed = await _processDownloadDecisions.ProcessDecisions(decisions);
|
|
|
|
|
|
|
|
downloadedCount += processed.Grabbed.Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
_logger.ProgressInfo("Completed missing search for {0} albums. {1} reports downloaded.", albums.Count, downloadedCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Execute(AlbumSearchCommand message)
|
|
|
|
{
|
|
|
|
foreach (var albumId in message.AlbumIds)
|
|
|
|
{
|
|
|
|
var decisions = _releaseSearchService.AlbumSearch(albumId, false, message.Trigger == CommandTrigger.Manual, false).GetAwaiter().GetResult();
|
|
|
|
var processed = _processDownloadDecisions.ProcessDecisions(decisions).GetAwaiter().GetResult();
|
|
|
|
|
|
|
|
_logger.ProgressInfo("Album search completed. {0} reports downloaded.", processed.Grabbed.Count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Execute(MissingAlbumSearchCommand message)
|
|
|
|
{
|
|
|
|
List<Album> albums;
|
|
|
|
|
|
|
|
if (message.ArtistId.HasValue)
|
|
|
|
{
|
|
|
|
var artistId = message.ArtistId.Value;
|
|
|
|
|
|
|
|
var pagingSpec = new PagingSpec<Album>
|
|
|
|
{
|
|
|
|
Page = 1,
|
|
|
|
PageSize = 100000,
|
|
|
|
SortDirection = SortDirection.Ascending,
|
|
|
|
SortKey = "Id"
|
|
|
|
};
|
|
|
|
|
|
|
|
pagingSpec.FilterExpressions.Add(v => v.Monitored == true && v.Artist.Value.Monitored == true);
|
|
|
|
|
|
|
|
albums = _albumService.AlbumsWithoutFiles(pagingSpec).Records.Where(e => e.ArtistId.Equals(artistId)).ToList();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var pagingSpec = new PagingSpec<Album>
|
|
|
|
{
|
|
|
|
Page = 1,
|
|
|
|
PageSize = 100000,
|
|
|
|
SortDirection = SortDirection.Ascending,
|
|
|
|
SortKey = "Id"
|
|
|
|
};
|
|
|
|
|
|
|
|
pagingSpec.FilterExpressions.Add(v => v.Monitored == true && v.Artist.Value.Monitored == true);
|
|
|
|
|
|
|
|
albums = _albumService.AlbumsWithoutFiles(pagingSpec).Records.ToList();
|
|
|
|
}
|
|
|
|
|
|
|
|
var queue = _queueService.GetQueue().Where(q => q.Album != null).Select(q => q.Album.Id);
|
|
|
|
var missing = albums.Where(e => !queue.Contains(e.Id)).ToList();
|
|
|
|
|
|
|
|
SearchForMissingAlbums(missing, message.Trigger == CommandTrigger.Manual).GetAwaiter().GetResult();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Execute(CutoffUnmetAlbumSearchCommand message)
|
|
|
|
{
|
|
|
|
Expression<Func<Album, bool>> filterExpression;
|
|
|
|
|
|
|
|
filterExpression = v =>
|
|
|
|
v.Monitored == true &&
|
|
|
|
v.Artist.Value.Monitored == true;
|
|
|
|
|
|
|
|
var pagingSpec = new PagingSpec<Album>
|
|
|
|
{
|
|
|
|
Page = 1,
|
|
|
|
PageSize = 100000,
|
|
|
|
SortDirection = SortDirection.Ascending,
|
|
|
|
SortKey = "Id"
|
|
|
|
};
|
|
|
|
|
|
|
|
pagingSpec.FilterExpressions.Add(filterExpression);
|
|
|
|
|
|
|
|
var albums = _albumCutoffService.AlbumsWhereCutoffUnmet(pagingSpec).Records.ToList();
|
|
|
|
|
|
|
|
var queue = _queueService.GetQueue().Where(q => q.Album != null).Select(q => q.Album.Id);
|
|
|
|
var missing = albums.Where(e => !queue.Contains(e.Id)).ToList();
|
|
|
|
|
|
|
|
SearchForMissingAlbums(missing, message.Trigger == CommandTrigger.Manual).GetAwaiter().GetResult();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|