diff --git a/code/backend/Cleanuparr.Api/Features/MalwareBlocker/Contracts/Requests/UpdateMalwareBlockerConfigRequest.cs b/code/backend/Cleanuparr.Api/Features/MalwareBlocker/Contracts/Requests/UpdateMalwareBlockerConfigRequest.cs
index ffd660ac..291a8d5c 100644
--- a/code/backend/Cleanuparr.Api/Features/MalwareBlocker/Contracts/Requests/UpdateMalwareBlockerConfigRequest.cs
+++ b/code/backend/Cleanuparr.Api/Features/MalwareBlocker/Contracts/Requests/UpdateMalwareBlockerConfigRequest.cs
@@ -1,5 +1,6 @@
using System.Collections.Generic;
+using Cleanuparr.Domain.Enums;
using Cleanuparr.Persistence.Models.Configuration.MalwareBlocker;
namespace Cleanuparr.Api.Features.MalwareBlocker.Contracts.Requests;
@@ -8,6 +9,8 @@ public sealed record UpdateMalwareBlockerConfigRequest
{
public bool Enabled { get; init; }
+ public MalwareBlockerTriggerMode TriggerMode { get; init; } = MalwareBlockerTriggerMode.Schedule;
+
public string CronExpression { get; init; } = "0/5 * * * * ?";
public bool UseAdvancedScheduling { get; init; }
@@ -35,6 +38,7 @@ public sealed record UpdateMalwareBlockerConfigRequest
public ContentBlockerConfig ApplyTo(ContentBlockerConfig config)
{
config.Enabled = Enabled;
+ config.TriggerMode = TriggerMode;
config.CronExpression = CronExpression;
config.UseAdvancedScheduling = UseAdvancedScheduling;
config.IgnorePrivate = IgnorePrivate;
diff --git a/code/backend/Cleanuparr.Domain/Enums/MalwareBlockerTriggerMode.cs b/code/backend/Cleanuparr.Domain/Enums/MalwareBlockerTriggerMode.cs
new file mode 100644
index 00000000..14713d0a
--- /dev/null
+++ b/code/backend/Cleanuparr.Domain/Enums/MalwareBlockerTriggerMode.cs
@@ -0,0 +1,8 @@
+namespace Cleanuparr.Domain.Enums;
+
+public enum MalwareBlockerTriggerMode
+{
+ Schedule,
+ Webhook,
+ Both,
+}
diff --git a/code/backend/Cleanuparr.Persistence/Migrations/Data/20260616155757_AddMalwareBlockerTriggerMode.Designer.cs b/code/backend/Cleanuparr.Persistence/Migrations/Data/20260616155757_AddMalwareBlockerTriggerMode.Designer.cs
new file mode 100644
index 00000000..3b22f8fa
--- /dev/null
+++ b/code/backend/Cleanuparr.Persistence/Migrations/Data/20260616155757_AddMalwareBlockerTriggerMode.Designer.cs
@@ -0,0 +1,2267 @@
+//
+using System;
+using System.Collections.Generic;
+using Cleanuparr.Persistence;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+
+#nullable disable
+
+namespace Cleanuparr.Persistence.Migrations.Data
+{
+ [DbContext(typeof(DataContext))]
+ [Migration("20260616155757_AddMalwareBlockerTriggerMode")]
+ partial class AddMalwareBlockerTriggerMode
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder.HasAnnotation("ProductVersion", "10.0.1");
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Arr.ArrConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("FailedImportMaxStrikes")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_max_strikes");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("type");
+
+ b.HasKey("Id")
+ .HasName("pk_arr_configs");
+
+ b.ToTable("arr_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Arr.ArrInstance", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("ApiKey")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("api_key");
+
+ b.Property("ArrConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("arr_config_id");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("ExternalUrl")
+ .HasColumnType("TEXT")
+ .HasColumnName("external_url");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Url")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("url");
+
+ b.Property("Version")
+ .HasColumnType("REAL")
+ .HasColumnName("version");
+
+ b.HasKey("Id")
+ .HasName("pk_arr_instances");
+
+ b.HasIndex("ArrConfigId")
+ .HasDatabaseName("ix_arr_instances_arr_config_id");
+
+ b.ToTable("arr_instances", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.BlacklistSync.BlacklistSyncConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("BlacklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("blacklist_path");
+
+ b.Property("CronExpression")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("cron_expression");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.HasKey("Id")
+ .HasName("pk_blacklist_sync_configs");
+
+ b.ToTable("blacklist_sync_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.DeadTorrentConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("MaxStrikes")
+ .HasColumnType("INTEGER")
+ .HasColumnName("max_strikes");
+
+ b.Property("TargetCategory")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("target_category");
+
+ b.Property("UseTag")
+ .HasColumnType("INTEGER")
+ .HasColumnName("use_tag");
+
+ b.HasKey("Id")
+ .HasName("pk_dead_torrent_configs");
+
+ b.HasIndex("DownloadClientConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_dead_torrent_configs_download_client_config_id");
+
+ b.ToTable("dead_torrent_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.DelugeSeedingRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DeleteSourceFiles")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_source_files");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("MaxRatio")
+ .HasColumnType("REAL")
+ .HasColumnName("max_ratio");
+
+ b.Property("MaxSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("max_seed_time");
+
+ b.Property("MinSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("min_seed_time");
+
+ b.Property("MinSeeders")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_seeders");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("PrivacyType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("privacy_type");
+
+ b.Property("TrackerPatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tracker_patterns");
+
+ b.HasKey("Id")
+ .HasName("pk_deluge_seeding_rules");
+
+ b.HasIndex("DownloadClientConfigId")
+ .HasDatabaseName("ix_deluge_seeding_rules_download_client_config_id");
+
+ b.ToTable("deluge_seeding_rules", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.DownloadCleanerConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("CronExpression")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("cron_expression");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.PrimitiveCollection("IgnoredDownloads")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("ignored_downloads");
+
+ b.Property("UseAdvancedScheduling")
+ .HasColumnType("INTEGER")
+ .HasColumnName("use_advanced_scheduling");
+
+ b.HasKey("Id")
+ .HasName("pk_download_cleaner_configs");
+
+ b.ToTable("download_cleaner_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.OrphanedFilesConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("ExcludePatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("exclude_patterns");
+
+ b.Property("MinFileAgeHours")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_file_age_hours");
+
+ b.Property("OrphanedDirectory")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("orphaned_directory");
+
+ b.Property("PurgeAfterHours")
+ .HasColumnType("INTEGER")
+ .HasColumnName("purge_after_hours");
+
+ b.Property("ScanDirectories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("scan_directories");
+
+ b.HasKey("Id")
+ .HasName("pk_orphaned_files_configs");
+
+ b.HasIndex("DownloadClientConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_orphaned_files_configs_download_client_config_id");
+
+ b.ToTable("orphaned_files_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.QBitSeedingRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DeleteSourceFiles")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_source_files");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("MaxRatio")
+ .HasColumnType("REAL")
+ .HasColumnName("max_ratio");
+
+ b.Property("MaxSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("max_seed_time");
+
+ b.Property("MinSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("min_seed_time");
+
+ b.Property("MinSeeders")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_seeders");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("PrivacyType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("privacy_type");
+
+ b.Property("TagsAll")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags_all");
+
+ b.Property("TagsAny")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags_any");
+
+ b.Property("TrackerPatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tracker_patterns");
+
+ b.HasKey("Id")
+ .HasName("pk_q_bit_seeding_rules");
+
+ b.HasIndex("DownloadClientConfigId")
+ .HasDatabaseName("ix_q_bit_seeding_rules_download_client_config_id");
+
+ b.ToTable("q_bit_seeding_rules", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.RTorrentSeedingRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DeleteSourceFiles")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_source_files");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("MaxRatio")
+ .HasColumnType("REAL")
+ .HasColumnName("max_ratio");
+
+ b.Property("MaxSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("max_seed_time");
+
+ b.Property("MinSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("min_seed_time");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("PrivacyType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("privacy_type");
+
+ b.Property("TrackerPatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tracker_patterns");
+
+ b.HasKey("Id")
+ .HasName("pk_r_torrent_seeding_rules");
+
+ b.HasIndex("DownloadClientConfigId")
+ .HasDatabaseName("ix_r_torrent_seeding_rules_download_client_config_id");
+
+ b.ToTable("r_torrent_seeding_rules", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.TransmissionSeedingRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DeleteSourceFiles")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_source_files");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("MaxRatio")
+ .HasColumnType("REAL")
+ .HasColumnName("max_ratio");
+
+ b.Property("MaxSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("max_seed_time");
+
+ b.Property("MinSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("min_seed_time");
+
+ b.Property("MinSeeders")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_seeders");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("PrivacyType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("privacy_type");
+
+ b.Property("TagsAll")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags_all");
+
+ b.Property("TagsAny")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags_any");
+
+ b.Property("TrackerPatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tracker_patterns");
+
+ b.HasKey("Id")
+ .HasName("pk_transmission_seeding_rules");
+
+ b.HasIndex("DownloadClientConfigId")
+ .HasDatabaseName("ix_transmission_seeding_rules_download_client_config_id");
+
+ b.ToTable("transmission_seeding_rules", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.UTorrentSeedingRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DeleteSourceFiles")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_source_files");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("MaxRatio")
+ .HasColumnType("REAL")
+ .HasColumnName("max_ratio");
+
+ b.Property("MaxSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("max_seed_time");
+
+ b.Property("MinSeedTime")
+ .HasColumnType("REAL")
+ .HasColumnName("min_seed_time");
+
+ b.Property("MinSeeders")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_seeders");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("PrivacyType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("privacy_type");
+
+ b.Property("TrackerPatterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tracker_patterns");
+
+ b.HasKey("Id")
+ .HasName("pk_u_torrent_seeding_rules");
+
+ b.HasIndex("DownloadClientConfigId")
+ .HasDatabaseName("ix_u_torrent_seeding_rules_download_client_config_id");
+
+ b.ToTable("u_torrent_seeding_rules", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadCleaner.UnlinkedConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.PrimitiveCollection("Categories")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("categories");
+
+ b.Property("DownloadClientConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_client_config_id");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.PrimitiveCollection("IgnoredRootDirs")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("ignored_root_dirs");
+
+ b.Property("TargetCategory")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("target_category");
+
+ b.Property("UseTag")
+ .HasColumnType("INTEGER")
+ .HasColumnName("use_tag");
+
+ b.HasKey("Id")
+ .HasName("pk_unlinked_configs");
+
+ b.HasIndex("DownloadClientConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_unlinked_configs_download_client_config_id");
+
+ b.ToTable("unlinked_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.DownloadClientConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("DownloadDirectorySource")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_directory_source");
+
+ b.Property("DownloadDirectoryTarget")
+ .HasColumnType("TEXT")
+ .HasColumnName("download_directory_target");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("ExternalUrl")
+ .HasColumnType("TEXT")
+ .HasColumnName("external_url");
+
+ b.Property("Host")
+ .HasColumnType("TEXT")
+ .HasColumnName("host");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("Password")
+ .HasColumnType("TEXT")
+ .HasColumnName("password");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("type");
+
+ b.Property("TypeName")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("type_name");
+
+ b.Property("UrlBase")
+ .HasColumnType("TEXT")
+ .HasColumnName("url_base");
+
+ b.Property("Username")
+ .HasColumnType("TEXT")
+ .HasColumnName("username");
+
+ b.HasKey("Id")
+ .HasName("pk_download_clients");
+
+ b.ToTable("download_clients", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.General.GeneralConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("DisplaySupportBanner")
+ .HasColumnType("INTEGER")
+ .HasColumnName("display_support_banner");
+
+ b.Property("DryRun")
+ .HasColumnType("INTEGER")
+ .HasColumnName("dry_run");
+
+ b.Property("EncryptionKey")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("encryption_key");
+
+ b.Property("HttpCertificateValidation")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("http_certificate_validation");
+
+ b.Property("HttpMaxRetries")
+ .HasColumnType("INTEGER")
+ .HasColumnName("http_max_retries");
+
+ b.Property("HttpTimeout")
+ .HasColumnType("INTEGER")
+ .HasColumnName("http_timeout");
+
+ b.PrimitiveCollection("IgnoredDownloads")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("ignored_downloads");
+
+ b.Property("StatusCheckEnabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("status_check_enabled");
+
+ b.Property("StrikeInactivityWindowHours")
+ .HasColumnType("INTEGER")
+ .HasColumnName("strike_inactivity_window_hours");
+
+ b.ComplexProperty(typeof(Dictionary), "Auth", "Cleanuparr.Persistence.Models.Configuration.General.GeneralConfig.Auth#AuthConfig", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("DisableAuthForLocalAddresses")
+ .HasColumnType("INTEGER")
+ .HasColumnName("auth_disable_auth_for_local_addresses");
+
+ b1.Property("TrustForwardedHeaders")
+ .HasColumnType("INTEGER")
+ .HasColumnName("auth_trust_forwarded_headers");
+
+ b1.Property("TrustedNetworks")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("auth_trusted_networks");
+ });
+
+ b.ComplexProperty(typeof(Dictionary), "Log", "Cleanuparr.Persistence.Models.Configuration.General.GeneralConfig.Log#LoggingConfig", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("ArchiveEnabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_archive_enabled");
+
+ b1.Property("ArchiveRetainedCount")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_archive_retained_count");
+
+ b1.Property("ArchiveTimeLimitHours")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_archive_time_limit_hours");
+
+ b1.Property("Level")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("log_level");
+
+ b1.Property("RetainedFileCount")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_retained_file_count");
+
+ b1.Property("RollingSizeMB")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_rolling_size_mb");
+
+ b1.Property("TimeLimitHours")
+ .HasColumnType("INTEGER")
+ .HasColumnName("log_time_limit_hours");
+ });
+
+ b.HasKey("Id")
+ .HasName("pk_general_configs");
+
+ b.ToTable("general_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("CronExpression")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("cron_expression");
+
+ b.Property("DeleteIfAnyFileBlocked")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_if_any_file_blocked");
+
+ b.Property("DeletePrivate")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_private");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("IgnorePrivate")
+ .HasColumnType("INTEGER")
+ .HasColumnName("ignore_private");
+
+ b.PrimitiveCollection("IgnoredDownloads")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("ignored_downloads");
+
+ b.Property("ProcessNoContentId")
+ .HasColumnType("INTEGER")
+ .HasColumnName("process_no_content_id");
+
+ b.Property("TriggerMode")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("trigger_mode");
+
+ b.Property("UseAdvancedScheduling")
+ .HasColumnType("INTEGER")
+ .HasColumnName("use_advanced_scheduling");
+
+ b.ComplexProperty(typeof(Dictionary), "Lidarr", "Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig.Lidarr#BlocklistSettings", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("BlocklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("lidarr_blocklist_path");
+
+ b1.Property("BlocklistType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("lidarr_blocklist_type");
+
+ b1.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("lidarr_enabled");
+ });
+
+ b.ComplexProperty(typeof(Dictionary), "Radarr", "Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig.Radarr#BlocklistSettings", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("BlocklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("radarr_blocklist_path");
+
+ b1.Property("BlocklistType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("radarr_blocklist_type");
+
+ b1.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("radarr_enabled");
+ });
+
+ b.ComplexProperty(typeof(Dictionary), "Readarr", "Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig.Readarr#BlocklistSettings", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("BlocklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("readarr_blocklist_path");
+
+ b1.Property("BlocklistType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("readarr_blocklist_type");
+
+ b1.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("readarr_enabled");
+ });
+
+ b.ComplexProperty(typeof(Dictionary), "Sonarr", "Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig.Sonarr#BlocklistSettings", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("BlocklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("sonarr_blocklist_path");
+
+ b1.Property("BlocklistType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("sonarr_blocklist_type");
+
+ b1.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("sonarr_enabled");
+ });
+
+ b.ComplexProperty(typeof(Dictionary), "Whisparr", "Cleanuparr.Persistence.Models.Configuration.MalwareBlocker.ContentBlockerConfig.Whisparr#BlocklistSettings", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("BlocklistPath")
+ .HasColumnType("TEXT")
+ .HasColumnName("whisparr_blocklist_path");
+
+ b1.Property("BlocklistType")
+ .HasColumnType("INTEGER")
+ .HasColumnName("whisparr_blocklist_type");
+
+ b1.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("whisparr_enabled");
+ });
+
+ b.HasKey("Id")
+ .HasName("pk_content_blocker_configs");
+
+ b.ToTable("content_blocker_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.AppriseConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("Key")
+ .IsRequired()
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("key");
+
+ b.Property("Mode")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("mode");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("ServiceUrls")
+ .HasMaxLength(4000)
+ .HasColumnType("TEXT")
+ .HasColumnName("service_urls");
+
+ b.Property("Tags")
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("tags");
+
+ b.Property("Url")
+ .IsRequired()
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("url");
+
+ b.HasKey("Id")
+ .HasName("pk_apprise_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_apprise_configs_notification_config_id");
+
+ b.ToTable("apprise_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.DiscordConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("AvatarUrl")
+ .IsRequired()
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("avatar_url");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("Username")
+ .IsRequired()
+ .HasMaxLength(80)
+ .HasColumnType("TEXT")
+ .HasColumnName("username");
+
+ b.Property("WebhookUrl")
+ .IsRequired()
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("webhook_url");
+
+ b.HasKey("Id")
+ .HasName("pk_discord_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_discord_configs_notification_config_id");
+
+ b.ToTable("discord_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.GotifyConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("ApplicationToken")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("TEXT")
+ .HasColumnName("application_token");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("Priority")
+ .HasColumnType("INTEGER")
+ .HasColumnName("priority");
+
+ b.Property("ServerUrl")
+ .IsRequired()
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("server_url");
+
+ b.HasKey("Id")
+ .HasName("pk_gotify_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_gotify_configs_notification_config_id");
+
+ b.ToTable("gotify_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.NotifiarrConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("ApiKey")
+ .IsRequired()
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("api_key");
+
+ b.Property("ChannelId")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("TEXT")
+ .HasColumnName("channel_id");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.HasKey("Id")
+ .HasName("pk_notifiarr_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_notifiarr_configs_notification_config_id");
+
+ b.ToTable("notifiarr_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.NotificationConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("CreatedAt")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("created_at");
+
+ b.Property("IsEnabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("is_enabled");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("TEXT")
+ .HasColumnName("name");
+
+ b.Property("OnCategoryChanged")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_category_changed");
+
+ b.Property("OnDownloadCleaned")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_download_cleaned");
+
+ b.Property("OnFailedImportStrike")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_failed_import_strike");
+
+ b.Property("OnQueueItemDeleted")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_queue_item_deleted");
+
+ b.Property("OnSearchItemGrabbed")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_search_item_grabbed");
+
+ b.Property("OnSearchTriggered")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_search_triggered");
+
+ b.Property("OnSlowStrike")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_slow_strike");
+
+ b.Property("OnStalledStrike")
+ .HasColumnType("INTEGER")
+ .HasColumnName("on_stalled_strike");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("type");
+
+ b.Property("UpdatedAt")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("updated_at");
+
+ b.HasKey("Id")
+ .HasName("pk_notification_configs");
+
+ b.HasIndex("Name")
+ .IsUnique()
+ .HasDatabaseName("ix_notification_configs_name");
+
+ b.ToTable("notification_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.NtfyConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("AccessToken")
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("access_token");
+
+ b.Property("AuthenticationType")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("authentication_type");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("Password")
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("password");
+
+ b.Property("Priority")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("priority");
+
+ b.Property("ServerUrl")
+ .IsRequired()
+ .HasMaxLength(500)
+ .HasColumnType("TEXT")
+ .HasColumnName("server_url");
+
+ b.PrimitiveCollection("Tags")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags");
+
+ b.PrimitiveCollection("Topics")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("topics");
+
+ b.Property("Username")
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("username");
+
+ b.HasKey("Id")
+ .HasName("pk_ntfy_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_ntfy_configs_notification_config_id");
+
+ b.ToTable("ntfy_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.PushoverConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("ApiToken")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("TEXT")
+ .HasColumnName("api_token");
+
+ b.Property("Devices")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("devices");
+
+ b.Property("Expire")
+ .HasColumnType("INTEGER")
+ .HasColumnName("expire");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("Priority")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("priority");
+
+ b.Property("Retry")
+ .HasColumnType("INTEGER")
+ .HasColumnName("retry");
+
+ b.Property("Sound")
+ .HasMaxLength(50)
+ .HasColumnType("TEXT")
+ .HasColumnName("sound");
+
+ b.Property("Tags")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("tags");
+
+ b.Property("UserKey")
+ .IsRequired()
+ .HasMaxLength(50)
+ .HasColumnType("TEXT")
+ .HasColumnName("user_key");
+
+ b.HasKey("Id")
+ .HasName("pk_pushover_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_pushover_configs_notification_config_id");
+
+ b.ToTable("pushover_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.Notification.TelegramConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("BotToken")
+ .IsRequired()
+ .HasMaxLength(255)
+ .HasColumnType("TEXT")
+ .HasColumnName("bot_token");
+
+ b.Property("ChatId")
+ .IsRequired()
+ .HasMaxLength(100)
+ .HasColumnType("TEXT")
+ .HasColumnName("chat_id");
+
+ b.Property("NotificationConfigId")
+ .HasColumnType("TEXT")
+ .HasColumnName("notification_config_id");
+
+ b.Property("SendSilently")
+ .HasColumnType("INTEGER")
+ .HasColumnName("send_silently");
+
+ b.Property("TopicId")
+ .HasMaxLength(100)
+ .HasColumnType("TEXT")
+ .HasColumnName("topic_id");
+
+ b.HasKey("Id")
+ .HasName("pk_telegram_configs");
+
+ b.HasIndex("NotificationConfigId")
+ .IsUnique()
+ .HasDatabaseName("ix_telegram_configs_notification_config_id");
+
+ b.ToTable("telegram_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.QueueCleaner.QueueCleanerConfig", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("CronExpression")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("cron_expression");
+
+ b.Property("DownloadingMetadataMaxStrikes")
+ .HasColumnType("INTEGER")
+ .HasColumnName("downloading_metadata_max_strikes");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.PrimitiveCollection("IgnoredDownloads")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("ignored_downloads");
+
+ b.Property("ProcessNoContentId")
+ .HasColumnType("INTEGER")
+ .HasColumnName("process_no_content_id");
+
+ b.Property("UseAdvancedScheduling")
+ .HasColumnType("INTEGER")
+ .HasColumnName("use_advanced_scheduling");
+
+ b.ComplexProperty(typeof(Dictionary), "FailedImport", "Cleanuparr.Persistence.Models.Configuration.QueueCleaner.QueueCleanerConfig.FailedImport#FailedImportConfig", b1 =>
+ {
+ b1.IsRequired();
+
+ b1.Property("ChangeCategory")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_change_category");
+
+ b1.Property("DeletePrivate")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_delete_private");
+
+ b1.Property("IgnorePrivate")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_ignore_private");
+
+ b1.Property("MaxStrikes")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_max_strikes");
+
+ b1.Property("PatternMode")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("failed_import_pattern_mode");
+
+ b1.PrimitiveCollection("Patterns")
+ .IsRequired()
+ .HasColumnType("TEXT")
+ .HasColumnName("failed_import_patterns");
+
+ b1.Property("SkipIfNotFoundInClient")
+ .HasColumnType("INTEGER")
+ .HasColumnName("failed_import_skip_if_not_found_in_client");
+ });
+
+ b.HasKey("Id")
+ .HasName("pk_queue_cleaner_configs");
+
+ b.ToTable("queue_cleaner_configs", (string)null);
+ });
+
+ modelBuilder.Entity("Cleanuparr.Persistence.Models.Configuration.QueueCleaner.SlowRule", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("TEXT")
+ .HasColumnName("id");
+
+ b.Property("ChangeCategory")
+ .HasColumnType("INTEGER")
+ .HasColumnName("change_category");
+
+ b.Property("DeletePrivateTorrentsFromClient")
+ .HasColumnType("INTEGER")
+ .HasColumnName("delete_private_torrents_from_client");
+
+ b.Property("Enabled")
+ .HasColumnType("INTEGER")
+ .HasColumnName("enabled");
+
+ b.Property("IgnoreAboveSize")
+ .HasColumnType("TEXT")
+ .HasColumnName("ignore_above_size");
+
+ b.Property("MaxCompletionPercentage")
+ .HasColumnType("INTEGER")
+ .HasColumnName("max_completion_percentage");
+
+ b.Property("MaxStrikes")
+ .HasColumnType("INTEGER")
+ .HasColumnName("max_strikes");
+
+ b.Property("MaxTimeHours")
+ .HasColumnType("REAL")
+ .HasColumnName("max_time_hours");
+
+ b.Property("MinCompletionPercentage")
+ .HasColumnType("INTEGER")
+ .HasColumnName("min_completion_percentage");
+
+ b.Property