using System.Collections.Generic; using System.Linq; using FizzWare.NBuilder; using FluentAssertions; using Moq; using NUnit.Framework; using NzbDrone.Core.Books; using NzbDrone.Core.Datastore; using NzbDrone.Core.DecisionEngine; using NzbDrone.Core.DecisionEngine.Specifications; using NzbDrone.Core.IndexerSearch.Definitions; using NzbDrone.Core.Parser; using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Test.Framework; using NzbDrone.Test.Common; namespace NzbDrone.Core.Test.DecisionEngineTests { [TestFixture] public class DownloadDecisionMakerFixture : CoreTest { private List _reports; private RemoteBook _remoteBook; private Mock _pass1; private Mock _pass2; private Mock _pass3; private Mock _fail1; private Mock _fail2; private Mock _fail3; private Mock _failDelayed1; [SetUp] public void Setup() { _pass1 = new Mock(); _pass2 = new Mock(); _pass3 = new Mock(); _fail1 = new Mock(); _fail2 = new Mock(); _fail3 = new Mock(); _failDelayed1 = new Mock(); _pass1.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Accept); _pass2.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Accept); _pass3.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Accept); _fail1.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Reject("fail1")); _fail2.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Reject("fail2")); _fail3.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Reject("fail3")); _failDelayed1.Setup(c => c.IsSatisfiedBy(It.IsAny(), null)).Returns(Decision.Reject("failDelayed1")); _failDelayed1.SetupGet(c => c.Priority).Returns(SpecificationPriority.Disk); _reports = new List { new ReleaseInfo { Title = "Coldplay-A Head Full Of Dreams-CD-FLAC-2015-PERFECT" } }; _remoteBook = new RemoteBook { Author = new Author(), Books = new List { new Book() } }; Mocker.GetMock() .Setup(c => c.Map(It.IsAny(), It.IsAny())) .Returns(_remoteBook); } private void GivenSpecifications(params Mock[] mocks) { Mocker.SetConstant>(mocks.Select(c => c.Object)); } [Test] public void should_call_all_specifications() { GivenSpecifications(_pass1, _pass2, _pass3, _fail1, _fail2, _fail3); Subject.GetRssDecision(_reports).ToList(); _fail1.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); _fail2.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); _fail3.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); _pass1.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); _pass2.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); _pass3.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); } [Test] public void should_call_delayed_specifications_if_non_delayed_passed() { GivenSpecifications(_pass1, _failDelayed1); Subject.GetRssDecision(_reports).ToList(); _failDelayed1.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Once()); } [Test] public void should_not_call_delayed_specifications_if_non_delayed_failed() { GivenSpecifications(_fail1, _failDelayed1); Subject.GetRssDecision(_reports).ToList(); _failDelayed1.Verify(c => c.IsSatisfiedBy(_remoteBook, null), Times.Never()); } [Test] public void should_return_rejected_if_single_specs_fail() { GivenSpecifications(_fail1); var result = Subject.GetRssDecision(_reports); result.Single().Approved.Should().BeFalse(); } [Test] public void should_return_rejected_if_one_of_specs_fail() { GivenSpecifications(_pass1, _fail1, _pass2, _pass3); var result = Subject.GetRssDecision(_reports); result.Single().Approved.Should().BeFalse(); } [Test] public void should_return_pass_if_all_specs_pass() { GivenSpecifications(_pass1, _pass2, _pass3); var result = Subject.GetRssDecision(_reports); result.Single().Approved.Should().BeTrue(); } [Test] public void should_have_same_number_of_rejections_as_specs_that_failed() { GivenSpecifications(_pass1, _pass2, _pass3, _fail1, _fail2, _fail3); var result = Subject.GetRssDecision(_reports); result.Single().Rejections.Should().HaveCount(3); } [Test] public void should_not_attempt_to_map_book_if_not_parsable() { GivenSpecifications(_pass1, _pass2, _pass3); _reports[0].Title = "Not parsable"; Subject.GetRssDecision(_reports).ToList(); Mocker.GetMock().Verify(c => c.Map(It.IsAny(), It.IsAny()), Times.Never()); _pass1.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass2.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass3.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); } [Test] public void should_not_attempt_to_map_book_if_author_title_is_blank() { GivenSpecifications(_pass1, _pass2, _pass3); _reports[0].Title = "2013 - Night Visions"; var results = Subject.GetRssDecision(_reports).ToList(); Mocker.GetMock().Verify(c => c.Map(It.IsAny(), It.IsAny()), Times.Never()); _pass1.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass2.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass3.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); results.Should().BeEmpty(); } [Test] public void should_return_rejected_result_for_unparsable_search() { GivenSpecifications(_pass1, _pass2, _pass3); _reports[0].Title = "1937 - Snow White and the Seven Dwarves"; var author = new Author { Name = "Some Author" }; var books = new List { new Book { Title = "Some Book" } }; Subject.GetSearchDecision(_reports, new BookSearchCriteria { Author = author, Books = books }).ToList(); Mocker.GetMock().Verify(c => c.Map(It.IsAny(), It.IsAny()), Times.Never()); _pass1.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass2.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass3.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); } [Test] public void should_not_attempt_to_make_decision_if_author_is_unknown() { GivenSpecifications(_pass1, _pass2, _pass3); _remoteBook.Author = null; Subject.GetRssDecision(_reports); _pass1.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass2.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); _pass3.Verify(c => c.IsSatisfiedBy(It.IsAny(), null), Times.Never()); } [Test] public void broken_report_shouldnt_blowup_the_process() { GivenSpecifications(_pass1); Mocker.GetMock().Setup(c => c.Map(It.IsAny(), It.IsAny())) .Throws(); _reports = new List { new ReleaseInfo { Title = "Coldplay-A Head Full Of Dreams-CD-FLAC-2015-PERFECT" }, new ReleaseInfo { Title = "Coldplay-A Head Full Of Dreams-CD-FLAC-2015-PERFECT" }, new ReleaseInfo { Title = "Coldplay-A Head Full Of Dreams-CD-FLAC-2015-PERFECT" } }; Subject.GetRssDecision(_reports); Mocker.GetMock().Verify(c => c.Map(It.IsAny(), It.IsAny()), Times.Exactly(_reports.Count)); ExceptionVerification.ExpectedErrors(3); } [Test] public void should_return_unknown_author_rejection_if_author_is_unknown() { GivenSpecifications(_pass1, _pass2, _pass3); _remoteBook.Author = null; var result = Subject.GetRssDecision(_reports); result.Should().HaveCount(1); } [Test] public void should_only_include_reports_for_requested_books() { var author = Builder.CreateNew().Build(); var books = Builder.CreateListOfSize(2) .All() .With(v => v.AuthorId, author.Id) .With(v => v.Author, new LazyLoaded(author)) .BuildList(); var criteria = new AuthorSearchCriteria { Books = books.Take(1).ToList() }; var reports = books.Select(v => new ReleaseInfo() { Title = string.Format("{0}-{1}[FLAC][2017][DRONE]", author.Name, v.Title) }).ToList(); Mocker.GetMock() .Setup(v => v.Map(It.IsAny(), It.IsAny())) .Returns((p, c) => new RemoteBook { DownloadAllowed = true, ParsedBookInfo = p, Author = author, Books = books.Where(v => v.Title == p.BookTitle).ToList() }); Mocker.SetConstant>(new List { Mocker.Resolve() }); var decisions = Subject.GetSearchDecision(reports, criteria); var approvedDecisions = decisions.Where(v => v.Approved).ToList(); approvedDecisions.Count.Should().Be(1); } [Test] public void should_not_allow_download_if_author_is_unknown() { GivenSpecifications(_pass1, _pass2, _pass3); _remoteBook.Author = null; var result = Subject.GetRssDecision(_reports); result.Should().HaveCount(1); result.First().RemoteBook.DownloadAllowed.Should().BeFalse(); } [Test] public void should_not_allow_download_if_no_books_found() { GivenSpecifications(_pass1, _pass2, _pass3); _remoteBook.Books = new List(); var result = Subject.GetRssDecision(_reports); result.Should().HaveCount(1); result.First().RemoteBook.DownloadAllowed.Should().BeFalse(); } [Test] public void should_return_a_decision_when_exception_is_caught() { GivenSpecifications(_pass1); Mocker.GetMock().Setup(c => c.Map(It.IsAny(), It.IsAny())) .Throws(); _reports = new List { new ReleaseInfo { Title = "Alien Ant Farm - TruAnt (FLAC) DRONE" }, }; Subject.GetRssDecision(_reports).Should().HaveCount(1); ExceptionVerification.ExpectedErrors(1); } } }