|  | @@ -0,0 +1,186 @@
 | 
	
		
			
				|  |  | +using MediaBrowser.Controller.Entities.TV;
 | 
	
		
			
				|  |  | +using MediaBrowser.Controller.Library;
 | 
	
		
			
				|  |  | +using MediaBrowser.Controller.Providers;
 | 
	
		
			
				|  |  | +using MediaBrowser.Controller.Resolvers;
 | 
	
		
			
				|  |  | +using MediaBrowser.Model.Configuration;
 | 
	
		
			
				|  |  | +using MediaBrowser.Model.Entities;
 | 
	
		
			
				|  |  | +using MediaBrowser.Model.Logging;
 | 
	
		
			
				|  |  | +using System;
 | 
	
		
			
				|  |  | +using System.Collections.Generic;
 | 
	
		
			
				|  |  | +using System.IO;
 | 
	
		
			
				|  |  | +using System.Linq;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +namespace MediaBrowser.Server.Implementations.FileSorting
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    public class TvFileSorter
 | 
	
		
			
				|  |  | +    {
 | 
	
		
			
				|  |  | +        private readonly ILibraryManager _libraryManager;
 | 
	
		
			
				|  |  | +        private readonly ILogger _logger;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        public TvFileSorter(ILibraryManager libraryManager, ILogger logger)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            _libraryManager = libraryManager;
 | 
	
		
			
				|  |  | +            _logger = logger;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        public void Sort(string path, FileSortingOptions options)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            var minFileBytes = options.MinFileSizeMb * 1024 * 1024;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            var allSeries = _libraryManager.RootFolder
 | 
	
		
			
				|  |  | +                .RecursiveChildren.OfType<Series>()
 | 
	
		
			
				|  |  | +                .Where(i => i.LocationType == LocationType.FileSystem)
 | 
	
		
			
				|  |  | +                .ToList();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            var eligibleFiles = new DirectoryInfo(path)
 | 
	
		
			
				|  |  | +                .EnumerateFiles("*", SearchOption.AllDirectories)
 | 
	
		
			
				|  |  | +                .Where(i => EntityResolutionHelper.IsVideoFile(i.FullName) && i.Length >= minFileBytes)
 | 
	
		
			
				|  |  | +                .ToList();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            foreach (var file in eligibleFiles)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                SortFile(file.FullName, options, allSeries);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (options.LeftOverFileExtensionsToDelete.Length > 0)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                DeleteLeftOverFiles(path, options.LeftOverFileExtensionsToDelete);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (options.DeleteEmptyFolders)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                DeleteEmptyFolders(path);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private void SortFile(string path, FileSortingOptions options, IEnumerable<Series> allSeries)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            _logger.Info("Sorting file {0}", path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            var seriesName = TVUtils.GetSeriesNameFromEpisodeFile(path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (!string.IsNullOrEmpty(seriesName))
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                var season = TVUtils.GetSeasonNumberFromEpisodeFile(path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                if (season.HasValue)
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    // Passing in true will include a few extra regex's
 | 
	
		
			
				|  |  | +                    var episode = TVUtils.GetEpisodeNumberFromFile(path, true);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                    if (episode.HasValue)
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        _logger.Debug("Extracted information from {0}. Series name {1}, Season {2}, Episode {3}", path, seriesName, season, episode);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                        SortFile(path, seriesName, season.Value, episode.Value, options, allSeries);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    else
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        _logger.Warn("Unable to determine episode number from {0}", path);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +                else
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    _logger.Warn("Unable to determine season number from {0}", path);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            else
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                _logger.Warn("Unable to determine series name from {0}", path);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private void SortFile(string path, string seriesName, int seasonNumber, int episodeNumber, FileSortingOptions options, IEnumerable<Series> allSeries)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            var series = GetMatchingSeries(seriesName, allSeries);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (series == null)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                _logger.Warn("Unable to find series in library matching name {0}", seriesName);
 | 
	
		
			
				|  |  | +                return;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            _logger.Info("Sorting file {0} into series {1}", path, series.Path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            // Proceed to sort the file
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private Series GetMatchingSeries(string seriesName, IEnumerable<Series> allSeries)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            int? yearInName;
 | 
	
		
			
				|  |  | +            var nameWithoutYear = seriesName;
 | 
	
		
			
				|  |  | +            NameParser.ParseName(nameWithoutYear, out nameWithoutYear, out yearInName);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return allSeries.Select(i => GetMatchScore(nameWithoutYear, yearInName, i))
 | 
	
		
			
				|  |  | +                .Where(i => i.Item2 > 0)
 | 
	
		
			
				|  |  | +                .OrderByDescending(i => i.Item2)
 | 
	
		
			
				|  |  | +                .Select(i => i.Item1)
 | 
	
		
			
				|  |  | +                .FirstOrDefault();
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private Tuple<Series, int> GetMatchScore(string sortedName, int? year, Series series)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            var score = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            if (year.HasValue)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                if (series.ProductionYear.HasValue && year.Value == series.ProductionYear.Value)
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    score++;
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            // TODO: Improve this
 | 
	
		
			
				|  |  | +            if (string.Equals(sortedName, series.Name, StringComparison.OrdinalIgnoreCase))
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                score++;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return new Tuple<Series, int>(series, score);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private void DeleteLeftOverFiles(string path, IEnumerable<string> extensions)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            var eligibleFiles = new DirectoryInfo(path)
 | 
	
		
			
				|  |  | +                .EnumerateFiles("*", SearchOption.AllDirectories)
 | 
	
		
			
				|  |  | +                .Where(i => extensions.Contains(i.Extension, StringComparer.OrdinalIgnoreCase))
 | 
	
		
			
				|  |  | +                .ToList();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            foreach (var file in eligibleFiles)
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                try
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    File.Delete(file.FullName);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +                catch (IOException ex)
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    _logger.ErrorException("Error deleting file {0}", ex, file.FullName);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private void DeleteEmptyFolders(string path)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            try
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                foreach (var d in Directory.EnumerateDirectories(path))
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    DeleteEmptyFolders(d);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                var entries = Directory.EnumerateFileSystemEntries(path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                if (!entries.Any())
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    try
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        Directory.Delete(path);
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                    catch (UnauthorizedAccessException) { }
 | 
	
		
			
				|  |  | +                    catch (DirectoryNotFoundException) { }
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            catch (UnauthorizedAccessException) { }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 |