I have a series of class libraries that are used in asp.net-core
middleware, and in an IHostedService
.
To fetch the user context, I can inject IHttpContextAccessor
to grab the HttpContext
user:
public class MyLibrary
{
public MyLibrary(IHttpContextAccessor accessor)
{
// set the accessor - no problem
}
public async Task DoWorkAsync(SomeObject payload)
{
// get the user from the accessor
// do some work
}
}
To be a little more abstract, I have an IUserAccessor
with an HttpUserAccessor
implementation:
public class HttpUserAccessor: IUserAccessor
{
IHttpContextAccessor _httpaccessor;
public HttpUserAccessor(IHttpContextAccessor accessor)
{
_httpaccessor = accessor;
}
public string GetUser()
{
// return user from _httpaccessor
}
}
and then MyLibrary
does not need an IHttpContextAccessor
dependency:
public class MyLibrary
{
public MyLibrary(IUserAccessor accessor)
{
// set the accessor - no problem
}
public async Task DoWorkAsync(SomeObject payload)
{
// get the user from the accessor
// do some work
}
}
My IHostedService
is popping message from a queue, where the message includes:
- a user context, and
- a serialized
SomeObject
to pass toMyLibrary.DoWorkAsync
So, something like:
public class MyHostedService : IHostedService
{
IServiceScopeProvider _serviceScopeFactory;
public MyHostedService(IServiceScopeFactory serviceScopeFactory)
{
_serviceScopeFactory = servicesScopeFactory;
}
public Task StartAsync(CancellationToken cancellationToken)
{ ... }
public Task StopAsync(CancellationToken cancellationToken)
{ ... }
public async Task ExecuteAsync(CancellationToken stoppingToken)
{
foreach (var message in queue)
{
using (var scope = _serviceScopeFactory.CreateScope())
{
// todo: tell IUserAccessor what message.User is!
var payload = // create a SomeObject from the queue message
var mylibrary = _services.GetRequiredService<MyLibrary>();
await myLibrary.DoWorkAsync(payload);
}
}
}
}
So, my question is, how does MyHostedService
store message.User
in such a way that a custom IUserAccessor
can access it in a thread-safe manner via DI?
The thing you're looking for is
AsyncLocal<T>
- it's like a thread-local variable but scoped to a (possibly asynchronous) code block instead of a thread.I tend to prefer a "provider" + "accessor" pairing for this: one type that provides the value, and a separate type that reads the value. This is logically the same thing as a React Context in the JS world, though the implementation is quite different.
One tricky thing about
AsyncLocal<T>
is that you need to overwrite its value on any change. In this case, that's not really a problem (no message processing will want to update the "user"), but in the general case it's important to keep in mind. I prefer storing immutable types in theAsyncLocal<T>
to ensure they aren't mutated directly instead of overwriting the value. In this case, your "user" is astring
, which is already immutable, so that's perfect.First, you'll need to define the actual
AsyncLocal<T>
to hold the user value and define some low-level accessors. I strongly recommend usingIDisposable
to ensure theAsyncLocal<T>
value is unset properly at the end of the scope:Then you can define a provider:
and the accessor is similarly simple:
You'll want to set up your DI to point
IUserAccessor
toAsyncLocalUser.Accessor
. You can also optionally addAsyncLocalUser.Provider
to your DI, or you can just create it directly.Usage would go something like this: