Skip to content

Commit

Permalink
E2E tests with EKS
Browse files Browse the repository at this point in the history
  • Loading branch information
Slava Lysunkin committed Nov 19, 2024
1 parent a21cdec commit 5e87262
Show file tree
Hide file tree
Showing 10 changed files with 160 additions and 9 deletions.
5 changes: 5 additions & 0 deletions templates/provider/hmc/templates/rbac/controller/roles.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -191,6 +191,11 @@ rules:
- clusterprofiles
- clustersummaries
verbs: {{ include "rbac.editorVerbs" . | nindent 4 }}
- apiGroups:
- controlplane.cluster.x-k8s.io
resources:
- awsmanagedcontrolplanes
verbs: {{ include "rbac.viewerVerbs" . | nindent 4 }}
- apiGroups:
- hmc.mirantis.com
resources:
Expand Down
6 changes: 4 additions & 2 deletions test/e2e/e2e_suite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -217,8 +217,10 @@ func collectLogArtifacts(kc *kubeclient.KubeClient, clusterName string, provider
"describe", "cluster", clusterName, "--namespace", internalutils.DefaultSystemNamespace, "--show-conditions=all")
output, err := utils.Run(cmd)
if err != nil {
utils.WarnError(fmt.Errorf("failed to get clusterctl log: %w", err))
return
if !strings.Contains(err.Error(), "unable to verify clusterctl version") {
utils.WarnError(fmt.Errorf("failed to get clusterctl log: %w", err))
return
}
}
err = os.WriteFile(filepath.Join("test/e2e", host+"-"+"clusterctl.log"), output, 0o644)
if err != nil {
Expand Down
12 changes: 12 additions & 0 deletions test/e2e/kubeclient/kubeclient.go
Original file line number Diff line number Diff line change
Expand Up @@ -279,3 +279,15 @@ func (kc *KubeClient) ListK0sControlPlanes(
Resource: "k0scontrolplanes",
}, clusterName)
}

func (kc *KubeClient) ListAWSManagedControlPlanes(
ctx context.Context, clusterName string,
) ([]unstructured.Unstructured, error) {
GinkgoHelper()

return kc.listResource(ctx, schema.GroupVersionResource{
Group: "controlplane.cluster.x-k8s.io",
Version: "v1beta2",
Resource: "awsmanagedcontrolplanes",
}, clusterName)
}
6 changes: 6 additions & 0 deletions test/e2e/managedcluster/managedcluster.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@ const (
TemplateAzureStandaloneCP Template = "azure-standalone-cp"
TemplateVSphereStandaloneCP Template = "vsphere-standalone-cp"
TemplateVSphereHostedCP Template = "vsphere-hosted-cp"
TemplateEKSCP Template = "aws-eks-cp"
)

//go:embed resources/aws-standalone-cp.yaml.tpl
Expand All @@ -70,6 +71,9 @@ var vsphereStandaloneCPManagedClusterTemplateBytes []byte
//go:embed resources/vsphere-hosted-cp.yaml.tpl
var vsphereHostedCPManagedClusterTemplateBytes []byte

//go:embed resources/aws-eks-cp.yaml.tpl
var eksCPManagedClusterTemplateBytes []byte

func FilterAllProviders() []string {
return []string{
utils.HMCControllerLabel,
Expand Down Expand Up @@ -134,6 +138,8 @@ func GetUnstructured(templateName Template) *unstructured.Unstructured {
managedClusterTemplateBytes = azureHostedCPManagedClusterTemplateBytes
case TemplateAzureStandaloneCP:
managedClusterTemplateBytes = azureStandaloneCPManagedClusterTemplateBytes
case TemplateEKSCP:
managedClusterTemplateBytes = eksCPManagedClusterTemplateBytes
default:
Fail(fmt.Sprintf("Unsupported template: %s", templateName))
}
Expand Down
10 changes: 9 additions & 1 deletion test/e2e/managedcluster/providervalidator.go
Original file line number Diff line number Diff line change
Expand Up @@ -65,14 +65,22 @@ func NewProviderValidator(template Template, clusterName string, action Validati
case TemplateAWSStandaloneCP, TemplateAWSHostedCP:
resourcesToValidate["ccm"] = validateCCM
resourceOrder = append(resourceOrder, "ccm")
case TemplateEKSCP:
resourcesToValidate["control-planes"] = validateAWSManagedControlPlanes
delete(resourcesToValidate, "csi-driver")
case TemplateAzureStandaloneCP, TemplateVSphereStandaloneCP:
delete(resourcesToValidate, "csi-driver")
}
} else {
validateCPDeletedFunc := validateK0sControlPlanesDeleted
if template == TemplateEKSCP {
validateCPDeletedFunc = validateAWSManagedControlPlanesDeleted
}

resourcesToValidate = map[string]resourceValidationFunc{
"clusters": validateClusterDeleted,
"machinedeployments": validateMachineDeploymentsDeleted,
"control-planes": validateK0sControlPlanesDeleted,
"control-planes": validateCPDeletedFunc,
}
resourceOrder = []string{"clusters", "machinedeployments", "control-planes"}
}
Expand Down
16 changes: 16 additions & 0 deletions test/e2e/managedcluster/resources/aws-eks-cp.yaml.tpl
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
apiVersion: hmc.mirantis.com/v1alpha1
kind: ManagedCluster
metadata:
name: ${MANAGED_CLUSTER_NAME}-eks
spec:
template: aws-eks-0-0-2
credential: ${AWS_CLUSTER_IDENTITY}-cred
config:
region: ${AWS_REGION}
workersNumber: ${WORKERS_NUMBER:=1}
clusterIdentity:
name: ${AWS_CLUSTER_IDENTITY}-cred
namespace: ${NAMESPACE}
publicIP: ${AWS_PUBLIC_IP:=true}
worker:
instanceType: ${AWS_INSTANCE_TYPE:=t3.small}
18 changes: 18 additions & 0 deletions test/e2e/managedcluster/validate_deleted.go
Original file line number Diff line number Diff line change
Expand Up @@ -101,3 +101,21 @@ func validateK0sControlPlanesDeleted(ctx context.Context, kc *kubeclient.KubeCli

return nil
}

func validateAWSManagedControlPlanesDeleted(ctx context.Context, kc *kubeclient.KubeClient, clusterName string) error {
controlPlanes, err := kc.ListAWSManagedControlPlanes(ctx, clusterName)
if err != nil && !apierrors.IsNotFound(err) {
return err
}

var cpNames []string
if len(controlPlanes) > 0 {
for _, cp := range controlPlanes {
cpNames = append(cpNames, cp.GetName())

return fmt.Errorf("AWS Managed control planes still exist: %s", cpNames)
}
}

return nil
}
28 changes: 24 additions & 4 deletions test/e2e/managedcluster/validate_deployed.go
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ func validateCluster(ctx context.Context, kc *kubeclient.KubeClient, clusterName
Fail(err.Error())
}

return utils.ValidateConditionsTrue(cluster)
return utils.NewConditionsValidator().IfTrue(cluster)
}

func validateMachines(ctx context.Context, kc *kubeclient.KubeClient, clusterName string) error {
Expand All @@ -79,7 +79,7 @@ func validateMachines(ctx context.Context, kc *kubeclient.KubeClient, clusterNam
Fail(err.Error())
}

if err := utils.ValidateConditionsTrue(&md); err != nil {
if err := utils.NewConditionsValidator().IfTrue(&md); err != nil {
return err
}
}
Expand All @@ -90,7 +90,7 @@ func validateMachines(ctx context.Context, kc *kubeclient.KubeClient, clusterNam
Fail(err.Error())
}

if err := utils.ValidateConditionsTrue(&machine); err != nil {
if err := utils.NewConditionsValidator().IfTrue(&machine); err != nil {
return err
}
}
Expand All @@ -113,7 +113,7 @@ func validateK0sControlPlanes(ctx context.Context, kc *kubeclient.KubeClient, cl

// k0s does not use the metav1.Condition type for status.conditions,
// instead it uses a custom type so we can't use
// ValidateConditionsTrue here, instead we'll check for "ready: true".
// ordinary conditions validation here, instead we'll check for "ready: true".
objStatus, found, err := unstructured.NestedFieldCopy(controlPlane.Object, "status")
if !found {
return fmt.Errorf("no status found for %s: %s", objKind, objName)
Expand All @@ -139,6 +139,26 @@ func validateK0sControlPlanes(ctx context.Context, kc *kubeclient.KubeClient, cl
return nil
}

func validateAWSManagedControlPlanes(ctx context.Context, kc *kubeclient.KubeClient, clusterName string) error {
controlPlanes, err := kc.ListAWSManagedControlPlanes(ctx, clusterName)
if err != nil {
return err
}

for _, controlPlane := range controlPlanes {
if err := utils.ValidateObjectNamePrefix(&controlPlane, clusterName); err != nil {
Fail(err.Error())
}

// EKSControlPlaneCreating condition very often has READY=False, SEVERITY=Info and REASON=created (this is fine).
if err := utils.NewConditionsValidator(utils.WithExcluded([]string{"EKSControlPlaneCreating"})).IfTrue(&controlPlane); err != nil {
return err
}
}

return nil
}

// validateCSIDriver validates that the provider CSI driver is functioning
// by creating a PVC and verifying it enters "Bound" status.
func validateCSIDriver(ctx context.Context, kc *kubeclient.KubeClient, clusterName string) error {
Expand Down
41 changes: 41 additions & 0 deletions test/e2e/provider_aws_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -186,4 +186,45 @@ var _ = Describe("AWS Templates", Label("provider:cloud", "provider:aws"), Order
time.Second).Should(Succeed())
*/
})

It("should work with an EKS provider", func() {
// Deploy a standalone cluster and verify it is running/ready.
GinkgoT().Setenv(managedcluster.EnvVarAWSInstanceType, "t3.small")

cmd := exec.Command("kubectl", "get", "clustertemplates", "-n", "hmc-system", "-o", "yaml")
output, err := utils.Run(cmd)
_, _ = fmt.Fprintln(GinkgoWriter, string(output))
Expect(err).NotTo(HaveOccurred())

templateBy(managedcluster.TemplateEKSCP, "creating a ManagedCluster for EKS")
sd := managedcluster.GetUnstructured(managedcluster.TemplateEKSCP)
clusterName = sd.GetName()

standaloneDeleteFunc = kc.CreateManagedCluster(context.Background(), sd)

templateBy(managedcluster.TemplateEKSCP, "waiting for infrastructure to deploy successfully")
deploymentValidator := managedcluster.NewProviderValidator(
managedcluster.TemplateEKSCP,
clusterName,
managedcluster.ValidationActionDeploy,
)

Eventually(func() error {
return deploymentValidator.Validate(context.Background(), kc)
}).WithTimeout(60 * time.Minute).WithPolling(30 * time.Second).Should(Succeed())

// --- clean up ---
templateBy(managedcluster.TemplateAWSStandaloneCP, "deleting the ManagedCluster for EKS")
Expect(standaloneDeleteFunc()).NotTo(HaveOccurred())

deletionValidator := managedcluster.NewProviderValidator(
managedcluster.TemplateAWSStandaloneCP,
clusterName,
managedcluster.ValidationActionDelete,
)
Eventually(func() error {
return deletionValidator.Validate(context.Background(), kc)
}).WithTimeout(15 * time.Minute).WithPolling(10 *
time.Second).Should(Succeed())
})
})
27 changes: 25 additions & 2 deletions test/utils/utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ import (
. "github.com/onsi/ginkgo/v2"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/utils/strings/slices"

"github.com/Mirantis/hmc/internal/utils/status"
)
Expand Down Expand Up @@ -111,10 +112,28 @@ func GetProjectDir() (string, error) {
return wd, nil
}

// ValidateConditionsTrue iterates over the conditions of the given
type ConditionsValidator struct {
excludedConditions []string
}

func NewConditionsValidator(options ...func(*ConditionsValidator)) *ConditionsValidator {
cv := &ConditionsValidator{}
for _, o := range options {
o(cv)
}
return cv
}

func WithExcluded(excludedConditions []string) func(*ConditionsValidator) {
return func(cv *ConditionsValidator) {
cv.excludedConditions = excludedConditions
}
}

// IfTrue iterates over the conditions of the given
// unstructured object and returns an error if any of the conditions are not
// true. Conditions are expected to be of type metav1.Condition.
func ValidateConditionsTrue(unstrObj *unstructured.Unstructured) error {
func (cv *ConditionsValidator) IfTrue(unstrObj *unstructured.Unstructured) error {
objKind, objName := status.ObjKindName(unstrObj)

conditions, err := status.ConditionsFromUnstructured(unstrObj)
Expand All @@ -129,6 +148,10 @@ func ValidateConditionsTrue(unstrObj *unstructured.Unstructured) error {
continue
}

if slices.Contains(cv.excludedConditions, c.Type) {
continue
}

errs = errors.Join(errors.New(ConvertConditionsToString(c)), errs)
}

Expand Down

0 comments on commit 5e87262

Please sign in to comment.