mirror of
https://github.com/Radarr/Radarr.git
synced 2024-09-17 15:02:34 +02:00
Fixed: Performance issue when scanning large root folder
This commit is contained in:
parent
942d239092
commit
4606503818
@ -142,6 +142,13 @@ public bool FolderWritable(string path)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool FolderEmpty(string path)
|
||||||
|
{
|
||||||
|
Ensure.That(path, () => path).IsValidPath();
|
||||||
|
|
||||||
|
return Directory.EnumerateDirectories(path).Empty();
|
||||||
|
}
|
||||||
|
|
||||||
public string[] GetDirectories(string path)
|
public string[] GetDirectories(string path)
|
||||||
{
|
{
|
||||||
Ensure.That(path, () => path).IsValidPath();
|
Ensure.That(path, () => path).IsValidPath();
|
||||||
|
@ -21,6 +21,7 @@ public interface IDiskProvider
|
|||||||
bool FileExists(string path);
|
bool FileExists(string path);
|
||||||
bool FileExists(string path, StringComparison stringComparison);
|
bool FileExists(string path, StringComparison stringComparison);
|
||||||
bool FolderWritable(string path);
|
bool FolderWritable(string path);
|
||||||
|
bool FolderEmpty(string path);
|
||||||
string[] GetDirectories(string path);
|
string[] GetDirectories(string path);
|
||||||
string[] GetFiles(string path, SearchOption searchOption);
|
string[] GetFiles(string path, SearchOption searchOption);
|
||||||
long GetFolderSize(string path);
|
long GetFolderSize(string path);
|
||||||
|
@ -5,9 +5,12 @@
|
|||||||
using Moq;
|
using Moq;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
using NzbDrone.Common.Disk;
|
using NzbDrone.Common.Disk;
|
||||||
|
using NzbDrone.Common.Extensions;
|
||||||
|
using NzbDrone.Core.Configuration;
|
||||||
using NzbDrone.Core.MediaFiles;
|
using NzbDrone.Core.MediaFiles;
|
||||||
using NzbDrone.Core.MediaFiles.MovieImport;
|
using NzbDrone.Core.MediaFiles.MovieImport;
|
||||||
using NzbDrone.Core.Movies;
|
using NzbDrone.Core.Movies;
|
||||||
|
using NzbDrone.Core.RootFolders;
|
||||||
using NzbDrone.Core.Test.Framework;
|
using NzbDrone.Core.Test.Framework;
|
||||||
using NzbDrone.Test.Common;
|
using NzbDrone.Test.Common;
|
||||||
|
|
||||||
@ -17,28 +20,58 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
|
|||||||
public class ScanFixture : CoreTest<DiskScanService>
|
public class ScanFixture : CoreTest<DiskScanService>
|
||||||
{
|
{
|
||||||
private Movie _movie;
|
private Movie _movie;
|
||||||
|
private string _rootFolder;
|
||||||
|
private string _otherMovieFolder;
|
||||||
|
|
||||||
[SetUp]
|
[SetUp]
|
||||||
public void Setup()
|
public void Setup()
|
||||||
{
|
{
|
||||||
|
_rootFolder = @"C:\Test\Movies".AsOsAgnostic();
|
||||||
|
_otherMovieFolder = @"C:\Test\Movies\OtherMovie".AsOsAgnostic();
|
||||||
|
var movieFolder = @"C:\Test\Movies\Movie".AsOsAgnostic();
|
||||||
|
|
||||||
_movie = Builder<Movie>.CreateNew()
|
_movie = Builder<Movie>.CreateNew()
|
||||||
.With(s => s.Path = @"C:\Test\Movies\Movie".AsOsAgnostic())
|
.With(s => s.Path = movieFolder)
|
||||||
.Build();
|
.Build();
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(s => s.FolderExists(It.IsAny<string>()))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<IDiskProvider>()
|
||||||
.Setup(s => s.GetParentFolder(It.IsAny<string>()))
|
.Setup(s => s.GetParentFolder(It.IsAny<string>()))
|
||||||
.Returns((string path) => Directory.GetParent(path).FullName);
|
.Returns((string path) => Directory.GetParent(path).FullName);
|
||||||
|
|
||||||
|
Mocker.GetMock<IRootFolderService>()
|
||||||
|
.Setup(s => s.GetBestRootFolderPath(It.IsAny<string>()))
|
||||||
|
.Returns(_rootFolder);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void GivenParentFolderExists()
|
private void GivenRootFolder(params string[] subfolders)
|
||||||
{
|
{
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<IDiskProvider>()
|
||||||
.Setup(s => s.FolderExists(It.IsAny<string>()))
|
.Setup(s => s.FolderExists(_rootFolder))
|
||||||
.Returns(true);
|
.Returns(true);
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
Mocker.GetMock<IDiskProvider>()
|
||||||
.Setup(s => s.GetDirectories(It.IsAny<string>()))
|
.Setup(s => s.GetDirectories(_rootFolder))
|
||||||
.Returns(new string[] { @"C:\Test\Movies\Movie2".AsOsAgnostic() });
|
.Returns(subfolders);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(s => s.FolderEmpty(_rootFolder))
|
||||||
|
.Returns(subfolders.Empty());
|
||||||
|
|
||||||
|
foreach (var folder in subfolders)
|
||||||
|
{
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Setup(s => s.FolderExists(folder))
|
||||||
|
.Returns(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void GivenMovieFolder()
|
||||||
|
{
|
||||||
|
GivenRootFolder(_movie.Path);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void GivenFiles(IEnumerable<string> files)
|
private void GivenFiles(IEnumerable<string> files)
|
||||||
@ -55,6 +88,12 @@ public void should_not_scan_if_movie_root_folder_does_not_exist()
|
|||||||
|
|
||||||
ExceptionVerification.ExpectedWarns(1);
|
ExceptionVerification.ExpectedWarns(1);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.GetFiles(_movie.Path, SearchOption.AllDirectories), Times.Never());
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.CreateFolder(_movie.Path), Times.Never());
|
||||||
|
|
||||||
Mocker.GetMock<IMediaFileTableCleanupService>()
|
Mocker.GetMock<IMediaFileTableCleanupService>()
|
||||||
.Verify(v => v.Clean(It.IsAny<Movie>(), It.IsAny<List<string>>()), Times.Never());
|
.Verify(v => v.Clean(It.IsAny<Movie>(), It.IsAny<List<string>>()), Times.Never());
|
||||||
}
|
}
|
||||||
@ -62,26 +101,44 @@ public void should_not_scan_if_movie_root_folder_does_not_exist()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_if_movie_root_folder_is_empty()
|
public void should_not_scan_if_movie_root_folder_is_empty()
|
||||||
{
|
{
|
||||||
Mocker.GetMock<IDiskProvider>()
|
GivenRootFolder();
|
||||||
.Setup(s => s.FolderExists(It.IsAny<string>()))
|
|
||||||
.Returns(true);
|
|
||||||
|
|
||||||
Mocker.GetMock<IDiskProvider>()
|
|
||||||
.Setup(s => s.GetDirectories(It.IsAny<string>()))
|
|
||||||
.Returns(new string[0]);
|
|
||||||
|
|
||||||
Subject.Scan(_movie);
|
Subject.Scan(_movie);
|
||||||
|
|
||||||
ExceptionVerification.ExpectedWarns(1);
|
ExceptionVerification.ExpectedWarns(1);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.GetFiles(_movie.Path, SearchOption.AllDirectories), Times.Never());
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.CreateFolder(_movie.Path), Times.Never());
|
||||||
|
|
||||||
Mocker.GetMock<IMediaFileTableCleanupService>()
|
Mocker.GetMock<IMediaFileTableCleanupService>()
|
||||||
.Verify(v => v.Clean(It.IsAny<Movie>(), new List<string>()), Times.Never());
|
.Verify(v => v.Clean(It.IsAny<Movie>(), It.IsAny<List<string>>()), Times.Never());
|
||||||
|
|
||||||
|
Mocker.GetMock<IMakeImportDecision>()
|
||||||
|
.Verify(v => v.GetImportDecisions(It.IsAny<List<string>>(), _movie), Times.Never());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_create_if_movie_folder_does_not_exist_but_create_folder_enabled()
|
||||||
|
{
|
||||||
|
GivenRootFolder(_otherMovieFolder);
|
||||||
|
|
||||||
|
Mocker.GetMock<IConfigService>()
|
||||||
|
.Setup(s => s.CreateEmptyMovieFolders)
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
Subject.Scan(_movie);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.CreateFolder(_movie.Path), Times.Once());
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_extras_subfolder()
|
public void should_not_scan_extras_subfolder()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -98,10 +155,42 @@ public void should_not_scan_extras_subfolder()
|
|||||||
.Verify(v => v.GetImportDecisions(It.Is<List<string>>(l => l.Count == 1), _movie), Times.Once());
|
.Verify(v => v.GetImportDecisions(It.Is<List<string>>(l => l.Count == 1), _movie), Times.Once());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_not_create_if_movie_folder_does_not_exist_and_create_folder_disabled()
|
||||||
|
{
|
||||||
|
GivenRootFolder(_otherMovieFolder);
|
||||||
|
|
||||||
|
Mocker.GetMock<IConfigService>()
|
||||||
|
.Setup(s => s.CreateEmptyMovieFolders)
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
Subject.Scan(_movie);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.CreateFolder(_movie.Path), Times.Never());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void should_clean_but_not_import_if_movie_folder_does_not_exist()
|
||||||
|
{
|
||||||
|
GivenRootFolder(_otherMovieFolder);
|
||||||
|
|
||||||
|
Subject.Scan(_movie);
|
||||||
|
|
||||||
|
Mocker.GetMock<IDiskProvider>()
|
||||||
|
.Verify(v => v.FolderExists(_movie.Path), Times.Once());
|
||||||
|
|
||||||
|
Mocker.GetMock<IMediaFileTableCleanupService>()
|
||||||
|
.Verify(v => v.Clean(It.IsAny<Movie>(), It.IsAny<List<string>>()), Times.Once());
|
||||||
|
|
||||||
|
Mocker.GetMock<IMakeImportDecision>()
|
||||||
|
.Verify(v => v.GetImportDecisions(It.IsAny<List<string>>(), _movie), Times.Never());
|
||||||
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_AppleDouble_subfolder()
|
public void should_not_scan_AppleDouble_subfolder()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -119,9 +208,10 @@ public void should_not_scan_AppleDouble_subfolder()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_scan_extras_movie_and_subfolders()
|
public void should_scan_extras_movie_and_subfolders()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
|
||||||
_movie.Path = @"C:\Test\Movies\Extras".AsOsAgnostic();
|
_movie.Path = @"C:\Test\Movies\Extras".AsOsAgnostic();
|
||||||
|
|
||||||
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
Path.Combine(_movie.Path, "Extras", "file1.mkv").AsOsAgnostic(),
|
Path.Combine(_movie.Path, "Extras", "file1.mkv").AsOsAgnostic(),
|
||||||
@ -141,7 +231,7 @@ public void should_scan_extras_movie_and_subfolders()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_subfolders_that_start_with_period()
|
public void should_not_scan_subfolders_that_start_with_period()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -160,7 +250,7 @@ public void should_not_scan_subfolders_that_start_with_period()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_subfolder_of_season_folder_that_starts_with_a_period()
|
public void should_not_scan_subfolder_of_season_folder_that_starts_with_a_period()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -180,7 +270,7 @@ public void should_not_scan_subfolder_of_season_folder_that_starts_with_a_period
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_Synology_eaDir()
|
public void should_not_scan_Synology_eaDir()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -197,7 +287,7 @@ public void should_not_scan_Synology_eaDir()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_not_scan_thumb_folder()
|
public void should_not_scan_thumb_folder()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -214,9 +304,10 @@ public void should_not_scan_thumb_folder()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_scan_dotHack_folder()
|
public void should_scan_dotHack_folder()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
|
||||||
_movie.Path = @"C:\Test\TV\.hack".AsOsAgnostic();
|
_movie.Path = @"C:\Test\TV\.hack".AsOsAgnostic();
|
||||||
|
|
||||||
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
Path.Combine(_movie.Path, "Season 1", "file1.mkv").AsOsAgnostic(),
|
Path.Combine(_movie.Path, "Season 1", "file1.mkv").AsOsAgnostic(),
|
||||||
@ -230,9 +321,9 @@ public void should_scan_dotHack_folder()
|
|||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void should_find_files_at_root_of_series_folder()
|
public void should_find_files_at_root_of_movie_folder()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
@ -249,7 +340,7 @@ public void should_find_files_at_root_of_series_folder()
|
|||||||
[Test]
|
[Test]
|
||||||
public void should_exclude_osx_metadata_files()
|
public void should_exclude_osx_metadata_files()
|
||||||
{
|
{
|
||||||
GivenParentFolderExists();
|
GivenMovieFolder();
|
||||||
|
|
||||||
GivenFiles(new List<string>
|
GivenFiles(new List<string>
|
||||||
{
|
{
|
||||||
|
@ -75,23 +75,28 @@ public void Scan(Movie movie)
|
|||||||
{
|
{
|
||||||
var rootFolder = _rootFolderService.GetBestRootFolderPath(movie.Path);
|
var rootFolder = _rootFolderService.GetBestRootFolderPath(movie.Path);
|
||||||
|
|
||||||
if (!_diskProvider.FolderExists(rootFolder))
|
var movieFolderExists = _diskProvider.FolderExists(movie.Path);
|
||||||
{
|
|
||||||
_logger.Warn("Movies' root folder ({0}) doesn't exist.", rootFolder);
|
|
||||||
_eventAggregator.PublishEvent(new MovieScanSkippedEvent(movie, MovieScanSkippedReason.RootFolderDoesNotExist));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_diskProvider.GetDirectories(rootFolder).Empty())
|
if (!movieFolderExists)
|
||||||
{
|
{
|
||||||
_logger.Warn("Movies' root folder ({0}) is empty.", rootFolder);
|
if (!_diskProvider.FolderExists(rootFolder))
|
||||||
_eventAggregator.PublishEvent(new MovieScanSkippedEvent(movie, MovieScanSkippedReason.RootFolderIsEmpty));
|
{
|
||||||
return;
|
_logger.Warn("Movie's root folder ({0}) doesn't exist.", rootFolder);
|
||||||
|
_eventAggregator.PublishEvent(new MovieScanSkippedEvent(movie, MovieScanSkippedReason.RootFolderDoesNotExist));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_diskProvider.FolderEmpty(rootFolder))
|
||||||
|
{
|
||||||
|
_logger.Warn("Movie's root folder ({0}) is empty.", rootFolder);
|
||||||
|
_eventAggregator.PublishEvent(new MovieScanSkippedEvent(movie, MovieScanSkippedReason.RootFolderIsEmpty));
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_logger.ProgressInfo("Scanning disk for {0}", movie.Title);
|
_logger.ProgressInfo("Scanning disk for {0}", movie.Title);
|
||||||
|
|
||||||
if (!_diskProvider.FolderExists(movie.Path))
|
if (!movieFolderExists)
|
||||||
{
|
{
|
||||||
if (_configService.CreateEmptyMovieFolders)
|
if (_configService.CreateEmptyMovieFolders)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user