5

使用 ASP.NET Core 3.0 我已经能够使用这种IHostedService方法......

确定 Kestrel 绑定到的端口

... 在运行时确定 Kestrel 的动态端口。

IServerAddressesFeature此处记录了 ASP.NET 3.0:

https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.hosting.server.features.iserveraddressesfeature?view=aspnetcore-3.0

但是,当将版本更改为 ASP.NET Core 3.1 时,页面会重定向回 ASP.NET 3.0,并提示该文档不适用于 ASP.NET Core 3.1。IServerAddressesFeature不再工作了吗?与 ASP.NET Core 3.1 一起使用IServerAddressesFeature仍然可以编译,但返回的端口ServerAddresses始终为零。

程序:

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

    public static IWebHost BuildWebHost() =>

        WebHost.CreateDefaultBuilder()
            .UseKestrel()
            .UseUrls("http://127.0.0.1:0") // port zero to use random dynamic port
            .UseStartup<Startup>()
            .Build();
}

后来,什么时候Configure叫...

    public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
    {
        HostedService.ServerAddresses = app.ServerFeatures.Get<Microsoft.AspNetCore.Hosting.Server.Features.IServerAddressesFeature>();

...服务器地址分配给HostedService.ServerAddresses静态变量,如链接示例中所示。但是,ServerAddresses仅包含端口为零的环回地址:"http://127.0.0.1:0"

我忽略了什么吗?是否有不同的、正确的方法来解决 v3.1 中的问题?如何使用 ASP.NET Core 3.1 配置 Kestrel 以使用随机动态端口并确定它在运行时(在任何控制器操作发生之前)是哪个端口?

更新

这是一个丑陋的解决方法,有助于确定端口。动态端口分配的时间或顺序似乎发生了变化。从该HostedService.StartAsync方法返回并稍后读取服务器地址似乎就足够了。肯定有更好的方法吗?

        public Task StartAsync(CancellationToken cancellationToken)
        {
            System.Threading.Tasks.Task.Run(async () =>
            {
                int port = 0;
                while (port == 0)
                {
                    await System.Threading.Tasks.Task.Delay(250);
                    var address = ServerAddresses.Addresses.FirstOrDefault();
                    if (string.IsNullOrEmpty(address))
                        continue;
                    // address is always in form http://127.0.0.1:port
                    var pos = address.LastIndexOf(':');
                    if (pos > 0)
                    {
                        var portString = address.Substring(pos + 1);
                        port = int.Parse(portString);
                    }
                }
                // have determined the dynamic port now
            });
            return System.Threading.Tasks.Task.CompletedTask;
        }
4

1 回答 1

5

IHostedService 方法不起作用的原因是在执行 IHostedServices 时 .Net Core 3 发生了变化。在 .Net Core 2 中,IHostedService 在主机启动后执行,因此服务器地址信息随时可用。在 .Net Core 3 中,IHostedService 在主机构建之后运行,但在主机启动之前并且地址尚不可用。 这个博客很好地解释了发生了什么变化。

以下从此处复制的获取绑定地址的方法适用于 .Net Core 2 和 3。

您可以致电IWebHost.Start()而不是按照此处IWebHost.Run()的建议致电。这将允许您的方法继续执行,以便您可以从. 请记住,您的应用程序将立即关闭,除非您明确告诉它不要使用.MainIWebHost.ServerFeaturesIWebHost.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();
    }
于 2020-03-03T19:02:45.220 回答