Resolving Autofac components in background Tasks in ASP.NET

前端 未结 2 883
执笔经年
执笔经年 2020-12-19 10:20

Using Autofac in ASP.NET along with the ContainerDisposalModule, how can i support fire and forget calls that have component dependencies that need to be resolved? The prob

相关标签:
2条回答
  • 2020-12-19 10:43

    Answer posted by Alex adapted to current Autofac and MVC versions:

    • Use InstancePerRequest for a database context
    • Add ILifetimeScope as dependency to get to the container
    • SingleInstance ensures it's the root lifetime scope
    • Use HostingEnvironment.QueueBackgroundWorkItem to reliably run something in the background
    • Use MatchingScopeLifetimeTags.RequestLifetimeScopeTag to avoid having to know the tagname autofac uses for PerRequest lifetimes

    https://groups.google.com/forum/#!topic/autofac/gJYDDls981A https://groups.google.com/forum/#!topic/autofac/yGQWjVbPYGM

    Gist: https://gist.github.com/janv8000/35e6250c8efc00288d21

    Global.asax.cs:

    protected void Application_Start() {
      //Other registrations
      builder.RegisterType<ListingService>();
      builder.RegisterType<WebsiteContext>().As<IWebsiteContext>().InstancePerRequest();  //WebsiteContext is a EF DbContext
      builder.RegisterType<AsyncRunner>().As<IAsyncRunner>().SingleInstance();
    }
    

    AsyncRunner.cs

    public interface IAsyncRunner
    {
        void Run<T>(Action<T> action);
    }
    
    public class AsyncRunner : IAsyncRunner
    {
        public ILifetimeScope LifetimeScope { get; set; }
    
        public AsyncRunner(ILifetimeScope lifetimeScope)
        {
            Guard.NotNull(() => lifetimeScope, lifetimeScope);
            LifetimeScope = lifetimeScope;
        }
    
        public void Run<T>(Action<T> action)
        {
            HostingEnvironment.QueueBackgroundWorkItem(ct =>
            {
                // Create a nested container which will use the same dependency
                // registrations as set for HTTP request scopes.
                using (var container = LifetimeScope.BeginLifetimeScope(MatchingScopeLifetimeTags.RequestLifetimeScopeTag))
                {
                    var service = container.Resolve<T>();
                    action(service);
                }
            });
        }
    }
    

    Controller

    public Controller(IAsyncRunner asyncRunner)
    {
      Guard.NotNull(() => asyncRunner, asyncRunner);
      AsyncRunner = asyncRunner;
    }
    
    public ActionResult Index()
    {
      //Snip
      AsyncRunner.Run<ListingService>(listingService => listingService.RenderListing(listingGenerationArguments, Thread.CurrentThread.CurrentCulture));
      //Snip
    }
    

    ListingService

    public class ListingService : IListingService
    {
      public ListingService(IWebsiteContext context)
      {
        Guard.NotNull(() => context, context);
        Context = context;
      }
    }
    
    0 讨论(0)
  • 2020-12-19 10:45

    You need to create a new lifetime scope that is independent of the request lifetime scope. The blog post below shows an example of how to do this using MVC but the same concept can be applied to WebForms.

    http://aboutcode.net/2010/11/01/start-background-tasks-from-mvc-actions-using-autofac.html

    If you need to ensure that the async work is definitely performed after the request is finished then this is not a good approach. In such cases I would recommend posting a message onto a queue during the request allowing a separate process to pick it up and perform the work.

    0 讨论(0)
提交回复
热议问题