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.Download; using NzbDrone.Core.Download.TrackedDownloads; using NzbDrone.Core.MediaFiles; using NzbDrone.Core.MediaFiles.MovieImport; using NzbDrone.Core.Movies; using NzbDrone.Core.Parser; using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Test.Framework; using NzbDrone.Test.Common; namespace NzbDrone.Core.Test.MediaFiles { [TestFixture] public class DownloadedMoviesImportServiceFixture : CoreTest { private string _droneFactory = "c:\\drop\\".AsOsAgnostic(); private string[] _subFolders = new[] { "c:\\root\\foldername".AsOsAgnostic() }; private string[] _videoFiles = new[] { "c:\\root\\foldername\\47.ronin.2013.ext".AsOsAgnostic() }; private TrackedDownload _trackedDownload; [SetUp] public void Setup() { ParseMovieTitle(); // UseRealParsingService(); Mocker.GetMock().Setup(c => c.GetVideoFiles(It.IsAny(), It.IsAny())) .Returns(_videoFiles); Mocker.GetMock().Setup(c => c.FilterPaths(It.IsAny(), It.IsAny>(), It.IsAny())) .Returns, bool>((b, s, e) => s.ToList()); 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()); var downloadItem = Builder.CreateNew() .With(v => v.DownloadId = "sab1") .With(v => v.Status = DownloadItemStatus.Downloading) .Build(); var remoteMovie = Builder.CreateNew() .With(v => v.Movie = new Movie()) .Build(); _trackedDownload = new TrackedDownload { DownloadItem = downloadItem, RemoteMovie = remoteMovie, State = TrackedDownloadState.Downloading }; } private void GivenValidMovie() { Mocker.GetMock() .Setup(s => s.GetMovie(It.IsAny())) .Returns(Builder.CreateNew().Build()); } private void GivenSuccessfulImport() { var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true, true)) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(imported.Select(i => new ImportResult(i)).ToList()) .Callback(() => WasImportedResponse()); } private void WasImportedResponse() { Mocker.GetMock().Setup(c => c.GetVideoFiles(It.IsAny(), It.IsAny())) .Returns(System.Array.Empty()); } [Test] public void should_search_for_series_using_folder_name() { Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock().Verify(c => c.GetMovie("foldername"), Times.Once()); } [Test] public void should_skip_if_file_is_in_use_by_another_process() { GivenValidMovie(); Mocker.GetMock().Setup(c => c.IsFileLocked(It.IsAny())) .Returns(true); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); VerifyNoImport(); } [Test] public void should_skip_if_no_series_found() { Mocker.GetMock().Setup(c => c.GetMovie("foldername")).Returns((Movie)null); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(c => c.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny()), Times.Never()); VerifyNoImport(); } [Test] public void should_not_import_if_folder_is_a_series_path() { GivenValidMovie(); Mocker.GetMock() .Setup(s => s.MoviePathExists(It.IsAny())) .Returns(true); Mocker.GetMock() .Setup(c => c.GetVideoFiles(It.IsAny(), It.IsAny())) .Returns(System.Array.Empty()); Subject.ProcessRootFolder(new DirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.GetVideoFiles(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_after_import() { GivenValidMovie(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = false; Subject.ProcessPath(_droneFactory, ImportMode.Auto, _trackedDownload.RemoteMovie.Movie, _trackedDownload.DownloadItem); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); } [Test] public void should_delete_folder_if_importmode_move() { GivenValidMovie(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = false; Subject.ProcessPath(_droneFactory, ImportMode.Move, _trackedDownload.RemoteMovie.Movie, _trackedDownload.DownloadItem); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Once()); } [Test] public void should_not_delete_folder_if_importmode_copy() { GivenValidMovie(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = true; Subject.ProcessPath(_droneFactory, ImportMode.Copy, _trackedDownload.RemoteMovie.Movie, _trackedDownload.DownloadItem); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); } [Test] public void should_not_delete_folder_if_files_were_imported_and_video_files_remain() { GivenValidMovie(); var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true)) .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() { GivenValidMovie(); var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true)) .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())) .Returns(DetectSampleResult.Sample); 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 = "47.ronin.2013.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.GetMovie(folderName), Times.Once()); Mocker.GetMock() .Verify(v => v.GetMovie(It.Is(s => s.StartsWith(prefix))), Times.Never()); } [Test] public void should_return_importresult_on_unknown_movie() { 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.LocalMovie.Should().NotBeNull(); result.First().ImportDecision.LocalMovie.Path.Should().Be(fileName); result.First().Result.Should().Be(ImportResultType.Rejected); } [Test] public void should_not_delete_if_there_is_large_rar_file() { GivenValidMovie(); var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true)) .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())) .Returns(DetectSampleResult.Sample); 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() { GivenValidMovie(); var folderName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\[HorribleSubs] American Psycho (2000) [720p]".AsOsAgnostic(); var fileName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\[HorribleSubs] American Psycho (2000) [720p]\[HorribleSubs] American Psycho (2000) [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 localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Subject.ProcessPath(fileName); Mocker.GetMock() .Verify(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny(), true), Times.Once()); } [Test] public void should_not_use_folder_if_file_import() { GivenValidMovie(); var fileName = @"C:\media\ba09030e-1234-1234-1234-123456789abc\Torrents\[HorribleSubs] 47 Ronin (2013) [720p].mkv".AsOsAgnostic(); Mocker.GetMock().Setup(c => c.FolderExists(fileName)) .Returns(false); Mocker.GetMock().Setup(c => c.FileExists(fileName)) .Returns(true); var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); var result = Subject.ProcessPath(fileName); Mocker.GetMock() .Verify(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true), 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] 47 Ronin (2013) [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.GetMovie(It.IsAny()), Times.Never()); ExceptionVerification.ExpectedErrors(1); } [Test] public void should_not_delete_if_no_files_were_imported() { GivenValidMovie(); var localMovie = new LocalMovie(); var imported = new List(); imported.Add(new ImportDecision(localMovie)); Mocker.GetMock() .Setup(s => s.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), null, true)) .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())) .Returns(DetectSampleResult.Sample); 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()); } } }