From f55e38557a6bbb9f09d3ed36d98ef7529c6d4188 Mon Sep 17 00:00:00 2001 From: Matthias Vach Date: Mon, 4 Dec 2023 15:34:10 +0100 Subject: [PATCH 1/2] Fix failing copy_ami_driver tests --- driver/copy_ami_driver_test.go | 146 ++++++++++++++++++++------------- 1 file changed, 91 insertions(+), 55 deletions(-) diff --git a/driver/copy_ami_driver_test.go b/driver/copy_ami_driver_test.go index aa1ac844..b08096f0 100644 --- a/driver/copy_ami_driver_test.go +++ b/driver/copy_ami_driver_test.go @@ -16,25 +16,21 @@ import ( uuid "github.com/satori/go.uuid" ) +type AmiCopyConfig struct { + amiId string + encrypted bool + kmsKeyId string + sharedWithAccounts []string +} + var _ = Describe("CopyAmiDriver", func() { It("copies an existing AMI to a new region while preserving its properties", func() { - copyAmi(false, "", amiFixtureID, func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { - snapshotIDptr := getSnapshotID(reqOutput) - - snapshotAttributes, err := ec2Client.DescribeSnapshotAttribute(&ec2.DescribeSnapshotAttributeInput{ - SnapshotId: snapshotIDptr, - Attribute: aws.String("createVolumePermission"), - }) - Expect(err).ToNot(HaveOccurred()) - - Expect(len(snapshotAttributes.CreateVolumePermissions)).To(Equal(1)) - Expect(*snapshotAttributes.CreateVolumePermissions[0].Group).To(Equal("all")) - }) - }) - - Context("when encrypted flag is set to true", func() { - It("does NOT make snapshot public", func() { - copyAmi(true, "", amiFixtureID, func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { + copyAmi( + AmiCopyConfig{ + amiId: amiFixtureID, + encrypted: false, + kmsKeyId: ""}, + func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { snapshotIDptr := getSnapshotID(reqOutput) snapshotAttributes, err := ec2Client.DescribeSnapshotAttribute(&ec2.DescribeSnapshotAttributeInput{ @@ -43,72 +39,112 @@ var _ = Describe("CopyAmiDriver", func() { }) Expect(err).ToNot(HaveOccurred()) - Expect(len(snapshotAttributes.CreateVolumePermissions)).To(Equal(0)) + Expect(len(snapshotAttributes.CreateVolumePermissions)).To(Equal(1)) + Expect(*snapshotAttributes.CreateVolumePermissions[0].Group).To(Equal("all")) }) - }) + }) - It("encrypts destination AMI using default AWS KMS key", func() { - copyAmi(true, "", amiFixtureID, func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { - respSnapshots, err := ec2Client.DescribeSnapshots(&ec2.DescribeSnapshotsInput{SnapshotIds: []*string{reqOutput.Images[0].BlockDeviceMappings[0].Ebs.SnapshotId}}) - Expect(err).ToNot(HaveOccurred()) + Context("when encrypted flag is set to true", func() { + It("does NOT make snapshot public", func() { + copyAmi( + AmiCopyConfig{ + amiId: amiFixtureID, + encrypted: true, + kmsKeyId: "", + }, + func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { + snapshotIDptr := getSnapshotID(reqOutput) + + snapshotAttributes, err := ec2Client.DescribeSnapshotAttribute(&ec2.DescribeSnapshotAttributeInput{ + SnapshotId: snapshotIDptr, + Attribute: aws.String("createVolumePermission"), + }) + Expect(err).ToNot(HaveOccurred()) - Expect(*respSnapshots.Snapshots[0].Encrypted).To(BeTrue()) - }) + Expect(len(snapshotAttributes.CreateVolumePermissions)).To(Equal(0)) + }) }) - Context("when kms_key_id is provided", func() { - It("encrypts destination AMI using provided kms key", func() { - copyAmi(true, kmsKeyId, amiFixtureID, func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { + It("encrypts destination AMI using default AWS KMS key", func() { + copyAmi( + AmiCopyConfig{ + amiId: amiFixtureID, + encrypted: true, + kmsKeyId: "", + }, + func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { respSnapshots, err := ec2Client.DescribeSnapshots(&ec2.DescribeSnapshotsInput{SnapshotIds: []*string{reqOutput.Images[0].BlockDeviceMappings[0].Ebs.SnapshotId}}) Expect(err).ToNot(HaveOccurred()) Expect(*respSnapshots.Snapshots[0].Encrypted).To(BeTrue()) - Expect(*respSnapshots.Snapshots[0].KmsKeyId).To(Equal(kmsKeyId)) }) + }) + + Context("when kms_key_id is provided", func() { + It("encrypts destination AMI using the kms key in the destination region", func() { + destinationRegionKmsKeyId := strings.Replace(multiregionKmsKeyId, creds.Region, destinationRegion, 1) + copyAmi( + AmiCopyConfig{ + amiId: privateAmiFixtureID, + encrypted: true, + kmsKeyId: destinationRegionKmsKeyId, + }, + func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { + respSnapshots, err := ec2Client.DescribeSnapshots(&ec2.DescribeSnapshotsInput{SnapshotIds: []*string{reqOutput.Images[0].BlockDeviceMappings[0].Ebs.SnapshotId}}) + Expect(err).ToNot(HaveOccurred()) + + Expect(*respSnapshots.Snapshots[0].Encrypted).To(BeTrue()) + Expect(*respSnapshots.Snapshots[0].KmsKeyId).To(Equal(destinationRegionKmsKeyId)) + }) }) }) }) Context("when shared_with_accounts is provided", func() { It("shares the AMI with other accounts", func() { - copyAmi(true, multiregionKmsKeyId, privateAmiFixtureID, func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { - attribute := "launchPermission" - output, err := ec2Client.DescribeImageAttribute(&ec2.DescribeImageAttributeInput{ - ImageId: reqOutput.Images[0].ImageId, - Attribute: &attribute, + destinationRegionKmsKeyId := strings.Replace(multiregionKmsKeyId, creds.Region, destinationRegion, 1) + copyAmi(AmiCopyConfig{ + amiId: privateAmiFixtureID, + encrypted: true, + kmsKeyId: destinationRegionKmsKeyId, + sharedWithAccounts: []string{awsAccount}, + }, + func(ec2Client *ec2.EC2, reqOutput *ec2.DescribeImagesOutput) { + attribute := "launchPermission" + output, err := ec2Client.DescribeImageAttribute(&ec2.DescribeImageAttributeInput{ + ImageId: reqOutput.Images[0].ImageId, + Attribute: &attribute, + }) + Expect(err).ToNot(HaveOccurred()) + Expect(*output.LaunchPermissions[0].UserId).To(Equal(awsAccount)) }) - Expect(err).ToNot(HaveOccurred()) - Expect(*output.LaunchPermissions[0].UserId).To(Equal(awsAccount)) - }) }) }) }) -func copyAmi(encrypted bool, kmsKeyId string, amiId string, cb ...func(*ec2.EC2, *ec2.DescribeImagesOutput)) { +func copyAmi(amiCopyConfig AmiCopyConfig, cb ...func(*ec2.EC2, *ec2.DescribeImagesOutput)) { accessibility := resources.PublicAmiAccessibility - if encrypted { + if amiCopyConfig.encrypted { accessibility = resources.PrivateAmiAccessibility } - var sharedWithAccounts []string - if kmsKeyId != "" { - sharedWithAccounts = []string{awsAccount} - } else { - sharedWithAccounts = []string{} + amiProperties := resources.AmiProperties{ + Name: fmt.Sprintf("BOSH-%s", strings.ToUpper(uuid.NewV4().String())), + VirtualizationType: resources.HvmAmiVirtualization, + Description: "bosh cpi test ami", + Accessibility: accessibility, + Encrypted: amiCopyConfig.encrypted, + KmsKeyId: amiCopyConfig.kmsKeyId, + } + if len(amiCopyConfig.sharedWithAccounts) > 0 { + amiProperties.SharedWithAccounts = amiCopyConfig.sharedWithAccounts } amiDriverConfig := resources.AmiDriverConfig{ - ExistingAmiID: amiId, + ExistingAmiID: amiCopyConfig.amiId, DestinationRegion: destinationRegion, - AmiProperties: resources.AmiProperties{ - Name: fmt.Sprintf("BOSH-%s", strings.ToUpper(uuid.NewV4().String())), - VirtualizationType: resources.HvmAmiVirtualization, - Description: "bosh cpi test ami", - Accessibility: accessibility, - Encrypted: encrypted, - KmsKeyId: kmsKeyId, - SharedWithAccounts: sharedWithAccounts, - }, + AmiProperties: amiProperties, + KmsKey: resources.KmsKey{ARN: amiCopyConfig.kmsKeyId}, } amiCopyDriver := driverset.NewStandardRegionDriverSet(GinkgoWriter, creds).CopyAmiDriver() @@ -133,7 +169,7 @@ func copyAmi(encrypted bool, kmsKeyId string, amiId string, cb ...func(*ec2.EC2, Expect(*firstImage.Name).To(Equal(amiDriverConfig.Name)) Expect(*firstImage.Architecture).To(Equal(resources.AmiArchitecture)) Expect(*firstImage.VirtualizationType).To(Equal(amiDriverConfig.VirtualizationType)) - if !encrypted { + if !amiCopyConfig.encrypted { Expect(*firstImage.Public).To(BeTrue()) } From 09286939c5757856e17ffdd3e73807e2987cd020 Mon Sep 17 00:00:00 2001 From: Matthias Vach Date: Mon, 4 Dec 2023 15:34:10 +0100 Subject: [PATCH 2/2] Prevent flaky copy_ami_driver tests Currently copy_ami_driver_test depends on the kms_driver_test since the kms_driver_test replicates a key that is used in the copy_ami_driver_test This can result in flakey tests. --- driver/copy_ami_driver_test.go | 4 ++-- driver/driver_suite_test.go | 9 ++++++--- driver/kms_driver_test.go | 4 ++-- 3 files changed, 10 insertions(+), 7 deletions(-) diff --git a/driver/copy_ami_driver_test.go b/driver/copy_ami_driver_test.go index b08096f0..cba4b669 100644 --- a/driver/copy_ami_driver_test.go +++ b/driver/copy_ami_driver_test.go @@ -82,7 +82,7 @@ var _ = Describe("CopyAmiDriver", func() { Context("when kms_key_id is provided", func() { It("encrypts destination AMI using the kms key in the destination region", func() { - destinationRegionKmsKeyId := strings.Replace(multiregionKmsKeyId, creds.Region, destinationRegion, 1) + destinationRegionKmsKeyId := strings.ReplaceAll(multiRegionKey, creds.Region, destinationRegion) copyAmi( AmiCopyConfig{ amiId: privateAmiFixtureID, @@ -102,7 +102,7 @@ var _ = Describe("CopyAmiDriver", func() { Context("when shared_with_accounts is provided", func() { It("shares the AMI with other accounts", func() { - destinationRegionKmsKeyId := strings.Replace(multiregionKmsKeyId, creds.Region, destinationRegion, 1) + destinationRegionKmsKeyId := strings.ReplaceAll(multiRegionKey, creds.Region, destinationRegion) copyAmi(AmiCopyConfig{ amiId: privateAmiFixtureID, encrypted: true, diff --git a/driver/driver_suite_test.go b/driver/driver_suite_test.go index 9aae34af..354f276f 100644 --- a/driver/driver_suite_test.go +++ b/driver/driver_suite_test.go @@ -20,7 +20,7 @@ var ebsVolumeID, ebsSnapshotID string var machineImagePath, machineImageFormat string var s3MachineImageUrl, s3MachineImageFormat string -var kmsKeyId, multiregionKmsKeyId string +var kmsKeyId, multiRegionKey, multiRegionKeyReplicationTest string var awsAccount string @@ -78,8 +78,11 @@ var _ = SynchronizedBeforeSuite( kmsKeyId = os.Getenv("AWS_KMS_KEY_ID") Expect(kmsKeyId).ToNot(BeEmpty(), "AWS_KMS_KEY_ID must be set") - multiregionKmsKeyId = os.Getenv("MULTI_REGION_AWS_KMS_KEY_ID") - Expect(multiregionKmsKeyId).ToNot(BeEmpty(), "MULTI_REGION_AWS_KMS_KEY_ID must be set") + multiRegionKey = os.Getenv("MULTI_REGION_KEY") + Expect(multiRegionKey).ToNot(BeEmpty(), "MULTI_REGION_KEY must be set") + + multiRegionKeyReplicationTest = os.Getenv("MULTI_REGION_KEY_REPLICATION_TEST") + Expect(multiRegionKeyReplicationTest).ToNot(BeEmpty(), "MULTI_REGION_KEY_REPLICATION_TEST must be set") awsAccount = os.Getenv("AWS_ACCOUNT") Expect(awsAccount).ToNot(BeEmpty(), "AWS_ACCOUNT must be set") diff --git a/driver/kms_driver_test.go b/driver/kms_driver_test.go index 691c96a6..a7840eb2 100644 --- a/driver/kms_driver_test.go +++ b/driver/kms_driver_test.go @@ -58,7 +58,7 @@ var _ = Describe("KmsDriver", func() { It("replicates a given kms key to another region", func() { driverConfig := resources.KmsReplicateKeyDriverConfig{ - KmsKeyId: multiregionKmsKeyId, + KmsKeyId: multiRegionKeyReplicationTest, SourceRegion: creds.Region, TargetRegion: destinationRegion, } @@ -74,7 +74,7 @@ var _ = Describe("KmsDriver", func() { //defer cleanup of the created key replica, sadly we can only schedule it to be deleted after 7 days //therefore this test will reuse the replicated key for 7 days and only afterward create a new one defer func(aliasCreationResult resources.KmsKey) { - destinationKeyId := strings.ReplaceAll(kmsKeyId, originalRegion, destinationRegion) + destinationKeyId := strings.ReplaceAll(multiRegionKeyReplicationTest, originalRegion, destinationRegion) awsSession, _ := session.NewSession(creds.GetAwsConfig()) kmsClient := kms.New(awsSession)