-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmonitor.go
138 lines (116 loc) · 3.59 KB
/
monitor.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 nested
import (
"math"
"math/rand"
"sync"
)
// A Monitor is a basic implementation of the nested service finite state machine.
//
// An empty Monitor is ready to use and in the Not Ready state. A Monitor must not be copied after first use.
type Monitor struct {
sync.Mutex
state State // current state
err error // current error state, if the state is not ready
errCount int // number of consecutive errors
callbacks map[Token]func(Event)
}
// Verifies that a Monitor implements the Service interface. Note that the Service interface does NOT include the
// SetState() method. This is by design, as SetState() should only be called by the packiage implementing the service
// and not by consumers of the service.
var _ Service = &Monitor{}
// GetState returns the current state of the service.
func (m *Monitor) GetState() State {
m.Lock()
defer m.Unlock()
return m.state
}
// Err returns the error from the most recent Err state, or nil if the Monitor has never been in the error state.
func (m *Monitor) Err() error {
m.Lock()
defer m.Unlock()
return m.err
}
// ErrCount returns the number of consective errors recorded for this service. If the service is not in Error state,
// ErrCount returns 0.
func (m *Monitor) ErrCount() int {
m.Lock()
defer m.Unlock()
return m.errCount
}
// Stop sets the service to stopped. If there are registered observers, all observers are called before returning.
func (m *Monitor) Stop() {
m.setState(Stopped, nil)
}
// RegisterCallback registers a function which will be called any time there is a state change. Returns a token that
// can be used to deregister it later.
func (m *Monitor) RegisterCallback(f func(Event)) Token {
m.Lock()
defer m.Unlock()
if m.callbacks == nil {
m.callbacks = make(map[Token]func(Event))
}
// Choose a random token that we haven't used.
var token Token
for ok := true; ok; {
token = Token(rand.Uint32())
_, ok = m.callbacks[token]
}
m.callbacks[token] = f
return token
}
// Deregister removes a registered callback. Does nothing if there is no callback registered with the provided token.
func (m *Monitor) DeregisterCallback(token Token) {
m.Lock()
defer m.Unlock()
delete(m.callbacks, token)
}
// SetReady sets the monitor state to Ready. If there are registered observers, all observers are called before returning.
// Panics if the monitor is already stopped.
func (m *Monitor) SetReady() {
m.setState(Ready, nil)
}
// SetReady sets the monitor state to Error. If there are registered observers, all observers are called before returning.
// Panics if the monitor is already stopped.
func (m *Monitor) SetError(err error) {
m.setState(Error, err)
}
func (m *Monitor) setState(newState State, newErr error) {
// Initialize the wait group first so that wg.Wait() runs after the lock is released. That way, if we block
// on any of the observer callbacks, we do so without holding the lock.
var wg sync.WaitGroup
defer wg.Wait()
m.Lock()
defer m.Unlock()
if newState == m.state && newState != Error {
return // nothing to do
}
if newState == Error {
if m.errCount < math.MaxInt {
m.errCount++
}
} else {
m.errCount = 0
}
if m.state == Stopped {
panic("cannot transition from stopped state")
}
ev := Event{
OldState: m.state,
NewState: newState,
Error: newErr,
ErrCount: m.errCount,
}
m.state = newState
if newState == Error {
m.err = newErr
}
// Notify all observers.
wg.Add(len(m.callbacks))
for _, cb := range m.callbacks {
// Run these in the background so as not to block while holding the lock.
go func(f func(Event)) {
f(ev)
wg.Done()
}(cb)
}
}