mirror of
https://github.com/edgelesssys/constellation.git
synced 2024-12-27 00:19:36 -05:00
a87b7894db
* add current chart add current helm chart * disable service controller for aws ccm * add new iam roles * doc AWS internet LB + add to LB test * pass clusterName to helm for AWS LB * fix update-aws-lb chart to also include .helmignore * move chart outside services * working state * add subnet tags for AWS subnet discovery * fix .helmignore load rule with file in subdirectory * upgrade iam profile * revert new loader impl since cilium is not correctly loaded * install chart if not already present during `upgrade apply` * cleanup PR + fix build + add todos cleanup PR + add todos * shared helm pkg for cli install and bootstrapper * add link to eks docs * refactor iamMigrationCmd * delete unused helm.symwallk * move iammigrate to upgrade pkg * fixup! delete unused helm.symwallk * add to upgradecheck * remove nodeSelector from go code (Otto) * update iam docs and sort permission + remove duplicate roles * fix bug in `upgrade check` * better upgrade check output when svc version upgrade not possible * pr feedback * remove force flag in upgrade_test * use upgrader.GetUpgradeID instead of extra type * remove todos + fix check * update doc lb (leo) * remove bootstrapper helm package * Update cli/internal/cmd/upgradecheck.go Co-authored-by: Daniel Weiße <66256922+daniel-weisse@users.noreply.github.com> * final nits * add docs for e2e upgrade test setup * Apply suggestions from code review Co-authored-by: Daniel Weiße <66256922+daniel-weisse@users.noreply.github.com> * Update cli/internal/helm/loader.go Co-authored-by: Daniel Weiße <66256922+daniel-weisse@users.noreply.github.com> * Update cli/internal/cmd/tfmigrationclient.go Co-authored-by: Daniel Weiße <66256922+daniel-weisse@users.noreply.github.com> * fix daniel review * link to the iam permissions instead of manually updating them (agreed with leo) * disable iam upgrade in upgrade apply --------- Co-authored-by: Daniel Weiße <66256922+daniel-weisse@users.noreply.github.com> Co-authored-by: Malte Poll
160 lines
4.3 KiB
Go
160 lines
4.3 KiB
Go
/*
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
package helm
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/constants"
|
|
"github.com/edgelesssys/constellation/v2/internal/logger"
|
|
"github.com/edgelesssys/constellation/v2/internal/retry"
|
|
"go.uber.org/zap"
|
|
"helm.sh/helm/v3/pkg/action"
|
|
"helm.sh/helm/v3/pkg/chart"
|
|
"helm.sh/helm/v3/pkg/chart/loader"
|
|
"helm.sh/helm/v3/pkg/cli"
|
|
"k8s.io/apimachinery/pkg/util/wait"
|
|
)
|
|
|
|
const (
|
|
// timeout is the maximum time given to the helm Installer.
|
|
timeout = 10 * time.Minute
|
|
// maximumRetryAttempts is the maximum number of attempts to retry a helm install.
|
|
maximumRetryAttempts = 3
|
|
)
|
|
|
|
// Installer is a wrapper for a helm install action.
|
|
type Installer struct {
|
|
*action.Install
|
|
log *logger.Logger
|
|
}
|
|
|
|
// NewInstaller creates a new Installer with the given logger.
|
|
func NewInstaller(log *logger.Logger, kubeconfig string) (*Installer, error) {
|
|
settings := cli.New()
|
|
settings.KubeConfig = kubeconfig
|
|
|
|
actionConfig := &action.Configuration{}
|
|
if err := actionConfig.Init(settings.RESTClientGetter(), constants.HelmNamespace,
|
|
"secret", log.Infof); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
action := action.NewInstall(actionConfig)
|
|
action.Namespace = constants.HelmNamespace
|
|
action.Timeout = timeout
|
|
|
|
return &Installer{
|
|
action,
|
|
log,
|
|
}, nil
|
|
}
|
|
|
|
// InstallChart is the generic install function for helm charts.
|
|
// When timeout is nil, the default timeout is used.
|
|
func (h *Installer) InstallChart(ctx context.Context, release Release) error {
|
|
return h.InstallChartWithValues(ctx, release, nil)
|
|
}
|
|
|
|
// InstallChartWithValues is the generic install function for helm charts with custom values.
|
|
func (h *Installer) InstallChartWithValues(ctx context.Context, release Release, extraValues map[string]any) error {
|
|
mergedVals := MergeMaps(release.Values, extraValues)
|
|
h.ReleaseName = release.ReleaseName
|
|
if err := h.SetWaitMode(release.WaitMode); err != nil {
|
|
return err
|
|
}
|
|
return h.install(ctx, release.Chart, mergedVals)
|
|
}
|
|
|
|
// install tries to install the given chart and aborts after ~5 tries.
|
|
// The function will wait 30 seconds before retrying a failed installation attempt.
|
|
// After 3 tries, the retrier will be canceled and the function returns with an error.
|
|
func (h *Installer) install(ctx context.Context, chartRaw []byte, values map[string]any) error {
|
|
var retries int
|
|
retriable := func(err error) bool {
|
|
// abort after maximumRetryAttempts tries.
|
|
if retries >= maximumRetryAttempts {
|
|
return false
|
|
}
|
|
retries++
|
|
// only retry if atomic is set
|
|
// otherwise helm doesn't uninstall
|
|
// the release on failure
|
|
if !h.Atomic {
|
|
return false
|
|
}
|
|
// check if error is retriable
|
|
return wait.Interrupted(err) ||
|
|
strings.Contains(err.Error(), "connection refused")
|
|
}
|
|
|
|
reader := bytes.NewReader(chartRaw)
|
|
chart, err := loader.LoadArchive(reader)
|
|
if err != nil {
|
|
return fmt.Errorf("helm load archive: %w", err)
|
|
}
|
|
|
|
doer := installDoer{
|
|
h,
|
|
chart,
|
|
values,
|
|
h.log,
|
|
}
|
|
retrier := retry.NewIntervalRetrier(doer, 30*time.Second, retriable)
|
|
|
|
retryLoopStartTime := time.Now()
|
|
if err := retrier.Do(ctx); err != nil {
|
|
return fmt.Errorf("helm install: %w", err)
|
|
}
|
|
retryLoopFinishDuration := time.Since(retryLoopStartTime)
|
|
h.log.With(zap.String("chart", chart.Name()), zap.Duration("duration", retryLoopFinishDuration)).Infof("Helm chart installation finished")
|
|
|
|
return nil
|
|
}
|
|
|
|
// SetWaitMode sets the wait mode of the installer.
|
|
func (h *Installer) SetWaitMode(waitMode WaitMode) error {
|
|
switch waitMode {
|
|
case WaitModeNone:
|
|
h.Wait = false
|
|
h.Atomic = false
|
|
case WaitModeWait:
|
|
h.Wait = true
|
|
h.Atomic = false
|
|
case WaitModeAtomic:
|
|
h.Wait = true
|
|
h.Atomic = true
|
|
default:
|
|
return fmt.Errorf("unknown wait mode %q", waitMode)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// installDoer is a help struct to enable retrying helm's install action.
|
|
type installDoer struct {
|
|
Installer *Installer
|
|
chart *chart.Chart
|
|
values map[string]any
|
|
log *logger.Logger
|
|
}
|
|
|
|
// Do logs which chart is installed and tries to install it.
|
|
func (i installDoer) Do(ctx context.Context) error {
|
|
i.log.With(zap.String("chart", i.chart.Name())).Infof("Trying to install Helm chart")
|
|
|
|
if _, err := i.Installer.RunWithContext(ctx, i.chart, i.values); err != nil {
|
|
i.log.With(zap.Error(err), zap.String("chart", i.chart.Name())).Errorf("Helm chart installation failed")
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|