I'm developing an ASP.NET Core MVC web application where I have these two tasks that should be running as background services:
- Set the user status as "Expired" if EndOfSubscription date is == DateTime.Now
- Before 1 month of EndOfSubscription date send a reminder e-mail to this user
After searching, I found that I can use service worker to implement this. But I'm totally confused how to use this service worker in existing ASP.NET Core MVC web application where I need to access my models and database.
Should I isolate these tasks in a separate service worker project? But in this case should I share the same database for both projects?
Can someone guide me with main steps in this kind of situations?
Thank you in advance.
Service worker or Worker service?
The tutorial Background tasks with hosted services in ASP.NET Core shows how to create and use a BackgroundService but is a bit ... overengineered. The article tries to show too many things at the same time and ends up missing some essential things.
A better introduction is Steve Gordon's What are Worker Services?.
The background service
All that's needed to create a background service, is a class that implements the IHostedService interface. Instead of implementing all the interface methods, it's easier to inherit from the BackgroundService base class and override just the
ExecuteAsync
method.The article's example shows this method doesn't need to be anything fancy:
That's just a loop with a delay. This will run until the web app terminates and signals the
stoppingToken
. This service will be created by the DI container, so it can have service dependencies likeILogger
or any other singleton service.Registering the service
The background service needs to be registered as a service in
ConfigureServices
, the same way any other service is registered. If you have a console application, you configure it in the host'sConfigureServices
call. If you have a web application, you need to register it inStartup.ConfigureServices
:This registers
Worker
as a service that can be constructed by the DI container and adds it to the list of hosted services that will start once.Run()
is called in the web app'sMain
:Using DbContext and other scoped services
Adding a DbContext as a dependency is trickier, since
DbContext
is a scoped service. We can't just inject aDbContext
instance and store it in a field - a DbContext is meant to be used as a Unit-of-Work, something that collects all changes made for a single scenario and either commit all of them to the database or discard them. It's meant to be used inside ausing
block. If we dispose the singleDbContext
instance we injected though, where do we get a new one?To solve this, we have to inject the DI service,
IServiceProvider
, create a scope explicitly and get our DbContext from this scope:The
OrdersContext
will be disposed when the scope exits and any unsaved changes will be discarded.Nothing says the entire code needs to be inside
ExecuteAsync
. Once the code starts getting too long, we can easily extract the important code into a separate method :