-
Notifications
You must be signed in to change notification settings - Fork 792
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
HybridCache: Unit Testing Support #5763
Comments
Related, I think it would be nice if there was some sort of |
That's ... not that simple. I can imagine a home-brew implementation something like: sealed class MockHybridCache : HybridCache
{
private readonly ConcurrentDictionary<string, object> _values = new();
public object? this[string key]
{
get => _values.TryGetValue(key, out var val) ? val : null;
set => _values[key] = value;
}
public override ValueTask<T> GetOrCreateAsync<TState, T>(string key, TState state, Func<TState, CancellationToken, ValueTask<T>> factory,
HybridCacheEntryOptions? options = null, IEnumerable<string>? tags = null, CancellationToken cancellationToken = default)
=> _values.TryGetValue(key, out var value) ? new((T)value) : factory(state, cancellationToken);
public override ValueTask RemoveAsync(string key, CancellationToken cancellationToken = default) { _values.TryRemove(key); return default; }
public override ValueTask RemoveByTagAsync(string tag, CancellationToken cancellationToken = default) => default;
public override ValueTask SetAsync<T>(string key, T value, HybridCacheEntryOptions? options = null, IEnumerable<string>? tags = null,
CancellationToken cancellationToken = default) => default;
} But I suspect the requirements on a per-scenario basis would be more bespoke and unique. If anything, I think we'd need to actually try and gather some requirements, i.e. how is this meant to be used? what features does it need to support? I'd be disinclined to simply rush in and throw together random ideas, without exploring it first. So: what do we need here?
That's pretty easy to implement if needed: sealed class FakeHybridCache : HybridCache
{
public override ValueTask<T> GetOrCreateAsync<TState, T>(string key, TState state, Func<TState, CancellationToken, ValueTask<T>> factory,
HybridCacheEntryOptions? options = null, IEnumerable<string>? tags = null, CancellationToken cancellationToken = default)
=> factory(state, cancellationToken);
public override ValueTask RemoveAsync(string key, CancellationToken cancellationToken = default) => default;
public override ValueTask RemoveByTagAsync(string tag, CancellationToken cancellationToken = default) => default;
public override ValueTask SetAsync<T>(string key, T value, HybridCacheEntryOptions? options = null, IEnumerable<string>? tags = null,
CancellationToken cancellationToken = default) => default;
} I'm ... not sure how useful that would be to add, and I certainly wouldn't want to make that the base implementation, as that's a foot-shotgun waiting to happen. |
My use cases are:
The first case is more important than anything, so that when I'm using the abstraction I can stub in what should be returned without having to do anything more complex than |
Actually, for preloading data: Does that just leave the "what was accessed"? We do have some inbuilt metrics, but these are currently using the usual metric APIs - so we don't retain much data that we can access directly. Since we defer L1 to |
Implementing an official Start with the abstraction, and implement a new, fake, implementation of A hypothetical If this proves too much for |
Currently, writing tests with the
HybridCache
abstraction is a little clunky. Essentially, you have to mock out the more complexGetOrCreateAsync()
method. While this isn't hard to write a simple set of extensions to do so (which I'll include below), having something akin to whatTimeProvider
has, Microsoft.Extensions.TimeProvider.Testing that provides aFakeTimeProvider
, would be most welcome.I don't foresee a hypothetical
FakeHybridCache
having to do much, as the only contracted method isGetOrCreateAsync()
. There's no need to add support for building out fake memory / distributed caches to control cache hits/misses. Rather, having a simple way to control what gets returned when called with a specific key (or tags perhaps too) would be all that's required.Here's a little something I whipped up, using
NSubstitute
, to support both "I want to mock some fake return data from my cache" and "I want to assert my cache was called / used." Obviously this is quick and dirty and I just passnull
or "any argument allowed" for the majority of things, but you can see where something like this would be beneficial in an official capacity. (Not strictly forNSubstitute
, mind; methods that simply let us set up fake data or test that calls happened (a simple counter) would be all that's needed.)And my unit test code has something like this:
There's tons to improve here, but here's a super basic not-fully-implemented
FakeHybridCache
that could be a jumping off point.The text was updated successfully, but these errors were encountered: