eShopOnContainers 知多少[11]:服务间通信之gRPC

1. 引言

最近翻看最新3.0 eShopOncontainers源码,发现其在架构选型中补充了 gRPC 进行服务间通信。那就索性也写一篇,作为系列的补充。

2. gRPC

老规矩,先来理一下gRPC的基本概念。gRPC是Google开源的RPC框架,比肩dubbo、thrift、brpc。其优势在于:

  1. 基于proto buffer:二进制协议,具有高性能的序列化机制。相较于JSON(文本协议)而言,首先从数据包上就有60%-80%的减小,其次其解包速度仅需要简单的数学运算完成,无需复杂的词法语法分析,具有8倍以上的性能提升。

  2. 基于proto 文件:可以更方便的在客户端和服务端之间进行交互。

  3. gRPC语言无关性: 所有服务都是使用原型文件定义的。这些文件基于protobuffer语言,并定义服务的接口。基于原型文件,可以为每种语言生成用于创建服务端和客户端的代码。其中protoc编译工具就支持将其生成C #代码。从.NET Core 3 中,gRPC在工具和框架中深度集成,开发者会有更好的开发体验。

  4. 支持数据流。

3. gRPC 在 eShopOncontainers 的应用

首先来理一下eShopOncontainers 中服务间同步通信的技术选型,主要还是是基于HTTP/REST,gRPC作为补充。

在eShopOncontainers中Ordering API、Catalog API、Basket API微服务通过gRPC端点暴露服务。其中Mobile Shopping、Web Shopping BFFs使用gRPC客户端访问服务。以下以Ordering API gRPC 服务举例说明。

订单微服务中定义了一个gRPC服务,用于从购物车创建订单。

3.1 服务端实现

proto文件定义如下:

syntax = "proto3";
option csharp_namespace = "GrpcOrdering";
package OrderingApi;
service OrderingGrpc {rpc CreateOrderDraftFromBasketData(CreateOrderDraftCommand) returns (OrderDraftDTO) {}
}
message CreateOrderDraftCommand {string buyerId = 1;repeated BasketItem items = 2;
}
message BasketItem {string id = 1;int32 productId = 2;string productName = 3;double unitPrice = 4;double oldUnitPrice = 5;int32 quantity = 6;string pictureUrl = 7;
}
message OrderDraftDTO {double total = 1;repeated OrderItemDTO orderItems = 2;
}
message OrderItemDTO {int32 productId = 1;string productName = 2;double unitPrice = 3;double discount = 4;int32 units = 5;string pictureUrl = 6;
}

服务实现,主要是借助Mediator充当CommandBus进行命令分发,具体实现如下:

namespace GrpcOrdering
{public class OrderingService : OrderingGrpc.OrderingGrpcBase{private readonly IMediator _mediator;private readonly ILogger<OrderingService> _logger;public OrderingService(IMediator mediator, ILogger<OrderingService> logger){_mediator = mediator;_logger = logger;}public override async Task<OrderDraftDTO> CreateOrderDraftFromBasketData(CreateOrderDraftCommand createOrderDraftCommand, ServerCallContext context){_logger.LogInformation("Begin gRPC call from method {Method} for ordering get order draft {CreateOrderDraftCommand}", context.Method, createOrderDraftCommand);_logger.LogTrace("----- Sending command: {CommandName} - {IdProperty}: {CommandId} ({@Command})",createOrderDraftCommand.GetGenericTypeName(),nameof(createOrderDraftCommand.BuyerId),createOrderDraftCommand.BuyerId,createOrderDraftCommand);var command = new AppCommand.CreateOrderDraftCommand(createOrderDraftCommand.BuyerId,this.MapBasketItems(createOrderDraftCommand.Items));var data = await _mediator.Send(command);if (data != null){context.Status = new Status(StatusCode.OK, $" ordering get order draft {createOrderDraftCommand} do exist");return this.MapResponse(data);}else{context.Status = new Status(StatusCode.NotFound, $" ordering get order draft {createOrderDraftCommand} do not exist");}return new OrderDraftDTO();}public OrderDraftDTO MapResponse(AppCommand.OrderDraftDTO order){var result = new OrderDraftDTO(){Total = (double)order.Total,};order.OrderItems.ToList().ForEach(i => result.OrderItems.Add(new OrderItemDTO(){Discount = (double)i.Discount,PictureUrl = i.PictureUrl,ProductId = i.ProductId,ProductName = i.ProductName,UnitPrice = (double)i.UnitPrice,Units = i.Units,}));return result;}public IEnumerable<ApiModels.BasketItem> MapBasketItems(RepeatedField<BasketItem> items){return items.Select(x => new ApiModels.BasketItem(){Id = x.Id,ProductId = x.ProductId,ProductName = x.ProductName,UnitPrice = (decimal)x.UnitPrice,OldUnitPrice = (decimal)x.OldUnitPrice,Quantity = x.Quantity,PictureUrl = x.PictureUrl,});}}
}

同时,服务端还要注册gRPC的请求处理管道:

app.UseEndpoints(endpoints =>
{endpoints.MapDefaultControllerRoute();endpoints.MapControllers();endpoints.MapGrpcService<OrderingService>();
});

3.2 客户端调用

接下来看下客户端[web.bff.shopping]怎么消费的:

public class OrderingService : IOrderingService{private readonly UrlsConfig _urls;private readonly ILogger<OrderingService> _logger;public readonly HttpClient _httpClient;public OrderingService(HttpClient httpClient, IOptions<UrlsConfig> config, ILogger<OrderingService> logger){_urls = config.Value;_httpClient = httpClient;_logger = logger;}public async Task<OrderData> GetOrderDraftAsync(BasketData basketData){return await GrpcCallerService.CallService(_urls.GrpcOrdering, async channel =>{var client = new OrderingGrpc.OrderingGrpcClient(channel);_logger.LogDebug(" gRPC client created, basketData={@basketData}", basketData);var command = MapToOrderDraftCommand(basketData);var response = await client.CreateOrderDraftFromBasketDataAsync(command);_logger.LogDebug(" gRPC response: {@response}", response);return MapToResponse(response, basketData);});}private OrderData MapToResponse(GrpcOrdering.OrderDraftDTO orderDraft, BasketData basketData){if (orderDraft == null){return null;}var data = new OrderData{Buyer = basketData.BuyerId,Total = (decimal)orderDraft.Total,};orderDraft.OrderItems.ToList().ForEach(o => data.OrderItems.Add(new OrderItemData{Discount = (decimal)o.Discount,PictureUrl = o.PictureUrl,ProductId = o.ProductId,ProductName = o.ProductName,UnitPrice = (decimal)o.UnitPrice,Units = o.Units,}));return data;}private CreateOrderDraftCommand MapToOrderDraftCommand(BasketData basketData){var command = new CreateOrderDraftCommand{BuyerId = basketData.BuyerId,};basketData.Items.ForEach(i => command.Items.Add(new BasketItem{Id = i.Id,OldUnitPrice = (double)i.OldUnitPrice,PictureUrl = i.PictureUrl,ProductId = i.ProductId,ProductName = i.ProductName,Quantity = i.Quantity,UnitPrice = (double)i.UnitPrice,}));return command;}}

其中, GrpcCallerService是对gRPC Client的一层封装,主要是为了解决未启用TLS无法使用gRPC的问题。

4. 不启用TLS使用gRPC

我们已经知道gRpc 是基于HTTP2.0 协议。然而,连接的建立,默认并不是一步到位直接基于HTTP2.0建立连接的。客户端是先基于HTTP1.1进行协议协商,协商成功后,确认服务端支持HTTP2.0后,才会建立HTT2.0连接,协议协商需要TLS的ALPN协议来实现。流程如下:

这意味着,默认情况下,您需要启用TLS协议才能完成HTTP2.0协议协商,进而才能使用gRPC。

然而,在微服务架构中,并不是所有服务都需要启用安全传输层协议,尤其是微服务间的内部调用。那么在微服务内部如何使用gRPC进行通信呢?

客户端绕过协议协商,直连HTTP2.0(前提是:服务端必须支持HTTP2.0)

服务端配置如下:

WebHost.CreateDefaultBuilder(args).ConfigureKestrel(options =>{options.Listen(IPAddress.Any, ports.httpPort, listenOptions =>{listenOptions.Protocols = HttpProtocols.Http1AndHttp2; //同时监听协议HTTP1,HTTP2});options.Listen(IPAddress.Any, ports.gRPCPort, listenOptions =>{listenOptions.Protocols = HttpProtocols.Http2; // gRPC端口仅监听HTTP2.0});})

客户端需要添加以下设置,这些设置只能在客户端开始时设置一次:

AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", true);
AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2Support", true);

知道了这些,再回过来看 GrpcCallerService的实现,就一目了然了。

public static class GrpcCallerService
{public static async Task<TResponse> CallService<TResponse>(string urlGrpc, Func<GrpcChannel, Task<TResponse>> func)
{AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", true);AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2Support", true);var channel = GrpcChannel.ForAddress(urlGrpc);/*using var httpClientHandler = new HttpClientHandler{ServerCertificateCustomValidationCallback = (message, cert, chain, errors) => { return true; }};*/Log.Information(@"Creating gRPC client base address urlGrpc ={@urlGrpc}, BaseAddress={@BaseAddress} ", urlGrpc, channel.Target);try{return await func(channel);}catch (RpcException e){Log.Error("Error calling via gRPC: {Status} - {Message}", e.Status, e.Message);return default;}finally{AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", false);AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2Support", false);}}public static async Task CallService(string urlGrpc, Func<GrpcChannel, Task> func)
{AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", true);AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2Support", true);/*using var httpClientHandler = new HttpClientHandler{ServerCertificateCustomValidationCallback = (message, cert, chain, errors) => { return true; }};*/var channel = GrpcChannel.ForAddress(urlGrpc);Log.Debug("Creating gRPC client base address {@httpClient.BaseAddress} ", channel.Target);try{await func(channel);}catch (RpcException e){Log.Error("Error calling via gRPC: {Status} - {Message}", e.Status, e.Message);}finally{AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport", false);AppContext.SetSwitch("System.Net.Http.SocketsHttpHandler.Http2Support", false);}}
}

5. 最后

本文简要介绍了 eShopOnContainers 如何通过集成 gRPC 来完善服务间同步通信机制,希望对你在对微服务进行RPC相关技术选型时有一定的启示和帮助。

参考资料:

  1. [HTTP2.0笔记之连接建立:http://www.blogjava.net/yongboy/archive/2015/03/18/423570.html]

  2. [eShopOnContainers/wiki/gRPC:https://github.com/dotnet-architecture/eShopOnContainers/wiki/gRPC]

  3. [Google Protocol Buffer 的使用和原理:https://www.ibm.com/developerworks/cn/linux/l-cn-gpb/index.html]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/308827.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[mybatis]映射文件_参数处理_#取值时指定参数相关规则

jdbcType通常需要在某种特定的条件下被设置&#xff1a; 在我们数据为null的时候&#xff0c;有些数据库可能不能识别mybatis对null的默认处理。比如Oracle&#xff08;报错&#xff09;&#xff1b;jdbcType OTHER&#xff1a;无效的类型&#xff1b;因为mybatis对所有的null映…

龙芯开源社区上线.NET主页

龙芯团队从2019年7 月份开始着手.NET Core的MIPS64支持研发&#xff0c;经过将近一年的研发&#xff0c;在2020年6月18日完成了里程碑性的工作&#xff0c;在github CoreCLR 仓库&#xff1a;https://github.com/gsvm/coreclr&#xff0c; 随后受到.NET社区的很大参与热情鼓舞之…

[mybatis]映射文件_select_返回集合(List,Map)

select返回List public interface EmployeeMapper {public List<Employee> getEmpsByLastNameLike(String lastName);}<!-- public List<Employee> getEmpsByLastNameLike(String lastName);--><!--resultType;如果返回的是一个集合&#xff0c;要写集…

免费 | 数千个 Azure 官方高清矢量图标大放送!

点击上方关注“汪宇杰博客” ^_^导语我们平时绘制云应用架构图时&#xff0c;经常要用 Azure 各项服务的图标。不知各位会自己去截图&#xff0c;还是在浏览器 F12 里使劲抠图&#xff1f;是不是做PPT 5分钟&#xff0c;扣图2小时&#xff1f;现在用不着自己使劲啦&#xff0c;…

使用sqlserver搭建高可用双机热备的Quartz集群部署

一般拿 Timer 和 Quartz 相比较的&#xff0c;简直就是对 Quartz 的侮辱&#xff0c;两者的功能根本就不在一个层级上&#xff0c;如本篇介绍的Quartz强大的集群机制&#xff0c;可以采用基于sqlserver&#xff0c;mysql的集群方案&#xff0c;当然还可以在第三方插件的基础上实…

从Memcached到Redis,从RabbitMQ到Kafka,为了高并发,这些年我们不容易!

周一上班又撕了一页台历&#xff0c;随即心里头却是一惊。如今高考落下帷幕&#xff0c;不曾想2020竟已过了一半&#xff01;这一年又是疫情又是洪水&#xff0c;我们的目光被各种重大事件所吸引&#xff0c;此时审视一下自己&#xff0c;突然觉得不踏实了&#xff01;一场疫情…

[mybatis]动态sql_if_where_trim判断OGNL

OGNL if 要求&#xff1a;查询员工&#xff0c;要求&#xff0c;携带了哪个字段查询条件就带上这个字段的值 //携带了哪个字段查询条件就带上这个字段的值public List<Employee> getEmpsByConditionIf(Employee employee);<!-- public List<Employee> getEm…

[mybatis]动态sql_choose_分支选择

choose 如果带了id就用id查&#xff0c;如果带了lastName就用lastName查&#xff1b;只会进入其中一个 public List<Employee> getEmpByConditionChoose(Employee employee);<!-- public List<Employee> getEmpByConditionChoose(Employee employee);-->…

7.15周三晚8点,dotnet课堂全新起航,张善友/陈计节/刘腾飞我们一起来聊聊abp的故事...

直播主题&#xff1a;我们和Abp的故事直播嘉宾&#xff1a;张善友&#xff0c;陈计节&#xff0c;刘腾飞直播话题张善友&#xff1a;我是如何使用Abp的刘腾飞&#xff1a;利用Abp的模块化解决单体和分布式混合架构陈计节&#xff1a;Abp开源项目的DevOps实践Abp VNext 处于被低…

ASP.NET Core端点路由 作用原理

端点路由(Endpoint Routing)最早出现在ASP.NET Core2.2&#xff0c;在ASP.NET Core3.0提升为一等公民。Endpoint Routing的动机在端点路由出现之前&#xff0c;我们一般在请求处理管道的末尾&#xff0c;定义MVC中间件解析路由。这种方式意味着在处理管道中&#xff0c;MVC中间…

程序员修神之路--分布式系统设计理念这么难学?

点击“蓝字”关注我们吧福利&#xff1a;有件小事想和大家说一下菜菜哥&#xff0c;问你个问题&#xff0c;为什么现在的系统都设计为分布式系统呢&#xff1f;这个问题问得好&#xff0c;就像为什么程序员会慢慢脱发一样神奇01PART分布式系统身为二十一世纪的一名程序员&#…

使用keepalived搭建双机热备高可用一览

很多时候大家为了部署高可用方案都是前端配一个 nginx&#xff0c;如果nginx挂掉怎么办&#xff0c;比如下面这张图&#xff1a;你可以清楚的看到&#xff0c;如果 192.168.2.100 这台机器挂掉了&#xff0c;那么整个集群就下线了&#xff0c;这个问题该怎么解决呢&#xff1f;…

Linus通过了Linux中避免master/slave等术语的提案

Linux 内核维护者 Dan Williams 曾于 7 月初提交一份提案&#xff0c;建议逐步取消 master/slave 和 blacklist/whitelist 术语。近日&#xff0c;Linus Torvalds 则在 Linux 5.8 版本库的拉取请求中批准了该提议。自此&#xff0c;Linux 开发人员则需要使用新的术语来替代 mas…

[mybatis]缓存_一级缓存_一级缓存失效的四种情况

1.sqlSession不同 Testpublic void test05() throws IOException {SqlSessionFactory sqlSessionFactory getSqlSessionFactory();SqlSession sqlSession01 sqlSessionFactory.openSession();try{EmployeeMapper mapper01 sqlSession01.getMapper(EmployeeMapper.class);Emp…

LINQ :最终统治了​所有的语言!

LINQ&#xff1a;最终统治了所有的语言&#xff01;让我们看看LINQ如何彻底改变了.NET中访问数据的方式.NET与其他技术栈的不同之处之一绝对是LINQ&#xff0c;它是Language Integrated Query的首字母缩写。实际上&#xff0c;它是随.NET Framework 3.5和Visual Studio 2008引入…

[SpringBoot2]文件上传_单文件与多文件上传的使用

<form role"form" th:action"{/upload}" method"post" enctype"multipart/form-data"><div class"form-group"><label for"exampleInputEmail1">邮箱</label><input type"email&…

使用Docker运行SQL Server

现在.net core已经跨平台了&#xff0c;大家也都用上了linux用上了docker。跟.net经常配套使用的SQL SERVER以前一直是windows only&#xff0c;但是从SQL Server 2017开始已经支持运行在docker上&#xff0c;也就说现在SQL Serer已经可以运行在linux下了。下面在Ubuntu 16.4上…

[mybatis]映射文件_select_resultMap_关联查询_association分步查询延迟加载

association分步查询 场景一 查询Employee的同时查询员工对应的部门 EmployeeDepartment 一个员工有与之对应的部门信息 Employee表: Department表&#xff1a; public interface DepartmentMapper {public Department getDeptById(Integer id);}public interface EmployeeMap…

C#中你想象的Task,很简单?

【导读】网上关于Task的文章如数家珍&#xff0c;不过有一部分并未谈到一个根本的问题&#xff0c;所创建的Task一定在线程池上运行&#xff1f;如何合理的使用Task&#xff1f;这里并不会去重新讲解每一个APi的使用&#xff0c;没有任何意义&#xff0c;这属于包括我在内的各位…