Add support for Flood (#2104)
parent
f890fa1697
commit
600975873b
@ -0,0 +1,226 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using FluentValidation.Results;
|
||||
using NLog;
|
||||
using NzbDrone.Common.Disk;
|
||||
using NzbDrone.Common.Http;
|
||||
using NzbDrone.Core.Configuration;
|
||||
using NzbDrone.Core.Download.Clients.Flood.Models;
|
||||
using NzbDrone.Core.MediaFiles.TorrentInfo;
|
||||
using NzbDrone.Core.Parser.Model;
|
||||
using NzbDrone.Core.RemotePathMappings;
|
||||
using NzbDrone.Core.ThingiProvider;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood
|
||||
{
|
||||
public class Flood : TorrentClientBase<FloodSettings>
|
||||
{
|
||||
private readonly IFloodProxy _proxy;
|
||||
|
||||
public Flood(IFloodProxy proxy,
|
||||
ITorrentFileInfoReader torrentFileInfoReader,
|
||||
IHttpClient httpClient,
|
||||
IConfigService configService,
|
||||
IDiskProvider diskProvider,
|
||||
IRemotePathMappingService remotePathMappingService,
|
||||
Logger logger)
|
||||
: base(torrentFileInfoReader, httpClient, configService, diskProvider, remotePathMappingService, logger)
|
||||
{
|
||||
_proxy = proxy;
|
||||
}
|
||||
|
||||
private static IEnumerable<string> HandleTags(RemoteAlbum remoteAlbum, FloodSettings settings)
|
||||
{
|
||||
var result = new HashSet<string>();
|
||||
|
||||
if (settings.Tags.Any())
|
||||
{
|
||||
result.UnionWith(settings.Tags);
|
||||
}
|
||||
|
||||
if (settings.AdditionalTags.Any())
|
||||
{
|
||||
foreach (var additionalTag in settings.AdditionalTags)
|
||||
{
|
||||
switch (additionalTag)
|
||||
{
|
||||
case (int)AdditionalTags.Artist:
|
||||
result.Add(remoteAlbum.Artist.Name);
|
||||
break;
|
||||
case (int)AdditionalTags.Quality:
|
||||
result.Add(remoteAlbum.ParsedAlbumInfo.Quality.Quality.ToString());
|
||||
break;
|
||||
case (int)AdditionalTags.ReleaseGroup:
|
||||
result.Add(remoteAlbum.ParsedAlbumInfo.ReleaseGroup);
|
||||
break;
|
||||
case (int)AdditionalTags.Year:
|
||||
result.Add(remoteAlbum.ParsedAlbumInfo.ArtistTitleInfo.Year.ToString());
|
||||
break;
|
||||
case (int)AdditionalTags.Indexer:
|
||||
result.Add(remoteAlbum.Release.Indexer);
|
||||
break;
|
||||
default:
|
||||
throw new DownloadClientException("Unexpected additional tag ID");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public override string Name => "Flood";
|
||||
public override ProviderMessage Message => new ProviderMessage("Lidarr is unable to remove torrents that have finished seeding when using Flood", ProviderMessageType.Warning);
|
||||
|
||||
protected override string AddFromTorrentFile(RemoteAlbum remoteAlbum, string hash, string filename, byte[] fileContent)
|
||||
{
|
||||
_proxy.AddTorrentByFile(Convert.ToBase64String(fileContent), HandleTags(remoteAlbum, Settings), Settings);
|
||||
|
||||
return hash;
|
||||
}
|
||||
|
||||
protected override string AddFromMagnetLink(RemoteAlbum remoteAlbum, string hash, string magnetLink)
|
||||
{
|
||||
_proxy.AddTorrentByUrl(magnetLink, HandleTags(remoteAlbum, Settings), Settings);
|
||||
|
||||
return hash;
|
||||
}
|
||||
|
||||
public override IEnumerable<DownloadClientItem> GetItems()
|
||||
{
|
||||
var items = new List<DownloadClientItem>();
|
||||
|
||||
var list = _proxy.GetTorrents(Settings);
|
||||
|
||||
foreach (var torrent in list)
|
||||
{
|
||||
var properties = torrent.Value;
|
||||
|
||||
if (!Settings.Tags.All(tag => properties.Tags.Contains(tag)))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
var item = new DownloadClientItem
|
||||
{
|
||||
DownloadClientInfo = DownloadClientItemClientInfo.FromDownloadClient(this),
|
||||
DownloadId = torrent.Key,
|
||||
Title = properties.Name,
|
||||
OutputPath = _remotePathMappingService.RemapRemoteToLocal(Settings.Host, new OsPath(properties.Directory)),
|
||||
Category = properties.Tags.Count > 0 ? properties.Tags[0] : null,
|
||||
RemainingSize = properties.SizeBytes - properties.BytesDone,
|
||||
TotalSize = properties.SizeBytes,
|
||||
SeedRatio = properties.Ratio,
|
||||
Message = properties.Message,
|
||||
};
|
||||
|
||||
if (properties.Eta > 0)
|
||||
{
|
||||
item.RemainingTime = TimeSpan.FromSeconds(properties.Eta);
|
||||
}
|
||||
|
||||
if (properties.Status.Contains("error"))
|
||||
{
|
||||
item.Status = DownloadItemStatus.Warning;
|
||||
}
|
||||
else if (properties.Status.Contains("seeding") || properties.Status.Contains("complete"))
|
||||
{
|
||||
item.Status = DownloadItemStatus.Completed;
|
||||
}
|
||||
else if (properties.Status.Contains("downloading"))
|
||||
{
|
||||
item.Status = DownloadItemStatus.Downloading;
|
||||
}
|
||||
else if (properties.Status.Contains("stopped"))
|
||||
{
|
||||
item.Status = DownloadItemStatus.Paused;
|
||||
}
|
||||
|
||||
item.CanMoveFiles = item.CanBeRemoved = false;
|
||||
|
||||
items.Add(item);
|
||||
}
|
||||
|
||||
return items;
|
||||
}
|
||||
|
||||
public override DownloadClientItem GetImportItem(DownloadClientItem item, DownloadClientItem previousImportAttempt)
|
||||
{
|
||||
var result = item.Clone();
|
||||
|
||||
var contentPaths = _proxy.GetTorrentContentPaths(item.DownloadId, Settings);
|
||||
|
||||
if (contentPaths.Count < 1)
|
||||
{
|
||||
throw new DownloadClientUnavailableException($"Failed to fetch list of contents of torrent: {item.DownloadId}");
|
||||
}
|
||||
|
||||
if (contentPaths.Count == 1)
|
||||
{
|
||||
// For single-file torrent, OutputPath should be the path of file.
|
||||
result.OutputPath = item.OutputPath + new OsPath(contentPaths[0]);
|
||||
}
|
||||
else
|
||||
{
|
||||
// For multi-file torrent, OutputPath should be the path of base directory of torrent.
|
||||
var baseDirectoryPaths = contentPaths.ConvertAll(path =>
|
||||
path.Split(new char[] { '\\', '/' }, StringSplitOptions.RemoveEmptyEntries)[0]);
|
||||
|
||||
// Check first segment (directory) of paths of contents. If all contents share the same directory, use that directory.
|
||||
if (baseDirectoryPaths.TrueForAll(path => path == baseDirectoryPaths[0]))
|
||||
{
|
||||
result.OutputPath = item.OutputPath + new OsPath(baseDirectoryPaths[0]);
|
||||
}
|
||||
|
||||
// Otherwise, OutputPath is already the base directory.
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
public override void MarkItemAsImported(DownloadClientItem downloadClientItem)
|
||||
{
|
||||
if (Settings.PostImportTags.Any())
|
||||
{
|
||||
var list = _proxy.GetTorrents(Settings);
|
||||
|
||||
if (list.ContainsKey(downloadClientItem.DownloadId))
|
||||
{
|
||||
_proxy.SetTorrentsTags(downloadClientItem.DownloadId,
|
||||
new HashSet<string>(list[downloadClientItem.DownloadId].Tags.Concat(Settings.PostImportTags)),
|
||||
Settings);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public override void RemoveItem(string downloadId, bool deleteData)
|
||||
{
|
||||
_proxy.DeleteTorrent(downloadId, deleteData, Settings);
|
||||
}
|
||||
|
||||
public override DownloadClientInfo GetStatus()
|
||||
{
|
||||
return new DownloadClientInfo
|
||||
{
|
||||
IsLocalhost = Settings.Host == "127.0.0.1" || Settings.Host == "::1" || Settings.Host == "localhost",
|
||||
OutputRootFolders = new List<OsPath> { _remotePathMappingService.RemapRemoteToLocal(Settings.Host, new OsPath(Settings.Destination)) }
|
||||
};
|
||||
}
|
||||
|
||||
protected override void Test(List<ValidationFailure> failures)
|
||||
{
|
||||
try
|
||||
{
|
||||
_proxy.AuthVerify(Settings);
|
||||
}
|
||||
catch (DownloadClientAuthenticationException ex)
|
||||
{
|
||||
failures.Add(new ValidationFailure("Password", ex.Message));
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
failures.Add(new ValidationFailure("Host", ex.Message));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,213 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Net;
|
||||
using NLog;
|
||||
using NzbDrone.Common.Cache;
|
||||
using NzbDrone.Common.Http;
|
||||
using NzbDrone.Common.Serializer;
|
||||
using NzbDrone.Core.Download.Clients.Flood.Types;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood
|
||||
{
|
||||
public interface IFloodProxy
|
||||
{
|
||||
void AuthVerify(FloodSettings settings);
|
||||
void AddTorrentByUrl(string url, IEnumerable<string> tags, FloodSettings settings);
|
||||
void AddTorrentByFile(string file, IEnumerable<string> tags, FloodSettings settings);
|
||||
void DeleteTorrent(string hash, bool deleteData, FloodSettings settings);
|
||||
Dictionary<string, Torrent> GetTorrents(FloodSettings settings);
|
||||
List<string> GetTorrentContentPaths(string hash, FloodSettings settings);
|
||||
void SetTorrentsTags(string hash, IEnumerable<string> tags, FloodSettings settings);
|
||||
}
|
||||
|
||||
public class FloodProxy : IFloodProxy
|
||||
{
|
||||
private readonly IHttpClient _httpClient;
|
||||
private readonly Logger _logger;
|
||||
private readonly ICached<Dictionary<string, string>> _authCookieCache;
|
||||
|
||||
public FloodProxy(IHttpClient httpClient, ICacheManager cacheManager, Logger logger)
|
||||
{
|
||||
_httpClient = httpClient;
|
||||
_logger = logger;
|
||||
_authCookieCache = cacheManager.GetCache<Dictionary<string, string>>(GetType(), "authCookies");
|
||||
}
|
||||
|
||||
private string BuildUrl(FloodSettings settings)
|
||||
{
|
||||
return $"{(settings.UseSsl ? "https://" : "http://")}{settings.Host}:{settings.Port}/{settings.UrlBase}";
|
||||
}
|
||||
|
||||
private string BuildCachedCookieKey(FloodSettings settings)
|
||||
{
|
||||
return $"{BuildUrl(settings)}:{settings.Username}";
|
||||
}
|
||||
|
||||
private HttpRequestBuilder BuildRequest(FloodSettings settings)
|
||||
{
|
||||
var requestBuilder = new HttpRequestBuilder(HttpUri.CombinePath(BuildUrl(settings), "/api"))
|
||||
{
|
||||
LogResponseContent = true,
|
||||
NetworkCredential = new NetworkCredential(settings.Username, settings.Password)
|
||||
};
|
||||
|
||||
requestBuilder.Headers.ContentType = "application/json";
|
||||
requestBuilder.SetCookies(AuthAuthenticate(requestBuilder, settings));
|
||||
|
||||
return requestBuilder;
|
||||
}
|
||||
|
||||
private HttpResponse HandleRequest(HttpRequest request, FloodSettings settings)
|
||||
{
|
||||
try
|
||||
{
|
||||
return _httpClient.Execute(request);
|
||||
}
|
||||
catch (HttpException ex)
|
||||
{
|
||||
if (ex.Response.StatusCode == HttpStatusCode.Forbidden ||
|
||||
ex.Response.StatusCode == HttpStatusCode.Unauthorized)
|
||||
{
|
||||
_authCookieCache.Remove(BuildCachedCookieKey(settings));
|
||||
throw new DownloadClientAuthenticationException("Failed to authenticate with Flood.");
|
||||
}
|
||||
|
||||
throw new DownloadClientException("Unable to connect to Flood, please check your settings");
|
||||
}
|
||||
catch
|
||||
{
|
||||
throw new DownloadClientException("Unable to connect to Flood, please check your settings");
|
||||
}
|
||||
}
|
||||
|
||||
private Dictionary<string, string> AuthAuthenticate(HttpRequestBuilder requestBuilder, FloodSettings settings, bool force = false)
|
||||
{
|
||||
var cachedCookies = _authCookieCache.Find(BuildCachedCookieKey(settings));
|
||||
|
||||
if (cachedCookies == null || force)
|
||||
{
|
||||
var authenticateRequest = requestBuilder.Resource("/auth/authenticate").Post().Build();
|
||||
|
||||
var body = new Dictionary<string, object>
|
||||
{
|
||||
{ "username", settings.Username },
|
||||
{ "password", settings.Password }
|
||||
};
|
||||
authenticateRequest.SetContent(body.ToJson());
|
||||
|
||||
var response = HandleRequest(authenticateRequest, settings);
|
||||
cachedCookies = response.GetCookies();
|
||||
_authCookieCache.Set(BuildCachedCookieKey(settings), cachedCookies);
|
||||
}
|
||||
|
||||
return cachedCookies;
|
||||
}
|
||||
|
||||
public void AuthVerify(FloodSettings settings)
|
||||
{
|
||||
var verifyRequest = BuildRequest(settings).Resource("/auth/verify").Build();
|
||||
|
||||
verifyRequest.Method = HttpMethod.GET;
|
||||
|
||||
HandleRequest(verifyRequest, settings);
|
||||
}
|
||||
|
||||
public void AddTorrentByFile(string file, IEnumerable<string> tags, FloodSettings settings)
|
||||
{
|
||||
var addRequest = BuildRequest(settings).Resource("/torrents/add-files").Post().Build();
|
||||
|
||||
var body = new Dictionary<string, object>
|
||||
{
|
||||
{ "files", new List<string> { file } },
|
||||
{ "tags", tags.ToList() }
|
||||
};
|
||||
|
||||
if (settings.Destination != null)
|
||||
{
|
||||
body.Add("destination", settings.Destination);
|
||||
}
|
||||
|
||||
if (!settings.AddPaused)
|
||||
{
|
||||
body.Add("start", true);
|
||||
}
|
||||
|
||||
addRequest.SetContent(body.ToJson());
|
||||
|
||||
HandleRequest(addRequest, settings);
|
||||
}
|
||||
|
||||
public void AddTorrentByUrl(string url, IEnumerable<string> tags, FloodSettings settings)
|
||||
{
|
||||
var addRequest = BuildRequest(settings).Resource("/torrents/add-urls").Post().Build();
|
||||
|
||||
var body = new Dictionary<string, object>
|
||||
{
|
||||
{ "urls", new List<string> { url } },
|
||||
{ "tags", tags.ToList() }
|
||||
};
|
||||
|
||||
if (settings.Destination != null)
|
||||
{
|
||||
body.Add("destination", settings.Destination);
|
||||
}
|
||||
|
||||
if (!settings.AddPaused)
|
||||
{
|
||||
body.Add("start", true);
|
||||
}
|
||||
|
||||
addRequest.SetContent(body.ToJson());
|
||||
|
||||
HandleRequest(addRequest, settings);
|
||||
}
|
||||
|
||||
public void DeleteTorrent(string hash, bool deleteData, FloodSettings settings)
|
||||
{
|
||||
var deleteRequest = BuildRequest(settings).Resource("/torrents/delete").Post().Build();
|
||||
|
||||
var body = new Dictionary<string, object>
|
||||
{
|
||||
{ "hashes", new List<string> { hash } },
|
||||
{ "deleteData", deleteData }
|
||||
};
|
||||
deleteRequest.SetContent(body.ToJson());
|
||||
|
||||
HandleRequest(deleteRequest, settings);
|
||||
}
|
||||
|
||||
public Dictionary<string, Torrent> GetTorrents(FloodSettings settings)
|
||||
{
|
||||
var getTorrentsRequest = BuildRequest(settings).Resource("/torrents").Build();
|
||||
|
||||
getTorrentsRequest.Method = HttpMethod.GET;
|
||||
|
||||
return Json.Deserialize<TorrentListSummary>(HandleRequest(getTorrentsRequest, settings).Content).Torrents;
|
||||
}
|
||||
|
||||
public List<string> GetTorrentContentPaths(string hash, FloodSettings settings)
|
||||
{
|
||||
var contentsRequest = BuildRequest(settings).Resource($"/torrents/{hash}/contents").Build();
|
||||
|
||||
contentsRequest.Method = HttpMethod.GET;
|
||||
|
||||
return Json.Deserialize<List<TorrentContent>>(HandleRequest(contentsRequest, settings).Content).ConvertAll(content => content.Path);
|
||||
}
|
||||
|
||||
public void SetTorrentsTags(string hash, IEnumerable<string> tags, FloodSettings settings)
|
||||
{
|
||||
var tagsRequest = BuildRequest(settings).Resource("/torrents/tags").Build();
|
||||
|
||||
tagsRequest.Method = HttpMethod.PATCH;
|
||||
|
||||
var body = new Dictionary<string, object>
|
||||
{
|
||||
{ "hashes", new List<string> { hash } },
|
||||
{ "tags", tags.ToList() }
|
||||
};
|
||||
tagsRequest.SetContent(body.ToJson());
|
||||
|
||||
HandleRequest(tagsRequest, settings);
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,75 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using FluentValidation;
|
||||
using NzbDrone.Core.Annotations;
|
||||
using NzbDrone.Core.Download.Clients.Flood.Models;
|
||||
using NzbDrone.Core.ThingiProvider;
|
||||
using NzbDrone.Core.Validation;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood
|
||||
{
|
||||
public class FloodSettingsValidator : AbstractValidator<FloodSettings>
|
||||
{
|
||||
public FloodSettingsValidator()
|
||||
{
|
||||
RuleFor(c => c.Host).ValidHost();
|
||||
RuleFor(c => c.Port).InclusiveBetween(1, 65535);
|
||||
}
|
||||
}
|
||||
|
||||
public class FloodSettings : IProviderConfig
|
||||
{
|
||||
private static readonly FloodSettingsValidator Validator = new FloodSettingsValidator();
|
||||
|
||||
public FloodSettings()
|
||||
{
|
||||
UseSsl = false;
|
||||
Host = "localhost";
|
||||
Port = 3000;
|
||||
Tags = new string[]
|
||||
{
|
||||
"lidarr"
|
||||
};
|
||||
AdditionalTags = Enumerable.Empty<int>();
|
||||
AddPaused = false;
|
||||
}
|
||||
|
||||
[FieldDefinition(0, Label = "Use SSL", Type = FieldType.Checkbox)]
|
||||
public bool UseSsl { get; set; }
|
||||
|
||||
[FieldDefinition(1, Label = "Host", Type = FieldType.Textbox)]
|
||||
public string Host { get; set; }
|
||||
|
||||
[FieldDefinition(2, Label = "Port", Type = FieldType.Textbox)]
|
||||
public int Port { get; set; }
|
||||
|
||||
[FieldDefinition(3, Label = "Url Base", Type = FieldType.Textbox, HelpText = "Optionally adds a prefix to Flood API, such as [protocol]://[host]:[port]/[urlBase]api")]
|
||||
public string UrlBase { get; set; }
|
||||
|
||||
[FieldDefinition(4, Label = "Username", Type = FieldType.Textbox, Privacy = PrivacyLevel.UserName)]
|
||||
public string Username { get; set; }
|
||||
|
||||
[FieldDefinition(5, Label = "Password", Type = FieldType.Password, Privacy = PrivacyLevel.Password)]
|
||||
public string Password { get; set; }
|
||||
|
||||
[FieldDefinition(6, Label = "Destination", Type = FieldType.Textbox, HelpText = "Manually specifies download destination")]
|
||||
public string Destination { get; set; }
|
||||
|
||||
[FieldDefinition(7, Label = "Tags", Type = FieldType.Tag, HelpText = "Initial tags of a download. To be recognized, a download must have all initial tags. This avoids conflicts with unrelated downloads.")]
|
||||
public IEnumerable<string> Tags { get; set; }
|
||||
|
||||
[FieldDefinition(8, Label = "Post-Import Tags", Type = FieldType.Tag, HelpText = "Appends tags after a download is imported.", Advanced = true)]
|
||||
public IEnumerable<string> PostImportTags { get; set; }
|
||||
|
||||
[FieldDefinition(9, Label = "Additional Tags", Type = FieldType.Select, SelectOptions = typeof(AdditionalTags), HelpText = "Adds properties of media as tags. Hints are examples.", Advanced = true)]
|
||||
public IEnumerable<int> AdditionalTags { get; set; }
|
||||
|
||||
[FieldDefinition(10, Label = "Add Paused", Type = FieldType.Checkbox)]
|
||||
public bool AddPaused { get; set; }
|
||||
|
||||
public NzbDroneValidationResult Validate()
|
||||
{
|
||||
return new NzbDroneValidationResult(Validator.Validate(this));
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,22 @@
|
||||
using NzbDrone.Core.Annotations;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood.Models
|
||||
{
|
||||
public enum AdditionalTags
|
||||
{
|
||||
[FieldOption(Hint = "Elvis Presley")]
|
||||
Artist = 0,
|
||||
|
||||
[FieldOption(Hint = "MP3-320")]
|
||||
Quality = 1,
|
||||
|
||||
[FieldOption(Hint = "Example-Raws")]
|
||||
ReleaseGroup = 2,
|
||||
|
||||
[FieldOption(Hint = "2020")]
|
||||
Year = 3,
|
||||
|
||||
[FieldOption(Hint = "Torznab")]
|
||||
Indexer = 4,
|
||||
}
|
||||
}
|
@ -0,0 +1,35 @@
|
||||
using System.Collections.Generic;
|
||||
using Newtonsoft.Json;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood.Types
|
||||
{
|
||||
public sealed class Torrent
|
||||
{
|
||||
[JsonProperty(PropertyName = "bytesDone")]
|
||||
public long BytesDone { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "directory")]
|
||||
public string Directory { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "eta")]
|
||||
public long Eta { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "message")]
|
||||
public string Message { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "name")]
|
||||
public string Name { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "ratio")]
|
||||
public float Ratio { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "sizeBytes")]
|
||||
public long SizeBytes { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "status")]
|
||||
public List<string> Status { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "tags")]
|
||||
public List<string> Tags { get; set; }
|
||||
}
|
||||
}
|
@ -0,0 +1,10 @@
|
||||
using Newtonsoft.Json;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood.Types
|
||||
{
|
||||
public sealed class TorrentContent
|
||||
{
|
||||
[JsonProperty(PropertyName = "path")]
|
||||
public string Path { get; set; }
|
||||
}
|
||||
}
|
@ -0,0 +1,14 @@
|
||||
using System.Collections.Generic;
|
||||
using Newtonsoft.Json;
|
||||
|
||||
namespace NzbDrone.Core.Download.Clients.Flood.Types
|
||||
{
|
||||
public sealed class TorrentListSummary
|
||||
{
|
||||
[JsonProperty(PropertyName = "id")]
|
||||
public long Id { get; set; }
|
||||
|
||||
[JsonProperty(PropertyName = "torrents")]
|
||||
public Dictionary<string, Torrent> Torrents { get; set; }
|
||||
}
|
||||
}
|
Loading…
Reference in new issue