forked from cloudprober/cloudprober
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserviceimpl_test.go
230 lines (197 loc) · 6.75 KB
/
serviceimpl_test.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
// Copyright 2019 The Cloudprober Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package prober
import (
"context"
"reflect"
"sort"
"testing"
"time"
"github.com/cloudprober/cloudprober/metrics"
pb "github.com/cloudprober/cloudprober/prober/proto"
"github.com/cloudprober/cloudprober/probes"
"github.com/cloudprober/cloudprober/probes/options"
probes_configpb "github.com/cloudprober/cloudprober/probes/proto"
testdatapb "github.com/cloudprober/cloudprober/probes/testdata"
targetspb "github.com/cloudprober/cloudprober/targets/proto"
"google.golang.org/protobuf/proto"
)
func testProber() *Prober {
pr := &Prober{
Probes: make(map[string]*probes.ProbeInfo),
probeCancelFunc: make(map[string]context.CancelFunc),
grpcStartProbeCh: make(chan string),
}
// Start a never-ending loop to clear pr.grpcStartProbeCh channel.
go func() {
for {
select {
case name := <-pr.grpcStartProbeCh:
pr.startProbe(context.Background(), name)
}
}
}()
return pr
}
// testProbe implements the probes.Probe interface, while providing
// facilities to examine the probe status for the purpose of testing.
// Since cloudprober has to be aware of the probe type, we add testProbe to
// cloudprober as an EXTENSION probe type (done through the init() function
// below).
type testProbe struct {
intialized bool
runningStatusCh chan bool
}
func (p *testProbe) Init(name string, opts *options.Options) error {
p.intialized = true
p.runningStatusCh = make(chan bool)
return nil
}
func (p *testProbe) Start(ctx context.Context, dataChan chan *metrics.EventMetrics) {
p.runningStatusCh <- true
// If context is done (used to stop a running probe before removing it),
// change probe state to not-running.
<-ctx.Done()
p.runningStatusCh <- false
close(p.runningStatusCh)
}
// We use an EXTENSION probe for testing. Following has the same effect as:
// This has the same effect as using the following in your config:
// probe {
// name: "<name>"
// targets {
// dummy_targets{}
// }
// [cloudprober.probes.testdata.fancy_probe] {
// name: "fancy"
// }
// }
func testProbeDef(name string) *probes_configpb.ProbeDef {
probeDef := &probes_configpb.ProbeDef{
Name: proto.String(name),
Type: probes_configpb.ProbeDef_EXTENSION.Enum(),
Targets: &targetspb.TargetsDef{
Type: &targetspb.TargetsDef_DummyTargets{},
},
}
proto.SetExtension(probeDef, testdatapb.E_FancyProbe, &testdatapb.FancyProbe{Name: proto.String("fancy-" + name)})
return probeDef
}
// verifyProbeRunningStatus is a helper function to verify probe's running
// status.
func verifyProbeRunningStatus(t *testing.T, p *testProbe, expectedRunning bool) {
t.Helper()
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
var running, timedout bool
select {
case running = <-p.runningStatusCh:
case <-ctx.Done():
timedout = true
}
if timedout {
t.Errorf("timed out while waiting for test probe's running status")
return
}
if running != expectedRunning {
t.Errorf("Running status: expected=%v, got=%v", expectedRunning, running)
}
}
func TestAddProbe(t *testing.T) {
pr := testProber()
// Test AddProbe()
// Empty probe config should return an error
_, err := pr.AddProbe(context.Background(), &pb.AddProbeRequest{})
if err == nil {
t.Error("empty probe config didn't result in error")
}
// Add a valid test probe
testProbeName := "test-probe"
_, err = pr.AddProbe(context.Background(), &pb.AddProbeRequest{ProbeConfig: testProbeDef(testProbeName)})
if err != nil {
t.Errorf("error while adding the probe: %v", err)
}
if pr.Probes[testProbeName] == nil {
t.Errorf("test-probe not added to the probes database")
}
p := pr.Probes[testProbeName].Probe.(*testProbe)
if !p.intialized {
t.Errorf("test probe not initialized. p.initialized=%v", p.intialized)
}
verifyProbeRunningStatus(t, p, true)
// Add test probe again, should result in error
_, err = pr.AddProbe(context.Background(), &pb.AddProbeRequest{ProbeConfig: testProbeDef(testProbeName)})
if err == nil {
t.Error("empty probe config didn't result in error")
}
}
func TestListProbes(t *testing.T) {
pr := testProber()
// Add couple of probes for testing
testProbes := []string{"test-probe-1", "test-probe-2"}
for _, name := range testProbes {
pr.AddProbe(context.Background(), &pb.AddProbeRequest{ProbeConfig: testProbeDef(name)})
}
// Test ListProbes()
resp, err := pr.ListProbes(context.Background(), &pb.ListProbesRequest{})
if err != nil {
t.Errorf("error while list probes: %v", err)
}
respProbes := resp.GetProbe()
if len(respProbes) != len(testProbes) {
t.Errorf("didn't get correct number of probe in ListProbes response. Got %d probes, expected %d probes", len(respProbes), len(testProbes))
}
var respProbeNames []string
for _, p := range respProbes {
respProbeNames = append(respProbeNames, p.GetName())
t.Logf("Probe Config: %+v", p.GetConfig())
}
sort.Strings(respProbeNames)
if !reflect.DeepEqual(respProbeNames, testProbes) {
t.Errorf("Probes in ListProbes() response: %v, expected: %s", respProbeNames, testProbes)
}
}
func TestRemoveProbes(t *testing.T) {
pr := testProber()
testProbeName := "test-probe"
// Remove a non-existent probe, should result in error.
_, err := pr.RemoveProbe(context.Background(), &pb.RemoveProbeRequest{ProbeName: &testProbeName})
if err == nil {
t.Error("removing non-existent probe didn't result in error")
}
// Add a probe for testing
_, err = pr.AddProbe(context.Background(), &pb.AddProbeRequest{ProbeConfig: testProbeDef(testProbeName)})
if err != nil {
t.Errorf("error while adding test probe: %v", err)
}
p := pr.Probes[testProbeName].Probe.(*testProbe)
// Clear probe's running status channel before removing the probe and thus
// causing another running status update
<-p.runningStatusCh
_, err = pr.RemoveProbe(context.Background(), &pb.RemoveProbeRequest{ProbeName: &testProbeName})
if err != nil {
t.Errorf("error while removing probe: %v", err)
}
if pr.Probes[testProbeName] != nil {
t.Errorf("test probe still in the probes database: %v", pr.Probes[testProbeName])
}
// Verify that probe is not running anymore
verifyProbeRunningStatus(t, p, false)
}
func init() {
// Register extension probe.
probes.RegisterProbeType(200, func() probes.Probe {
return &testProbe{}
})
}