With `delete_old_custom_formats: false` and `replace_existing_custom_formats: false`, if you comment out a CF in your configuration, sync, uncomment it and sync again, you get an error about duplicate CFs. This is because, once a CF is removed from the configuration, it's also removed from the cache. This change makes the cache more flexible. As long as a CF (created by Recyclarr) exists either in the config OR in the service itself, it will be kept in the cache. This means that temporarily disabling CFs in configuration won't cause ownership issues.pull/201/head
parent
f06a2c829a
commit
f05ff6e04b
@ -0,0 +1,230 @@
|
|||||||
|
using Recyclarr.Cli.Cache;
|
||||||
|
using Recyclarr.Cli.Pipelines.CustomFormat;
|
||||||
|
using Recyclarr.TrashLib.TestLibrary;
|
||||||
|
|
||||||
|
namespace Recyclarr.Cli.Tests.Cache;
|
||||||
|
|
||||||
|
[TestFixture]
|
||||||
|
[Parallelizable(ParallelScope.All)]
|
||||||
|
public class CustomFormatCacheTest
|
||||||
|
{
|
||||||
|
[Test]
|
||||||
|
public void New_updated_and_changed_are_added()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData
|
||||||
|
{
|
||||||
|
NewCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("one", "1", 1),
|
||||||
|
NewCf.Data("two", "2", 2)
|
||||||
|
},
|
||||||
|
UpdatedCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("three", "3", 3)
|
||||||
|
},
|
||||||
|
UnchangedCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("four", "4", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache();
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Deleted_cfs_are_removed()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData
|
||||||
|
{
|
||||||
|
NewCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("one", "1", 1),
|
||||||
|
NewCf.Data("two", "2", 2)
|
||||||
|
},
|
||||||
|
DeletedCustomFormats =
|
||||||
|
{
|
||||||
|
new TrashIdMapping("3", "three", 3)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache
|
||||||
|
{
|
||||||
|
TrashIdMappings = new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Cfs_not_in_service_are_removed()
|
||||||
|
{
|
||||||
|
var serviceCfs = new[]
|
||||||
|
{
|
||||||
|
NewCf.Data("one", "1", 1),
|
||||||
|
NewCf.Data("two", "2", 2)
|
||||||
|
};
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache
|
||||||
|
{
|
||||||
|
TrashIdMappings = new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = cache.RemoveStale(serviceCfs);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Cache_update_skips_custom_formats_with_zero_id()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData
|
||||||
|
{
|
||||||
|
NewCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("one", "1", 1),
|
||||||
|
NewCf.Data("zero", "0")
|
||||||
|
},
|
||||||
|
UpdatedCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("two", "2", 2)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache();
|
||||||
|
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Existing_matching_mappings_should_be_replaced()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData
|
||||||
|
{
|
||||||
|
NewCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("one_new", "1", 1),
|
||||||
|
NewCf.Data("two_new", "2", 2)
|
||||||
|
},
|
||||||
|
UpdatedCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("three_new", "3", 3)
|
||||||
|
},
|
||||||
|
UnchangedCustomFormats =
|
||||||
|
{
|
||||||
|
NewCf.Data("four_new", "4", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache
|
||||||
|
{
|
||||||
|
TrashIdMappings = new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one_new", 1),
|
||||||
|
new TrashIdMapping("2", "two_new", 2),
|
||||||
|
new TrashIdMapping("3", "three_new", 3),
|
||||||
|
new TrashIdMapping("4", "four_new", 4)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Duplicate_mappings_should_be_removed()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData();
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache
|
||||||
|
{
|
||||||
|
TrashIdMappings = new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("12", "one2", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void Mappings_ordered_by_id()
|
||||||
|
{
|
||||||
|
var transactions = new CustomFormatTransactionData();
|
||||||
|
|
||||||
|
var cache = new CustomFormatCache
|
||||||
|
{
|
||||||
|
TrashIdMappings = new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = cache.Update(transactions);
|
||||||
|
|
||||||
|
result.TrashIdMappings.Should().BeEquivalentTo(new[]
|
||||||
|
{
|
||||||
|
new TrashIdMapping("1", "one", 1),
|
||||||
|
new TrashIdMapping("2", "two", 2),
|
||||||
|
new TrashIdMapping("3", "three", 3),
|
||||||
|
new TrashIdMapping("4", "four", 4)
|
||||||
|
}, o => o.WithStrictOrdering());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue