using System.Linq; using System; using System.Collections.Generic; using System.IO; using NLog; using NzbDrone.Common; using NzbDrone.Core.Providers; using NzbDrone.Core.Repository; namespace NzbDrone.Core.RootFolders { public interface IRootFolderService { List All(); RootFolder Add(RootFolder rootDir); void Remove(long rootDirId); List GetUnmappedFolders(string path); Dictionary FreeSpaceOnDrives(); } public class RootFolderService : IRootFolderService { private static readonly Logger Logger = LogManager.GetCurrentClassLogger(); private readonly IRootFolderRepository _rootFolderRepository; private readonly DiskProvider _diskProvider; private readonly SeriesProvider _seriesProvider; public RootFolderService(IRootFolderRepository rootFolderRepository, SeriesProvider seriesProvider, DiskProvider diskProvider) { _rootFolderRepository = rootFolderRepository; _diskProvider = diskProvider; _seriesProvider = seriesProvider; } public virtual List All() { var rootFolders = _rootFolderRepository.All(); rootFolders.ForEach(folder => { folder.FreeSpace = _diskProvider.FreeDiskSpace(new DirectoryInfo(folder.Path)); folder.UnmappedFolders = GetUnmappedFolders(folder.Path); }); return rootFolders; } public virtual RootFolder Add(RootFolder rootFolder) { if (String.IsNullOrWhiteSpace(rootFolder.Path) || !Path.IsPathRooted(rootFolder.Path)) throw new ArgumentException("Invalid path"); if (!_diskProvider.FolderExists(rootFolder.Path)) throw new DirectoryNotFoundException("Can't add root directory that doesn't exist."); if (All().Exists(r => DiskProvider.PathEquals(r.Path, rootFolder.Path))) throw new InvalidOperationException("Root directory already exist."); _rootFolderRepository.Add(rootFolder); rootFolder.FreeSpace = _diskProvider.FreeDiskSpace(new DirectoryInfo(rootFolder.Path)); rootFolder.UnmappedFolders = GetUnmappedFolders(rootFolder.Path); return rootFolder; } public virtual void Remove(long rootDirId) { _rootFolderRepository.Delete(rootDirId); } public virtual List GetUnmappedFolders(string path) { Logger.Debug("Generating list of unmapped folders"); if (String.IsNullOrEmpty(path)) throw new ArgumentException("Invalid path provided", "path"); var results = new List(); if (!_diskProvider.FolderExists(path)) { Logger.Debug("Path supplied does not exist: {0}", path); return results; } foreach (string seriesFolder in _diskProvider.GetDirectories(path)) { if (!_seriesProvider.SeriesPathExists(seriesFolder)) { results.Add(new DirectoryInfo(seriesFolder.Normalize()).Name); } } Logger.Debug("{0} unmapped folders detected.", results.Count); return results; } public virtual Dictionary FreeSpaceOnDrives() { var freeSpace = new Dictionary(); var rootDirs = All(); foreach (var rootDir in rootDirs) { var pathRoot = _diskProvider.GetPathRoot(rootDir.Path); if (!freeSpace.ContainsKey(pathRoot)) { try { freeSpace.Add(pathRoot, _diskProvider.FreeDiskSpace(new DirectoryInfo(rootDir.Path))); } catch (Exception ex) { Logger.WarnException("Error getting fromm space for: " + pathRoot, ex); } } } return freeSpace; } } }