-
Notifications
You must be signed in to change notification settings - Fork 787
/
gc_helm.go
247 lines (223 loc) · 6.67 KB
/
gc_helm.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
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
package cmd
import (
"bytes"
"errors"
"fmt"
"io"
"io/ioutil"
"os"
"regexp"
"sort"
"strconv"
"github.com/ghodss/yaml"
"gopkg.in/AlecAivazis/survey.v1/terminal"
"github.com/spf13/cobra"
v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"github.com/jenkins-x/jx/pkg/jx/cmd/templates"
"github.com/jenkins-x/jx/pkg/log"
)
// GetOptions is the start of the data required to perform the operation. As new fields are added, add them here instead of
// referencing the cmd.Flags()
type GCHelmOptions struct {
CommonOptions
RevisionHistoryLimit int
OutDir string
DryRun bool
NoBackup bool
}
var (
GCHelmLong = templates.LongDesc(`
Garbage collect Helm ConfigMaps. To facilitate rollbacks, Helm leaves a history of chart versions in place in Kubernetes and these should be pruned at intervals to avoid consuming excessive system resources.
`)
GCHelmExample = templates.Examples(`
jx garbage collect helm
jx gc helm
`)
)
// NewCmdGCHelm a command object for the "garbage collect" command
func NewCmdGCHelm(f Factory, in terminal.FileReader, out terminal.FileWriter, errOut io.Writer) *cobra.Command {
options := &GCHelmOptions{
CommonOptions: CommonOptions{
Factory: f,
In: in,
Out: out,
Err: errOut,
},
}
cmd := &cobra.Command{
Use: "helm",
Short: "garbage collection for Helm ConfigMaps",
Long: GCHelmLong,
Example: GCHelmExample,
Run: func(cmd *cobra.Command, args []string) {
options.Cmd = cmd
options.Args = args
err := options.Run()
CheckErr(err)
},
}
options.addCommonFlags(cmd)
cmd.Flags().IntVarP(&options.RevisionHistoryLimit, "revision-history-limit", "", 10, "Minimum number of versions per release to keep")
cmd.Flags().StringVarP(&options.OutDir, optionOutputDir, "o", "configmaps", "Relative directory to output backup to. Defaults to ./configmaps")
cmd.Flags().BoolVarP(&options.DryRun, "dry-run", "", false, "Does not perform the delete operation on Kubernetes")
cmd.Flags().BoolVarP(&options.NoBackup, "no-backup", "", false, "Does not perform the backup operation to store files locally")
return cmd
}
func (o *GCHelmOptions) Run() error {
kubeClient, _, err := o.KubeClient()
if err != nil {
return err
}
kubeNamespace := "kube-system"
cms, err := kubeClient.CoreV1().ConfigMaps(kubeNamespace).List(metav1.ListOptions{LabelSelector: "OWNER=TILLER"})
if err != nil {
return err
}
if len(cms.Items) == 0 {
// no configmaps found so lets return gracefully
if o.Verbose {
log.Info("no config maps found\n")
}
return nil
}
releases := ExtractReleases(cms)
if o.Verbose {
log.Info(fmt.Sprintf("Found %d releases.\n", len(releases)))
log.Info(fmt.Sprintf("Releases: %v\n", releases))
}
for _, release := range releases {
if o.Verbose {
log.Info(fmt.Sprintf("Checking %s. ", release))
}
versions := ExtractVersions(cms, release)
if o.Verbose {
log.Info(fmt.Sprintf("Found %d.\n", len(versions)))
log.Info(fmt.Sprintf("%v\n", versions))
}
to_delete := VersionsToDelete(versions, o.RevisionHistoryLimit)
if len(to_delete) > 0 {
if o.DryRun {
log.Infoln("Would delete:")
log.Infof("%v\n", to_delete)
} else {
// Backup and delete
if o.NoBackup == false {
// First make sure that destination path exists
err3 := os.MkdirAll(o.OutDir, 0755)
if err3 != nil {
// Failed to create path
return err3
}
}
for _, version := range to_delete {
cm, err1 := ExtractConfigMap(cms, version)
if err1 == nil {
if o.NoBackup == false {
// Create backup for ConfigMap about to be deleted
filename := o.OutDir + "/" + version + ".yaml"
log.Info(fmt.Sprintf("Backing up %v. ", filename))
y, err2 := yaml.Marshal(cm)
if err2 != nil {
// Failed to Marshall to YAML
return err2
}
// Add apiVersion and Kind
var b bytes.Buffer
b.WriteString("apiVersion: v1\nkind: ConfigMap\n")
b.Write(y)
err4 := ioutil.WriteFile(filename, b.Bytes(), 0644)
if err4 == nil {
log.Info("Success. ")
} else {
// Failed to write backup so abort
return err4
}
}
// Now delete
var opts *metav1.DeleteOptions
err5 := kubeClient.CoreV1().ConfigMaps(kubeNamespace).Delete(version, opts)
if err5 == nil {
log.Info(fmt.Sprintf("ConfigMap %v deleted.\n", version))
} else {
// Failed to delete
return err5
}
} else {
// Failed to find a ConfigMap that we know was in memory. Unlikely to occur.
log.Warn(fmt.Sprintf("Failed to find ConfigMap %s. \n", version))
}
}
}
} else {
if o.Verbose {
log.Info("Nothing to do.")
}
}
}
return nil
}
// ExtractReleases Extract a set of releases from a list of ConfigMaps
func ExtractReleases(cms *v1.ConfigMapList) []string {
found := make(map[string]bool)
for _, cm := range cms.Items {
if cmname, ok := cm.Labels["NAME"]; ok {
// Collect unique names
if _, seen := found[cmname]; !seen {
found[cmname] = true
}
}
}
// Return a set of unique helm releases
releases := []string{}
for key, _ := range found {
releases = append(releases, key)
}
return releases
}
// ExtractVersions Extract a set of versions of a named release from a list of ConfigMaps
func ExtractVersions(cms *v1.ConfigMapList, release string) []string {
found := []string{}
for _, cm := range cms.Items {
if release == cm.Labels["NAME"] {
found = append(found, cm.Name)
}
}
return found
}
// VersionsToDelete returns a slice of strings
func VersionsToDelete(versions []string, desired int) []string {
if desired >= len(versions) {
// nothing to delete
return []string{}
}
sort.Sort(ByVersion(versions))
return versions[:len(versions)-desired]
}
// ExtractConfigMap extracts a configmap
func ExtractConfigMap(cms *v1.ConfigMapList, version string) (v1.ConfigMap, error) {
for _, cm := range cms.Items {
if version == cm.Name {
return cm, nil
}
}
return v1.ConfigMap{}, errors.New("Not found")
}
// Components for sorting versions by numeric version number where version name ends in .vddd where ddd is an arbitrary sequence of digits
type ByVersion []string
func (a ByVersion) Len() int { return len(a) }
func (a ByVersion) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
func (a ByVersion) Less(i, j int) bool {
r, _ := regexp.Compile(`.v\d*$`)
loc := r.FindStringIndex(a[i])
if loc == nil {
return false
}
trim := loc[0] + 2 // start of numeric
version_number_i, err_i := strconv.Atoi(a[i][trim:])
version_number_j, err_j := strconv.Atoi(a[j][trim:])
if (err_i == nil) && (err_j == nil) {
return version_number_i < version_number_j
}
return false
}