确定端口Kestrel绑定到

时间:2018-04-17 06:31:31

标签: asp.net-core asp.net-core-2.0 kestrel-http-server

我正在使用ASP.NET Core空(web)模板编写一个简单的ASP.NET Core服务。

默认情况下,它绑定到端口5000,但我希望它绑定到系统上的随机可用端口。

我可以将BuildWebHost修改为:

    public static IWebHost BuildWebHost(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
            .UseStartup<Startup>()
            .UseUrls("http://*:0") // This enables binding to random port
            .Build();

它绑定到一个随机端口,但我如何从应用程序中确定我正在收听哪个端口?

2 个答案:

答案 0 :(得分:4)

可以通过IServerAddressesFeature.Addresses集合访问ASP.NET Core应用程序的托管地址。

主要的挑战是调用代码,该代码将在合适的时间分析此集合。调用IWebHost.Run()时会发生实际的端口绑定(来自Program.Main())。因此,您无法在Startup.Configure()方法中访问托管地址,因为此阶段尚未分配端口。并且在调用IWebHost.Run()后您失去控制权,因为此调用不会返回,直到Web主机关闭。

根据我的理解,分析绑定端口的最合适方法是通过IHostedService的实现。这是工作样本:

public class GetBindingHostedService : IHostedService
{
    public static IServerAddressesFeature ServerAddresses { get; set; }

    public Task StartAsync(CancellationToken cancellationToken)
    {
        var address = ServerAddresses.Addresses.Single();
        var match = Regex.Match(address, @"^.+:(\d+)$");
        if (match.Success)
        {
            int port = Int32.Parse(match.Groups[1].Value);
            Console.WriteLine($"Bound port is {port}");
        }

        return Task.CompletedTask;
    }

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

Startup课程中:

public class Startup
{

    //  ...

    public void ConfigureServices(IServiceCollection services)
    {
        services.AddMvc();
        services.AddSingleton<IHostedService, GetBindingHostedService>();
    }

    public void Configure(IApplicationBuilder app, IHostingEnvironment env)
    {
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }

        app.UseMvc();

        GetBindingHostedService.ServerAddresses = app.ServerFeatures.Get<IServerAddressesFeature>();
    }
}

IServerAddressesFeature的实例通过GetBindingHostedService中的丑陋静态属性传递。我不知道如何将其注入服务中。

Sample Project on GitHub

总的来说,我对这种解决方案不满意。它完成了这项工作,但它看起来要复杂得多。

答案 1 :(得分:1)

您可以呼叫IWebHost.Start()而不是建议的here呼叫IWebHost.Run()。这将允许您继续执行Main方法,因此您可以从IWebHost.ServerFeatures获取所需的信息。请记住,您的应用程序将立即关闭,除非您明确告知不要使用IWebHost.WaitForShutdown()

 public static void Main(string[] args)
    {
        var host = new WebHostBuilder()
            .UseStartup<Startup>()
            .UseUrls("http://*:0") // This enables binding to random port
            .Build();

        host.Start();

        foreach(var address in host.ServerFeatures.Get<IServerAddressesFeature>().Addresses)
        {
            var uri = new Uri(address);
            var port = uri.Port;

            Console.WriteLine($"Bound to port: {port}");
        }

        //Tell the host to block the thread just as host.Run() would have.
        host.WaitForShutdown();
    }