在asp.net核心应用程序中实现了IHostedService,如何在没有IIS首次请求的情况下运行它?

我已经在asp.net核心网站中实现了IHostedService。效果很好,但问题是我希望在托管服务器启动或IIS服务重新启动时启动它,但除非有第一个对网站的请求进入,否则它不会启动。

  • 该网站托管在IIS 10.0.18版上
  • AppPool处于“始终运行”模式
  • “ PreloadEnabled”在网站上为“ True”。
  • 将“ .NET CLR版本”设置为“无托管代码”或“ v4.0.xxxxxx”没有帮助。
  • dotnet核心版本为2.2,并安装了dotnet核心托管捆绑软件。

更新1: @Arthur建议的“应用程序初始化模块”没有帮助。既不在站点级别,也不在服务器级别。

我使用的配置:

    <applicationInitialization
         doAppInitAfterRestart="true"
         skipManagedmodules="false"
         remapManagedRequestsTo="init.htm">
        <add initializationPage="/init.htm" hostName="localhost"/>
    </applicationInitialization>

更新2:这是我实现界面的方式

internal class PaymentQueueService : IHostedService,IDisposable
{
    private readonly ILogger _logService;
    private Timer _timerEnqueue;

    public PaymentQueueService(ILogger logService)
    {
        _logService = logService;
    }

    public Task StartAsync(CancellationToken cancellationToken)
    {
        _logService.LogInformation("Starting processing payments.");

        _timerEnqueue = new Timer(EnqueuePayments,null,TimeSpan.Zero,TimeSpan.FromSeconds(10));

        return Task.CompletedTask;
    }

    private void EnqueuePayments(object state)
    {
        _logService.LogInformation("Enqueueing Payments.");
    }

    public Task StopAsync(CancellationToken cancellationToken)
    {
        _logService.LogInformation("Stopping processing payments.");

        _timerEnqueue?.Change(Timeout.Infinite,0);

        return Task.CompletedTask;
    }

    public void Dispose()
    {
        _timerEnqueue?.Dispose();
    }
}

main.cs文件中的Program类:

public class Program
    {
        public static void Main(string[] args)
        {
            CreateWebHostBuilder(args).Build().Run();
        }

        public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
            WebHost.CreateDefaultBuilder(args).ConfigureServices(services =>
            {
                services.AddHostedService<PaymentQueueService>();
            }).Configure((IApplicationBuilder app) =>
            {

                app.UseMvc();
            })
                .UseStartup<Startup>();
    }

启动类:

public class Startup
    {
        public Startup(IConfiguration configuration)
        {
            Configuration = configuration;
        }

        public IConfiguration Configuration { get; }

        // This method gets called by the runtime. Use this method to add services to the container.
        public void ConfigureServices(IServiceCollection services)
        {

        }

        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app,IHostEnvironment env)
        {

        }
    }
qinhao8438 回答:在asp.net核心应用程序中实现了IHostedService,如何在没有IIS首次请求的情况下运行它?

由于建议的“应用程序初始化模块”不起作用,因此您可以考虑与客户端进行通话

  

当第一个请求到达时,模块启动ASP.NET Core应用程序的进程,如果关闭或崩溃,则重新启动应用程序。

public class Program {
    static Lazy<HttpClient> client = new Lazy<HttpClient>();
    public static async Task Main(string[] args) {
        var host = CreateWebHostBuilder(args).Start();//non blocking start
        using (host) {
            bool started = false;
            do {
                var response = await client.Value.GetAsync("site root");
                started = response.IsSuccessStatusCode;
                await Task.Delay(someDelayHere);
            } while (!started);

            host.WaitForShutdown();
        }
    }

    public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
            .ConfigureServices(services => {
                services.AddHostedService<PaymentQueueService>();
            })
            .Configure((IApplicationBuilder app) => {
                app.UseMvc();
            })
            .UseStartup<Startup>();
}

注意:

  

为防止托管进程外的应用程序超时,请使用以下两种方法之一:

     
      
  • 通过外部服务对应用程序执行Ping操作,以使其保持运行状态。
  •   
  • 如果该应用仅托管后台服务,请避免IIS托管,而使用Windows服务托管ASP.NET Core应用。
  •   
,

将其托管为“ OutOfProcess”可以解决问题,但请记住,这不是首选方式。我已经使用给定的代码在GitHub上创建了一个仓库,您可以检查上一次提交的更改。

来源:https://github.com/tvdias/SO58831661

要深入了解InProcess和OutOfProcess之间的区别,我建议阅读以下页面:https://weblog.west-wind.com/posts/2019/Mar/16/ASPNET-Core-Hosting-on-IIS-with-ASPNET-Core-22

,

这是我创建webservices的方式。在Startup.cs中注入我的IHostedServicesContainerStructureMap的一类。我是从一个现有项目中复制它的,所以它不适合您的样本的100%。

public class Program
{
    public static void Main(string[] args)
    {
        Config.Directories.EnsureDirectoryTree();

        var isService = !(Debugger.IsAttached || args.Contains("--console"));
        var webHostService = MyWebHostService.BuildWebHostService(args.Where(arg => arg != "--console").ToArray());

        if (isService)
        {
            ServiceBase.Run(webHostService);
        }
        else
        {
            webHostService.InitializeBackend();
            webHostService.Host.Run();
        }
    }
}

public class MyWebHostService : WebHostService
{
    public static readonly Logger Logger = LogManager.GetCurrentClassLogger();

    public IWebHost Host { get; }

    public static ZlsWebHostService BuildWebHostService(string[] args)
    {
        ConfigureNLog();

        Logger.Info("{0} (v{1}) starting...",Config.ServiceName,GetApplicationVersion());

        // restore config files
        Config.Files.EnsureRestored();

        var host = CreateWebHostBuilder(args).Build();
        return new ZlsWebHostService(host);
    }

    public static IWebHostBuilder CreateWebHostBuilder(string[] args)
    {
        var pathToExe = Assembly.GetExecutingAssembly().Location;
        var pathToContentRoot = Path.GetDirectoryName(pathToExe);

        return WebHost.CreateDefaultBuilder()
            .UseKestrel()
            .UseContentRoot(pathToContentRoot)
            .ConfigureAppConfiguration((context,config) =>
            {
                config.SetBasePath(Config.Directories.ActiveConfig);
                config.AddJsonFile(Config.Files.KestrelConfig.RelativePath,true,true);})
            .ConfigureLogging((hostingContext,logging) =>
            {
                logging.ClearProviders();
                logging.AddConfiguration(hostingContext.Configuration.GetSection("Logging"));
                logging.AddConsole();
                if(hostingContext.HostingEnvironment.IsDevelopment())
                    logging.AddDebug();
            })
            .UseNLog()
            .UseStartup<Startup>();
    }

    public MyWebHostService(IWebHost host) : base(host)
    {
        this.Host = host;
    }

    protected override void OnStarting(string[] args)
    {
        InitializeBackend();

        base.OnStarting(args);
    }

    protected override void OnStarted()
    {
        Logger.Info("{0} started...",Config.ServiceName);
        base.OnStarted();
    }

    protected override void OnStopped()
    {
        Logger.Info("{0} stopped...",Config.ServiceName);
        base.OnStopped();
    }

...

}

public class Startup
{
...

    // This method gets called by the runtime. Use this method to add services to the container.
    // For more information on how to configure your application,visit https://go.microsoft.com/fwlink/?LinkID=398940
    public IServiceProvider ConfigureServices(IServiceCollection services)
    {
        ...

        services.AddSingleton<PaymentQueueService>();

        ...

        var container = new Container(c =>
        {
            c.IncludeRegistry<MyFooRegistry>();
            c.Populate(services);
        });
        return container.GetInstance<IServiceProvider>();
    }

...

}
本文链接:https://www.f2er.com/3112310.html

大家都在问