-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathresolver.go
138 lines (129 loc) · 3.46 KB
/
resolver.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
package discovery
import (
"context"
"encoding/json"
"go.etcd.io/etcd/api/v3/mvccpb"
clientv3 "go.etcd.io/etcd/client/v3"
"google.golang.org/grpc/resolver"
"sync/atomic"
"time"
)
type Resolver struct {
ctx context.Context
prefix string
client *clientv3.Client
cc resolver.ClientConn
serviceMap map[string]ServiceInfo
running atomic.Bool
}
func NewResolver(ctx context.Context, eCfg EtcdConfig, prefix string) *Resolver {
cli, err := clientv3.New(clientv3.Config{
DialTimeout: time.Second * 3,
Endpoints: eCfg.Endpoints,
Username: eCfg.Username,
Password: eCfg.Password,
})
if err != nil {
panic(err)
}
r := &Resolver{
ctx: ctx,
prefix: prefix,
client: cli,
}
// 防止连完etcd程序已经停止了
select {
case <-ctx.Done():
panic("stopped by " + ctx.Err().Error())
default:
}
return r
}
func (r *Resolver) Build(_ resolver.Target, cc resolver.ClientConn, _ resolver.BuildOptions) (resolver.Resolver, error) {
Log.Printf("Resolver Build\n")
r.cc = cc
if r.running.Load() {
Log.Printf("Resolver already running\n")
r.update()
return r, nil
}
go r.run()
return r, nil
}
func (r *Resolver) Scheme() string {
return "etcd"
}
func (r *Resolver) ResolveNow(resolver.ResolveNowOptions) {}
func (r *Resolver) Close() {}
func (r *Resolver) update() {
Log.Printf("Resolver update prefix=%s services_count=%d\n", r.prefix, len(r.serviceMap))
if len(r.serviceMap) == 0 {
return
}
addresses := make([]resolver.Address, 0, len(r.serviceMap))
for _, srv := range r.serviceMap {
Log.Printf("Resolver update prefix=%s name=%s addr=%s\n", r.prefix, srv.Name, srv.Internal)
addresses = append(addresses, resolver.Address{
Addr: srv.Internal,
ServerName: srv.Name,
})
}
err := r.cc.UpdateState(resolver.State{Addresses: addresses})
if err != nil {
Log.Printf("Resolver UpdateState err %v\n", err.Error())
}
}
func (r *Resolver) run() {
r.running.Store(true)
r.serviceMap = make(map[string]ServiceInfo)
// 首次获取所有的服务,超时3s
timeoutCtx, fn := context.WithTimeout(context.Background(), time.Second*3)
defer fn()
resp, err := r.client.Get(timeoutCtx, r.prefix, clientv3.WithPrefix())
if err != nil {
panic(err)
}
if len(resp.Kvs) == 0 {
Log.Printf("当前没有服务 prefix:%s\n", r.prefix)
}
for _, kv := range resp.Kvs {
var info ServiceInfo
err = json.Unmarshal(kv.Value, &info)
if err != nil {
panic(err)
}
r.serviceMap[string(kv.Key)] = info
}
r.update()
//log.Debugf("resolver prefix:%s", r.prefix)
rch := r.client.Watch(context.Background(), r.prefix, clientv3.WithPrefix(), clientv3.WithPrevKV())
for {
select {
case <-r.ctx.Done():
Log.Printf("Resolver exit\n")
return
case n := <-rch:
for _, ev := range n.Events {
key := string(ev.Kv.Key)
switch ev.Type {
case mvccpb.PUT:
var info ServiceInfo
if err := json.Unmarshal(ev.Kv.Value, &info); err != nil {
Log.Printf("ev.Kv.Value json.Unmarshal error:%v\n", err)
continue
}
Log.Printf("[增加服务] key:%v server:%v id:%d internal:%v external:%s\n",
key, info.Name, info.ID, info.Internal, info.External)
r.serviceMap[key] = info
r.update()
case mvccpb.DELETE:
srvInfo := r.serviceMap[key]
Log.Printf("[移除服务] key:%v server:%v id:%d internal:%s external:%s\n", key,
srvInfo.Name, srvInfo.ID, srvInfo.Internal, srvInfo.External)
delete(r.serviceMap, string(ev.PrevKv.Key))
r.update()
}
}
}
}
}