-
Notifications
You must be signed in to change notification settings - Fork 6
Expand file tree
/
Copy pathunit.go
More file actions
126 lines (116 loc) · 3.04 KB
/
unit.go
File metadata and controls
126 lines (116 loc) · 3.04 KB
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 client
import (
"fmt"
"io/ioutil"
"os"
"path"
"regexp"
"strconv"
"strings"
"github.com/coreos/fleet/unit"
)
// path hierarchy for finding systemd service templates
var rootPaths = []string{"/var/lib/deis/units", "units", "../units"}
// getUnits returns a list of units filtered by target
func (c *FleetClient) getUnits(target string) (units []string, err error) {
jobs, err := c.Fleet.Jobs()
if err != nil {
return
}
var r *regexp.Regexp
if strings.HasSuffix(target, "-data") {
r = regexp.MustCompile(`deis\-(` + target + `)\.service`)
} else if strings.Contains(target, ".") {
r = regexp.MustCompile(`deis\-(` + target + `)\.service`)
} else {
r = regexp.MustCompile(`deis\-(` + target + `)\.([\d]+)\.service`)
}
for _, j := range jobs {
match := r.MatchString(j.Name)
if match {
units = append(units, j.Name)
}
}
return
}
// nextUnit returns the next unit number for a given component
func (c *FleetClient) nextUnit(component string) (num int, err error) {
units, err := c.getUnits(component)
if err != nil {
return
}
num, err = nextUnitNum(units)
if err != nil {
return
}
return
}
// lastUnit returns the last unit number for a given component
func (c *FleetClient) lastUnit(component string) (num int, err error) {
units, err := c.getUnits(component)
if err != nil {
return
}
num, err = lastUnitNum(units)
if err != nil {
return
}
return
}
// NewUnit takes a component type and returns a Fleet unit
// that includes the relevant systemd service template
func NewUnit(component string) (u *unit.Unit, err error) {
template, err := readTemplate(component)
if err != nil {
return
}
u, err = unit.NewUnit(string(template))
if err != nil {
return
}
return
}
// NewDataUnit takes a component type and returns a Fleet unit
// that is hard-scheduled to a machine ID
func NewDataUnit(component string, machineID string) (u *unit.Unit, err error) {
template, err := readTemplate(component)
if err != nil {
return
}
// replace CHANGEME with random machineID
replaced := strings.Replace(string(template), "CHANGEME", machineID, 1)
u, err = unit.NewUnit(replaced)
if err != nil {
return
}
return
}
// formatUnitName returns a properly formatted systemd service name
// using the given component type and number
func formatUnitName(component string, num int) (unitName string, err error) {
if num == 0 {
return "deis-" + component + ".service", nil
} else {
return "deis-" + component + "." + strconv.Itoa(num) + ".service", nil
}
}
// readTemplate returns the contents of a systemd template for the given component
func readTemplate(component string) (out []byte, err error) {
templateName := "deis-" + component + ".service"
var templateFile string
for _, rootPath := range rootPaths {
filename := path.Join(rootPath, templateName)
if _, err := os.Stat(filename); err == nil {
templateFile = filename
break
}
}
if templateFile == "" {
return nil, fmt.Errorf("Could not find unit template for %v", component)
}
out, err = ioutil.ReadFile(templateFile)
if err != nil {
return
}
return
}