This repository has been archived by the owner on Sep 4, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 594
/
backend_etcd.go
126 lines (111 loc) · 3.18 KB
/
backend_etcd.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
package discover
import (
"encoding/json"
"fmt"
"strings"
"sync"
"github.com/coreos/etcd/store"
"github.com/coreos/go-etcd/etcd"
)
const (
KeyPrefix = "/discover"
)
type EtcdBackend struct {
Client *etcd.Client
}
func servicePath(name, addr string) string {
if addr != "" {
return fmt.Sprintf("%s/services/%s/%s", KeyPrefix, name, addr)
} else {
return fmt.Sprintf("%s/services/%s", KeyPrefix, name)
}
}
func (b *EtcdBackend) Subscribe(name string) (UpdateStream, error) {
stream := &etcdStream{ch: make(chan *ServiceUpdate), stop: make(chan bool)}
watch := b.getStateChanges(name, stream.stop)
responses, err := b.getCurrentState(name)
if err != nil {
return nil, err
}
go func() {
for _, u := range responses {
if update := b.responseToUpdate(u); update != nil {
stream.ch <- update
}
}
stream.ch <- &ServiceUpdate{}
for u := range watch {
if update := b.responseToUpdate(u); update != nil {
stream.ch <- update
}
}
}()
return stream, nil
}
type etcdStream struct {
ch chan *ServiceUpdate
stop chan bool
stopOnce sync.Once
}
func (s *etcdStream) Chan() chan *ServiceUpdate { return s.ch }
func (s *etcdStream) Close() { s.stopOnce.Do(func() { close(s.stop) }) }
func (b *EtcdBackend) responseToUpdate(resp *store.Response) *ServiceUpdate {
// expected key structure: /PREFIX/services/NAME/ADDR
splitKey := strings.SplitN(resp.Key, "/", 5)
if len(splitKey) < 5 {
return nil
}
serviceName := splitKey[3]
serviceAddr := splitKey[4]
if "GET" == resp.Action || ("SET" == resp.Action && (resp.NewKey || resp.Value != resp.PrevValue)) {
// GET is because getCurrentState returns responses of Action GET.
// some SETs are heartbeats, so we ignore SETs where value didn't change.
var serviceAttrs map[string]string
err := json.Unmarshal([]byte(resp.Value), &serviceAttrs)
if err != nil {
return nil
}
return &ServiceUpdate{
Name: serviceName,
Addr: serviceAddr,
Online: true,
Attrs: serviceAttrs,
}
} else if "DELETE" == resp.Action {
return &ServiceUpdate{
Name: serviceName,
Addr: serviceAddr,
}
} else {
return nil
}
}
func (b *EtcdBackend) getCurrentState(name string) ([]*store.Response, error) {
return b.Client.Get(servicePath(name, ""))
}
func (b *EtcdBackend) getStateChanges(name string, stop chan bool) chan *store.Response {
watch := make(chan *store.Response)
go b.Client.Watch(servicePath(name, ""), 0, watch, stop)
return watch
}
func (b *EtcdBackend) Register(name, addr string, attrs map[string]string) error {
attrsJson, err1 := json.Marshal(attrs)
if err1 != nil {
return err1
}
_, err2 := b.Client.Set(servicePath(name, addr), string(attrsJson), HeartbeatIntervalSecs+MissedHearbeatTTL)
return err2
}
func (b *EtcdBackend) Heartbeat(name, addr string) error {
resp, err1 := b.Client.Get(servicePath(name, addr))
if err1 != nil {
return err1
}
// ignore test failure, it doesn't need a heartbeat if it was just set.
_, _, err2 := b.Client.TestAndSet(servicePath(name, addr), resp[0].Value, resp[0].Value, HeartbeatIntervalSecs+MissedHearbeatTTL)
return err2
}
func (b *EtcdBackend) Unregister(name, addr string) error {
_, err := b.Client.Delete(servicePath(name, addr))
return err
}