Fixed: Performance issue when scanning large root folder

pull/4421/head
Taloth Saldono 5 years ago committed by Qstick
parent 942d239092
commit 4606503818

@ -142,6 +142,13 @@ namespace NzbDrone.Common.Disk
} }
} }
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 @@ namespace NzbDrone.Common.Disk
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 FizzWare.NBuilder;
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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
.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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
} }
[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 @@ namespace NzbDrone.Core.Test.MediaFiles.DiskScanServiceTests
[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 @@ namespace NzbDrone.Core.MediaFiles
{ {
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…
Cancel
Save