ARO-RP/cmd/aro/update_ocp_versions.go

252 строки
7.2 KiB
Go

package main
// Copyright (c) Microsoft Corporation.
// Licensed under the Apache License 2.0.
import (
"context"
"encoding/base64"
"encoding/json"
"fmt"
"os"
"github.com/sirupsen/logrus"
"github.com/Azure/ARO-RP/pkg/api"
"github.com/Azure/ARO-RP/pkg/database"
"github.com/Azure/ARO-RP/pkg/env"
"github.com/Azure/ARO-RP/pkg/metrics/statsd"
"github.com/Azure/ARO-RP/pkg/util/version"
)
// Corresponds to configuration.openShiftVersions in RP-Config
type OpenShiftVersions struct {
DefaultStream map[string]string
InstallStreams map[string]string
}
func getEnvironmentData(envKey string, envData any) error {
var err error
jsonData := []byte(os.Getenv(envKey))
// For Azure DevOps pipelines, the JSON data is Base64-encoded
// since it's embedded in JSON-formatted build artifacts. But
// let's not force that on local development mode.
if !env.IsLocalDevelopmentMode() {
jsonData, err = base64.StdEncoding.DecodeString(string(jsonData))
if err != nil {
return fmt.Errorf("%s: Failed to decode base64: %w", envKey, err)
}
}
if err = json.Unmarshal(jsonData, envData); err != nil {
return fmt.Errorf("%s: Failed to parse JSON: %w", envKey, err)
}
return nil
}
func getOpenShiftVersions() (*OpenShiftVersions, error) {
const envKey = envOpenShiftVersions
var openShiftVersions OpenShiftVersions
if err := getEnvironmentData(envKey, &openShiftVersions); err != nil {
return nil, err
}
// The DefaultStream map must have exactly one entry.
numDefaultStreams := len(openShiftVersions.DefaultStream)
if numDefaultStreams != 1 {
return nil, fmt.Errorf("%s: DefaultStream must have exactly 1 entry, found %d", envKey, numDefaultStreams)
}
return &openShiftVersions, nil
}
func getInstallerImageDigests() (map[string]string, error) {
// INSTALLER_IMAGE_DIGESTS is the mapping of a minor version to
// the aro-installer wrapper digest. This allows us to utilize
// Azure Safe Deployment Practices (SDP) instead of pushing the
// version tag and deploying to all regions at once.
const envKey = envInstallerImageDigests
var installerImageDigests map[string]string
if err := getEnvironmentData(envKey, &installerImageDigests); err != nil {
return nil, err
}
return installerImageDigests, nil
}
func appendOpenShiftVersions(ocpVersions []api.OpenShiftVersion, installStreams map[string]string, installerImageName string, installerImageDigests map[string]string, isDefault bool) ([]api.OpenShiftVersion, error) {
for fullVersion, openShiftPullspec := range installStreams {
openShiftVersion, err := version.ParseVersion(fullVersion)
if err != nil {
return nil, err
}
fullVersion = openShiftVersion.String() // trimmed of whitespace
minorVersion := openShiftVersion.MinorVersion()
installerDigest, ok := installerImageDigests[minorVersion]
if !ok {
return nil, fmt.Errorf("no installer digest for version %s", minorVersion)
}
installerPullspec := fmt.Sprintf("%s:%s@%s", installerImageName, minorVersion, installerDigest)
ocpVersions = append(ocpVersions, api.OpenShiftVersion{
Properties: api.OpenShiftVersionProperties{
Version: fullVersion,
OpenShiftPullspec: openShiftPullspec,
InstallerPullspec: installerPullspec,
Enabled: true,
Default: isDefault,
},
})
}
return ocpVersions, nil
}
func getLatestOCPVersions(ctx context.Context, log *logrus.Entry) ([]api.OpenShiftVersion, error) {
env, err := env.NewCoreForCI(ctx, log)
if err != nil {
return nil, err
}
dstAcr := os.Getenv("DST_ACR_NAME")
acrDomainSuffix := "." + env.Environment().ContainerRegistryDNSSuffix
installerImageName := dstAcr + acrDomainSuffix + "/aro-installer"
openShiftVersions, err := getOpenShiftVersions()
if err != nil {
return nil, err
}
installerImageDigests, err := getInstallerImageDigests()
if err != nil {
return nil, err
}
ocpVersions := make([]api.OpenShiftVersion, 0, len(openShiftVersions.DefaultStream)+len(openShiftVersions.InstallStreams))
ocpVersions, err = appendOpenShiftVersions(ocpVersions, openShiftVersions.DefaultStream, installerImageName, installerImageDigests, true)
if err != nil {
return nil, err
}
ocpVersions, err = appendOpenShiftVersions(ocpVersions, openShiftVersions.InstallStreams, installerImageName, installerImageDigests, false)
if err != nil {
return nil, err
}
return ocpVersions, nil
}
func getVersionsDatabase(ctx context.Context, log *logrus.Entry) (database.OpenShiftVersions, error) {
_env, err := env.NewCore(ctx, log, env.COMPONENT_UPDATE_OCP_VERSIONS)
if err != nil {
return nil, err
}
if err = env.ValidateVars("DST_ACR_NAME"); err != nil {
return nil, err
}
if !_env.IsLocalDevelopmentMode() {
if err = env.ValidateVars("MDM_ACCOUNT", "MDM_NAMESPACE"); err != nil {
return nil, err
}
}
m := statsd.New(ctx, log.WithField("component", "update-ocp-versions"), _env, os.Getenv("MDM_ACCOUNT"), os.Getenv("MDM_NAMESPACE"), os.Getenv("MDM_STATSD_SOCKET"))
dbc, err := database.NewDatabaseClientFromEnv(ctx, _env, log, m, nil)
if err != nil {
return nil, fmt.Errorf("failed creating database client: %w", err)
}
dbName, err := env.DBName(_env)
if err != nil {
return nil, err
}
dbOpenShiftVersions, err := database.NewOpenShiftVersions(ctx, dbc, dbName)
if err != nil {
return nil, err
}
return dbOpenShiftVersions, nil
}
func updateOpenShiftVersions(ctx context.Context, dbOpenShiftVersions database.OpenShiftVersions, log *logrus.Entry) error {
existingVersions, err := dbOpenShiftVersions.ListAll(ctx)
if err != nil {
return err
}
latestVersions, err := getLatestOCPVersions(ctx, log)
if err != nil {
return err
}
newVersions := make(map[string]api.OpenShiftVersion)
for _, doc := range latestVersions {
newVersions[doc.Properties.Version] = doc
}
for _, doc := range existingVersions.OpenShiftVersionDocuments {
existing, found := newVersions[doc.OpenShiftVersion.Properties.Version]
if found {
log.Printf("Found Version %q, patching", existing.Properties.Version)
_, err := dbOpenShiftVersions.Patch(ctx, doc.ID, func(inFlightDoc *api.OpenShiftVersionDocument) error {
inFlightDoc.OpenShiftVersion = &existing
return nil
})
if err != nil {
return err
}
log.Printf("Version %q found", existing.Properties.Version)
delete(newVersions, existing.Properties.Version)
continue
}
log.Printf("Version %q not found, deleting", doc.OpenShiftVersion.Properties.Version)
// Delete via changefeed
_, err := dbOpenShiftVersions.Patch(ctx, doc.ID,
func(d *api.OpenShiftVersionDocument) error {
d.OpenShiftVersion.Deleting = true
d.TTL = 60
return nil
})
if err != nil {
return err
}
}
for _, doc := range newVersions {
log.Printf("Version %q not found in database, creating", doc.Properties.Version)
newDoc := api.OpenShiftVersionDocument{
ID: dbOpenShiftVersions.NewUUID(),
OpenShiftVersion: &doc,
}
_, err := dbOpenShiftVersions.Create(ctx, &newDoc)
if err != nil {
return err
}
}
return nil
}
func updateOCPVersions(ctx context.Context, log *logrus.Entry) error {
dbOpenShiftVersions, err := getVersionsDatabase(ctx, log)
if err != nil {
return err
}
err = updateOpenShiftVersions(ctx, dbOpenShiftVersions, log)
if err != nil {
return err
}
return nil
}