mirror of
https://github.com/Cleanuparr/Cleanuparr.git
synced 2026-01-28 23:52:07 -05:00
192 lines
7.2 KiB
C#
192 lines
7.2 KiB
C#
using System.Collections.Concurrent;
|
|
using System.Text.RegularExpressions;
|
|
using Common.Configuration.Arr;
|
|
using Common.Configuration.ContentBlocker;
|
|
using Common.Configuration.DownloadClient;
|
|
using Domain.Enums;
|
|
using Domain.Models.Arr.Queue;
|
|
using Infrastructure.Configuration;
|
|
using Infrastructure.Helpers;
|
|
using Infrastructure.Services;
|
|
using Infrastructure.Verticals.Arr;
|
|
using Infrastructure.Verticals.Arr.Interfaces;
|
|
using Infrastructure.Verticals.DownloadClient;
|
|
using Infrastructure.Verticals.Jobs;
|
|
using Infrastructure.Verticals.Notifications;
|
|
using MassTransit;
|
|
using Microsoft.Extensions.Caching.Memory;
|
|
using Microsoft.Extensions.Logging;
|
|
using LogContext = Serilog.Context.LogContext;
|
|
|
|
namespace Infrastructure.Verticals.ContentBlocker;
|
|
|
|
public sealed class ContentBlocker : GenericHandler
|
|
{
|
|
private ContentBlockerConfig _config;
|
|
private readonly BlocklistProvider _blocklistProvider;
|
|
private readonly IIgnoredDownloadsService _ignoredDownloadsService;
|
|
private readonly IConfigManager _configManager;
|
|
|
|
public ContentBlocker(
|
|
ILogger<ContentBlocker> logger,
|
|
IConfigManager configManager,
|
|
IMemoryCache cache,
|
|
IBus messageBus,
|
|
ArrClientFactory arrClientFactory,
|
|
ArrQueueIterator arrArrQueueIterator,
|
|
BlocklistProvider blocklistProvider,
|
|
INotificationPublisher notifier,
|
|
IIgnoredDownloadsService ignoredDownloadsService,
|
|
DownloadServiceFactory downloadServiceFactory
|
|
) : base(
|
|
logger, cache, messageBus,
|
|
arrClientFactory, arrArrQueueIterator,
|
|
downloadServiceFactory, notifier
|
|
)
|
|
{
|
|
_configManager = configManager;
|
|
_blocklistProvider = blocklistProvider;
|
|
_ignoredDownloadsService = ignoredDownloadsService;
|
|
|
|
_config = configManager.GetConfiguration<ContentBlockerConfig>();
|
|
_downloadClientConfig = configManager.GetConfiguration<DownloadClientConfig>();
|
|
_sonarrConfig = configManager.GetConfiguration<SonarrConfig>();
|
|
_radarrConfig = configManager.GetConfiguration<RadarrConfig>();
|
|
_lidarrConfig = configManager.GetConfiguration<LidarrConfig>();
|
|
}
|
|
|
|
public override async Task ExecuteAsync()
|
|
{
|
|
if (_downloadClientConfig.Clients.Count is 0)
|
|
{
|
|
_logger.LogWarning("No download clients configured");
|
|
return;
|
|
}
|
|
|
|
bool blocklistIsConfigured = _config.Sonarr.Enabled && !string.IsNullOrEmpty(_config.Sonarr.Path) ||
|
|
_config.Radarr.Enabled && !string.IsNullOrEmpty(_config.Radarr.Path) ||
|
|
_config.Lidarr.Enabled && !string.IsNullOrEmpty(_config.Lidarr.Path);
|
|
|
|
if (!blocklistIsConfigured)
|
|
{
|
|
_logger.LogWarning("no blocklist is configured");
|
|
return;
|
|
}
|
|
|
|
await _blocklistProvider.LoadBlocklistsAsync();
|
|
await base.ExecuteAsync();
|
|
}
|
|
|
|
protected override async Task ProcessInstanceAsync(ArrInstance instance, InstanceType instanceType, ArrConfig config)
|
|
{
|
|
IReadOnlyList<string> ignoredDownloads = await _ignoredDownloadsService.GetIgnoredDownloadsAsync();
|
|
|
|
using var _ = LogContext.PushProperty("InstanceName", instanceType.ToString());
|
|
|
|
IArrClient arrClient = _arrClientFactory.GetClient(instanceType);
|
|
BlocklistType blocklistType = _blocklistProvider.GetBlocklistType(instanceType);
|
|
ConcurrentBag<string> patterns = _blocklistProvider.GetPatterns(instanceType);
|
|
ConcurrentBag<Regex> regexes = _blocklistProvider.GetRegexes(instanceType);
|
|
|
|
await _arrArrQueueIterator.Iterate(arrClient, instance, async items =>
|
|
{
|
|
var groups = items
|
|
.GroupBy(x => x.DownloadId)
|
|
.ToList();
|
|
|
|
foreach (var group in groups)
|
|
{
|
|
if (group.Any(x => !arrClient.IsRecordValid(x)))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
QueueRecord record = group.First();
|
|
|
|
if (record.Protocol is not "torrent")
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(record.DownloadId))
|
|
{
|
|
_logger.LogDebug("skip | download id is null for {title}", record.Title);
|
|
continue;
|
|
}
|
|
|
|
if (ignoredDownloads.Contains(record.DownloadId, StringComparer.InvariantCultureIgnoreCase))
|
|
{
|
|
_logger.LogInformation("skip | {title} | ignored", record.Title);
|
|
continue;
|
|
}
|
|
|
|
_logger.LogTrace("processing | {name}", record.Title);
|
|
|
|
string downloadRemovalKey = CacheKeys.DownloadMarkedForRemoval(record.DownloadId, instance.Url);
|
|
|
|
if (_cache.TryGetValue(downloadRemovalKey, out bool _))
|
|
{
|
|
_logger.LogDebug("skip | already marked for removal | {title}", record.Title);
|
|
continue;
|
|
}
|
|
|
|
_logger.LogDebug("searching unwanted files for {title}", record.Title);
|
|
bool found = false;
|
|
|
|
foreach (var downloadService in _downloadServices)
|
|
{
|
|
try
|
|
{
|
|
BlockFilesResult result = await downloadService
|
|
.BlockUnwantedFilesAsync(record.DownloadId, blocklistType, patterns, regexes, ignoredDownloads);
|
|
|
|
if (!result.Found)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
found = true;
|
|
|
|
if (!result.ShouldRemove)
|
|
{
|
|
break;
|
|
}
|
|
|
|
_logger.LogDebug("all files are marked as unwanted | {hash}", record.Title);
|
|
|
|
bool removeFromClient = true;
|
|
|
|
if (result.IsPrivate && !_config.DeletePrivate)
|
|
{
|
|
removeFromClient = false;
|
|
}
|
|
|
|
await PublishQueueItemRemoveRequest(
|
|
downloadRemovalKey,
|
|
instanceType,
|
|
instance,
|
|
record,
|
|
group.Count() > 1,
|
|
removeFromClient,
|
|
DeleteReason.AllFilesBlocked
|
|
);
|
|
|
|
break;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(
|
|
ex,
|
|
"Error blocking unwanted files for {hash} with download client",
|
|
record.DownloadId);
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
{
|
|
_logger.LogWarning("skip | download not found {title}", record.Title);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
} |