A library for implementing the Command Pattern, providing of a set
of base classes for operations and a mediator/invoker/dispatcher class for invoking them. Useful for abstracting data
access and API calls as either queries (for read operations) or commands (for write operations). Query
and Command
classes declare the type of context they require for execution, and the type of the result, and optionally the type of cache
options they require for caching the result. Parameters are modelled as properties on the Query
and Command
classes.
Define a query object:
public class PostById : AsyncQuery<HttpClient, Post>
{
// The context here is an HttpClient, but it could be an IDbConnection or anything you want
public override async Task<Post> Query(HttpClient context, CancellationToken cancellationToken = default)
{
var response = await context.GetAsync($"https://jsonplaceholder.typicode.com/posts/{Id}", cancellationToken);
return await response.Content.ReadAsAsync<Post>(cancellationToken);
}
// We could also add a constructor and make this a readonly property
public int Id { get; set; }
}
Invoke it:
// Allow IMagneto to fetch the appropriate context from the IServiceProvider
var post = await _magneto.QueryAsync(new PostById { Id = 1 });
// Or pass the context yourself by using IMediary
var post = await _mediary.QueryAsync(new PostById { Id = 1 }, _httpClient);
When using the provided base classes for queries and commands, they behave like value objects which allows easier mocking in unit tests (just make sure to model all the parameters as public properties):
// Moq
magnetoMock.Setup(x => x.QueryAsync(new PostById { Id = 1 })).ReturnsAsync(new Post());
// NSubstitute
magneto.QueryAsync(new PostById { Id = 1 }).Returns(new Post());
// Both of the above will be "strict" mocks that only match if the "Id" property is equal to 1
Use a ValueTuple for the context (if you require more than one service and don't want to create a class to hold them):
public class SaveAlbum : SyncCommand<(IFileProvider, JsonSerializerSettings)>
{
public override void Execute((IFileProvider, JsonSerializerSettings) context)
{
// Here we deconstruct the context into two local variables
var (fileProvider, jsonSerializerSettings) = context;
...
}
public Album Album { get; set; }
}
Leverage built-in caching by deriving from a base class and specifying the type of cache-entry options:
public class CommentsByPostId : AsyncCachedQuery<HttpClient, MemoryCacheEntryOptions, Comment[]>
{
// Here we get to specify which parameters comprise the cache key
protected override void CacheKey(ICache cache) => cache.VaryBy = PostId;
// Here we get to specify the caching policy (absolute/sliding)
protected override MemoryCacheEntryOptions CacheEntryOptions(HttpClient context) =>
new MemoryCacheEntryOptions().SetAbsoluteExpiration(TimeSpan.FromSeconds(30));
protected override async Task<Comment[]> Query(HttpClient context, CancellationToken cancellationToken = default)
{
var response = await context.GetAsync($"https://jsonplaceholder.typicode.com/posts/{PostId}/comments", cancellationToken);
return await response.Content.ReadAsAsync<Comment[]>(cancellationToken);
}
public int PostId { get; set; }
}
Cache intermediate results and transform to a final result:
public class UserById : AsyncTransformedCachedQuery<HttpClient, DistributedCacheEntryOptions, User[], User>
{
protected override DistributedCacheEntryOptions CacheEntryOptions(HttpClient context) =>
new DistributedCacheEntryOptions().SetAbsoluteExpiration(TimeSpan.FromSeconds(30));
protected override async Task<User[]> Query(HttpClient context, CancellationToken cancellationToken = default)
{
var response = await context.GetAsync("https://jsonplaceholder.typicode.com/users", cancellationToken);
return await response.Content.ReadAsAsync<User[]>(cancellationToken);
}
protected override Task<User> TransformCachedResult(User[] cachedResult, CancellationToken cancellationToken = default) =>
Task.FromResult(cachedResult.Single(x => x.Id == Id));
public int Id { get; set; }
}
Easily skip reading from the cache if required, by passing the argument CacheOption.Refresh
(the fresh result
will be written to the cache):
var postComments = await _magneto.QueryAsync(new CommentsByPostId { PostId = id }, CacheOption.Refresh);
Otherwise, when invoking a cached query, pass the argument CacheOption.Default
(or use the default
keyword). This
parameter is intentionally left as required rather than optional, to make it clear that a cached query is being used:
var postComments = await _magneto.QueryAsync(new CommentsByPostId { PostId = id }, CacheOption.Default);
Easily evict a previously cached result for a query:
var commentsByPostById = new CommentsByPostId { PostId = 1 };
var comments = await _magneto.QueryAsync(commentsByPostById, CacheOption.Default);
...
await _magneto.EvictCachedResultAsync(commentsByPostById);
When using a distributed cache store, changes to a previously cached result for a query can be updated:
var commentsByPostById = new CommentsByPostId { PostId = 1 };
var comments = await _magneto.QueryAsync(commentsByPostById, CacheOption.Refresh);
...
comments[0].Votes++;
await _magneto.UpdateCachedResultAsync(commentsByPostById);
Register a decorator to apply cross-cutting concerns:
// In application startup
...
services.AddSingleton<IDecorator, ApplicationInsightsDecorator>();
...
// Decorator implementation
public class ApplicationInsightsDecorator : IDecorator
{
readonly TelemetryClient _telemetryClient;
public ApplicationInsightsDecorator(TelemetryClient telemetryClient) => _telemetryClient = telemetryClient;
public TResult Decorate<TResult>(string operationName, Func<TResult> invoke)
{
try
{
var stopwatch = Stopwatch.StartNew();
var result = invoke();
var elapsed = stopwatch.Elapsed.TotalMilliseconds;
_telemetryClient.TrackMetric(operationName, elapsed);
return result;
}
catch (Exception e)
{
_telemetryClient.TrackException(e);
throw;
}
}
...
}
See the bundled sample application for further examples of usage.