forked from eripa/prometheus-zfs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
zpool.go
111 lines (101 loc) · 2.58 KB
/
zpool.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
package main
import (
"errors"
"fmt"
"log"
"os/exec"
"strconv"
"strings"
)
type zpool struct {
name string
capacity int64
healthy bool
status string
online int64
faulted int64
}
func (z *zpool) checkHealth(output string) (err error) {
output = strings.Trim(output, "\n")
if output == "ONLINE" {
z.healthy = true
} else if output == "DEGRADED" || output == "FAULTED" {
z.healthy = false
} else {
z.healthy = false // just to make sure
err = errors.New("Unknown status")
}
return err
}
func (z *zpool) getCapacity(output string) (err error) {
s := strings.Split(output, "%")[0]
z.capacity, err = strconv.ParseInt(s, 0, 8)
if err != nil {
return err
}
return err
}
func (z *zpool) getProviders(output string) (err error) {
nonProviderLines := []string{
z.name,
"state:",
"mirror-",
"raid0-",
"raid10-",
"raidz-",
"raidz2-",
"raidz3-",
}
lines := strings.Split(output, "\n")
z.status = strings.Split(lines[1], " ")[2]
// Count all providers, ONLINE and FAULTED
var fcount int64
var dcount int64
for _, line := range lines {
if (strings.Contains(line, "FAULTED") || strings.Contains(line, "UNAVAIL")) && !substringInSlice(line, nonProviderLines) {
fcount = fcount + 1
} else if strings.Contains(line, "ONLINE") && !substringInSlice(line, nonProviderLines) {
dcount = dcount + 1
}
}
z.faulted = fcount
z.online = dcount
if z.status != "ONLINE" && z.status != "DEGRADED" && z.status != "FAULTED" {
z.faulted = 1 // fake faulted if there is a parsing error or other status
err = errors.New("Error parsing faulted/unavailable providers")
}
return
}
func (z *zpool) getStatus() {
output := runZpoolCommand([]string{"status", z.name})
err := z.getProviders(output)
if err != nil {
log.Fatal("Error parsing zpool status")
}
output = runZpoolCommand([]string{"list", "-H", "-o", "health", z.name})
err = z.checkHealth(output)
if err != nil {
log.Fatal("Error parsing zpool list -H -o health ", z.name)
}
output = runZpoolCommand([]string{"list", "-H", "-o", "cap", z.name})
err = z.getCapacity(output)
if err != nil {
log.Fatal("Error parsing zpool capacity")
}
}
func checkExistance(pool string) (err error) {
output := runZpoolCommand([]string{"list", pool})
if strings.Contains(fmt.Sprintf("%s", output), "no such pool") {
err = errors.New("No such pool")
}
return
}
func runZpoolCommand(args []string) string {
zpoolPath, err := exec.LookPath("zpool")
if err != nil {
log.Fatal("Could not find zpool in PATH")
}
cmd := exec.Command(zpoolPath, args...)
out, _ := cmd.CombinedOutput()
return fmt.Sprintf("%s", out)
}