using System.Collections.Generic; using System.IO; using System.IO.Abstractions; using System.IO.Abstractions.TestingHelpers; 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.TrackImport; using NzbDrone.Core.Music; 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 DownloadedTracksImportServiceFixture : FileSystemTest { private string _droneFactory = "c:\\drop\\".AsOsAgnostic(); private string[] _subFolders = new[] { "c:\\drop\\foldername".AsOsAgnostic() }; private string[] _audioFiles = new[] { "c:\\drop\\foldername\\01 the first track.ext".AsOsAgnostic() }; private TrackedDownload _trackedDownload; [SetUp] public void Setup() { GivenAudioFiles(_audioFiles, 10); Mocker.GetMock().Setup(c => c.GetAudioFiles(It.IsAny(), It.IsAny())) .Returns(_audioFiles.Select(x => DiskProvider.GetFileInfo(x)).ToArray()); Mocker.GetMock().Setup(c => c.FilterFiles(It.IsAny(), It.IsAny>())) .Returns>((b, s) => s.ToList()); 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 remoteAlbum = Builder.CreateNew() .With(v => v.Artist = new Artist()) .Build(); _trackedDownload = new TrackedDownload { DownloadItem = downloadItem, RemoteAlbum = remoteAlbum, State = TrackedDownloadState.Downloading }; } private void GivenAudioFiles(string[] files, long filesize) { foreach (var file in files) { FileSystem.AddFile(file, new MockFileData("".PadRight((int)filesize))); } } private void GivenValidArtist() { Mocker.GetMock() .Setup(s => s.GetArtist(It.IsAny())) .Returns(Builder.CreateNew().Build()); } private void GivenSuccessfulImport() { var localTrack = new LocalTrack(); var imported = new List>(); imported.Add(new ImportDecision(localTrack)); Mocker.GetMock() .Setup(v => v.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny())) .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.GetAudioFiles(It.IsAny(), It.IsAny())) .Returns(System.Array.Empty()); } [Test] public void should_search_for_artist_using_folder_name() { Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); Mocker.GetMock().Verify(c => c.GetArtist("foldername"), Times.Once()); } [Test] public void should_skip_if_file_is_in_use_by_another_process() { GivenValidArtist(); foreach (var file in _audioFiles) { FileSystem.AddFile(file, new MockFileData("".PadRight(10)) { AllowedFileShare = FileShare.None }); } Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); VerifyNoImport(); } [Test] public void should_skip_if_no_artist_found() { Mocker.GetMock().Setup(c => c.GetArtist("foldername")).Returns((Artist)null); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(c => c.GetImportDecisions(It.IsAny>(), 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(System.Array.Empty()); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_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(DiskProvider.GetDirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.GetFolderSize(It.IsAny()), Times.Never()); } [Test] public void should_not_delete_folder_if_files_were_imported_and_audio_files_remain() { GivenValidArtist(); var localTrack = new LocalTrack(); var imported = new List>(); imported.Add(new ImportDecision(localTrack)); Mocker.GetMock() .Setup(v => v.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>>(), true, null, ImportMode.Auto)) .Returns(imported.Select(i => new ImportResult(i)).ToList()); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); ExceptionVerification.ExpectedWarns(1); } [TestCase("_UNPACK_")] [TestCase("_FAILED_")] public void should_remove_unpack_from_folder_name(string prefix) { var folderName = "Alien Ant Farm - Truant (2003)"; FileSystem.AddDirectory(string.Format(@"C:\drop\{0}{1}", prefix, folderName).AsOsAgnostic()); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_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() { var fileName = @"C:\folder\file.mkv".AsOsAgnostic(); FileSystem.AddFile(fileName, new MockFileData(string.Empty)); var result = Subject.ProcessPath(fileName); result.Should().HaveCount(1); result.First().ImportDecision.Should().NotBeNull(); result.First().ImportDecision.Item.Should().NotBeNull(); result.First().ImportDecision.Item.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 localTrack = new LocalTrack(); var imported = new List>(); imported.Add(new ImportDecision(localTrack)); Mocker.GetMock() .Setup(v => v.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>>(), true, null, ImportMode.Auto)) .Returns(imported.Select(i => new ImportResult(i)).ToList()); GivenAudioFiles(new[] { _audioFiles.First().Replace(".ext", ".rar") }, 15.Megabytes()); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); DiskProvider.FolderExists(_subFolders[0]).Should().BeTrue(); ExceptionVerification.ExpectedWarns(1); } [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(); 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 localTrack = new LocalTrack(); var imported = new List>(); imported.Add(new ImportDecision(localTrack)); Mocker.GetMock() .Setup(v => v.GetImportDecisions(It.IsAny>(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(imported); Mocker.GetMock() .Setup(s => s.Import(It.IsAny>>(), true, null, ImportMode.Auto)) .Returns(new List()); Subject.ProcessRootFolder(DiskProvider.GetDirectoryInfo(_droneFactory)); DiskProvider.FolderExists(_subFolders[0]).Should().BeTrue(); Mocker.GetMock() .Verify(v => v.DeleteFolder(It.IsAny(), true), Times.Never()); } [Test] public void should_not_delete_folder_after_import() { GivenValidArtist(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = false; Subject.ProcessPath(_droneFactory, ImportMode.Auto, _trackedDownload.RemoteAlbum.Artist, _trackedDownload.DownloadItem); DiskProvider.FolderExists(_subFolders[0]).Should().BeTrue(); } [Test] public void should_delete_folder_if_importmode_move() { GivenValidArtist(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = false; Subject.ProcessPath(_droneFactory, ImportMode.Move, _trackedDownload.RemoteAlbum.Artist, _trackedDownload.DownloadItem); DiskProvider.FolderExists(_subFolders[0]).Should().BeFalse(); } [Test] public void should_not_delete_folder_if_importmode_copy() { GivenValidArtist(); GivenSuccessfulImport(); _trackedDownload.DownloadItem.CanMoveFiles = true; Subject.ProcessPath(_droneFactory, ImportMode.Copy, _trackedDownload.RemoteAlbum.Artist, _trackedDownload.DownloadItem); DiskProvider.FolderExists(_subFolders[0]).Should().BeTrue(); } 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()); } } }