|
|
|
/*
|
|
|
|
Copyright 2023 The Flux authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/manifoldco/promptui"
|
|
|
|
apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
|
|
|
|
|
|
kustomizev1 "github.com/fluxcd/kustomize-controller/api/v1"
|
|
|
|
sourcev1 "github.com/fluxcd/source-controller/api/v1"
|
|
|
|
|
|
|
|
"github.com/fluxcd/flux2/v2/pkg/manifestgen"
|
|
|
|
)
|
|
|
|
|
|
|
|
// bootstrapLabels are labels put on a resource by kustomize-controller. These labels on the CRD indicates
|
|
|
|
// that flux has been bootstrapped.
|
|
|
|
var bootstrapLabels = []string{
|
|
|
|
fmt.Sprintf("%s/name", kustomizev1.GroupVersion.Group),
|
|
|
|
fmt.Sprintf("%s/namespace", kustomizev1.GroupVersion.Group),
|
|
|
|
}
|
|
|
|
|
|
|
|
// fluxClusterInfo contains information about an existing flux installation on a cluster.
|
|
|
|
type fluxClusterInfo struct {
|
|
|
|
// bootstrapped indicates that Flux was installed using the `flux bootstrap` command.
|
|
|
|
bootstrapped bool
|
|
|
|
// managedBy is the name of the tool being used to manage the installation of Flux.
|
|
|
|
managedBy string
|
|
|
|
// partOf indicates which distribution the instance is a part of.
|
|
|
|
partOf string
|
|
|
|
// version is the Flux version number in semver format.
|
|
|
|
version string
|
|
|
|
}
|
|
|
|
|
|
|
|
// getFluxClusterInfo returns information on the Flux installation running on the cluster.
|
|
|
|
// If an error occurred, the returned error will be non-nil.
|
|
|
|
//
|
|
|
|
// This function retrieves the GitRepository CRD from the cluster and checks it
|
|
|
|
// for a set of labels used to determine the Flux version and how Flux was installed.
|
|
|
|
// It returns the NotFound error from the underlying library if it was unable to find
|
|
|
|
// the GitRepository CRD and this can be used to check if Flux is installed.
|
|
|
|
func getFluxClusterInfo(ctx context.Context, c client.Client) (fluxClusterInfo, error) {
|
|
|
|
var info fluxClusterInfo
|
|
|
|
crdMetadata := &metav1.PartialObjectMetadata{
|
|
|
|
TypeMeta: metav1.TypeMeta{
|
|
|
|
APIVersion: apiextensionsv1.SchemeGroupVersion.String(),
|
|
|
|
Kind: "CustomResourceDefinition",
|
|
|
|
},
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
Name: fmt.Sprintf("gitrepositories.%s", sourcev1.GroupVersion.Group),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if err := c.Get(ctx, client.ObjectKeyFromObject(crdMetadata), crdMetadata); err != nil {
|
|
|
|
return info, err
|
|
|
|
}
|
|
|
|
|
|
|
|
info.version = crdMetadata.Labels[manifestgen.VersionLabelKey]
|
|
|
|
|
|
|
|
var present bool
|
|
|
|
for _, l := range bootstrapLabels {
|
|
|
|
_, present = crdMetadata.Labels[l]
|
|
|
|
}
|
|
|
|
if present {
|
|
|
|
info.bootstrapped = true
|
|
|
|
}
|
|
|
|
|
|
|
|
// the `app.kubernetes.io/managed-by` label is not set by flux but might be set by other
|
|
|
|
// tools used to install Flux e.g Helm.
|
|
|
|
if manager, ok := crdMetadata.Labels["app.kubernetes.io/managed-by"]; ok {
|
|
|
|
info.managedBy = manager
|
|
|
|
}
|
|
|
|
|
|
|
|
if partOf, ok := crdMetadata.Labels[manifestgen.PartOfLabelKey]; ok {
|
|
|
|
info.partOf = partOf
|
|
|
|
}
|
|
|
|
return info, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// confirmFluxInstallOverride displays a prompt to the user so that they can confirm before overriding
|
|
|
|
// a Flux installation. It returns nil if the installation should continue,
|
|
|
|
// promptui.ErrAbort if the user doesn't confirm, or an error encountered.
|
|
|
|
func confirmFluxInstallOverride(info fluxClusterInfo) error {
|
|
|
|
// no need to display prompt if installation is managed by Flux
|
|
|
|
if installManagedByFlux(info.managedBy) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
display := fmt.Sprintf("Flux %s has been installed on this cluster with %s!", info.version, info.managedBy)
|
|
|
|
fmt.Fprintln(rootCmd.ErrOrStderr(), display)
|
|
|
|
prompt := promptui.Prompt{
|
|
|
|
Label: fmt.Sprintf("Are you sure you want to override the %s installation? Y/N", info.managedBy),
|
|
|
|
IsConfirm: true,
|
|
|
|
}
|
|
|
|
_, err := prompt.Run()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (info fluxClusterInfo) distribution() string {
|
|
|
|
distribution := info.version
|
|
|
|
if info.partOf != "" {
|
|
|
|
distribution = fmt.Sprintf("%s-%s", info.partOf, info.version)
|
|
|
|
}
|
|
|
|
return distribution
|
|
|
|
}
|
|
|
|
|
|
|
|
func installManagedByFlux(manager string) bool {
|
|
|
|
return manager == "" || manager == "flux"
|
|
|
|
}
|