Yesterday I blogged about using Castle.DynamicProxy to generate a proxy class and use AOP techniques for caching. The code I showed worked for synchronous method calls but fails when you want to proxy async method calls.
Let’s see how we can get this working for async…
Here is the async version of our repository:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public interface IAsyncProductRepository | |
{ | |
Task<IList<Product>> GetAll(); | |
} |
We need an extra NuGet package:
dotnet add Castle.Core.AsyncInterceptor
Now we have to change our interceptor to call a second async interceptor:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class AsyncCacheInterceptor : AsyncInterceptorBase | |
{ | |
private readonly IMemoryCache _cache; | |
public AsyncCacheInterceptor(IMemoryCache cache) | |
{ | |
_cache = cache; | |
} | |
protected override async Task InterceptAsync(IInvocation invocation, Func<IInvocation, Task> proceed) | |
{ | |
//Nothing to cache here --> method that returns a task has no return value | |
await proceed(invocation).ConfigureAwait(false); | |
} | |
protected override async Task<TResult> InterceptAsync<TResult>(IInvocation invocation, Func<IInvocation, Task<TResult>> proceed) | |
{ | |
var name = $"{invocation.Method.DeclaringType}_{invocation.Method.Name}"; | |
var args = string.Join(", ", invocation.Arguments.Select(a => (a ?? "").ToString())); | |
var cacheKey = $"{name}|{args}"; | |
if (!_cache.TryGetValue(cacheKey, out TResult returnValue)) | |
{ | |
returnValue = await proceed(invocation).ConfigureAwait(false); | |
_cache.Set(cacheKey, returnValue); | |
} | |
return returnValue; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class CacheInterceptor : IInterceptor | |
{ | |
private readonly AsyncCacheInterceptor _asyncInterceptor; | |
public CacheInterceptor(AsyncCacheInterceptor asyncInterceptor) | |
{ | |
_asyncInterceptor = asyncInterceptor; | |
} | |
public void Intercept(IInvocation invocation) | |
{ | |
_asyncInterceptor.ToInterceptor().Intercept(invocation); | |
} | |
} |
Our extension method that triggers the proxy creation remains the same:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public static class ServicesExtensions { | |
public static void AddProxiedScoped<TInterface, TImplementation>(this IServiceCollection services) | |
where TInterface : class | |
where TImplementation : class, TInterface | |
{ | |
services.AddScoped<TImplementation>(); | |
services.AddScoped(typeof(TInterface), serviceProvider => | |
{ | |
var proxyGenerator = serviceProvider.GetRequiredService<ProxyGenerator>(); | |
var actual = serviceProvider.GetRequiredService<TImplementation>(); | |
var interceptors = serviceProvider.GetServices<IInterceptor>().ToArray(); | |
return proxyGenerator.CreateInterfaceProxyWithTarget(typeof(TInterface), actual, interceptors); | |
}); | |
} | |
} |
Our registration code should be extended to register the async interceptor as well:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
services.AddSingleton(new ProxyGenerator()); | |
services.AddScoped<IInterceptor, CacheInterceptor>(); | |
services.AddScoped<AsyncCacheInterceptor>(); | |
services.AddMemoryCache(); | |
services.AddProxiedScoped<IAsyncProductRepository, ProductRepository>(); |