Skip to content

Commit 96a4ed3

Browse files
committed
fix: wire up git getter within cli
Also, disable git getter by default and add IsUntrustedRepository method back to helm package for backward-compatibility and convenience.
1 parent 2965850 commit 96a4ed3

11 files changed

+94
-55
lines changed

README.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -265,8 +265,9 @@ It exposes a `Helm` struct that provides a `Render()` function that returns the
265265
| `outputPathMapping[].selectors[].kind` | | Selects resources by kind. |
266266
| `outputPathMapping[].selectors[].namespace` | | Selects resources by namespace. |
267267
| `outputPathMapping[].selectors[].name` | | Selects resources by name. |
268-
| | `--output-replace` | If enabled replace the output directory or file (CLI-only). |
269-
| | `--trust-any-repo` | If enabled repositories that are not registered within `repositories.yaml` can be used as well (env var `KHELM_TRUST_ANY_REPO`). Within the kpt function this behaviour can be disabled by mounting `/helm/repository/repositories.yaml` or disabling network access. |
268+
| | `--output-replace` | If enabled, replace the output directory or file (CLI-only). |
269+
| | `--trust-any-repo` | If enabled, repositories that are not registered within `repositories.yaml` can be used as well (env var `KHELM_TRUST_ANY_REPO`). Within the kpt function this behaviour can be disabled by mounting `/helm/repository/repositories.yaml` or disabling network access. |
270+
| | `--enable-git-getter` | If enabled, support helm repository URLs with the git+https scheme (env var `KHELM_ENABLE_GIT_GETTER`). |
270271
| `debug` | `--debug` | Enables debug log and provides a stack trace on error. |
271272

272273
### Repository configuration
@@ -288,6 +289,8 @@ The following example points to an old version of cert-manager using a git URL:
288289
git+https://github.com/cert-manager/cert-manager@deploy/charts?ref=v0.6.2
289290
```
290291

292+
To enable this feature, set the `--enable-git-getter` option or the corresponding environment variable: `KHELM_ENABLE_GIT_GETTER=true`.
293+
291294
This feature is meant to be compatible with Helm's [helm-git](https://github.com/aslafy-z/helm-git#usage) plugin (but is reimplemented in Go).
292295
However currently khelm does not support `sparse` git checkouts (due to [lack of support in go-git](https://github.com/go-git/go-git/issues/90)).
293296

cmd/khelm/common.go

+1-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@ import (
99

1010
"github.com/mgoltzsche/khelm/v2/pkg/config"
1111
"github.com/mgoltzsche/khelm/v2/pkg/helm"
12-
"github.com/mgoltzsche/khelm/v2/pkg/repositories"
1312
"sigs.k8s.io/kustomize/kyaml/yaml"
1413
)
1514

@@ -24,7 +23,7 @@ func render(h *helm.Helm, req *config.ChartConfig) ([]*yaml.RNode, error) {
2423
}()
2524

2625
rendered, err := h.Render(ctx, req)
27-
if repositories.IsUntrustedRepository(err) {
26+
if helm.IsUntrustedRepository(err) {
2827
log.Printf("HINT: access to untrusted repositories can be enabled using env var %s=true or option --%s", envTrustAnyRepo, flagTrustAnyRepo)
2928
}
3029
return rendered, err

cmd/khelm/root.go

+25-5
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ package main
22

33
import (
44
"bytes"
5+
"context"
56
"fmt"
67
"io"
78
"log"
@@ -10,13 +11,16 @@ import (
1011
"strconv"
1112
"strings"
1213

14+
"github.com/mgoltzsche/khelm/v2/pkg/getter/git"
1315
"github.com/mgoltzsche/khelm/v2/pkg/helm"
1416
"github.com/spf13/cobra"
17+
helmgetter "helm.sh/helm/v3/pkg/getter"
1518
)
1619

1720
const (
1821
envKustomizePluginConfig = "KUSTOMIZE_PLUGIN_CONFIG_STRING"
1922
envKustomizePluginConfigRoot = "KUSTOMIZE_PLUGIN_CONFIG_ROOT"
23+
envEnableGitGetter = "KHELM_ENABLE_GIT_GETTER"
2024
envTrustAnyRepo = "KHELM_TRUST_ANY_REPO"
2125
envDebug = "KHELM_DEBUG"
2226
envHelmDebug = "HELM_DEBUG"
@@ -36,11 +40,15 @@ func Execute(reader io.Reader, writer io.Writer) error {
3640
helmDebug, _ := strconv.ParseBool(os.Getenv(envHelmDebug))
3741
h := helm.NewHelm()
3842
debug = debug || helmDebug
43+
enableGitGetter := false
3944
h.Settings.Debug = debug
4045
if trustAnyRepo, ok := os.LookupEnv(envTrustAnyRepo); ok {
4146
trust, _ := strconv.ParseBool(trustAnyRepo)
4247
h.TrustAnyRepository = &trust
4348
}
49+
if gitSupportStr, ok := os.LookupEnv(envEnableGitGetter); ok {
50+
enableGitGetter, _ = strconv.ParseBool(gitSupportStr)
51+
}
4452

4553
// Run as kustomize plugin (if kustomize-specific env var provided)
4654
if kustomizeGenCfgYAML, isKustomizePlugin := os.LookupEnv(envKustomizePluginConfig); isKustomizePlugin {
@@ -50,12 +58,16 @@ func Execute(reader io.Reader, writer io.Writer) error {
5058
return err
5159
}
5260

53-
logVersionPreRun := func(_ *cobra.Command, _ []string) {
61+
preRun := func(_ *cobra.Command, _ []string) {
5462
logVersion()
63+
if enableGitGetter {
64+
addGitGetterSupport(h)
65+
}
5566
}
5667
rootCmd := &cobra.Command{
57-
PreRun: logVersionPreRun,
68+
PersistentPreRun: preRun,
5869
}
70+
rootCmd.PersistentFlags().BoolVar(&enableGitGetter, "enable-git-getter", enableGitGetter, fmt.Sprintf("enable git+https helm repository URL scheme support (%s)", envEnableGitGetter))
5971
errBuf := bytes.Buffer{}
6072

6173
if filepath.Base(os.Args[0]) == "khelmfn" {
@@ -65,8 +77,7 @@ func Execute(reader io.Reader, writer io.Writer) error {
6577
rootCmd.SetOut(writer)
6678
rootCmd.SetErr(&errBuf)
6779
rootCmd.PersistentFlags().BoolVar(&debug, "debug", debug, fmt.Sprintf("enable debug log (%s)", envDebug))
68-
rootCmd.PreRun = func(_ *cobra.Command, _ []string) {
69-
logVersion()
80+
rootCmd.PreRun = func(cmd *cobra.Command, args []string) {
7081
fmt.Printf("# Reading kpt function input from stdin (use `%s template` to run without kpt)\n", os.Args[0])
7182
}
7283
}
@@ -88,7 +99,7 @@ In addition to helm's templating capabilities khelm allows to:
8899
templateCmd := templateCommand(h, writer)
89100
templateCmd.SetOut(writer)
90101
templateCmd.SetErr(&errBuf)
91-
templateCmd.PreRun = logVersionPreRun
102+
templateCmd.PreRun = preRun
92103
rootCmd.AddCommand(templateCmd)
93104

94105
// Run command
@@ -110,3 +121,12 @@ func logVersion() {
110121
func versionInfo() string {
111122
return fmt.Sprintf("%s (helm %s)", khelmVersion, helmVersion)
112123
}
124+
125+
func addGitGetterSupport(h *helm.Helm) {
126+
h.Getters = append(h.Getters, helmgetter.Provider{
127+
Schemes: git.Schemes,
128+
New: git.New(&h.Settings, h.Repositories, func(ctx context.Context, chartDir, repoDir string) (string, error) {
129+
return h.Package(ctx, chartDir, repoDir, chartDir)
130+
}),
131+
})
132+
}

cmd/khelm/template_test.go

+14
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,16 @@ func TestTemplateCommand(t *testing.T) {
9393
[]string{filepath.Join(exampleDir, "chart-hooks"), "--no-hooks"},
9494
1, "myvalue",
9595
},
96+
{
97+
"git-dependency",
98+
[]string{filepath.Join(exampleDir, "git-dependency"), "--enable-git-getter", "--trust-any-repo"},
99+
24, "ca-sync",
100+
},
101+
{
102+
"local-chart-with-transitive-remote-and-git-dependencies",
103+
[]string{filepath.Join(exampleDir, "localrefref-with-git"), "--enable-git-getter", "--trust-any-repo"},
104+
33, "admission.certmanager.k8s.io",
105+
},
96106
} {
97107
t.Run(c.name, func(t *testing.T) {
98108
var out bytes.Buffer
@@ -128,6 +138,10 @@ func TestTemplateCommandError(t *testing.T) {
128138
"reject cluster scoped resources",
129139
[]string{"cert-manager", "--repo=https://charts.jetstack.io", "--namespaced-only"},
130140
},
141+
{
142+
"reject git urls by default",
143+
[]string{"git-dependency", "--enable-git-getter=false"},
144+
},
131145
} {
132146
t.Run(c.name, func(t *testing.T) {
133147
os.Args = append([]string{"testee", "template"}, c.args...)

e2e/cli-tests.bats

+13-4
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,9 @@ teardown() {
6060
}
6161

6262
@test "CLI should accept git url as helm repository" {
63-
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" "$IMAGE" template cert-manager \
63+
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" \
64+
-e KHELM_ENABLE_GIT_GETTER=true \
65+
"$IMAGE" template cert-manager \
6466
--repo git+https://github.com/cert-manager/cert-manager@deploy/charts?ref=v0.6.2 \
6567
--output /out/manifest.yaml \
6668
--debug
@@ -70,14 +72,18 @@ teardown() {
7072

7173
@test "CLI should cache git repository" {
7274
mkdir $OUT_DIR/cache
73-
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" -v "$OUT_DIR/cache:/helm/cache" "$IMAGE" template cert-manager \
75+
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" -v "$OUT_DIR/cache:/helm/cache" \
76+
-e KHELM_ENABLE_GIT_GETTER=true \
77+
"$IMAGE" template cert-manager \
7478
--repo git+https://github.com/cert-manager/cert-manager@deploy/charts?ref=v0.6.2 \
7579
--output /out/manifest.yaml \
7680
--debug
7781
[ -f "$OUT_DIR/manifest.yaml" ]
7882
grep -q ca-sync "$OUT_DIR/manifest.yaml"
7983
rm -f "$OUT_DIR/manifest.yaml"
80-
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" -v "$OUT_DIR/cache:/helm/cache" --network=none "$IMAGE" template cert-manager \
84+
docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" -v "$OUT_DIR/cache:/helm/cache" \
85+
-e KHELM_ENABLE_GIT_GETTER=true \
86+
--network=none "$IMAGE" template cert-manager \
8187
--repo git+https://github.com/cert-manager/cert-manager@deploy/charts?ref=v0.6.2 \
8288
--output /out/manifest.yaml \
8389
--debug
@@ -86,7 +92,10 @@ teardown() {
8692
}
8793

8894
@test "CLI should reject git repository when not in repositories.yaml and trust-any disabled" {
89-
run -1 docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" -e KHELM_TRUST_ANY_REPO=false "$IMAGE" template cert-manager \
95+
run -1 docker run --rm -u $(id -u):$(id -g) -v "$OUT_DIR:/out" \
96+
-e KHELM_ENABLE_GIT_GETTER=true \
97+
-e KHELM_TRUST_ANY_REPO=false \
98+
"$IMAGE" template cert-manager \
9099
--repo git+https://github.com/cert-manager/cert-manager@deploy/charts?ref=v0.6.2 \
91100
--output /out/manifest.yaml \
92101
--debug

pkg/getter/git/gitgetter.go

+4-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,10 @@ import (
2020
helmyaml "sigs.k8s.io/yaml"
2121
)
2222

23-
var gitCheckout = gitCheckoutImpl
23+
var (
24+
Schemes = []string{"git+https", "git+ssh"}
25+
gitCheckout = gitCheckoutImpl
26+
)
2427

2528
type HelmPackageFunc func(ctx context.Context, path, repoDir string) (string, error)
2629

pkg/helm/helm.go

+6-4
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,10 @@ import (
1010
"helm.sh/helm/v3/pkg/helmpath"
1111
)
1212

13+
func IsUntrustedRepository(err error) bool {
14+
return repositories.IsUntrustedRepository(err)
15+
}
16+
1317
// Helm maintains the helm environment state
1418
type Helm struct {
1519
TrustAnyRepository *bool
@@ -25,12 +29,10 @@ func NewHelm() *Helm {
2529
// Fallback for old helm env var
2630
settings.RepositoryConfig = filepath.Join(helmHome, "repository", "repositories.yaml")
2731
}
28-
h := &Helm{Settings: *settings}
29-
h.Getters = getters(settings, h.repositories)
30-
return h
32+
return &Helm{Settings: *settings, Getters: getter.All(settings)}
3133
}
3234

33-
func (h *Helm) repositories() (repositories.Interface, error) {
35+
func (h *Helm) Repositories() (repositories.Interface, error) {
3436
if h.repos != nil {
3537
return h.repos, nil
3638
}

pkg/helm/load.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ func (h *Helm) loadChart(ctx context.Context, cfg *config.ChartConfig) (*chart.C
2929
fileExists := err == nil
3030
if cfg.Repository == "" {
3131
if fileExists {
32-
repos, err := h.repositories()
32+
repos, err := h.Repositories()
3333
if err != nil {
3434
return nil, err
3535
}
@@ -46,7 +46,7 @@ func (h *Helm) loadChart(ctx context.Context, cfg *config.ChartConfig) (*chart.C
4646

4747
func (h *Helm) loadRemoteChart(ctx context.Context, cfg *config.ChartConfig) (*chart.Chart, error) {
4848
repoURLs := map[string]struct{}{cfg.Repository: {}}
49-
repos, err := h.repositories()
49+
repos, err := h.Repositories()
5050
if err != nil {
5151
return nil, err
5252
}

pkg/helm/package.go

+24-2
Original file line numberDiff line numberDiff line change
@@ -10,15 +10,37 @@ import (
1010
"helm.sh/helm/v3/pkg/getter"
1111
)
1212

13-
func packageHelmChart(ctx context.Context, cfg *config.ChartConfig, destDir string, repos repositories.Interface, settings cli.EnvSettings, getters getter.Providers) (string, error) {
13+
type PackageOptions struct {
14+
ChartDir string
15+
BaseDir string
16+
DestDir string
17+
}
18+
19+
// Package builds and packages a local Helm chart.
20+
// Returns the tgz file path.
21+
func (h *Helm) Package(ctx context.Context, chartDir, baseDir, destDir string) (string, error) {
22+
repos, err := h.Repositories()
23+
if err != nil {
24+
return "", err
25+
}
26+
cfg := config.ChartConfig{
27+
LoaderConfig: config.LoaderConfig{
28+
Chart: chartDir,
29+
},
30+
BaseDir: baseDir,
31+
}
32+
return packageHelmChart(ctx, &cfg, repos, h.Settings, h.Getters)
33+
}
34+
35+
func packageHelmChart(ctx context.Context, cfg *config.ChartConfig, repos repositories.Interface, settings cli.EnvSettings, getters getter.Providers) (string, error) {
1436
// TODO: add unit test (there is an e2e/cli test for this though)
1537
_, err := buildAndLoadLocalChart(ctx, cfg, repos, settings, getters)
1638
if err != nil {
1739
return "", err
1840
}
1941
// See https://github.com/helm/helm/blob/v3.10.0/cmd/helm/package.go#L104
2042
client := action.NewPackage()
21-
client.Destination = destDir
43+
client.Destination = cfg.BaseDir
2244
chartPath := absPath(cfg.Chart, cfg.BaseDir)
2345
tgzFile, err := client.Run(chartPath, map[string]interface{}{})
2446
if err != nil {

pkg/helm/providers.go

-31
This file was deleted.

pkg/helm/render_test.go

-2
Original file line numberDiff line numberDiff line change
@@ -72,8 +72,6 @@ func TestRender(t *testing.T) {
7272
"chart-hooks-test",
7373
}},
7474
{"chart-hooks-disabled", "example/chart-hooks-disabled/generator.yaml", []string{"default"}, " key: myvalue", []string{"chart-hooks-disabled-myconfig"}},
75-
{"git-dependency", "example/git-dependency/generator.yaml", []string{"cert-manager", "kube-system"}, "ca-sync", nil},
76-
{"local-chart-with-transitive-remote-and-git-dependencies", "example/localrefref-with-git/generator.yaml", []string{"kube-system", "myotherns"}, "admission.certmanager.k8s.io", nil},
7775
} {
7876
t.Run(c.name, func(t *testing.T) {
7977
for _, cached := range []string{"", "cached "} {

0 commit comments

Comments
 (0)