163 lines
3.7 KiB
163 lines
3.7 KiB
package resolver
|
|
|
|
import (
|
|
"context"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/go-kratos/kratos/pkg/conf/env"
|
|
"github.com/go-kratos/kratos/pkg/log"
|
|
"github.com/go-kratos/kratos/pkg/naming"
|
|
wmeta "github.com/go-kratos/kratos/pkg/net/rpc/warden/internal/metadata"
|
|
|
|
"github.com/pkg/errors"
|
|
"google.golang.org/grpc/resolver"
|
|
)
|
|
|
|
const (
|
|
// Scheme is the scheme of discovery address
|
|
Scheme = "grpc"
|
|
)
|
|
|
|
var (
|
|
_ resolver.Resolver = &Resolver{}
|
|
_ resolver.Builder = &Builder{}
|
|
mu sync.Mutex
|
|
)
|
|
|
|
// Register register resolver builder if nil.
|
|
func Register(b naming.Builder) {
|
|
mu.Lock()
|
|
defer mu.Unlock()
|
|
if resolver.Get(b.Scheme()) == nil {
|
|
resolver.Register(&Builder{b})
|
|
}
|
|
}
|
|
|
|
// Set override any registered builder
|
|
func Set(b naming.Builder) {
|
|
mu.Lock()
|
|
defer mu.Unlock()
|
|
resolver.Register(&Builder{b})
|
|
}
|
|
|
|
// Builder is also a resolver builder.
|
|
// It's build() function always returns itself.
|
|
type Builder struct {
|
|
naming.Builder
|
|
}
|
|
|
|
// Build returns itself for Resolver, because it's both a builder and a resolver.
|
|
func (b *Builder) Build(target resolver.Target, cc resolver.ClientConn, opts resolver.BuildOption) (resolver.Resolver, error) {
|
|
var zone = env.Zone
|
|
ss := int64(50)
|
|
clusters := map[string]struct{}{}
|
|
str := strings.SplitN(target.Endpoint, "?", 2)
|
|
if len(str) == 0 {
|
|
return nil, errors.Errorf("warden resolver: parse target.Endpoint(%s) failed!err:=endpoint is empty", target.Endpoint)
|
|
} else if len(str) == 2 {
|
|
m, err := url.ParseQuery(str[1])
|
|
if err == nil {
|
|
for _, c := range m[naming.MetaCluster] {
|
|
clusters[c] = struct{}{}
|
|
}
|
|
zones := m[naming.MetaZone]
|
|
if len(zones) > 0 {
|
|
zone = zones[0]
|
|
}
|
|
if sub, ok := m["subset"]; ok {
|
|
if t, err := strconv.ParseInt(sub[0], 10, 64); err == nil {
|
|
ss = t
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
r := &Resolver{
|
|
nr: b.Builder.Build(str[0], naming.Filter(Scheme, clusters), naming.ScheduleNode(zone), naming.Subset(int(ss))),
|
|
cc: cc,
|
|
quit: make(chan struct{}, 1),
|
|
zone: zone,
|
|
}
|
|
go r.updateproc()
|
|
return r, nil
|
|
}
|
|
|
|
// Resolver watches for the updates on the specified target.
|
|
// Updates include address updates and service config updates.
|
|
type Resolver struct {
|
|
nr naming.Resolver
|
|
cc resolver.ClientConn
|
|
quit chan struct{}
|
|
|
|
clusters map[string]struct{}
|
|
zone string
|
|
subsetSize int64
|
|
}
|
|
|
|
// Close is a noop for Resolver.
|
|
func (r *Resolver) Close() {
|
|
select {
|
|
case r.quit <- struct{}{}:
|
|
r.nr.Close()
|
|
default:
|
|
}
|
|
}
|
|
|
|
// ResolveNow is a noop for Resolver.
|
|
func (r *Resolver) ResolveNow(o resolver.ResolveNowOption) {
|
|
}
|
|
|
|
func (r *Resolver) updateproc() {
|
|
event := r.nr.Watch()
|
|
for {
|
|
select {
|
|
case <-r.quit:
|
|
return
|
|
case _, ok := <-event:
|
|
if !ok {
|
|
return
|
|
}
|
|
}
|
|
if ins, ok := r.nr.Fetch(context.Background()); ok {
|
|
instances, _ := ins.Instances[r.zone]
|
|
if len(instances) == 0 {
|
|
for _, value := range ins.Instances {
|
|
instances = append(instances, value...)
|
|
}
|
|
}
|
|
r.newAddress(instances)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (r *Resolver) newAddress(instances []*naming.Instance) {
|
|
if len(instances) <= 0 {
|
|
return
|
|
}
|
|
addrs := make([]resolver.Address, 0, len(instances))
|
|
for _, ins := range instances {
|
|
var weight int64
|
|
if weight, _ = strconv.ParseInt(ins.Metadata[naming.MetaWeight], 10, 64); weight <= 0 {
|
|
weight = 10
|
|
}
|
|
var rpc string
|
|
for _, a := range ins.Addrs {
|
|
u, err := url.Parse(a)
|
|
if err == nil && u.Scheme == Scheme {
|
|
rpc = u.Host
|
|
}
|
|
}
|
|
addr := resolver.Address{
|
|
Addr: rpc,
|
|
Type: resolver.Backend,
|
|
ServerName: ins.AppID,
|
|
Metadata: wmeta.MD{Weight: uint64(weight), Color: ins.Metadata[naming.MetaColor]},
|
|
}
|
|
addrs = append(addrs, addr)
|
|
}
|
|
log.Info("resolver: finally get %d instances", len(addrs))
|
|
r.cc.NewAddress(addrs)
|
|
}
|
|
|