2023-03-03 06:43:33 -05:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2023-08-08 09:18:36 -04:00
|
|
|
Package helm provides a higher level interface to the Helm Go SDK.
|
2023-03-03 06:43:33 -05:00
|
|
|
|
|
|
|
It is used by the CLI to:
|
2023-08-08 09:18:36 -04:00
|
|
|
|
|
|
|
- load embedded charts
|
|
|
|
- install charts
|
|
|
|
- update helm releases
|
|
|
|
- get versions for installed helm releases
|
|
|
|
- create local backups before running service upgrades
|
|
|
|
|
|
|
|
The charts themselves are embedded in the CLI binary, and values are dynamically updated depending on configuration.
|
|
|
|
The charts can be found in “./charts/“.
|
|
|
|
Values should be added in the chart's "values.yaml“ file if they are static i.e. don't depend on user input,
|
|
|
|
otherwise they need to be dynamically created depending on a user's configuration.
|
|
|
|
|
|
|
|
Helm logic should not be implemented outside this package.
|
|
|
|
All values loading, parsing, installing, uninstalling, and updating of charts should be implemented here.
|
|
|
|
As such, the helm package requires to implement some CSP specific logic.
|
|
|
|
However, exported functions should be CSP agnostic and take a cloudprovider.Provider as argument.
|
|
|
|
As such, the number of exported functions should be kept minimal.
|
2023-03-03 06:43:33 -05:00
|
|
|
*/
|
|
|
|
package helm
|
2023-08-02 09:49:40 -04:00
|
|
|
|
2023-08-24 10:40:47 -04:00
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2023-11-29 08:55:10 -05:00
|
|
|
"time"
|
2023-08-24 10:40:47 -04:00
|
|
|
|
2023-12-01 06:51:51 -05:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/attestation/variant"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
|
2023-08-24 10:40:47 -04:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/config"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/constants"
|
2023-09-08 06:02:16 -04:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/file"
|
2023-08-24 10:40:47 -04:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/kms/uri"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/kubernetes/kubectl"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/semver"
|
2023-11-22 08:52:56 -05:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/state"
|
2023-12-01 06:51:51 -05:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/versions"
|
2023-08-24 10:40:47 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// AllowDestructive is a named bool to signal that destructive actions have been confirmed by the user.
|
|
|
|
AllowDestructive = true
|
|
|
|
// DenyDestructive is a named bool to signal that destructive actions have not been confirmed by the user yet.
|
|
|
|
DenyDestructive = false
|
|
|
|
)
|
|
|
|
|
|
|
|
type debugLog interface {
|
|
|
|
Debugf(format string, args ...any)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Client is a Helm client to apply charts.
|
|
|
|
type Client struct {
|
|
|
|
factory *actionFactory
|
|
|
|
cliVersion semver.Semver
|
|
|
|
log debugLog
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewClient returns a new Helm client.
|
2023-12-01 03:00:44 -05:00
|
|
|
func NewClient(kubeConfig []byte, log debugLog) (*Client, error) {
|
|
|
|
kubeClient, err := kubectl.NewFromConfig(kubeConfig)
|
2023-08-24 10:40:47 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("initializing kubectl: %w", err)
|
|
|
|
}
|
2023-12-01 03:00:44 -05:00
|
|
|
actionConfig, err := newActionConfig(kubeConfig, log)
|
2023-08-24 10:40:47 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("creating action config: %w", err)
|
|
|
|
}
|
|
|
|
lister := ReleaseVersionClient{actionConfig}
|
|
|
|
cliVersion := constants.BinaryVersion()
|
2023-09-08 17:09:02 -04:00
|
|
|
factory := newActionFactory(kubeClient, lister, actionConfig, log)
|
2023-08-24 10:40:47 -04:00
|
|
|
return &Client{factory, cliVersion, log}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Options are options for loading charts.
|
|
|
|
type Options struct {
|
|
|
|
Conformance bool
|
|
|
|
HelmWaitMode WaitMode
|
2023-12-01 06:51:51 -05:00
|
|
|
DeployCSIDriver bool
|
2023-08-24 10:40:47 -04:00
|
|
|
AllowDestructive bool
|
|
|
|
Force bool
|
2023-11-29 08:55:10 -05:00
|
|
|
ApplyTimeout time.Duration
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// PrepareApply loads the charts and returns the executor to apply them.
|
2023-09-25 10:19:43 -04:00
|
|
|
func (h Client) PrepareApply(
|
2023-12-01 06:51:51 -05:00
|
|
|
csp cloudprovider.Provider, attestationVariant variant.Variant, k8sVersion versions.ValidK8sVersion,
|
|
|
|
microserviceVersion semver.Semver, stateFile *state.State, flags Options, serviceAccURI string,
|
|
|
|
masterSecret uri.MasterSecret, openStackCfg *config.OpenStackConfig,
|
2023-09-08 06:02:16 -04:00
|
|
|
) (Applier, bool, error) {
|
2023-12-01 06:51:51 -05:00
|
|
|
releases, err := h.loadReleases(csp, attestationVariant, k8sVersion, masterSecret, stateFile, flags, serviceAccURI, openStackCfg)
|
2023-08-24 10:40:47 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, false, fmt.Errorf("loading Helm releases: %w", err)
|
|
|
|
}
|
2023-11-29 08:55:10 -05:00
|
|
|
|
2023-08-24 10:40:47 -04:00
|
|
|
h.log.Debugf("Loaded Helm releases")
|
2023-11-29 08:55:10 -05:00
|
|
|
actions, includesUpgrades, err := h.factory.GetActions(
|
2023-12-01 06:51:51 -05:00
|
|
|
releases, microserviceVersion, flags.Force, flags.AllowDestructive, flags.ApplyTimeout,
|
2023-11-29 08:55:10 -05:00
|
|
|
)
|
2023-08-24 10:40:47 -04:00
|
|
|
return &ChartApplyExecutor{actions: actions, log: h.log}, includesUpgrades, err
|
|
|
|
}
|
|
|
|
|
2023-09-25 10:19:43 -04:00
|
|
|
func (h Client) loadReleases(
|
2023-12-01 06:51:51 -05:00
|
|
|
csp cloudprovider.Provider, attestationVariant variant.Variant, k8sVersion versions.ValidK8sVersion, secret uri.MasterSecret,
|
|
|
|
stateFile *state.State, flags Options, serviceAccURI string, openStackCfg *config.OpenStackConfig,
|
2023-11-29 08:55:10 -05:00
|
|
|
) ([]release, error) {
|
2023-12-01 06:51:51 -05:00
|
|
|
helmLoader := newLoader(csp, attestationVariant, k8sVersion, stateFile, h.cliVersion)
|
2023-08-24 10:40:47 -04:00
|
|
|
h.log.Debugf("Created new Helm loader")
|
2023-12-01 06:51:51 -05:00
|
|
|
return helmLoader.loadReleases(flags.Conformance, flags.DeployCSIDriver, flags.HelmWaitMode, secret, serviceAccURI, openStackCfg)
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Applier runs the Helm actions.
|
|
|
|
type Applier interface {
|
|
|
|
Apply(ctx context.Context) error
|
2023-09-08 06:02:16 -04:00
|
|
|
SaveCharts(chartsDir string, fileHandler file.Handler) error
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// ChartApplyExecutor is a Helm action executor that applies all actions.
|
|
|
|
type ChartApplyExecutor struct {
|
|
|
|
actions []applyAction
|
|
|
|
log debugLog
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply applies the charts in order.
|
|
|
|
func (c ChartApplyExecutor) Apply(ctx context.Context) error {
|
|
|
|
for _, action := range c.actions {
|
|
|
|
c.log.Debugf("Applying %q", action.ReleaseName())
|
|
|
|
if err := action.Apply(ctx); err != nil {
|
|
|
|
return fmt.Errorf("applying %s: %w", action.ReleaseName(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-09-08 06:02:16 -04:00
|
|
|
// SaveCharts saves all Helm charts and their values to the given directory.
|
|
|
|
func (c ChartApplyExecutor) SaveCharts(chartsDir string, fileHandler file.Handler) error {
|
|
|
|
for _, action := range c.actions {
|
|
|
|
if err := action.SaveChart(chartsDir, fileHandler); err != nil {
|
|
|
|
return fmt.Errorf("saving chart %s: %w", action.ReleaseName(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-02 09:49:40 -04:00
|
|
|
// mergeMaps returns a new map that is the merger of it's inputs.
|
|
|
|
// Key collisions are resolved by taking the value of the second argument (map b).
|
|
|
|
// Taken from: https://github.com/helm/helm/blob/dbc6d8e20fe1d58d50e6ed30f09a04a77e4c68db/pkg/cli/values/options.go#L91-L108.
|
|
|
|
func mergeMaps(a, b map[string]any) map[string]any {
|
|
|
|
out := make(map[string]any, len(a))
|
|
|
|
for k, v := range a {
|
|
|
|
out[k] = v
|
|
|
|
}
|
|
|
|
for k, v := range b {
|
|
|
|
if v, ok := v.(map[string]any); ok {
|
|
|
|
if bv, ok := out[k]; ok {
|
|
|
|
if bv, ok := bv.(map[string]any); ok {
|
|
|
|
out[k] = mergeMaps(bv, v)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out[k] = v
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|