forked from adyatlov/bun-deprecated
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcheck_builder.go
136 lines (124 loc) · 3.6 KB
/
check_builder.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
package bun
import (
"fmt"
)
const errName = "bun.CheckBuilder.Build: Check Name should not be empty"
const errForEach = "bun.CheckBuilder.Build: At least one of the ForEach functions" +
"shoul be specified"
// MsgErr is a standard message used in the check summary when errors
// occures during the check.
const MsgErr = "Error(s) occurred while performing the check."
// CheckHost checks an individual host. It returns status, details, and error if
// the function cannot perform a check. If the returned error is not nil, then
// the status is ignored.
type CheckHost func(Host) (bool, interface{}, error)
// Aggregate check reults.
type Aggregate func(*Check, CheckBuilder)
// Result hols results of the CheckHost function.
type Result struct {
Host Host
OK bool
Details interface{}
Err error
}
// CheckBuilder helps to create checks.
type CheckBuilder struct {
Name string // Required
Description string // Optional
CollectFromMasters CheckHost // At least one of
CollectFromAgents CheckHost // the Collect... functions
CollectFromPublicAgents CheckHost // are required
ProblemSummary string // Optional
OKSummary string // Optional
Aggregate Aggregate // Implement if the default is not sufficient
Problems []Result // Do not set
OKs []Result // Do not set
}
// Build returns a Check
func (b *CheckBuilder) Build() Check {
if b.Name == "" {
panic(errName)
}
if b.CollectFromMasters == nil && b.CollectFromAgents == nil &&
b.CollectFromPublicAgents == nil {
panic(errForEach)
}
if b.ProblemSummary == "" {
b.ProblemSummary = "Problems were found."
}
if b.OKSummary == "" {
b.OKSummary = "No problems were found."
}
if b.Aggregate == nil {
panic("CheckBuilder.Aggregate should be set.")
}
return Check{
Name: b.Name,
Description: b.Description,
CheckFunc: b.checkFunc,
}
}
func formatMsg(h Host, msg string) string {
return fmt.Sprintf("%v %v: %v", h.Type, h.IP, msg)
}
// Default implementation of the Aggregate function.
// It assumes that the implementations of the CheckHost function return
// Result Details as a string or nil.
func DefaultAggregate(c *Check, b CheckBuilder) {
for _, r := range b.Problems {
if r.Details != nil {
c.Problems = append(c.Problems, formatMsg(r.Host, r.Details.(string)))
}
}
for _, r := range b.OKs {
if r.Details != nil {
c.OKs = append(c.OKs, formatMsg(r.Host, r.Details.(string)))
}
}
}
func (b *CheckBuilder) checkHosts(c *Check, h map[string]Host, ch CheckHost) {
for _, host := range h {
r := Result{}
r.Host = host
r.OK, r.Details, r.Err = ch(host)
if r.Err != nil {
c.Errors = append(c.Errors, formatMsg(r.Host, r.Err.Error()))
} else if r.OK {
b.OKs = append(b.OKs, r)
} else {
b.Problems = append(b.Problems, r)
}
}
}
// Implementation of the Check.CheckFunc
func (b *CheckBuilder) checkFunc(c *Check, bundle Bundle) {
if b.CollectFromMasters != nil {
b.checkHosts(c, bundle.Masters, b.CollectFromMasters)
}
if b.CollectFromAgents != nil {
b.checkHosts(c, bundle.Agents, b.CollectFromAgents)
}
if b.CollectFromPublicAgents != nil {
b.checkHosts(c, bundle.PublicAgents, b.CollectFromPublicAgents)
}
b.Aggregate(c, *b)
if len(c.Problems) > 0 {
c.Status = SProblem
if c.Summary == "" {
c.Summary = b.ProblemSummary
}
if len(c.Errors) > 0 {
c.Summary += " " + MsgErr
}
} else if len(c.Errors) == 0 {
c.Status = SOK
if c.Summary == "" {
c.Summary = b.OKSummary
}
} else {
c.Status = SUndefined
if c.Summary == "" {
c.Summary = MsgErr
}
}
}