using System; using System.Collections.Generic; using System.Linq; using System.Text.RegularExpressions; using FluentValidation.Results; using NLog; using NzbDrone.Common.Disk; using NzbDrone.Common.Extensions; using NzbDrone.Common.Http; using NzbDrone.Core.Configuration; using NzbDrone.Core.Parser.Model; using NzbDrone.Core.Validation; using NzbDrone.Core.RemotePathMappings; namespace NzbDrone.Core.Download.Clients.Sabnzbd { public class Sabnzbd : UsenetClientBase { private readonly ISabnzbdProxy _proxy; public Sabnzbd(ISabnzbdProxy proxy, IHttpClient httpClient, IConfigService configService, IDiskProvider diskProvider, IRemotePathMappingService remotePathMappingService, Logger logger) : base(httpClient, configService, diskProvider, remotePathMappingService, logger) { _proxy = proxy; } // patch can be a number (releases) or 'x' (git) private static readonly Regex VersionRegex = new Regex(@"(?\d+)\.(?\d+)\.(?\d+|x)(?.*)", RegexOptions.Compiled); protected override string AddFromNzbFile(RemoteEpisode remoteEpisode, string filename, byte[] fileContents) { var category = Settings.TvCategory; var priority = remoteEpisode.IsRecentEpisode() ? Settings.RecentTvPriority : Settings.OlderTvPriority; var response = _proxy.DownloadNzb(fileContents, filename, category, priority, Settings); if (response != null && response.Ids.Any()) { return response.Ids.First(); } return null; } protected override string AddFromNzbFile(RemoteMovie remoteMovie, string filename, byte[] fileContents) { var category = Settings.TvCategory; var priority = Settings.RecentTvPriority; var response = _proxy.DownloadNzb(fileContents, filename, category, priority, Settings); if (response != null && response.Ids.Any()) { return response.Ids.First(); } return null; } private IEnumerable GetQueue() { SabnzbdQueue sabQueue; try { sabQueue = _proxy.GetQueue(0, 0, Settings); } catch (DownloadClientException ex) { _logger.Warn("Couldn't get download queue. {0}", ex.Message); return Enumerable.Empty(); } var queueItems = new List(); foreach (var sabQueueItem in sabQueue.Items) { if (sabQueueItem.Status == SabnzbdDownloadStatus.Deleted) { continue; } var queueItem = new DownloadClientItem(); queueItem.DownloadClient = Definition.Name; queueItem.DownloadId = sabQueueItem.Id; queueItem.Category = sabQueueItem.Category; queueItem.Title = sabQueueItem.Title; queueItem.TotalSize = (long)(sabQueueItem.Size * 1024 * 1024); queueItem.RemainingSize = (long)(sabQueueItem.Sizeleft * 1024 * 1024); queueItem.RemainingTime = sabQueueItem.Timeleft; if (sabQueue.Paused || sabQueueItem.Status == SabnzbdDownloadStatus.Paused) { queueItem.Status = DownloadItemStatus.Paused; queueItem.RemainingTime = null; } else if (sabQueueItem.Status == SabnzbdDownloadStatus.Queued || sabQueueItem.Status == SabnzbdDownloadStatus.Grabbing || sabQueueItem.Status == SabnzbdDownloadStatus.Propagating) { queueItem.Status = DownloadItemStatus.Queued; } else { queueItem.Status = DownloadItemStatus.Downloading; } if (queueItem.Title.StartsWith("ENCRYPTED /")) { queueItem.Title = queueItem.Title.Substring(11); queueItem.IsEncrypted = true; } queueItems.Add(queueItem); } return queueItems; } private IEnumerable GetHistory() { SabnzbdHistory sabHistory; try { sabHistory = _proxy.GetHistory(0, _configService.DownloadClientHistoryLimit, Settings.TvCategory, Settings); } catch (DownloadClientException ex) { _logger.Error(ex, ex.Message); return Enumerable.Empty(); } var historyItems = new List(); foreach (var sabHistoryItem in sabHistory.Items) { if (sabHistoryItem.Status == SabnzbdDownloadStatus.Deleted) { continue; } var historyItem = new DownloadClientItem { DownloadClient = Definition.Name, DownloadId = sabHistoryItem.Id, Category = sabHistoryItem.Category, Title = sabHistoryItem.Title, TotalSize = sabHistoryItem.Size, RemainingSize = 0, RemainingTime = TimeSpan.Zero, Message = sabHistoryItem.FailMessage }; if (sabHistoryItem.Status == SabnzbdDownloadStatus.Failed) { if (sabHistoryItem.FailMessage.IsNotNullOrWhiteSpace() && sabHistoryItem.FailMessage.Equals("Unpacking failed, write error or disk is full?", StringComparison.InvariantCultureIgnoreCase)) { historyItem.Status = DownloadItemStatus.Warning; } else { historyItem.Status = DownloadItemStatus.Failed; } } else if (sabHistoryItem.Status == SabnzbdDownloadStatus.Completed) { historyItem.Status = DownloadItemStatus.Completed; } else // Verifying/Moving etc { historyItem.Status = DownloadItemStatus.Downloading; } var outputPath = _remotePathMappingService.RemapRemoteToLocal(Settings.Host, new OsPath(sabHistoryItem.Storage)); if (!outputPath.IsEmpty) { historyItem.OutputPath = outputPath; var parent = outputPath.Directory; while (!parent.IsEmpty) { if (parent.FileName == sabHistoryItem.Title) { historyItem.OutputPath = parent; } parent = parent.Directory; } } historyItems.Add(historyItem); } return historyItems; } public override string Name => "SABnzbd"; public override IEnumerable GetItems() { foreach (var downloadClientItem in GetQueue().Concat(GetHistory())) { if (downloadClientItem.Category == Settings.TvCategory || downloadClientItem.Category == "*" && Settings.TvCategory.IsNullOrWhiteSpace()) { yield return downloadClientItem; } } } public override void RemoveItem(string downloadId, bool deleteData) { if (GetQueue().Any(v => v.DownloadId == downloadId)) { _proxy.RemoveFrom("queue", downloadId, deleteData, Settings); } else { _proxy.RemoveFrom("history", downloadId, deleteData, Settings); } } protected IEnumerable GetCategories(SabnzbdConfig config) { var completeDir = new OsPath(config.Misc.complete_dir); if (!completeDir.IsRooted) { var queue = _proxy.GetQueue(0, 1, Settings); var defaultRootFolder = new OsPath(queue.DefaultRootFolder); completeDir = defaultRootFolder + completeDir; } foreach (var category in config.Categories) { var relativeDir = new OsPath(category.Dir.TrimEnd('*')); category.FullPath = completeDir + relativeDir; yield return category; } } public override DownloadClientStatus GetStatus() { var config = _proxy.GetConfig(Settings); var categories = GetCategories(config).ToArray(); var category = categories.FirstOrDefault(v => v.Name == Settings.TvCategory); if (category == null) { category = categories.FirstOrDefault(v => v.Name == "*"); } var status = new DownloadClientStatus { IsLocalhost = Settings.Host == "127.0.0.1" || Settings.Host == "localhost" }; if (category != null) { status.OutputRootFolders = new List { _remotePathMappingService.RemapRemoteToLocal(Settings.Host, category.FullPath) }; } return status; } protected override void Test(List failures) { failures.AddIfNotNull(TestConnectionAndVersion()); failures.AddIfNotNull(TestAuthentication()); failures.AddIfNotNull(TestGlobalConfig()); failures.AddIfNotNull(TestCategory()); } private bool HasVersion(int major, int minor, int patch = 0, string candidate = null) { candidate = candidate ?? string.Empty; var version = _proxy.GetVersion(Settings); var parsed = VersionRegex.Match(version); int actualMajor; int actualMinor; int actualPatch; string actualCandidate; if (!parsed.Success) { if (!version.Equals("develop", StringComparison.InvariantCultureIgnoreCase)) { return false; } actualMajor = 1; actualMinor = 1; actualPatch = 0; actualCandidate = null; } else { actualMajor = Convert.ToInt32(parsed.Groups["major"].Value); actualMinor = Convert.ToInt32(parsed.Groups["minor"].Value); actualPatch = Convert.ToInt32(parsed.Groups["patch"].Value.Replace("x", "")); actualCandidate = parsed.Groups["candidate"].Value.ToUpper(); } if (actualMajor > major) { return true; } else if (actualMajor < major) { return false; } if (actualMinor > minor) { return true; } else if (actualMinor < minor) { return false; } if (actualPatch > patch) { return true; } else if (actualPatch < patch) { return false; } if (actualCandidate.IsNullOrWhiteSpace()) { return true; } else if (candidate.IsNullOrWhiteSpace()) { return false; } else { return actualCandidate.CompareTo(candidate) > 0; } } private ValidationFailure TestConnectionAndVersion() { try { var version = _proxy.GetVersion(Settings); var parsed = VersionRegex.Match(version); if (!parsed.Success) { if (version.Equals("develop", StringComparison.InvariantCultureIgnoreCase)) { return new NzbDroneValidationFailure("Version", "Sabnzbd develop version, assuming version 1.1.0 or higher.") { IsWarning = true, DetailedDescription = "Radarr may not be able to support new features added to SABnzbd when running develop versions." }; } return new ValidationFailure("Version", "Unknown Version: " + version); } var major = Convert.ToInt32(parsed.Groups["major"].Value); var minor = Convert.ToInt32(parsed.Groups["minor"].Value); if (major >= 1) { return null; } if (minor >= 7) { return null; } return new ValidationFailure("Version", "Version 0.7.0+ is required, but found: " + version); } catch (Exception ex) { _logger.Error(ex, ex.Message); return new ValidationFailure("Host", "Unable to connect to SABnzbd"); } } private ValidationFailure TestAuthentication() { try { _proxy.GetConfig(Settings); } catch (Exception ex) { if (ex.Message.ContainsIgnoreCase("API Key Incorrect")) { return new ValidationFailure("APIKey", "API Key Incorrect"); } if (ex.Message.ContainsIgnoreCase("API Key Required")) { return new ValidationFailure("APIKey", "API Key Required"); } throw; } return null; } private ValidationFailure TestGlobalConfig() { var config = _proxy.GetConfig(Settings); if (config.Misc.pre_check && !HasVersion(1, 1)) { return new NzbDroneValidationFailure("", "Disable 'Check before download' option in Sabnbzd") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/switches/", Settings.Host, Settings.Port), DetailedDescription = "Using Check before download affects Radarr ability to track new downloads. Also Sabnzbd recommends 'Abort jobs that cannot be completed' instead since it's more effective." }; } return null; } private ValidationFailure TestCategory() { var config = _proxy.GetConfig(Settings); var category = GetCategories(config).FirstOrDefault((SabnzbdCategory v) => v.Name == Settings.TvCategory); if (category != null) { if (category.Dir.EndsWith("*")) { return new NzbDroneValidationFailure("TvCategory", "Enable Job folders") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/categories/", Settings.Host, Settings.Port), DetailedDescription = "Radarr prefers each download to have a separate folder. With * appended to the Folder/Path Sabnzbd will not create these job folders. Go to Sabnzbd to fix it." }; } } else { if (!Settings.TvCategory.IsNullOrWhiteSpace()) { return new NzbDroneValidationFailure("TvCategory", "Category does not exist") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/categories/", Settings.Host, Settings.Port), DetailedDescription = "The Category your entered doesn't exist in Sabnzbd. Go to Sabnzbd to create it." }; } } if (config.Misc.enable_tv_sorting) { if (!config.Misc.tv_categories.Any() || config.Misc.tv_categories.Contains(Settings.TvCategory) || (Settings.TvCategory.IsNullOrWhiteSpace() && config.Misc.tv_categories.Contains("Default"))) { return new NzbDroneValidationFailure("TvCategory", "Disable TV Sorting") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/sorting/", Settings.Host, Settings.Port), DetailedDescription = "You must disable Sabnzbd TV Sorting for the category Radarr uses to prevent import issues. Go to Sabnzbd to fix it." }; } } if (config.Misc.enable_movie_sorting) { if (!config.Misc.movie_categories.Any() || config.Misc.movie_categories.Contains(Settings.TvCategory) || (Settings.TvCategory.IsNullOrWhiteSpace() && config.Misc.movie_categories.Contains("Default"))) { return new NzbDroneValidationFailure("TvCategory", "Disable Movie Sorting") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/sorting/", Settings.Host, Settings.Port), DetailedDescription = "You must disable Sabnzbd Movie Sorting for the category Radarr uses to prevent import issues. Go to Sabnzbd to fix it." }; } } if (config.Misc.enable_date_sorting) { if (!config.Misc.date_categories.Any() || config.Misc.date_categories.Contains(Settings.TvCategory) || (Settings.TvCategory.IsNullOrWhiteSpace() && config.Misc.date_categories.Contains("Default"))) { return new NzbDroneValidationFailure("TvCategory", "Disable Date Sorting") { InfoLink = string.Format("http://{0}:{1}/sabnzbd/config/sorting/", Settings.Host, Settings.Port), DetailedDescription = "You must disable Sabnzbd Date Sorting for the category Radarr uses to prevent import issues. Go to Sabnzbd to fix it." }; } } return null; } } }