-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmain.go
233 lines (204 loc) · 6.56 KB
/
main.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
// Provides Helm functionality
//
// The main focus is to publish new Helm Chart versions on registries.
//
// For accomplishing this the https://helm.sh/ tool is used.
package main
import (
"context"
"dagger/helm/internal/dagger"
"fmt"
"os"
"strings"
)
const HELM_IMAGE string = "quay.io/puzzle/dagger-module-helm:latest"
type Helm struct{}
type PushOpts struct {
Registry string `yaml:"registry"`
Repository string `yaml:"repository"`
Oci bool `yaml:"oci"`
Username string `yaml:"username"`
Password *dagger.Secret
}
func (p PushOpts) getChartFqdn(name string) string {
if p.Oci {
return fmt.Sprintf("oci://%s/%s/%s", p.Registry, p.Repository, name)
}
return fmt.Sprintf("%s/%s/%s", p.Registry, p.Repository, name)
}
func (p PushOpts) getRepoFqdn() string {
if p.Oci {
return fmt.Sprintf("oci://%s/%s", p.Registry, p.Repository)
}
return fmt.Sprintf("%s/%s", p.Registry, p.Repository)
}
// Get and display the version of the Helm Chart located inside the given directory.
//
// Example usage: dagger call version --directory ./helm/examples/testdata/mychart/
func (h *Helm) Version(
// method call context
ctx context.Context,
// directory that contains the Helm Chart
directory *dagger.Directory,
) (string, error) {
c := h.createContainer(directory)
version, err := c.WithExec([]string{"sh", "-c", "helm show chart . | yq eval '.version' -"}).Stdout(ctx)
if err != nil {
return "", err
}
return strings.TrimSpace(version), nil
}
// Packages and pushes a Helm chart to a specified OCI-compatible registry with authentication.
//
// Returns true if the chart was successfully pushed, or false if the chart already exists, with error handling for push failures.
//
// Example usage:
//
// dagger call package-push \
// --registry registry.puzzle.ch \
// --repository helm \
// --username $REGISTRY_HELM_USER \
// --password env:REGISTRY_HELM_PASSWORD \
// --directory ./examples/testdata/mychart/
func (h *Helm) PackagePush(
// method call context
ctx context.Context,
// directory that contains the Helm Chart
directory *dagger.Directory,
// URL of the registry
registry string,
// name of the repository
repository string,
// registry login username
username string,
// registry login password
password *dagger.Secret,
) (bool, error) {
opts := PushOpts{
Registry: registry,
Repository: repository,
Oci: true,
Username: username,
Password: password,
}
fmt.Fprintf(os.Stdout, "☸️ Helm package and Push")
c := dag.Container().
From("registry.puzzle.ch/cicd/alpine-base:latest").
WithDirectory("/helm", directory).
WithWorkdir("/helm")
version, err := c.WithExec([]string{"sh", "-c", "helm show chart . | yq eval '.version' -"}).Stdout(ctx)
if err != nil {
return false, err
}
version = strings.TrimSpace(version)
name, err := c.WithExec([]string{"sh", "-c", "helm show chart . | yq eval '.name' -"}).Stdout(ctx)
if err != nil {
return false, err
}
name = strings.TrimSpace(name)
c, err = c.
WithEnvVariable("REGISTRY_URL", opts.Registry).
WithEnvVariable("REGISTRY_USERNAME", opts.Username).
WithSecretVariable("REGISTRY_PASSWORD", opts.Password).
WithExec([]string{"sh", "-c", `echo ${REGISTRY_PASSWORD} | helm registry login ${REGISTRY_URL} --username ${REGISTRY_USERNAME} --password-stdin`}).
Sync(ctx)
if err != nil {
return false, err
}
//TODO: Refactor with return
c, err = c.WithExec([]string{"sh", "-c", fmt.Sprintf("helm show chart %s --version %s; echo -n $? > /ec", opts.getChartFqdn(name), version)}).Sync(ctx)
if err != nil {
return false, err
}
exc, err := c.File("/ec").Contents(ctx)
if err != nil {
return false, err
}
if exc == "0" {
//Chart exists
return false, nil
}
_, err = c.WithExec([]string{"helm", "dependency", "update", "."}).
WithExec([]string{"helm", "package", "."}).
WithExec([]string{"sh", "-c", "ls"}).
WithExec([]string{"helm", "push", fmt.Sprintf("%s-%s.tgz", name, version), opts.getRepoFqdn()}).
Sync(ctx)
if err != nil {
return false, err
}
return true, nil
}
// Run Helm unittests with the given directory and files.
//
// Provide the helm chart directory with pointing to it with the `--directory` flag.
// Add the directory location with `"."` as `--args` parameter to tell helm unittest where to find the helm chart with the tests.
//
// Example usage: dagger call test --directory ./helm/examples/testdata/mychart/ --args "."
func (h *Helm) Test(
// method call context
ctx context.Context,
// directory that contains the Helm Chart
directory *dagger.Directory,
// Helm Unittest arguments
args []string,
) (string, error) {
c := h.createContainer(directory)
out, err := c.WithExec([]string{"sh", "-c", fmt.Sprintf("%s %s", "helm-unittest", strings.Join(args, " "))}).Stdout(ctx)
if err != nil {
return "", err
}
return out, nil
}
// Run Helm lint with the given directory.
//
// Provide the helm chart directory with pointing to it with the `--directory` flag.
// Use `--args` parameter to pass alternative chart locations or additional options to Helm lint - see https://helm.sh/docs/helm/helm_lint/#options
//
// Example usage: dagger call lint --directory ./helm/examples/testdata/mychart/ --args "--quiet"
func (h *Helm) Lint(
// method call context
ctx context.Context,
// directory that contains the Helm Chart
directory *dagger.Directory,
// Helm lint arguments
// +optional
args []string,
) (string, error) {
var c *dagger.Container
if h.hasMissingDependencies(ctx, directory) {
c = h.createContainer(directory).WithMountedDirectory("./charts", h.dependencyUpdate(directory))
} else {
c = h.createContainer(directory)
}
out, err := c.WithExec([]string{"sh", "-c", fmt.Sprintf("%s %s", "helm lint", strings.Join(args, " "))}).Stdout(ctx)
if err != nil {
return "", err
}
return out, nil
}
func (h *Helm) hasMissingDependencies(
// method call context
ctx context.Context,
// directory that contains the Helm Chart
directory *dagger.Directory,
) bool {
_, err := h.createContainer(directory).WithExec([]string{"sh", "-c", "helm dep list | grep missing"}).Stdout(ctx)
return err == nil
}
func (h *Helm) dependencyUpdate(
// directory that contains the Helm Chart
directory *dagger.Directory,
) *dagger.Directory {
c := h.createContainer(directory)
return c.WithExec([]string{"sh", "-c", "helm dep update"}).Directory("charts")
}
func (h *Helm) createContainer(
// directory that contains the Helm Chart
directory *dagger.Directory,
) *dagger.Container {
return dag.Container().
From(HELM_IMAGE).
WithDirectory("/helm", directory, dagger.ContainerWithDirectoryOpts{Owner: "1001"}).
WithWorkdir("/helm").
WithoutEntrypoint()
}