gpt4 book ai didi

c# - 在 Docker linux 容器中运行的 BackgroundService 类不会正常关闭

转载 作者:行者123 更新时间:2023-11-30 17:25:32 26 4
gpt4 key购买 nike

我创建了一个从 Microsoft.Extensions.Hosting.BackgroundService 继承的辅助服务,然后我通过 visual studio 调试器将其部署到我的 Windows 机器上的 docker linux 容器。我在 cancellationtoken.IsCancellationRequested 为真时发生的代码上放置了一个断点。然后我向容器发出“docker stop --time=30”,断点从未命中,30 秒后调试器强制停止。

我还尝试覆盖 StopAsync 方法并在其中放置一个断点,但它也没有被调用。我正在运行 .net core 3,最新版本的 docker desktop。我已经确认调用了 StartAsync。

这是我的程序文件。

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


public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.ConfigureAppConfiguration((hostingContext, config) =>
{
var builder = new ConfigurationBuilder().SetBasePath(Directory.GetCurrentDirectory())
.AddEnvironmentVariables().Build();
})
.ConfigureServices((hostContext, services) => { services.AddHostedService<Worker>(); });

}

如果有人知道我错过了什么,或者有一个尊重停止的非网络服务器服务的工作示例,我将非常感激。

添加我的 worker 的样子:

using System;
using System.Threading;
using System.Threading.Tasks;

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;

namespace MyWorker
{
public class Worker : BackgroundService
{
public Worker(ILogger<Worker> logger, IConfiguration configs)
{

}

public override Task StopAsync(CancellationToken cancellationToken)
{
return base.StopAsync(cancellationToken);
}

public override Task StartAsync(CancellationToken cancellationToken)
{
return base.StartAsync(cancellationToken);
}


protected override async Task ExecuteAsync(CancellationToken cancellationToken)
{
try
{
await DoWork(cancellationToken); //While loop that checks token in here
}

catch (Exception ex)
{

throw;
}
}
}
}

最佳答案

要收听 Ctrl+C、SIGINT 和 SIGTERM,您需要添加 Console Lifetime support , 通过 UseConsoleLifetime()RunConsoleAsync ,例如:

public static async Task Main(string[] args)
{
CreateHostBuilder(args).Build().RunConsoleAsync();
}

public static void Main(string[] args)
{
CreateHostBuilder(args).UseConsoleLifetime().Build().Run();
}

如果关闭时间比 ShutdownTimeout 中设置的超时时间长,将记录一条警告。

工作原理

了解控制台生命周期的工作原理有助于解决延迟问题。

如果你检查来源,RunConsoleAsync只不过是调用 UseConsoleLifetime().Build().RunAsync();

内部ConsoleLifetimeadds event listeners对于 CancelProcessExit 事件:

AppDomain.CurrentDomain.ProcessExit += OnProcessExit;
Console.CancelKeyPress += OnCancelKeyPress;

Ctrl+C will only forwart the Stop request to the host :

private void OnCancelKeyPress(object sender, ConsoleCancelEventArgs e)
{
e.Cancel = true;
ApplicationLifetime.StopApplication();
}

另一方面,如果关闭时间超过 HostOptions.ShutdownTimeout 中指定的超时时间,ProcessExit 处理程序将发出警告。 :

private void OnProcessExit(object sender, EventArgs e)
{
ApplicationLifetime.StopApplication();
if(!_shutdownBlock.WaitOne(HostOptions.ShutdownTimeout))
{
Logger.LogInformation("Waiting for the host to be disposed. Ensure all 'IHost' instances are wrapped in 'using' blocks.");
}
_shutdownBlock.WaitOne();
// On Linux if the shutdown is triggered by SIGTERM then that's signaled with the 143 exit code.
// Suppress that since we shut down gracefully. https://github.com/aspnet/AspNetCore/issues/6526
System.Environment.ExitCode = 0;
}

关于c# - 在 Docker linux 容器中运行的 BackgroundService 类不会正常关闭,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58331464/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com