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

Ami*_*mir 7 c# docker .net-core-3.0

我创建了一个从 Microsoft.Extensions.Hosting.BackgroundService 继承的工作服务,然后我通过 Visual Studio 调试器将其部署到我的 Windows 机器上的 docker linux 容器。我在取消令牌.IsCancellationRequested 为真时发生的代码上放置了一个断点。然后我向容器发出“docker stop --time=30”,断点永远不会被击中,30 秒后调试器强行停止。

我还尝试覆盖 StopAsync 方法并在其中放置一个断点,但也不会调用该方法。我正在运行 .net core 3,最新版本的 docker 桌面。我已经确认调用了 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>();    });

    }
Run Code Online (Sandbox Code Playgroud)

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

添加我的工人的样子:

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;
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

Pan*_*vos 6

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

public static async Task Main(string[] args)
{
    CreateHostBuilder(args).Build().RunConsoleAsync();
}
Run Code Online (Sandbox Code Playgroud)

或者

public static void Main(string[] args)
{
    CreateHostBuilder(args).UseConsoleLifeTime().Build().Run();
}
Run Code Online (Sandbox Code Playgroud)

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

这个怎么运作

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

如果查看源代码,RunConsoleAsync无非是调用UseConsoleLifeTime().Build().RunAsync();

内部ConsoleLifetime类为和事件添加事件侦听器:CancelProcessExit

AppDomain.CurrentDomain.ProcessExit += OnProcessExit;
Console.CancelKeyPress += OnCancelKeyPress;
Run Code Online (Sandbox Code Playgroud)

Ctrl+C只会将停止请求转发给主机

private void OnCancelKeyPress(object sender, ConsoleCancelEventArgs e)
{
    e.Cancel = true;
    ApplicationLifetime.StopApplication();
}
Run Code Online (Sandbox Code Playgroud)

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

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;
}
Run Code Online (Sandbox Code Playgroud)


Ami*_*mir 6

4 天后我终于弄明白了,问题不在于代码或 docker,问题在于 Visual Studio 调试器。如果您通过在 VS2019 中按 play 来运行代码,它将不会遵守 SIGTERM。如果您运行手动 docker run 命令,您将看到您的代码响应 DOCKER STOP 命令。

可能有办法让它最终工作,但我还没有找到。