C# 将gRPC服务绑定到aspnetcore中的特定端口
使用C# 将gRPC服务绑定到aspnetcore中的特定端口,c#,.net,asp.net-core,.net-core,grpc,C#,.net,Asp.net Core,.net Core,Grpc,使用aspnetcore 3.1和Grpc.aspnetcorenuget软件包,我成功地使Grpc服务与标准asp.net控制器一起运行,如中所述 但是,我希望将gRPC服务绑定到特定端口(例如5001),如果可能,最好通过配置而不是代码。这是因为我想限制我的gRPC服务的公开方式 最近的一次是在映射端点时使用RequireHost: // Startup.cs public void Configure(IApplicationBuilder app) { // ...
aspnetcore 3.1
和Grpc.aspnetcore
nuget软件包,我成功地使Grpc服务与标准asp.net控制器一起运行,如中所述
但是,我希望将gRPC服务绑定到特定端口(例如5001),如果可能,最好通过配置而不是代码。这是因为我想限制我的gRPC服务的公开方式
最近的一次是在映射端点时使用RequireHost
:
// Startup.cs
public void Configure(IApplicationBuilder app)
{
// ...
app.useEndpoints(endpoints =>
{
endpoints.MapGrpcService<MyService>()
.RequireHost("0.0.0.0:5001");
});
}
//Startup.cs
公共void配置(IApplicationBuilder应用程序)
{
// ...
app.useEndpoints(端点=>
{
endpoints.MapGrpcService()
.RequireHost(“0.0.0.0:5001”);
});
}
这似乎是我想要的,但我找不到任何关于它的文档,它需要每个服务的代码配置。也许有更好的方法?据我所知,没有其他方法为GRPC服务设置特定端口 grpc服务也在asp.net core kestrel服务器上运行,服务器将侦听端口而不是服务 如果您的asp.net核心应用程序只有GRPC服务,您可以将kestrel服务器的侦听端口设置为5001 如果您有多个服务,如MVC web api或其他,RequireHost是只允许特定端口访问grpc服务的最佳解决方案 如果要提示GRPC服务的路由系统需要指定的端口,可以使用以下端口:
routes.MapGrpcService<MyService>().RequireHost("*:5001");
routes.MapGrpcService().RequireHost(“*:5001”);
这对Kestrel(服务器端)有效:
public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.ConfigureWebHostDefaults(webBuilder =>
{
webBuilder.ConfigureKestrel(options =>
{
options.Listen(IPAddress.Loopback, 5000);
options.Listen(IPAddress.Loopback, 5005, configure => configure.UseHttps());
});
webBuilder.UseStartup<Startup>();
});
注:
当您拥有一个没有信任链的自签名证书时(主要是在开发时)
用于支持http。只有当请求使用您定义的端口时,才可以尝试使用
UseWhen
方法来使用MapGrpcService
端点
var grpcPort=5001;
app.UseWhen(context=>context.Connection.LocalPort==grpcPort,
生成器=>
{
builder.UseRouting();
builder.UseEndpoints(端点=>
{
endpoints.MapGrpcService();
});
});
这样做的好处是不重复.RequireHost(“*:5001”)
对于每个服务,尽管重复两次UseRouting
可能会导致奇怪的行为:例如,除非在builder.UseRouting()之后输入builder.UseAuthentication()
,否则身份验证可能无法工作
但是,如果您希望对REST和gRPC有一个不同的请求管道,则此行为非常有用。谢谢您的回答。关于*
的好提示,适用于任何试图实现此目标的人0.0.0.0
被Kestrel视为*
,而不是RequireHost。除非我遗漏了什么,否则这将简单地将所有内容(控制器和gRPC服务)绑定到端口5000和5005。我需要将我的gRPC服务绑定到一个特定的端口。@Lawrence:当你将所有内容打包到一个程序集中时,也许是这样。我不确定。但这样做是一种反模式。当您仅将gRPC打包到on assembly中时,它当然只绑定您的gRPC-services.PS@Lawrence:将ASP.NET核心微服务和gRPC服务放在一个程序集中听起来真的很奇怪。考虑重新考虑这一点。我们确实考虑了创建单独的程序集。我们的控制器层和gRPC服务是位于共享代码之上的薄层,将它们分开将使我们的部署数量翻倍,这似乎是一个很高的代价。。。Brando上面的答案对我们来说很好。感谢您提供了另一个选择-我们编写了一个扩展方法,简化了.RequireHost
重复,因此不会太痛苦。据我所见,多次调用UseEndpoints
是不受支持的(),所以我认为这可能也是这个方法的一个潜在障碍?这很奇怪,它对我有效。尽管gRPC和REST的请求管道完全不同确实有点奇怪,所以我明白他们为什么不建议使用它。
var httpHandler = new HttpClientHandler
{
ServerCertificateCustomValidationCallback =
HttpClientHandler.DangerousAcceptAnyServerCertificateValidator
};
AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", true);
using var channel = GrpcChannel.ForAddress("https://localhost:5005", new GrpcChannelOptions { HttpHandler = httpHandler } );
var client = new Greeter.GreeterClient(channel);
var httpHandler = new HttpClientHandler
{
ServerCertificateCustomValidationCallback =
HttpClientHandler.DangerousAcceptAnyServerCertificateValidator
};
AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", true);