.NET领域最硬核的gRPC 核心能力一把梭
前言,本文定位为.NET方向 grpc核心能力一把梭,全篇是姿势性和结论性的展示, 方便中高级程序员快速上手.NET Grpc。
有关grpc更深层次的前世今生、底层原理、困惑点释疑请听下回分解, 欢迎菜鸟老鸟们提出宝贵意见。
- grpc宏观目标: 高性能rpc框架
- grpc框架实现宏观目标的底层3协议
- http2通信协议, 基础能力
- proto buffer: 打解包协议==> 二进制
- proto buffer: 服务协议,IDL
- 通过脚手架项目分析grpc简单一元通信
- grpc打乒乓球实践双向流式通信
- grpc除了基于3大协议之外, 扩展点体现能力,扩展点在哪?
- 调用管道: 池化tcp、 tcp探活
- 负载均衡
- 元数据 metadata
- 拦截器
一. 宏观目标
gRPC是高性能的RPC框架, 有效地用于服务通信(不管是数据中心内部还是跨数据中心)。
科普rpc:
程序可以像调用本地函数和本地对象一样, 达成调用远程服务的效果,rpc屏蔽了底层的通信细节和打解包细节。
跟许多rpc协议一样, grpc也是基于IDL(interface define lauguage)来定义服务协议。
grpc是基于http/2协议的高性能的rpc框架。
二. grpc实现跨语言的rpc调用目标 基于三个协议:
- 底层传输协议: 基于http2 (多路复用、双向流式通信)
- 打解包协议: 基于proto Buffer 打包成二进制格式传输
- 接口协议: 基于契约优先的开发方式(契约以proto buffer格式定义), 可以使用protoc 编译器生产各种语言的本地代理类, 磨平了微服务平台中各语言的编程隔阂。
下图演示了C++ grpc服务, 被跨语言客户端调用, rpc服务提供方会在调用方产生服务代理stub, 客户端就像调用本地服务一样,产生远程调用的效果。
在大规模微服务中,C++grpc服务也可能作为调用的客户端, 于是这个服务上可能也存在其他服务提供方的服务代理stub, 上图没有体现。
三. 通过脚手架项目分析gRPC简单一元通信
我们将从使用gRPC服务模板
创建一个新的dotnet项目。
VS gRPC服务模板默认使用TLS 来创建gRRPC服务, 实际上不管是HTTP1.1 还是HTTP2, 都不强制要求使用TLS
如果服务一开始同时支持HTTP1.1+ HTTP2 但是没有TLS, 那么协商的结果将是 HTTP1.1+ TLS,这样的话gRPC调用将会失败。
3.1 The RPC Service Definition
protocol buffers
既用作服务的接口定义语言(记录服务定义和负载消息),又用作底层消息交换格式。 这个说法语上面的3大底层协议2,3 呼应。
① 使用protocol buffers
在.proto文件中定义服务接口。在其中,定义可远程调用的方法的入参和返回值类型。服务器实现此接口并运行gRPC服务器以处理客户端调用。
② 定义服务后,使用PB编译器protoc
从.proto文件生成指定语言的数据访问/传输类stub,该文件包含服务接口中消息和方法的实现。
syntax = "proto3"; // `syntax`指示使用的protocol buffers的版本
option csharp_namespace = "GrpcAuthor"; // `csharp_namespace`指示未来生成的存根文件所在的`命名空间`, 这是对应C#语言, java语言应填 java_package
package greet;
service Greeter {
rpc SayHello (HelloRequest) returns (HelloReply); // 一元rpc调用
}
message HelloRequest {
string name = 1;
}
message HelloReply {
string message = 1;
}
注释一看就懂。
接下来使用protoc
编译器和C#插件来对proto文件生成服务器或客户端代码。
- ① 由客户端和服务共享的强类型对象,表示消息的服务操作和数据元素, 这个是pb序列化协议的强类型对象。
- ②一个强类型基类,具有远程 gRPC 服务可以继承和扩展的所需网络管道: Greeter.GreeterBase
- ③一个客户端存根,其中包含调用远程 gRPC 服务所需的管道: Greeter.GreeterClient
运行时,每条消息都序列化为标准 Protobuf 二进制表示形式,在客户端和远程服务之间交换。
3.2 实现服务定义
脚手架项目使用Grpc.AspNetCore
NuGet包:所需的类由构建过程自动生成, 你只需要在项目.csproj文件中添加
<ItemGroup>
<Protobuf Include="Protos\greet.proto" GrpcServices="Server" />
</ItemGroup>
以下是继承②强基类而实现的grpc服务
public class GreeterService : Greeter.GreeterBase
{
private readonly ILogger<GreeterService> _logger;
public GreeterService(ILogger<GreeterService> logger)
{
_logger = logger;
}
public override Task<HelloReply> SayHello(HelloRequest request, ServerCallContext context)
{
return Task.FromResult(new HelloReply
{
Message = "Hello " + request.Name
});
}
}
最后在原http服务进程上注册Grpc端点
public void ConfigureServices(IServiceCollection services)
{
services.AddGrpc();
}
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
app.UseEndpoints(endpoints =>
{
endpoints.MapGrpcService<GreeterService>();
endpoints.MapGet("/", async context =>
{
await context.Response.WriteAsync("----http调用-------");
});
});
}
以上在localhost:5000端口同时支持了grpc调用和http调用。
--- 启动服务---...
3.3. 创建gRPC .NET客户端
Visual Studio创建一个名为GrpcAuthorClient的新控制台项目。
安装如下nuget包:
Install-Package Grpc.Net.Client // 包含.NET Core客户端;
Install-Package Google.Protobuf // 包含protobuf消息API;
Install-Package Grpc.Tools // 对Protobuf文件进行编译
① 拷贝服务端项目中的..proto文件
② 将选项csharp_namespace值修改为GrpcAuthorClient。
③ 更新.csproj文件的
<ItemGroup>
<Protobuf Include="Protos\author.proto" GrpcServices="Client" />
</ItemGroup>
④ Client主文件:
static void Main(string[] args)
{
var serverAddress = "https://localhost:5001";
using var channel = GrpcChannel.ForAddress(serverAddress);
var client = new Greeter.GreeterClient(channel);
var reply = client.SayHello(new HelloRequest { Name = "宋小宝!" });
Console.WriteLine(reply.Message.ToString());
Console.WriteLine("Press any key to exit...");
Console.ReadKey();
}
使用服务器地址创建GrpcChannel
,然后使用GrpcChannel对象实例化GreeterClient
;然后使用SayHello同步方法; 服务器响应时,打印结果。
脚手架例子就可以入门,下面聊一聊另外的核心功能
四. gRPC打乒乓球: 双向流式通信
除了上面的一元rpc调用(Unary RPC), 还有
- Client streaming RPC:客户端流式RPC,客户端以流形式(一系列消息)向服务器发起请求,客户端将等待服务器读取消息并返回响应,gRPC服务端能保证了收到的单个RPC调用中的消息顺序。
- Server streaming RPC :服务器流式RPC,客户端向服务器发送请求,并获取服务器流(一系列消息)。客户端从返回的流(一系列消息)中读取,直到没有更多消息为止, gRPC客户端能保证收到的单个RPC调用中的消息顺序。
- Bidirectional streaming RPC: 双向流式RPC,双方都使用读写流发送一系列消息。这两个流是独立运行的,因此客户端和服务器可以按照自己喜欢的顺序进行读写:例如,服务器可以在写响应之前等待接收所有客户端消息,或者可以先读取一条消息再写入一条消息,或读写的其他组合,同样每个流中的消息顺序都会保留。
针对脚手架项目,稍作修改成打乒乓球,考察gRpc双向流式通信、Cancellation机制、grpc元数据三个特性
双向流式可以不管对方是否回复,首先已方是可以持续发送的,己方可以等收到所有信息再回复,也可以收到一次回复一次,也可以自定义收到几次回复一次。
本次演示土乒乓球对攻,故
- 对攻用到 双向流,收到一次,回复一次。
- 强制设置1min的回合对攻必须分出胜负, 使用Cancellation控制回合制的结束
- 对攻双方是白云和黑土, 使用元数据约束
① 添加服务定义接口
rpc PingPongHello(stream HelloRequest) returns (stream HelloReply);
② 服务器实现
public override async Task PingPongHello(IAsyncStreamReader<HelloRequest> requestStream,IServerStreamWriter<HelloReply> responseStream, ServerCallContext context)
{
try
{
if ("baiyun" != context.RequestHeaders.Get("node").Value) // 接收请求头 header
{
context.Status = new Status(StatusCode.PermissionDenied,"黑土只和白云打乒乓球"); // 设置响应状态码
await Task.CompletedTask;
return;
}
await context.WriteResponseHeadersAsync(new Metadata{ // 发送响应头header
{ "node", "heitu" }
});
long round = 0L;
while (!context.CancellationToken.IsCancellationRequested)
{
var asyncRequests = requestStream.ReadAllAsync();
await foreach (var req in asyncRequests)
{
var send = Reverse(req.Name);
await responseStream.WriteAsync(new HelloReply
{
Message = send,
Id = req.Id + 1
});
Debug.WriteLine($" {context.Peer} : {req.Id} receive {req.Name}, send {req.Id + 1} {send}");
}
}
context.ResponseTrailers.Add("round", round.ToString()); // 统计一个回合里双方有多少次对攻
context.Status = new Status(StatusCode.OK,""); // 设置响应状态码
}
catch (RpcException ex)
{
Debug.WriteLine($"{ex.Message}");
}
catch (IOException ex)
{
Debug.WriteLine($"{ex.Message}");
}
catch(Exception ex)
{
Debug.WriteLine($"{ex.Message}");
}
finally
{
Debug.WriteLine($"乒乓球回合制结束");
}
}
③ 客户端
var serverAddress = "http://localhost:5000";
var handler = new SocketsHttpHandler
{
PooledConnectionIdleTimeout = Timeout.InfiniteTimeSpan,
KeepAlivePingDelay = TimeSpan.FromSeconds(60),
KeepAlivePingTimeout = TimeSpan.FromSeconds(30), // tcp心跳探活
EnableMultipleHttp2Connections = true // 启用并发tcp连接
};
using var channel = GrpcChannel.ForAddress(serverAddress, new GrpcChannelOptions {
Credentials = ChannelCredentials.Insecure,
MaxReceiveMessageSize = 1024 * 1024 * 10,
MaxSendMessageSize = 1024 * 1024 * 10,
HttpHandler = handler
});
var client = new Greeter.GreeterClient(channel);
Console.WriteLine($"开始打乒乓球,白云先发球");
using (var cancellationTokenSource = new CancellationTokenSource(60* 1000))
{
try
{
var md = new Metadata
{
{ "node", "baiyun" }
};
var duplexMessage = client.PingPongHello(md, null, cancellationTokenSource.Token);
var headers = await duplexMessage.ResponseHeadersAsync;
if ("heitu" != headers.Get("node").Value) // 接收请求头header
{
throw new RpcException(new Status(StatusCode.PermissionDenied, "白云只和黑土打乒乓球"));
}
await duplexMessage.RequestStream.WriteAsync(new HelloRequest { Id = 1, Name = "gridsum" }) ;
var asyncResp = duplexMessage.ResponseStream.ReadAllAsync();
await foreach (var resp in asyncResp)
{
var send = Reverse(resp.Message);
await duplexMessage.RequestStream.WriteAsync(new HelloRequest {Id= resp.Id, Name = send });
Console.WriteLine($"第{resp.Id}攻防,客户端收到 {resp.Message}, 客户端发送{send}");
}
var tr = duplexMessage.GetTrailers();
var round = tr.Get("round").Value.ToString();
Console.WriteLine($"打乒乓球回合结束,进行了 {round} 次攻防)");
}
catch (RpcException ex)
{
Console.WriteLine($"1min 乒乓球回合制结束,{ex.Message}");
}catch(InvalidOperationException ex)
{
Console.WriteLine($"1min 乒乓球回合制结束,{ex.Message}");
}
}
Console.WriteLine($"打乒乓球结束");
https://github.com/zaozaoniao/GrpcAuthor
五: grpc扩展点
grpc: 是基于http2 多路复用能力,在单tcp连接上发起高效rpc调用的框架。
根据grpc调用的生命周期: 可在如下阶段扩展能力
- 服务可寻址
- 附加在grpc header/trailer的元数据
- 连接/调用 凭证
- 连接/调用 重试机制----> 拦截器
- 调用状态码 : https://grpc.github.io/grpc/core/md_doc_statuscodes.html
下面挑选几个核心的扩展点着重聊一聊。
5.1 负载均衡
哪些调用能做负载均衡?
只有[gRPC调用]能实现对多服务提供方节点的负载平衡, 一旦建立了gRPC流式调用,所有通过该流式调用发送的消息都将发送到一个端点。
grpc负载均衡的时机?
grpc诞生的初衷是点对点通信,现在常用于内网服务之间的通信,在微服务背景下,服务调用也有负载均衡的问题,也正因为连接建立之后是“点对点通信”,所以不方便基于L4做负载均衡。
根据grpc的调用姿势, grpc的负载均衡可在如下环节:
① 客户端负载均衡 : 对于每次rpc call,选择一个服务终结点,直接调用无延迟, 但客户端需要周期性寻址 。
② L7做服务端负载均衡 : L7负载层能理解HTTP/2,并且能在一个HTTP/2连接上跨多个服务提供方节点将[多路复用的gRPC调用]分发给上游服务节点。使用代理比客户端负载平衡更简单,但会给gRPC调用增加额外的延迟。
常见的是客户端负载均衡。
5.2 调用通道
grpc 利用http2 使用单一tcp连接提供到指定主机端口上年的grpc调用,通道是与远程服务器的长期tcp连接的抽象。
客户端对象可以重用相同的通道,与实际rpc调用相比,创建通道是一项昂贵的操作,因此应该为尽可能多的调用重复使用单个通道。
-
根据http2 上默认并发流的限制(100), .NET支持在单tcp连接并发流到达上限的时候,产生新的tcp连接, 故通道是一个池化的tcp并发流的概念, grpc通道具有状态,包括已连接和空闲.
-
像websockets这类长时间利用tcp连接的机制一样,都需要心跳保活机制, 可以快速的进行grpc调用,而不用等待tcp连接建立而延迟。
-
可以指定通道参数来修改gRPC的默认行为,例如打开或关闭消息压缩, 添加连接凭据。
var handler = new SocketsHttpHandler
{
PooledConnectionIdleTimeout = Timeout.InfiniteTimeSpan,
KeepAlivePingDelay = TimeSpan.FromSeconds(60),
KeepAlivePingTimeout = TimeSpan.FromSeconds(30), // tcp心跳探活
EnableMultipleHttp2Connections = true // 启用并发tcp连接
};
var channel = GrpcChannel.ForAddress("https://localhost:5001", new GrpcChannelOptions
{
Credentials = ChannelCredentials.Insecure, // 连接凭据
HttpHandler = handler
});
https://learn.microsoft.com/en-us/aspnet/core/grpc/performance?view=aspnetcore-7.0
5.3 Metadata
元数据是以键值对列表的形式提供的有关特定RPC调用的信息(身份认证信息、访问令牌、代理信息),在grpc调用双方,一般元数据存储在header或trailer 中。
客户端发起调用时会有metadata参数可供使用:
// 上例中的 proto被编译之后产生了如下 sdk
public virtual HelloReply SayHello(HelloRequest request, Metadata headers = null, DateTime? deadline = null, CancellationToken cancellationToken = default(global::System.Threading.CancellationToken))
{
return SayHello(request, new CallOptions(headers, deadline, cancellationToken));
}
对于身份认证元数据,有更通用的方式:builder.Services.AddGrpcClient<Greeter.GreeterClient>().AddCallCredentials((x,y) =>{ })
grpc 服务端可发送的是 header 和trailer, trailer只能在服务端响应完毕发送, 至于为什么有header,还有trailer,请看再谈 gRPC 的 Trailers 设计, 总体而言grpc流式通信需要在调用结束 给客户端传递一些之前给不了的信息。
await context.WriteResponseHeadersAsync(new Metadata{ // 发送相应头 header
{ "node", "B" }
});
context.ResponseTrailers.Add("count", cnt); // 发送相应尾 trailer
context.Status = Status.DefaultSuccess; // 设置响应状态码
5.4 自定义拦截器和 可能使用到的HttpClient
拦截器与 .net httpclientDelegate 和 axio的请求拦截器类似,都是在发起调用的时候,做一些过滤或者追加的行为。
https://learn.microsoft.com/en-us/aspnet/core/grpc/interceptors?view=aspnetcore-8.0
builder.Services
.AddGrpcClient<Greeter.GreeterClient>(o =>
{
o.Address = new Uri("https://localhost:5001");
})
.AddInterceptor<LoggingInterceptor>(); // 默认在客户端之间共享
// 以下是一个客户端日志拦截器,在一元异步调用时拦截
public class ClientLoggingInterceptor : Interceptor
{
private readonly ILogger _logger;
public ClientLoggingInterceptor(ILoggerFactory loggerFactory)
{
_logger = loggerFactory.CreateLogger<ClientLoggingInterceptor>();
}
public override AsyncUnaryCall<TResponse> AsyncUnaryCall<TRequest, TResponse>(
TRequest request,
ClientInterceptorContext<TRequest, TResponse> context,
AsyncUnaryCallContinuation<TRequest, TResponse> continuation)
{
_logger.LogInformation("Starting call. Type/Method: {Type} / {Method}",
context.Method.Type, context.Method.Name); // 拦截动作: 在continuation之前做日志记录。
return continuation(request, context);
}
}
总结
gRPC是具有可插拔身份验证和负载平衡功能的高性能RPC框架。
使用protocol buffers定义结构化数据; 使用不同语言自动产生的代理sdk屏蔽底层通信和打接包细节, 完成了本地实现远程调用的效果 (调用方不care是远程通信)。
Additional Resources
本文来自博客园,作者:{有态度的马甲},转载请注明原文链接:https://www.cnblogs.com/JulianHuang/p/14441952.html
欢迎关注我的原创技术、职场公众号, 加好友谈天说地,一起进化