2023-08-24 10:40:47 -04:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
package helm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
2023-11-29 08:55:10 -05:00
|
|
|
"time"
|
2023-08-24 10:40:47 -04:00
|
|
|
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/compatibility"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/semver"
|
|
|
|
"helm.sh/helm/v3/pkg/action"
|
|
|
|
"helm.sh/helm/v3/pkg/chart"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ErrConfirmationMissing signals that an action requires user confirmation.
|
|
|
|
var ErrConfirmationMissing = errors.New("action requires user confirmation")
|
|
|
|
|
|
|
|
var errReleaseNotFound = errors.New("release not found")
|
|
|
|
|
|
|
|
type actionFactory struct {
|
|
|
|
versionLister releaseVersionLister
|
|
|
|
cfg *action.Configuration
|
|
|
|
kubeClient crdClient
|
|
|
|
log debugLog
|
|
|
|
}
|
|
|
|
|
|
|
|
type crdClient interface {
|
|
|
|
ApplyCRD(ctx context.Context, rawCRD []byte) error
|
|
|
|
}
|
|
|
|
|
|
|
|
// newActionFactory creates a new action factory for managing helm releases.
|
2023-09-08 17:09:02 -04:00
|
|
|
func newActionFactory(kubeClient crdClient, lister releaseVersionLister, actionConfig *action.Configuration, log debugLog) *actionFactory {
|
2023-08-24 10:40:47 -04:00
|
|
|
return &actionFactory{
|
|
|
|
versionLister: lister,
|
|
|
|
cfg: actionConfig,
|
|
|
|
kubeClient: kubeClient,
|
|
|
|
log: log,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetActions returns a list of actions to apply the given releases.
|
2023-11-29 08:55:10 -05:00
|
|
|
func (a actionFactory) GetActions(
|
|
|
|
releases []release, configTargetVersion semver.Semver, force, allowDestructive bool, timeout time.Duration,
|
|
|
|
) (actions []applyAction, includesUpgrade bool, err error) {
|
2023-08-24 10:40:47 -04:00
|
|
|
upgradeErrs := []error{}
|
|
|
|
for _, release := range releases {
|
2023-11-29 08:55:10 -05:00
|
|
|
err := a.appendNewAction(release, configTargetVersion, force, allowDestructive, timeout, &actions)
|
2023-08-24 10:40:47 -04:00
|
|
|
var invalidUpgrade *compatibility.InvalidUpgradeError
|
|
|
|
if errors.As(err, &invalidUpgrade) {
|
|
|
|
upgradeErrs = append(upgradeErrs, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
2023-11-29 08:55:10 -05:00
|
|
|
return actions, includesUpgrade, fmt.Errorf("creating action for %s: %w", release.releaseName, err)
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, action := range actions {
|
|
|
|
if _, ok := action.(*upgradeAction); ok {
|
|
|
|
includesUpgrade = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return actions, includesUpgrade, errors.Join(upgradeErrs...)
|
|
|
|
}
|
|
|
|
|
2023-11-29 08:55:10 -05:00
|
|
|
func (a actionFactory) appendNewAction(
|
|
|
|
release release, configTargetVersion semver.Semver, force, allowDestructive bool, timeout time.Duration, actions *[]applyAction,
|
|
|
|
) error {
|
|
|
|
newVersion, err := semver.New(release.chart.Metadata.Version)
|
2023-08-24 10:40:47 -04:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("parsing chart version: %w", err)
|
|
|
|
}
|
2023-09-08 17:09:02 -04:00
|
|
|
cliSupportsConfigVersion := configTargetVersion.Compare(newVersion) != 0
|
|
|
|
|
2023-11-29 08:55:10 -05:00
|
|
|
currentVersion, err := a.versionLister.currentVersion(release.releaseName)
|
2023-08-24 10:40:47 -04:00
|
|
|
if errors.Is(err, errReleaseNotFound) {
|
2023-09-08 17:09:02 -04:00
|
|
|
// Don't install a new release if the user's config specifies a different version than the CLI offers.
|
2023-11-29 08:55:10 -05:00
|
|
|
if !force && isCLIVersionedRelease(release.releaseName) && cliSupportsConfigVersion {
|
2023-09-08 17:09:02 -04:00
|
|
|
return compatibility.NewInvalidUpgradeError(
|
|
|
|
currentVersion.String(),
|
|
|
|
configTargetVersion.String(),
|
|
|
|
fmt.Errorf("this CLI only supports installing microservice version %s", newVersion),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-11-29 08:55:10 -05:00
|
|
|
a.log.Debugf("release %s not found, adding to new releases...", release.releaseName)
|
|
|
|
*actions = append(*actions, a.newInstall(release, timeout))
|
2023-08-24 10:40:47 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
2023-11-29 08:55:10 -05:00
|
|
|
return fmt.Errorf("getting version for %s: %w", release.releaseName, err)
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
2023-11-29 08:55:10 -05:00
|
|
|
a.log.Debugf("Current %s version: %s", release.releaseName, currentVersion)
|
|
|
|
a.log.Debugf("New %s version: %s", release.releaseName, newVersion)
|
2023-08-24 10:40:47 -04:00
|
|
|
|
|
|
|
if !force {
|
2023-09-08 17:09:02 -04:00
|
|
|
// For charts we package ourselves, the version is equal to the CLI version (charts are embedded in the binary).
|
|
|
|
// We need to make sure this matches with the version in a user's config, if an upgrade should be applied.
|
2023-11-29 08:55:10 -05:00
|
|
|
if isCLIVersionedRelease(release.releaseName) {
|
2023-09-08 17:09:02 -04:00
|
|
|
// If target version is not a valid upgrade, don't upgrade any charts.
|
|
|
|
if err := configTargetVersion.IsUpgradeTo(currentVersion); err != nil {
|
2023-11-29 08:55:10 -05:00
|
|
|
return fmt.Errorf("invalid upgrade for %s: %w", release.releaseName, err)
|
2023-09-08 17:09:02 -04:00
|
|
|
}
|
|
|
|
// Target version is newer than current version, so we should perform an upgrade.
|
|
|
|
// Now make sure the target version is equal to the the CLI version.
|
|
|
|
if cliSupportsConfigVersion {
|
|
|
|
return compatibility.NewInvalidUpgradeError(
|
|
|
|
currentVersion.String(),
|
|
|
|
configTargetVersion.String(),
|
|
|
|
fmt.Errorf("this CLI only supports upgrading to microservice version %s", newVersion),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// This may break for external chart dependencies if we decide to upgrade more than one minor version at a time.
|
|
|
|
if err := newVersion.IsUpgradeTo(currentVersion); err != nil {
|
2023-11-24 06:28:37 -05:00
|
|
|
// TODO(3u13r): Remove when Constellation v2.14 is released.
|
|
|
|
// We need to ignore that we jump from Cilium v1.12 to v1.15-pre. We have verified that this works.
|
2023-11-29 08:55:10 -05:00
|
|
|
if !(errors.Is(err, compatibility.ErrMinorDrift) && release.releaseName == "cilium") {
|
|
|
|
return fmt.Errorf("invalid upgrade for %s: %w", release.releaseName, err)
|
2023-11-24 06:28:37 -05:00
|
|
|
}
|
2023-09-08 17:09:02 -04:00
|
|
|
}
|
2023-08-24 10:40:47 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !allowDestructive &&
|
2023-11-29 08:55:10 -05:00
|
|
|
release.releaseName == certManagerInfo.releaseName {
|
2023-08-24 10:40:47 -04:00
|
|
|
return ErrConfirmationMissing
|
|
|
|
}
|
2023-11-29 08:55:10 -05:00
|
|
|
a.log.Debugf("Upgrading %s from %s to %s", release.releaseName, currentVersion, newVersion)
|
|
|
|
*actions = append(*actions, a.newUpgrade(release, timeout))
|
2023-08-24 10:40:47 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-29 08:55:10 -05:00
|
|
|
func (a actionFactory) newInstall(release release, timeout time.Duration) *installAction {
|
|
|
|
action := &installAction{helmAction: newHelmInstallAction(a.cfg, release, timeout), release: release, log: a.log}
|
2023-08-24 10:40:47 -04:00
|
|
|
return action
|
|
|
|
}
|
|
|
|
|
2023-11-29 08:55:10 -05:00
|
|
|
func (a actionFactory) newUpgrade(release release, timeout time.Duration) *upgradeAction {
|
|
|
|
action := &upgradeAction{helmAction: newHelmUpgradeAction(a.cfg, timeout), release: release, log: a.log}
|
|
|
|
if release.releaseName == constellationOperatorsInfo.releaseName {
|
2023-08-24 10:40:47 -04:00
|
|
|
action.preUpgrade = func(ctx context.Context) error {
|
2023-11-29 08:55:10 -05:00
|
|
|
if err := a.updateCRDs(ctx, release.chart); err != nil {
|
2023-08-24 10:40:47 -04:00
|
|
|
return fmt.Errorf("updating operator CRDs: %w", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return action
|
|
|
|
}
|
|
|
|
|
|
|
|
// updateCRDs walks through the dependencies of the given chart and applies
|
|
|
|
// the files in the dependencie's 'crds' folder.
|
|
|
|
// This function is NOT recursive!
|
|
|
|
func (a actionFactory) updateCRDs(ctx context.Context, chart *chart.Chart) error {
|
|
|
|
for _, dep := range chart.Dependencies() {
|
|
|
|
for _, crdFile := range dep.Files {
|
|
|
|
if strings.HasPrefix(crdFile.Name, "crds/") {
|
|
|
|
a.log.Debugf("Updating crd: %s", crdFile.Name)
|
|
|
|
err := a.kubeClient.ApplyCRD(ctx, crdFile.Data)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// isCLIVersionedRelease checks if the given release is versioned by the CLI,
|
|
|
|
// meaning that the version of the Helm release is equal to the version of the CLI that installed it.
|
|
|
|
func isCLIVersionedRelease(releaseName string) bool {
|
|
|
|
return releaseName == constellationOperatorsInfo.releaseName ||
|
|
|
|
releaseName == constellationServicesInfo.releaseName ||
|
|
|
|
releaseName == csiInfo.releaseName
|
|
|
|
}
|
|
|
|
|
|
|
|
// releaseVersionLister can list the versions of a helm release.
|
|
|
|
type releaseVersionLister interface {
|
|
|
|
currentVersion(release string) (semver.Semver, error)
|
|
|
|
}
|