Using QueueBackgroundWorkItem to Schedule Background Jobs from an ASP.NET Application in .NET 4.5.2
Starting with the recently released version 4.5.2 of the .NET Framework, ASP.NET now supports the HostingEnvironment.QueueBackgroundWorkItem
method found in the System.Web.Hosting
namespace. I want to quickly show you how you can use it to schedule background work items in an ASP.NET MVC application.
What Does QueueBackgroundWorkItem
Do? #
In the release notes, the QueueBackgroundWorkItem
method is summarized as follows:
The
HostingEnvironment.QueueBackgroundWorkItem
method lets you schedule small background work items. ASP.NET tracks these items and prevents IIS from abruptly terminating the worker process until all background work items have completed.
The summary (emphasis mine) highlights the reason for using QueueBackgroundWorkItem
: You won't have to worry about processes being shut down prematurely by IIS.
Note that QueueBackgroundWorkItem
can only be called inside an ASP.NET managed app domain. It won't work if the runtime host is either Internet Explorer or some Windows shell. For more information on app domains, please refer to Using Application Domains.
Basic Usage of the QueueBackgroundWorkItem
Method #
The QueueBackgroundWorkItem
method defines two overloads, each of which accepts a single parameter. You can pass either of the following delegate types:
Action<CancellationToken>
Func<CancellationToken, Task>
Here's how you could pass a lambda expression to the first overload:
HostingEnvironment.QueueBackgroundWorkItem(cancellationToken =>
{
// Some long-running job
});
The lambda can even be async
so that you can use all the goodness await
has to offer:
HostingEnvironment.QueueBackgroundWorkItem(async cancellationToken =>
{
var result = await LongRunningMethodAsync();
// Do something with result
// ...
});
Passing a method group to the overload accepting a Func
is possible as well:
private void QueueWorkItem()
{
Func<CancellationToken, Task> workItem = LongRunningMethodAsync;
HostingEnvironment.QueueBackgroundWorkItem(workItem);
}
private async Task LongRunningMethodAsync(CancellationToken cancellationToken)
{
// Some long-running job
}
Due to the way that the C# compiler does method group conversion, it's not possible to pass LongRunningMethodAsync
directly to QueueBackgroundWorkItem
. The issue is that the compiler utilizes overload resolution when converting the method group, and overload resolution doesn't take into account return types.
Since both Action
and Func
overloads accept a single parameter of type CancellationToken
, there's no way to distinguish the two method calls just by looking at their parameter types. The assignment to the workItem
variable provides the compiler with that missing type information. For more detail, make sure to read this great StackOverflow answer from Eric Lippert Himself™.
Posting to a Remote API from an ASP.NET MVC Controller #
Here's a more complete example of how QueueBackgroundWorkItem
can be used in an ASP.NET MVC controller. After creating some Foo
model, the controller registers a background work item which makes a (potentially) long-running call to a remote API:
public class FooController : Controller
{
[HttpPost]
public ActionResult Create(FooInputModel input)
{
// Process the input somehow
// ...
Action<CancellationToken> workItem = PostToRemoteService;
HostingEnvironment.QueueBackgroundWorkItem(workItem);
return View();
}
private async void PostToRemoteService(CancellationToken cancellationToken)
{
using (var client = new HttpClient())
{
var response = await client.PostAsync("https://example.com/endpoint",
new StringContent("..."), cancellationToken);
// Do something with response
// ...
}
}
// More action methods
// ...
}
That way, the controller can return an ActionResult
(in this case, a view) after the input has been processed. It doesn't have to wait until the HTTP request to the remote API has been made and a response is returned.
Of course, you can call the QueueBackgroundWorkItem
method from other ASP.NET application types as well, it is in no way specific to MVC.
Summary #
As you've seen, the new QueueBackgroundWorkItem
method is very easy to use with different delegate parameters. ASP.NET does the heavy lifting for us by preventing IIS from terminating worker processes when there are any pending background work items. Consequently, HostingEnvironment.QueueBackgroundWorkItem
is an ideal candidate for scheduling small background jobs in .NET 4.5.2.
For an overview of all the other new features that shipped with .NET 4.5.2, read the blog post from the .NET engineering team or the summary in the MSDN library: