using System.Collections.Generic; using System.IO; using System.Linq; using FizzWare.NBuilder; using FluentAssertions; using Moq; using NUnit.Framework; using NzbDrone.Common.Disk; using NzbDrone.Core.DecisionEngine; using NzbDrone.Core.Download; using NzbDrone.Core.MediaFiles; using NzbDrone.Core.MediaFiles.EpisodeImport; using NzbDrone.Core.MediaFiles.Events; using NzbDrone.Core.Messaging.Events; using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Profiles.Qualities; using NzbDrone.Core.Qualities; using NzbDrone.Core.Test.Framework; using NzbDrone.Core.Tv; using NzbDrone.Test.Common; using NzbDrone.Core.Languages; using NzbDrone.Core.Profiles.Languages; namespace NzbDrone.Core.Test.MediaFiles.EpisodeImport { [TestFixture] public class ImportApprovedEpisodesFixture : CoreTest { private List _rejectedDecisions; private List _approvedDecisions; private DownloadClientItem _downloadClientItem; [SetUp] public void Setup() { _rejectedDecisions = new List(); _approvedDecisions = new List(); var outputPath = @"C:\Test\Unsorted\TV\30.Rock.S01E01".AsOsAgnostic(); var series = Builder.CreateNew() .With(e => e.QualityProfile = new QualityProfile { Items = Qualities.QualityFixture.GetDefaultQualities() }) .With(l => l.LanguageProfile = new LanguageProfile { Cutoff = Language.Spanish, Languages = Languages.LanguageFixture.GetDefaultLanguages() }) .With(s => s.Path = @"C:\Test\TV\30 Rock".AsOsAgnostic()) .Build(); var episodes = Builder.CreateListOfSize(5) .Build(); _rejectedDecisions.Add(new ImportDecision(new LocalEpisode(), new Rejection("Rejected!"))); _rejectedDecisions.Add(new ImportDecision(new LocalEpisode(), new Rejection("Rejected!"))); _rejectedDecisions.Add(new ImportDecision(new LocalEpisode(), new Rejection("Rejected!"))); foreach (var episode in episodes) { _approvedDecisions.Add(new ImportDecision ( new LocalEpisode { Series = series, Episodes = new List { episode }, Path = Path.Combine(series.Path, "30 Rock - S01E01 - Pilot.avi"), Quality = new QualityModel(Quality.Bluray720p), ReleaseGroup = "DRONE" })); } Mocker.GetMock() .Setup(s => s.UpgradeEpisodeFile(It.IsAny(), It.IsAny(), It.IsAny())) .Returns(new EpisodeFileMoveResult()); _downloadClientItem = Builder.CreateNew() .With(d => d.OutputPath = new OsPath(outputPath)) .Build(); } private void GivenNewDownload() { _approvedDecisions.ForEach(a => a.LocalEpisode.Path = Path.Combine(_downloadClientItem.OutputPath.ToString(), Path.GetFileName(a.LocalEpisode.Path))); } private void GivenExistingFileOnDisk() { Mocker.GetMock() .Setup(s => s.GetFilesWithRelativePath(It.IsAny(), It.IsAny())) .Returns(new List()); } [Test] public void should_not_import_any_if_there_are_no_approved_decisions() { Subject.Import(_rejectedDecisions, false).Where(i => i.Result == ImportResultType.Imported).Should().BeEmpty(); Mocker.GetMock().Verify(v => v.Add(It.IsAny()), Times.Never()); } [Test] public void should_import_each_approved() { GivenExistingFileOnDisk(); Subject.Import(_approvedDecisions, false).Should().HaveCount(5); } [Test] public void should_only_import_approved() { GivenExistingFileOnDisk(); var all = new List(); all.AddRange(_rejectedDecisions); all.AddRange(_approvedDecisions); var result = Subject.Import(all, false); result.Should().HaveCount(all.Count); result.Where(i => i.Result == ImportResultType.Imported).Should().HaveCount(_approvedDecisions.Count); } [Test] public void should_only_import_each_episode_once() { GivenExistingFileOnDisk(); var all = new List(); all.AddRange(_approvedDecisions); all.Add(new ImportDecision(_approvedDecisions.First().LocalEpisode)); var result = Subject.Import(all, false); result.Where(i => i.Result == ImportResultType.Imported).Should().HaveCount(_approvedDecisions.Count); } [Test] public void should_move_new_downloads() { Subject.Import(new List { _approvedDecisions.First() }, true); Mocker.GetMock() .Verify(v => v.UpgradeEpisodeFile(It.IsAny(), _approvedDecisions.First().LocalEpisode, false), Times.Once()); } [Test] public void should_publish_EpisodeImportedEvent_for_new_downloads() { Subject.Import(new List { _approvedDecisions.First() }, true); Mocker.GetMock() .Verify(v => v.PublishEvent(It.IsAny()), Times.Once()); } [Test] public void should_not_move_existing_files() { GivenExistingFileOnDisk(); Subject.Import(new List { _approvedDecisions.First() }, false); Mocker.GetMock() .Verify(v => v.UpgradeEpisodeFile(It.IsAny(), _approvedDecisions.First().LocalEpisode, false), Times.Never()); } [Test] public void should_import_larger_files_first() { GivenExistingFileOnDisk(); var fileDecision = _approvedDecisions.First(); fileDecision.LocalEpisode.Size = 1.Gigabytes(); var sampleDecision = new ImportDecision (new LocalEpisode { Series = fileDecision.LocalEpisode.Series, Episodes = new List { fileDecision.LocalEpisode.Episodes.First() }, Path = @"C:\Test\TV\30 Rock\30 Rock - S01E01 - Pilot.avi".AsOsAgnostic(), Quality = new QualityModel(Quality.Bluray720p), Size = 80.Megabytes() }); var all = new List(); all.Add(fileDecision); all.Add(sampleDecision); var results = Subject.Import(all, false); results.Should().HaveCount(all.Count); results.Should().ContainSingle(d => d.Result == ImportResultType.Imported); results.Should().ContainSingle(d => d.Result == ImportResultType.Imported && d.ImportDecision.LocalEpisode.Size == fileDecision.LocalEpisode.Size); } [Test] public void should_copy_when_cannot_move_files_downloads() { GivenNewDownload(); _downloadClientItem.Title = "30.Rock.S01E01"; _downloadClientItem.CanMoveFiles = false; Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock() .Verify(v => v.UpgradeEpisodeFile(It.IsAny(), _approvedDecisions.First().LocalEpisode, true), Times.Once()); } [Test] public void should_use_override_importmode() { GivenNewDownload(); _downloadClientItem.Title = "30.Rock.S01E01"; _downloadClientItem.CanMoveFiles = false; Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem, ImportMode.Move); Mocker.GetMock() .Verify(v => v.UpgradeEpisodeFile(It.IsAny(), _approvedDecisions.First().LocalEpisode, false), Times.Once()); } [Test] public void should_use_file_name_only_for_download_client_item_without_a_job_folder() { var fileName = "Series.Title.S01E01.720p.HDTV.x264-Sonarr.mkv"; var path = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), fileName); _downloadClientItem.OutputPath = new OsPath(path); _approvedDecisions.First().LocalEpisode.Path = path; Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == fileName))); } [Test] public void should_use_folder_and_file_name_only_for_download_client_item_with_a_job_folder() { var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), name); _downloadClientItem.OutputPath = new OsPath(outputPath); _approvedDecisions.First().LocalEpisode.Path = Path.Combine(outputPath, name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}\\{name}.mkv".AsOsAgnostic()))); } [Test] public void should_include_intermediate_folders_for_download_client_item_with_a_job_folder() { var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), name); _downloadClientItem.OutputPath = new OsPath(outputPath); _approvedDecisions.First().LocalEpisode.Path = Path.Combine(outputPath, "subfolder", name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}\\subfolder\\{name}.mkv".AsOsAgnostic()))); } [Test] public void should_use_folder_info_release_title_to_find_relative_path() { var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), name); var localEpisode = _approvedDecisions.First().LocalEpisode; localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, "subfolder", name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, null); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}\\subfolder\\{name}.mkv".AsOsAgnostic()))); } [Test] public void should_get_relative_path_when_there_is_no_grandparent_windows() { WindowsOnly(); var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = @"C:\"; var localEpisode = _approvedDecisions.First().LocalEpisode; localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, null); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}.mkv".AsOsAgnostic()))); } [Test] public void should_get_relative_path_when_there_is_no_grandparent_mono() { PosixOnly(); var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = "/"; var localEpisode = _approvedDecisions.First().LocalEpisode; localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, null); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}.mkv".AsOsAgnostic()))); } [Test] public void should_get_relative_path_when_there_is_no_grandparent_for_UNC_path() { WindowsOnly(); var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = @"\\server\share"; var localEpisode = _approvedDecisions.First().LocalEpisode; localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, null); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}.mkv"))); } [Test] public void should_use_folder_info_release_title_to_find_relative_path_when_file_is_not_in_download_client_item_output_directory() { var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), name); var localEpisode = _approvedDecisions.First().LocalEpisode; _downloadClientItem.OutputPath = new OsPath(Path.Combine(@"C:\Test\Unsorted\TV-Other\".AsOsAgnostic(), name)); localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, "subfolder", name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}\\subfolder\\{name}.mkv".AsOsAgnostic()))); } [Test] public void should_delete_existing_metadata_files_with_the_same_path() { Mocker.GetMock() .Setup(s => s.GetFilesWithRelativePath(It.IsAny(), It.IsAny())) .Returns(Builder.CreateListOfSize(1).BuildList()); Subject.Import(new List { _approvedDecisions.First() }, false); Mocker.GetMock() .Verify(v => v.Delete(It.IsAny(), DeleteMediaFileReason.ManualOverride), Times.Once()); } [Test] public void should_use_folder_info_release_title_to_find_relative_path_when_download_client_item_has_an_empty_output_path() { var name = "Series.Title.S01E01.720p.HDTV.x264-Sonarr"; var outputPath = Path.Combine(@"C:\Test\Unsorted\TV\".AsOsAgnostic(), name); var localEpisode = _approvedDecisions.First().LocalEpisode; _downloadClientItem.OutputPath = new OsPath(); localEpisode.FolderEpisodeInfo = new ParsedEpisodeInfo { ReleaseTitle = name }; localEpisode.Path = Path.Combine(outputPath, "subfolder", name + ".mkv"); Subject.Import(new List { _approvedDecisions.First() }, true, _downloadClientItem); Mocker.GetMock().Verify(v => v.Add(It.Is(c => c.OriginalFilePath == $"{name}\\subfolder\\{name}.mkv".AsOsAgnostic()))); } [Test] public void should_include_scene_name_with_new_downloads() { var firstDecision = _approvedDecisions.First(); firstDecision.LocalEpisode.SceneName = "Series.Title.S01E01.dvdrip-DRONE"; Subject.Import(new List { _approvedDecisions.First() }, true); Mocker.GetMock() .Verify(v => v.UpgradeEpisodeFile(It.Is(e => e.SceneName == firstDecision.LocalEpisode.SceneName), _approvedDecisions.First().LocalEpisode, false), Times.Once()); } } }