.NET Core 3.1 控制台应用程序作为 Windows 服务

.NET Core 3.1 Console App as a Windows Service

我目前有一个相当大的控制台应用程序 运行 ASP.NET Core 3.1。现在,我的任务是将这项工作作为 Window 服务在我们的一台服务器上运行。我已准备好将其 运行 作为服务器本身的一项服务,但是,我目前坚持的一件事是如何在代码中实际更改它以使其 运行 作为不破坏服务。

我找到了一些教程,例如 this,它们确实解释了如何 运行 将控制台应用程序作为服务,但是,我发现的所有教程都是从一个新项目开始的。我的问题是我当前的项目已经写好了。我寻求帮助的主要问题是如何让我的项目作为 windows 服务工作,同时保持当前在 startup.cs 中的功能。对于上下文,这是我当前的 startup.cs 和 program.cs:

Startup.cs

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)
    {
        services.AddControllers();
        services.AddSignalR();
        services.AddTransient<SharePointUploader>();
        services.AddTransient<FileUploadService>();
        services.AddSingleton<UploaderHub>();
        //services.AddAuthentication(IISDefaults.AuthenticationScheme);
        services.AddAuthentication(NegotiateDefaults.AuthenticationScheme).AddNegotiate();
        services.AddAuthorization();
    }

    // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
    public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
    {
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }
        else
        {
            app.UseHttpsRedirection();
        }

        app.UseRouting();

        app.UseAuthentication();
        app.UseAuthorization();

        app.UseEndpoints(endpoints =>
        {
            endpoints.MapControllers();
            endpoints.MapHub<UploaderHub>("/uploadHub");
        });
    }
}

Program.cs

public class Program
{
    public static void Main(string[] args)
    {
        var logger = NLogBuilder.ConfigureNLog("nlog.config").GetCurrentClassLogger();
        try
        {
            logger.Debug("init main");
            CreateHostBuilder(args).Build().Run();
        }
        catch (Exception exception)
        {
            //NLog: catch setup errors
            logger.Error(exception, "Stopped program because of exception");
            throw;
        }
        finally
        {
            // Ensure to flush and stop internal timers/threads before application-exit (Avoid segmentation fault on Linux)
            NLog.LogManager.Shutdown();
        }
    }

    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseStartup<Startup>();
            })
            .ConfigureLogging(logging =>
            {
                logging.ClearProviders();
                logging.SetMinimumLevel(LogLevel.Trace);
            })
            .UseNLog();
}

我真的不明白当 运行 作为 windows 服务时它应该如何工作(基于上面链接的教程)。任何帮助将不胜感激。

使用 IWebHostBuilder 而不是 IHostBuilder:

public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
    WebHost.CreateDefaultBuilder(args)
        .ConfigureAppConfiguration((context, config) =>
        {
            // Configure the app here.
        })
        .UseNLog()
        .UseUrls("http://localhost:5001/;" +
                    "https://localhost:5002/;")
        .UseStartup<Startup>();

您还需要以下软件包:

Microsoft.AspNetCore.Hosting;
Microsoft.AspNetCore.Hosting.WindowsServices;

修改你的主要功能:

bool isService = !(Debugger.IsAttached || args.Contains("--console"));
var builder = CreateWebHostBuilder(args.Where(arg => arg != "--console").ToArray());
var host = builder.Build();

if (isService)
{
    host.RunAsService();
}
else
{
    host.Run();
}

要安装该服务,请使用工具 sc.exe。您可以 运行 通过将 --console 作为参数传递给应用程序,将应用程序作为控制台应用程序。对于调试,您还需要传递 --console。

我忘了回答这个问题,因为我问了几个小时后就解决了,但是您可以将“.UseWindowsService()”添加到 Host.CreateDefaultBuilder(args) 行。 例如:

 public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .UseWindowsService()                     //<==== THIS LINE
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseStartup<Startup>();
            })
            .ConfigureLogging(logging =>
            {
                logging.ClearProviders();
                logging.SetMinimumLevel(LogLevel.Trace);
            })
            .UseNLog();

就我而言,我的主机构建器设置中确实包含一个“UseWindowsService()”语句。但是,我将该配置分解为多行,问题是,在开发过程中的某个时刻,我 ALSO 放置了一个:

UseConsoleLifetime()

语句在代码中进一步混合。一旦我弄清楚发生了什么,使用以下部分代码块解决了这个问题:

        var hostBuilder = Host.CreateDefaultBuilder(args);
        if (WindowsServiceHelpers.IsWindowsService())
        {
            hostBuilder.UseWindowsService();
        }
        else
        {
            hostBuilder.UseConsoleLifetime();
        }

请注意,WindowsServiceHelpers 是“Microsoft.Extensions.Hosting.WindowsServices”命名空间中的静态 class。