java如何通过grpc连接etcd_grpc通过 etcd 实现服务发现与注册-源码分析

介绍

下面介绍 jupiter-0.2.7 版本中 grpc 通过 etcd 实现服务发现与注册。

服务发现与注册的实现解析

服务注册

服务注册的流程图:

aa9954741884a9b2f487a3950eded019.png

etcd的服务注册代码模块在 jupiter/pkg/registry/etcdv3 中。

下面让我们来看看实际的代码

// Registry register/unregister service

// registry impl should control rpc timeout

type Registry interface {

RegisterService(context.Context, *server.ServiceInfo) error

UnregisterService(context.Context, *server.ServiceInfo) error

ListServices(context.Context, string, string) ([]*server.ServiceInfo, error)

WatchServices(context.Context, string, string) (chan Endpoints, error)

io.Closer

}

复制代码

在 pkg/registry/registry.go 中定义了注册服务对象的接口。不同的服务只要实现了这些接口,jupiter 就能使用。

首先我们来看看注册方法

// RegisterService register service to registry

func (reg *etcdv3Registry) RegisterService(ctx context.Context, info *server.ServiceInfo) error {

err := reg.registerBiz(ctx, info)

...

}

// 业务信息注册

func (reg *etcdv3Registry) registerBiz(ctx context.Context, info *server.ServiceInfo) error {

...

// 提交信息到 etcd

_, err := reg.client.Put(readCtx, key, val, opOptions...)

...

}

复制代码

这里主要的部分是 reg.client.Put()  将服务信息提交到 etcd 中。其中的租约机制我会在之后单独写一篇文章介绍。这里主要还是关注如何注册。

源码中还有个 registerMetric() 方法,这个方法的目的是将服务信息在提交到etcd的 prometheus 前缀目录下,用于服务监控,用的也是 client.Put() 方法。这里具体就不展示代码了,感兴趣的同学可以去源码库中查看。

服务退出

// 删除服务

func (reg *etcdv3Registry) unregister(ctx context.Context, key string) error {

...

// 删除服务信息

_, err := reg.client.Delete(ctx, key)

...

}

复制代码

这里通过 client.Delete()  方法将服务信息从 etcd 中删除掉。

获取服务列表

// ListServices list service registered in registry with name `name`

func (reg *etcdv3Registry) ListServices(ctx context.Context, name string, scheme string) (services []*server.ServiceInfo, err error) {

// 服务信息key的前缀

target := fmt.Sprintf("/%s/%s/providers/%s://", reg.Prefix, name, scheme)

// 获取相关前缀的所有信息

getResp, getErr := reg.client.Get(ctx, target, clientv3.WithPrefix())

...

}

复制代码

通过 client.Get()  方法获取到相同前缀的服务信息。

服务信息变动监控

// WatchServices watch service change event, then return address list

func (reg *etcdv3Registry) WatchServices(ctx context.Context, name string, scheme string) (chan registry.Endpoints, error) {

prefix := fmt.Sprintf("/%s/%s/", reg.Prefix, name)

// 通过etcd客户端创建一个监控通道

watch, err := reg.client.WatchPrefix(context.Background(), prefix)

if err != nil {

return nil, err

}

...

xgo.Go(func() {

// 不断接收etcd发送过来的变动事件

for event := range watch.C() {

switch event.Type {

case mvccpb.PUT:

updateAddrList(al, prefix, scheme, event.Kv)

case mvccpb.DELETE:

deleteAddrList(al, prefix, scheme, event.Kv)

}

out := al.DeepCopy()

fmt.Printf("al => %p\n", al.Nodes)

fmt.Printf("snapshot => %p\n", out.Nodes)

select {

// 将更新后的服务信息发送出去,接收方是 resolver

case addresses

default:

xlog.Warnf("invalid")

}

}

})

// 返回一个地址通道,用于传递

return addresses, nil

}

复制代码

WatchServices()  方法主要是监控信息的变动事件,并将变动后的服务信息重新返回给 resolver。具体思路是通过 etcdClient.Watch()  方法创建一个监控通道,然后放入一个 goroutine来不断接收 etcd 推送过来的事件,维护本地的服务信息,并通过 resolver 最终返回到 grpclb 负载均衡器进行服务地址信息的更新。

服务发现

服务发现流程图:

641249cd61c3b251db846a33a07a9362.png

grpc 的 resolver 模块定义了两个接口

// Builder creates a resolver that will be used to watch name resolution updates.

type Builder interface {

Build(target Target, cc ClientConn, opts BuildOptions) (Resolver, error)

Scheme() string

}

// Resolver watches for the updates on the specified target.

// Updates include address updates and service config updates.

type Resolver interface {

ResolveNow(ResolveNowOptions)

Close()

}

复制代码

首先我们来看看 Builder 接口的具体实现

type baseBuilder struct {

name string

reg  registry.Registry

}

// Build ...

func (b *baseBuilder) Build(target resolver.Target, cc resolver.ClientConn, opts resolver.BuildOptions) (resolver.Resolver, error) {

endpoints, err := b.reg.WatchServices(context.Background(), target.Endpoint, "grpc")

if err != nil {

return nil, err

}

var stop = make(chan struct{})

xgo.Go(func() {

for {

select {

case endpoint :=

var state = resolver.State{

Addresses: make([]resolver.Address, 0),

...

}

for _, node := range endpoint.Nodes {

...

state.Addresses = append(state.Addresses, address)

}

cc.UpdateState(state)

case

return

}

}

})

return &baseResolver{

stop: stop,

}, nil

}

复制代码

这里Build 方法主要是通过 Registry 模块获得监控服务通道,然后将更新的服务信息再更新到 grpcClient 中去,保证 grpcClient 的负载均衡器的服务地址永远都是最新的。

如何将Builder的具体实现注册到 grpc 中

import "google.golang.org/grpc/resolver"

// Register ...

func Register(name string, reg registry.Registry) {

resolver.Register(&baseBuilder{

name: name,

reg:  reg,

})

}

复制代码

将 Registry模块注入到 Builder 对象中,然后注入到 grpc 的 resolver 模块中去。这样 grpcClient 在实际运行中就会调用 etcd 的服务发现功能了。

grpc 如何使用服务与发现的源码解析

这里在介绍一下jupiter框架在实际项目中如何使用服务发现与注册。

服务注册

func (app *Application) startServers() error {

var eg errgroup.Group

// start multi servers

for _, s := range app.servers {

s := s

eg.Go(func() (err error) {

_ = app.registerer.RegisterService(context.TODO(), s.Info())

defer app.registerer.UnregisterService(context.TODO(), s.Info())

...

})

}

return eg.Wait()

}

eng := engine.NewEngine()

eng.SetRegistry(compound_registry.New(

etcdv3_registry.StdConfig("default").Build(),

))

复制代码

在框架的 Application 模块中已经实现了服务的自动注册与删除。一般使用框架时不需要再调用。项目使用中只需要在创建 Application 对象时,将注册中心信息注入即可。

服务发现

// 服务发现需要初始化,拿到etcd中服务的信息

func (eng *Engine) initResolver() error {

resolver.Register("etcd", etcdv3.StdConfig("default").Build())

return nil

}

复制代码

服务发现也是类型的将注册中心信息注入即可。

文章系列

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

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

相关文章

spring jms 事务_Spring JMS:处理事务中的消息

spring jms 事务1.引言 这篇文章将向您展示使用JMS异步接收消息期间使用者执行过程中的错误如何导致消息丢失。 然后,我将解释如何使用本地事务解决此问题。 您还将看到这种解决方案在某些情况下可能导致消息重复(例如,当它将消息保存到数据…

使用AWS Lambda,S3和AWS CloudFront进行动态内容缓存

快速提供内容对于任何网站或应用程序具有更好的客户体验至关重要。 如果您将网站或应用程序托管在AWS Cloud中,那么无论从何处访问应用程序,都可以以较低的延迟快速提供内容。 AWS提供了CloudFront服务,用于将内容缓存在每个用户地理位置本地…

数据光端机设备性能指标介绍

作为安防监控工程,设备的可靠性应该是第一考虑要素。而数据光端机设备的可靠性是设备厂商在产品设计时就必需考虑的,但是,有些厂商可能会因为某些原因而不愿做或不知道怎么做这方面的工作,在这里着重从工程的角度简单地讨论以下问…

光电转换器有什么作用?光纤收发器如何保养?

光电转换器可以使原来的快速以太网平滑升级,并能充分保护用户原来的网络资源,它也可以称为光纤收发器。光电转换器可以实现交换机和计算机之间的互联,也可以作为传输中继,还可以进行单多模转换。光纤收发器在应用过程中&#xff0…

html中输出PHP的下拉列表,html中关于下拉列表select的图文代码详解

HTML中的下拉列表:Html代码VolvoSaabOpelAudi其中select是显示一个下拉列表(drop down list)出来,option是下拉列表中的项目(item),而option的文本内容(text content)是下拉列表项目中显示到页面上的值,value是真正需要提交到服务…

光纤收发器tx和rx的区别?

光纤收发器,是一种将短距离的双绞线电信号和长距离的光信号进行互换的以太网传输媒体转换单元,在很多地方也被称之为光电转换器(Fiber Converter)。产品一般应用在以太网电缆无法覆盖、必须使用光纤来延长传输距离的实际网络环境中…

光纤收发器元件级和整机测试内容介绍

对光纤收发器的测试可分为元件级和整机测试,元件级测试主要包括对光纤收发器内部关键器件在电工作的电性能测试。整机测试主要指将光纤收发器接入到以太局域网中,测试整机的功能、性能和特性。那么,具体要怎样测试光纤收发器才是一次完整的测…

如何判断光纤收发器是否有问题?

一般情况下,光纤收发器或光模块的发光功率如下:多模在10db--18db之间;单模20公里在-8db--15db之间;而单模60公里则在-5db--12db之间。但如是光纤收发器的发光功率出现在-30db--45db之间,那么,很有可能这个光…

activemq 实例_在一台计算机上运行多个ActiveMQ实例

activemq 实例几周前,我再次通过Mule ESB解决方案将Apache ActiveMQ用作JMS提供程序。 由于使用ActiveMQ已经有几年了,所以我认为最好检查一些(新)功能,例如故障转移传输和其他群集功能 。 为了能够测试这些最后的东西…

什么是光纤收发器?光纤收发器作用是什么?

许多朋友一听到光纤收发器这五个大字总会有困惑,比如说什么是光纤收发器,光纤收发器又有什么作用等等疑问。那么,什么是光纤收发器呢?光纤收发器又有什么作用呢?接下来我们就跟随飞畅科技的小编一起来详细了解下吧&…

在15分钟内使用Spring Boot和Spring Security构建一个Web应用程序

“我喜欢编写身份验证和授权代码。” 〜从来没有Java开发人员。 厌倦了一次又一次地建立相同的登录屏幕? 尝试使用Okta API进行托管身份验证,授权和多因素身份验证。 开发人员知道保护Web应用程序安全可能会很麻烦。 正确地做是很难的。 最糟糕的是&…

音频光端机的必备要素有哪些?

音频光端机就是发射端把传统的音频模拟信号转换成光信号,通过光纤传输到接收端,在接收端再转换成模拟信号的一种音频设备。那么,音频光端机有哪些必备要素呢?音频光端机的原理是什么呢?接下来我们就一起来详细了解下吧…

16路开关量光端机产品功能及接线图

飞畅科技生产的16路开关量光端机,可以使开关量信号通过光缆在光纤上传输双向控制。全数字光传输通道,确保高质量的信号传输。面板上有电源指示灯、光信号指示灯数据信号指示灯,可以直观的检测电源、光信号、数据信号状态。单向系列开关量光端…

matlab查找替换指令,使用matlab GUI在.txt文件中编辑特定数字使用搜索/替换

使用matlab GUI在.txt文件中编辑特定数字使用搜索/替换 我有一堆不同的模板.txt文件,我想由Matlab中的用户访问。这些模板看起来像这样,但是有一些主要的区别,我将解释:LOAD BOX 1 SUBJ M1_299633_D295158_JUN191910_Aut_ERROR2 E…

电话光端机安装步骤详解

光端机是光信号传输的终端设备,其主要作用是把传统的电话信号转换成光信号并在光纤上传输的设备,是随着光线技术的发展而出现的光端机产品。那么,电话光端机是怎么安装的呢?今天就由飞畅科技的小编来为大家介绍下电话光端机的详细…

php 站内搜索引擎,淘特站内搜索引擎For PhoCmsV9.4

一、产品特点:该产品是淘特站内搜索引擎For PhpCmsV9特别定制版,系统安装后,就可以索引、模糊查询PhpCmsV9的文章数据了(无需连接数据库哦)二、产品下载:三、使用说明:提示:本系统已和tomcat7打包&#xff…

交换机组网常见九大故障问题

交换机是一种用于电信号转发的网络设备。它可以为接入交换机的任意两个网络节点提供独享的电信号通路。最常见的交换机是以太网交换机。其他常见的还有电话语音交换机、光纤交换机等。那么,交换机组网常见九大故障问题有哪些呢?接下来我们就跟随飞畅科技…

python qtdesigner安装,PyCharm+Qt Designer+PyUIC安装配置教程详解

Qt Designer用于像VC的MFC一样拖放、设计控件PyUIC用于将Qt Designer生成的.ui文件转换成.py文件Qt Designer和PyUIC都包含在PyQt5中,所以我们只需要安装PyQt5塻块然后再指定Qt Designer和PyUIC即可为了避免篇幅过长,本文只讲安装配置,使用可…

可网管交换机的三种管理方式介绍

交换机的按是否可网管,分为可网管交换机和不可网管交换机,可网管交换机可以通过以下几种途径进行管理:通过RS-232串行口(或并行口)管理、通过网络浏览器管理和通过网络管理软件管理。接下来就由杭州飞畅科技来为大家详…

ping cat.flag.php,关于2020年强网杯-强网先锋-主动的赛题解析

原标题:关于2020年强网杯-强网先锋-主动的赛题解析一、基本信息(总概述)本题涉及知识点:命令执行正则匹配linux命令绕过二、基本环境和工具Linux系统PHPApacheFirefoxIndex.phpFlag.php三、Writeup1、根据题目要求还原实验环境首先在Linux虚拟机上安装Ap…