mirror of https://github.com/fluxcd/flux2.git
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			373 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
			
		
		
	
	
			373 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
| /*
 | |
| Copyright 2022 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 uninstall
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 
 | |
| 	appsv1 "k8s.io/api/apps/v1"
 | |
| 	corev1 "k8s.io/api/core/v1"
 | |
| 	networkingv1 "k8s.io/api/networking/v1"
 | |
| 	rbacv1 "k8s.io/api/rbac/v1"
 | |
| 	apiextensionsv1 "k8s.io/apiextensions-apiserver/pkg/apis/apiextensions/v1"
 | |
| 	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
 | |
| 	"k8s.io/apimachinery/pkg/util/errors"
 | |
| 	"sigs.k8s.io/controller-runtime/pkg/client"
 | |
| 
 | |
| 	"github.com/fluxcd/flux2/pkg/log"
 | |
| 	"github.com/fluxcd/flux2/pkg/manifestgen"
 | |
| 	helmv2 "github.com/fluxcd/helm-controller/api/v2beta1"
 | |
| 	autov1 "github.com/fluxcd/image-automation-controller/api/v1beta1"
 | |
| 	imagev1 "github.com/fluxcd/image-reflector-controller/api/v1beta2"
 | |
| 	kustomizev1 "github.com/fluxcd/kustomize-controller/api/v1beta2"
 | |
| 	notificationv1 "github.com/fluxcd/notification-controller/api/v1beta2"
 | |
| 	sourcev1 "github.com/fluxcd/source-controller/api/v1beta2"
 | |
| )
 | |
| 
 | |
| // Components removes all Kubernetes components that are part of Flux excluding the CRDs and namespace.
 | |
| func Components(ctx context.Context, logger log.Logger, kubeClient client.Client, namespace string, dryRun bool) error {
 | |
| 	var aggregateErr []error
 | |
| 	opts, dryRunStr := getDeleteOptions(dryRun)
 | |
| 	selector := client.MatchingLabels{manifestgen.PartOfLabelKey: manifestgen.PartOfLabelValue}
 | |
| 	{
 | |
| 		var list appsv1.DeploymentList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace(namespace), selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("Deployment/%s/%s deletion failed: %s", r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("Deployment/%s/%s deleted %s", r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list corev1.ServiceList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace(namespace), selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("Service/%s/%s deletion failed: %s", r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("Service/%s/%s deleted %s", r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list networkingv1.NetworkPolicyList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace(namespace), selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("NetworkPolicy/%s/%s deletion failed: %s", r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("NetworkPolicy/%s/%s deleted %s", r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list corev1.ServiceAccountList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace(namespace), selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("ServiceAccount/%s/%s deletion failed: %s", r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("ServiceAccount/%s/%s deleted %s", r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list rbacv1.ClusterRoleList
 | |
| 		if err := kubeClient.List(ctx, &list, selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("ClusterRole/%s deletion failed: %s", r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("ClusterRole/%s deleted %s", r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list rbacv1.ClusterRoleBindingList
 | |
| 		if err := kubeClient.List(ctx, &list, selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("ClusterRoleBinding/%s deletion failed: %s", r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("ClusterRoleBinding/%s deleted %s", r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return errors.Reduce(errors.Flatten(errors.NewAggregate(aggregateErr)))
 | |
| }
 | |
| 
 | |
| // Finalizers removes all finalizes on Kubernetes components that have been added by a Flux controller.
 | |
| func Finalizers(ctx context.Context, logger log.Logger, kubeClient client.Client, dryRun bool) error {
 | |
| 	var aggregateErr []error
 | |
| 	opts, dryRunStr := getUpdateOptions(dryRun)
 | |
| 	{
 | |
| 		var list sourcev1.GitRepositoryList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list sourcev1.OCIRepositoryList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list sourcev1.HelmRepositoryList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list sourcev1.HelmChartList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list sourcev1.BucketList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list kustomizev1.KustomizationList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list helmv2.HelmReleaseList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list notificationv1.AlertList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list notificationv1.ProviderList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list notificationv1.ReceiverList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list imagev1.ImagePolicyList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list imagev1.ImageRepositoryList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	{
 | |
| 		var list autov1.ImageUpdateAutomationList
 | |
| 		if err := kubeClient.List(ctx, &list, client.InNamespace("")); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				r.Finalizers = []string{}
 | |
| 				if err := kubeClient.Update(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("%s/%s/%s removing finalizers failed: %s", r.Kind, r.Namespace, r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("%s/%s/%s finalizers deleted %s", r.Kind, r.Namespace, r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return errors.Reduce(errors.Flatten(errors.NewAggregate(aggregateErr)))
 | |
| }
 | |
| 
 | |
| // CustomResourceDefinitions removes all Kubernetes CRDs that are a part of Flux.
 | |
| func CustomResourceDefinitions(ctx context.Context, logger log.Logger, kubeClient client.Client, dryRun bool) error {
 | |
| 	var aggregateErr []error
 | |
| 	opts, dryRunStr := getDeleteOptions(dryRun)
 | |
| 	selector := client.MatchingLabels{manifestgen.PartOfLabelKey: manifestgen.PartOfLabelValue}
 | |
| 	{
 | |
| 		var list apiextensionsv1.CustomResourceDefinitionList
 | |
| 		if err := kubeClient.List(ctx, &list, selector); err == nil {
 | |
| 			for _, r := range list.Items {
 | |
| 				if err := kubeClient.Delete(ctx, &r, opts); err != nil {
 | |
| 					logger.Failuref("CustomResourceDefinition/%s deletion failed: %s", r.Name, err.Error())
 | |
| 					aggregateErr = append(aggregateErr, err)
 | |
| 				} else {
 | |
| 					logger.Successf("CustomResourceDefinition/%s deleted %s", r.Name, dryRunStr)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return errors.Reduce(errors.Flatten(errors.NewAggregate(aggregateErr)))
 | |
| }
 | |
| 
 | |
| // Namespace removes the namespace Flux is installed in.
 | |
| func Namespace(ctx context.Context, logger log.Logger, kubeClient client.Client, namespace string, dryRun bool) error {
 | |
| 	var aggregateErr []error
 | |
| 	opts, dryRunStr := getDeleteOptions(dryRun)
 | |
| 	ns := corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: namespace}}
 | |
| 	if err := kubeClient.Delete(ctx, &ns, opts); err != nil {
 | |
| 		logger.Failuref("Namespace/%s deletion failed: %s", namespace, err.Error())
 | |
| 		aggregateErr = append(aggregateErr, err)
 | |
| 	} else {
 | |
| 		logger.Successf("Namespace/%s deleted %s", namespace, dryRunStr)
 | |
| 	}
 | |
| 	return errors.Reduce(errors.Flatten(errors.NewAggregate(aggregateErr)))
 | |
| }
 | |
| 
 | |
| func getDeleteOptions(dryRun bool) (*client.DeleteOptions, string) {
 | |
| 	opts := &client.DeleteOptions{}
 | |
| 	var dryRunStr string
 | |
| 	if dryRun {
 | |
| 		client.DryRunAll.ApplyToDelete(opts)
 | |
| 		dryRunStr = "(dry run)"
 | |
| 	}
 | |
| 
 | |
| 	return opts, dryRunStr
 | |
| }
 | |
| 
 | |
| func getUpdateOptions(dryRun bool) (*client.UpdateOptions, string) {
 | |
| 	opts := &client.UpdateOptions{}
 | |
| 	var dryRunStr string
 | |
| 	if dryRun {
 | |
| 		client.DryRunAll.ApplyToUpdate(opts)
 | |
| 		dryRunStr = "(dry run)"
 | |
| 	}
 | |
| 
 | |
| 	return opts, dryRunStr
 | |
| }
 |