这篇文章是记录自己查看客户端grpc负载均衡源码的过程,并没有太详细的讲解,参考价值不大,可以直接跳过,主要给自己看的。
一.主要接口:Balancer Resolver
1.Balancer定义
data:image/s3,"s3://crabby-images/ee129/ee1295072084b9934a62041c9cbc04719ffd71d3" alt=""
Resolver定义
data:image/s3,"s3://crabby-images/23e3d/23e3d295bb2c988e25d4bab737ad542b64ab2a7c" alt=""
具体位置为
data:image/s3,"s3://crabby-images/0db03/0db03adaee2fd492ffb0ca7c1ba22ee5d5a738b4" alt=""
1.grpc源码对解析器(resolver)Builder接口有三个结构体进行实现:dns,passthrough,unix。在clientconn.go文件中通过引用包在相应包的初始化方法中进行了注册
data:image/s3,"s3://crabby-images/bda5b/bda5b367a575702f9c224f8dab52abc37548a71a" alt=""
2.进入internal/resolver/passthrough/passthrough.go文件中init初始化函数中看到了将passthroughBuilder进行了注册。调用了resolver包的Register函数
func init() {
resolver.Register(&passthroughBuilder{})
}
3.Register将注册后的结构保存在全局变量m中,是一个map。key为Scheme,value为Builder接口。我们看到grpc默认的scheme是passthrough即透传什么也不做
package resolver
import (
"context"
"net"
"net/url"
"strings"
"google.golang.org/grpc/attributes"
"google.golang.org/grpc/credentials"
"google.golang.org/grpc/internal/pretty"
"google.golang.org/grpc/serviceconfig"
)
var (
// m is a map from scheme to resolver builder.
m = make(map[string]Builder)
// defaultScheme is the default scheme to use.
defaultScheme = "passthrough"
)
// TODO(bar) install dns resolver in init(){}.
// Register registers the resolver builder to the resolver map. b.Scheme will be
// used as the scheme registered with this builder.
//
// NOTE: this function must only be called during initialization time (i.e. in
// an init() function), and is not thread-safe. If multiple Resolvers are
// registered with the same name, the one registered last will take effect.
func Register(b Builder) {
m[b.Scheme()] = b
}
// Get returns the resolver builder registered with the given scheme.
//
// If no builder is register with the scheme, nil will be returned.
func Get(scheme string) Builder {
if b, ok := m[scheme]; ok {
return b
}
return nil
}
4.再看下passthroughBuilder,看到它实现了Buider接口,Build方法返回实现Resolver接口的结构passthroughResolver。其中passthroughBuilder用来创建解析器,passthroughResolver为创建的解析器
package passthrough
import (
"errors"
"google.golang.org/grpc/resolver"
)
const scheme = "passthrough"
type passthroughBuilder struct{}
func (*passthroughBuilder) Build(target resolver.Target, cc resolver.ClientConn, opts resolver.BuildOptions) (resolver.Resolver, error) {
if target.Endpoint() == "" && opts.Dialer == nil {
return nil, errors.New("passthrough: received empty target in Build()")
}
r := &passthroughResolver{
target: target,
cc: cc,
}
r.start()
return r, nil
}
func (*passthroughBuilder) Scheme() string {
return scheme
}
type passthroughResolver struct {
target resolver.Target
cc resolver.ClientConn
}
func (r *passthroughResolver) start() {
r.cc.UpdateState(resolver.State{Addresses: []resolver.Address{{Addr: r.target.Endpoint()}}})
}
func (*passthroughResolver) ResolveNow(o resolver.ResolveNowOptions) {}
func (*passthroughResolver) Close() {}
func init() {
resolver.Register(&passthroughBuilder{})
}
5.这段代码中返回的resolverBuilder就是从上面找到注册的builder,resolver.Get就是1.3中对应的代码
data:image/s3,"s3://crabby-images/67f37/67f37bf94c30ecab142dfd0c04fa196a5180d2a7" alt=""
data:image/s3,"s3://crabby-images/9d624/9d6240528009c2e34619d172a784408aaec4249b" alt=""
data:image/s3,"s3://crabby-images/2d79e/2d79ed109b3c7f51c909319a6b3fa6ebfba62c2c" alt=""
7.生成解析器的封装newCCResolverWrapper,并调用build将创建解析器,这个过程中调用了解析器的start方法,这个方法比较重要的是做了两件事情。1.触发选择balancer信号。2.触发状态修改(其中有负载均衡即连接的创建)
data:image/s3,"s3://crabby-images/c83ed/c83ed6db0ae8556fad5e43451564b33e68668d04" alt=""
data:image/s3,"s3://crabby-images/19628/19628b0fa4c823d1d47cdda2635b6f2543be8e50" alt=""
data:image/s3,"s3://crabby-images/7de7c/7de7c4ad498e1a14b77bf379c31ea710aac5b3c5" alt=""
data:image/s3,"s3://crabby-images/6e126/6e126ef4eb847295dac2c44d59b67daf7a63fcb8" alt=""
data:image/s3,"s3://crabby-images/6a82f/6a82f45cda92fb922d8b21e6b5f98f973ed10264" alt=""
data:image/s3,"s3://crabby-images/e4c01/e4c01ea13fa0a729b702fdcf96f6fc3cb11a302f" alt=""
data:image/s3,"s3://crabby-images/c6e15/c6e15cd101528126b619849fcfd9a719a69493fc" alt=""
data:image/s3,"s3://crabby-images/588dc/588dc06dd162df04316b9b5b1e02c04b55864358" alt=""
data:image/s3,"s3://crabby-images/78992/7899240c7a681a048ab69797b2dcf843f2466330" alt=""
data:image/s3,"s3://crabby-images/69872/69872c6c1c1058bb76b160fd52bc182cc68be140" alt=""
上面分别向ccBalancerWrapper的channel中发送了switchToUpdate和ccStateUpdate
data:image/s3,"s3://crabby-images/611ca/611ca069a0656c9f67f313fb966401d67f14be90" alt=""
balancer选择逻辑
data:image/s3,"s3://crabby-images/67837/67837c28a452057c15244b1d2f3c9353ce7e9fa6" alt=""
baseBalancer会将所有地址都创建一个连接
data:image/s3,"s3://crabby-images/e9e26/e9e2620067b68b6f8fe6b8a25930b59747c6f045" alt=""
invoke的时候,会调用picker选择连接
data:image/s3,"s3://crabby-images/71b9f/71b9fcefbb23c6f1cbc36e7f3fbdfebdf71bea30" alt=""
data:image/s3,"s3://crabby-images/164fd/164fdf836bbc96c64b67c0ff08e381fe73032150" alt=""
data:image/s3,"s3://crabby-images/d730b/d730b36d19d7324ec8a8dc5366bdf65ad5ff35bd" alt=""
data:image/s3,"s3://crabby-images/9be10/9be1041ff3efd7373b50e4e20542c8a52d82b6f8" alt=""
data:image/s3,"s3://crabby-images/812ee/812ee2c3493ec70b4b45eb58201c3c5ca0fa5bf7" alt=""
data:image/s3,"s3://crabby-images/7c90e/7c90ec09031c2be0f90531be549600047c3ab4c8" alt=""
以上路径最终调用这个方法,可以看到是轮询使用连接
data:image/s3,"s3://crabby-images/71f7e/71f7e1718e7db2e5b18ec5238064abb425f49a8e" alt=""