using System.Collections.Generic; using System.IO; using System.Linq; using FizzWare.NBuilder; using Moq; using NUnit.Framework; using NzbDrone.Common.Disk; using NzbDrone.Core.MediaFiles; using NzbDrone.Core.MediaFiles.TrackImport; using NzbDrone.Core.Parser; using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Qualities; using NzbDrone.Core.Test.Framework; using NzbDrone.Core.Music; using NzbDrone.Test.Common; using FluentAssertions; namespace NzbDrone.Core.Test.MediaFiles { [TestFixture] public class DownloadedTracksImportServiceFixture : CoreTest { private string _droneFactory = "c:\\drop\\".AsOsAgnostic(); private string[] _subFolders = new[] { "c:\\root\\foldername".AsOsAgnostic() }; private string[] _videoFiles = new[] { "c:\\root\\foldername\\30.rock.s01e01.ext".AsOsAgnostic() }; [SetUp] public void Setup() { Mocker.GetMock().Setup(c => c.GetAudioFiles(It.IsAny(), It.IsAny())) .Returns(_videoFiles); Mocker.GetMock().Setup(c => c.GetDirectories(It.IsAny())) .Returns(_subFolders); Mocker.GetMock().Setup(c => c.FolderExists(It.IsAny())) .Returns(true); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), true, null, ImportMode.Auto)) .Returns(new List()); } private void GivenValidArtist() { Mocker.GetMock() .Setup(s => s.GetArtist(It.IsAny())) .Returns(Builder.CreateNew().Build()); } [Test] public void should_search_for_artist_using_folder_name() { Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock().Verify(c => c.GetArtist("foldername"), Times.Once()); } [Test] public void should_skip_if_file_is_in_use_by_another_process() { GivenValidArtist(); Mocker.GetMock().Setup(c => c.IsFileLocked(It.IsAny())) .Returns(true); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); VerifyNoImport(); } [Test] public void should_skip_if_no_artist_found() { Mocker.GetMock().Setup(c => c.GetArtist("foldername")).Returns((Artist)null); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(c => c.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny()), Times.Never()); VerifyNoImport(); } [Test] public void should_not_import_if_folder_is_a_artist_path() { GivenValidArtist(); Mocker.GetMock() .Setup(s => s.ArtistPathExists(It.IsAny())) .Returns(true); Mocker.GetMock() .Setup(c => c.GetAudioFiles(It.IsAny(), It.IsAny())) .Returns(new string[0]); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.GetAudioFiles(It.IsAny(), true), Times.Never()); ExceptionVerification.ExpectedWarns(1); } [Test] public void should_not_delete_folder_if_no_files_were_imported() { Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), false, null, ImportMode.Auto)) .Returns(new List()); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.GetFolderSize(It.IsAny()), Times.Never()); } [Test] public void should_not_delete_folder_if_files_were_imported_and_video_files_remain() { GivenValidArtist(); var localTrack = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localTrack)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), null)) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), true, null, ImportMode.Auto)) .Returns(imported.Select(i => new ImportResult(i)).ToList()); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); ExceptionVerification.ExpectedWarns(1); } [Test] public void should_delete_folder_if_files_were_imported_and_only_sample_files_remain() { GivenValidArtist(); var localEpisode = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localEpisode)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), null)) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), true, null, ImportMode.Auto)) .Returns(imported.Select(i => new ImportResult(i)).ToList()); //Mocker.GetMock() // .Setup(s => s.IsSample(It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny())) // .Returns(true); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Once()); } [TestCase("_UNPACK_")] [TestCase("_FAILED_")] public void should_remove_unpack_from_folder_name(string prefix) { var folderName = "30.rock.s01e01.pilot.hdtv-lol"; var folders = new[] { string.Format(@"C:\Test\Unsorted\{0}{1}", prefix, folderName).AsOsAgnostic() }; Mocker.GetMock() .Setup(c => c.GetDirectories(It.IsAny())) .Returns(folders); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.GetArtist(folderName), Times.Once()); Mocker.GetMock() .Verify(v => v.GetArtist(It.Is(s => s.StartsWith(prefix))), Times.Never()); } [Test] public void should_return_importresult_on_unknown_artist() { Mocker.GetMock().Setup(c => c.FolderExists(It.IsAny())) .Returns(false); Mocker.GetMock().Setup(c => c.FileExists(It.IsAny())) .Returns(true); var fileName = @"C:\folder\file.mkv".AsOsAgnostic(); var result = Subject.ProcessPath(fileName); result.Should().HaveCount(1); result.First().ImportDecision.Should().NotBeNull(); result.First().ImportDecision.LocalTrack.Should().NotBeNull(); result.First().ImportDecision.LocalTrack.Path.Should().Be(fileName); result.First().Result.Should().Be(ImportResultType.Rejected); } [Test] public void should_not_delete_if_there_is_large_rar_file() { GivenValidArtist(); var localEpisode = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localEpisode)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), null)) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), true, null, ImportMode.Auto)) .Returns(imported.Select(i => new ImportResult(i)).ToList()); //Mocker.GetMock() // .Setup(s => s.IsSample(It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny())) // .Returns(true); Mocker.GetMock() .Setup(s => s.GetFiles(It.IsAny(), SearchOption.AllDirectories)) .Returns(new []{ _videoFiles.First().Replace(".ext", ".rar") }); Mocker.GetMock() .Setup(s => s.GetFileSize(It.IsAny())) .Returns(15.Megabytes()); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); ExceptionVerification.ExpectedWarns(1); } [Test] public void should_use_folder_if_folder_import() { GivenValidArtist(); var folderName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\[HorribleSubs] Maria the Virgin Witch - 09 [720p]".AsOsAgnostic(); var fileName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\[HorribleSubs] Maria the Virgin Witch - 09 [720p]\[HorribleSubs] Maria the Virgin Witch - 09 [720p].mkv".AsOsAgnostic(); Mocker.GetMock().Setup(c => c.FolderExists(folderName)) .Returns(true); Mocker.GetMock().Setup(c => c.GetFiles(folderName, SearchOption.TopDirectoryOnly)) .Returns(new[] { fileName }); var localEpisode = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localEpisode)); Subject.ProcessPath(fileName); Mocker.GetMock() .Verify(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.Is(v => v.TrackNumbers.First() == 9)), Times.Once()); } [Test] public void should_not_use_folder_if_file_import() { GivenValidArtist(); var fileName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\Torrents\[HorribleSubs] Maria the Virgin Witch - 09 [720p].mkv".AsOsAgnostic(); Mocker.GetMock().Setup(c => c.FolderExists(fileName)) .Returns(false); Mocker.GetMock().Setup(c => c.FileExists(fileName)) .Returns(true); var localEpisode = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localEpisode)); var result = Subject.ProcessPath(fileName); Mocker.GetMock() .Verify(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), null), Times.Once()); } [Test] public void should_not_process_if_file_and_folder_do_not_exist() { var folderName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\[HorribleSubs] Maria the Virgin Witch - 09 [720p]".AsOsAgnostic(); Mocker.GetMock().Setup(c => c.FolderExists(folderName)) .Returns(false); Mocker.GetMock().Setup(c => c.FileExists(folderName)) .Returns(false); Subject.ProcessPath(folderName).Should().BeEmpty(); Mocker.GetMock() .Verify(v => v.GetArtist(It.IsAny()), Times.Never()); ExceptionVerification.ExpectedErrors(1); } [Test] public void should_not_delete_if_no_files_were_imported() { GivenValidArtist(); var localEpisode = new LocalTrack(); var imported = new List(); imported.Add(new ImportDecision(localEpisode)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), null)) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), true, null, ImportMode.Auto)) .Returns(new List()); //Mocker.GetMock() // .Setup(s => s.IsSample(It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny(), // It.IsAny())) // .Returns(true); Mocker.GetMock() .Setup(s => s.GetFileSize(It.IsAny())) .Returns(15.Megabytes()); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); } private void VerifyNoImport() { Mocker.GetMock().Verify(c => c.Import(It.IsAny>(), true, null, ImportMode.Auto), Times.Never()); } private void VerifyImport() { Mocker.GetMock().Verify(c => c.Import(It.IsAny>(), true, null, ImportMode.Auto), Times.Once()); } } }