diff --git a/.changelog/24720.txt b/.changelog/24720.txt new file mode 100644 index 00000000000..bcbe59a7472 --- /dev/null +++ b/.changelog/24720.txt @@ -0,0 +1,3 @@ +```release-note:improvement +deps: Upgraded aws-sdk-go from v1 to v2 +``` diff --git a/client/fingerprint/env_aws.go b/client/fingerprint/env_aws.go index 4041a6b0950..74024b26f2d 100644 --- a/client/fingerprint/env_aws.go +++ b/client/fingerprint/env_aws.go @@ -4,19 +4,20 @@ package fingerprint import ( + "context" + "errors" "fmt" + "io" "net/http" - "net/url" - "os" "regexp" "strings" "time" - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/awserr" - "github.com/aws/aws-sdk-go/aws/ec2metadata" - "github.com/aws/aws-sdk-go/aws/session" - cleanhttp "github.com/hashicorp/go-cleanhttp" + "github.com/aws/aws-sdk-go-v2/config" + "github.com/aws/aws-sdk-go-v2/feature/ec2/imds" + smithyHttp "github.com/aws/smithy-go/transport/http" + + "github.com/hashicorp/go-cleanhttp" log "github.com/hashicorp/go-hclog" "github.com/hashicorp/nomad/nomad/structs" ) @@ -52,7 +53,7 @@ var ec2NetSpeedTable = map[*regexp.Regexp]int{ type EnvAWSFingerprint struct { StaticFingerprinter - // endpoint for EC2 metadata as expected by AWS SDK + // used to override IMDS endpoint for testing endpoint string logger log.Logger @@ -61,8 +62,7 @@ type EnvAWSFingerprint struct { // NewEnvAWSFingerprint is used to create a fingerprint from AWS metadata func NewEnvAWSFingerprint(logger log.Logger) Fingerprint { f := &EnvAWSFingerprint{ - logger: logger.Named("env_aws"), - endpoint: strings.TrimSuffix(os.Getenv("AWS_ENV_URL"), "/meta-data/"), + logger: logger.Named("env_aws"), } return f } @@ -77,12 +77,16 @@ func (f *EnvAWSFingerprint) Fingerprint(request *FingerprintRequest, response *F timeout = 1 * time.Millisecond } - ec2meta, err := ec2MetaClient(f.endpoint, timeout) + ctx, cancel := context.WithTimeout(context.TODO(), timeout) + defer cancel() + + imdsClient, err := f.imdsClient(ctx) if err != nil { - return fmt.Errorf("failed to setup ec2Metadata client: %v", err) + return fmt.Errorf("failed to setup IMDS client: %v", err) } - if !isAWS(ec2meta) { + if !isAWS(ctx, imdsClient) { + f.logger.Debug("error querying AWS IDMS URL, skipping") return nil } @@ -104,25 +108,26 @@ func (f *EnvAWSFingerprint) Fingerprint(request *FingerprintRequest, response *F } for k, unique := range keys { - resp, err := ec2meta.GetMetadata(k) - v := strings.TrimSpace(resp) - if v == "" { - f.logger.Debug("read an empty value", "attribute", k) - continue - } else if awsErr, ok := err.(awserr.RequestFailure); ok { - f.logger.Debug("could not read attribute value", "attribute", k, "error", awsErr) + resp, err := imdsClient.GetMetadata(ctx, &imds.GetMetadataInput{ + Path: k, + }) + if err := f.handleImdsError(err, k); err != nil { + return err + } + if resp == nil { continue - } else if awsErr, ok := err.(awserr.Error); ok { - // if it's a URL error, assume we're not in an AWS environment - // TODO: better way to detect AWS? Check xen virtualization? - if _, ok := awsErr.OrigErr().(*url.Error); ok { - return nil - } + } - // not sure what other errors it would return + v, err := readMetadataResponse(resp) + if err != nil { return err } + if v == "" { + f.logger.Debug("read an empty value", "attribute", k) + continue + } + // assume we want blank entries key := "platform.aws." + strings.ReplaceAll(k, "/", ".") if unique { @@ -144,7 +149,7 @@ func (f *EnvAWSFingerprint) Fingerprint(request *FingerprintRequest, response *F Device: "eth0", IP: val, CIDR: val + "/32", - MBits: f.throughput(request, ec2meta, val), + MBits: f.throughput(request, imdsClient, val), }, } } @@ -152,24 +157,24 @@ func (f *EnvAWSFingerprint) Fingerprint(request *FingerprintRequest, response *F // copy over IPv6 network specific information if val, ok := response.Attributes["unique.platform.aws.mac"]; ok && val != "" { k := "network/interfaces/macs/" + val + "/ipv6s" - addrsStr, err := ec2meta.GetMetadata(k) - addrsStr = strings.TrimSpace(addrsStr) - if addrsStr == "" { - f.logger.Debug("read an empty value", "attribute", k) - } else if awsErr, ok := err.(awserr.RequestFailure); ok { - f.logger.Debug("could not read attribute value", "attribute", k, "error", awsErr) - } else if awsErr, ok := err.(awserr.Error); ok { - // if it's a URL error, assume we're not in an AWS environment - // TODO: better way to detect AWS? Check xen virtualization? - if _, ok := awsErr.OrigErr().(*url.Error); ok { - return nil + resp, err := imdsClient.GetMetadata(ctx, &imds.GetMetadataInput{ + Path: k, + }) + if err := f.handleImdsError(err, k); err != nil { + return err + } + if resp != nil { + addrsStr, err := readMetadataResponse(resp) + if err != nil { + return err } - // not sure what other errors it would return - return err - } else { - addrs := strings.SplitN(addrsStr, "\n", 2) - response.AddAttribute("unique.platform.aws.public-ipv6", addrs[0]) + if addrsStr == "" { + f.logger.Debug("read an empty value", "attribute", k) + } else { + addrs := strings.SplitN(addrsStr, "\n", 2) + response.AddAttribute("unique.platform.aws.public-ipv6", addrs[0]) + } } } @@ -184,21 +189,41 @@ func (f *EnvAWSFingerprint) Fingerprint(request *FingerprintRequest, response *F return nil } -func (f *EnvAWSFingerprint) instanceType(ec2meta *ec2metadata.EC2Metadata) (string, error) { - response, err := ec2meta.GetMetadata("instance-type") +// See https://aws.github.io/aws-sdk-go-v2/docs/handling-errors for +// recommended error handling with aws-sdk-go-v2. +// See also: https://github.com/aws/aws-sdk-go-v2/issues/1306 +func (f *EnvAWSFingerprint) handleImdsError(err error, attr string) error { + var apiErr *smithyHttp.ResponseError + if errors.As(err, &apiErr) { + // In the event of a request error while fetching attributes, just log and return nil. + // This will happen if attributes do not exist for this instance (ex. ipv6, public-ipv4s). + f.logger.Debug("could not read attribute value", "attribute", attr, "error", err) + return nil + } + return err +} + +func (f *EnvAWSFingerprint) instanceType(client *imds.Client) (string, error) { + output, err := client.GetMetadata(context.TODO(), &imds.GetMetadataInput{ + Path: "instance-type", + }) + if err != nil { + return "", err + } + content, err := io.ReadAll(output.Content) if err != nil { return "", err } - return strings.TrimSpace(response), nil + return strings.TrimSpace(string(content)), nil } -func (f *EnvAWSFingerprint) throughput(request *FingerprintRequest, ec2meta *ec2metadata.EC2Metadata, ip string) int { +func (f *EnvAWSFingerprint) throughput(request *FingerprintRequest, client *imds.Client, ip string) int { throughput := request.Config.NetworkSpeed if throughput != 0 { return throughput } - throughput = f.linkSpeed(ec2meta) + throughput = f.linkSpeed(client) if throughput != 0 { return throughput } @@ -215,8 +240,8 @@ func (f *EnvAWSFingerprint) throughput(request *FingerprintRequest, ec2meta *ec2 } // EnvAWSFingerprint uses lookup table to approximate network speeds -func (f *EnvAWSFingerprint) linkSpeed(ec2meta *ec2metadata.EC2Metadata) int { - instanceType, err := f.instanceType(ec2meta) +func (f *EnvAWSFingerprint) linkSpeed(client *imds.Client) int { + instanceType, err := f.instanceType(client) if err != nil { f.logger.Error("error reading instance-type", "error", err) return 0 @@ -233,26 +258,51 @@ func (f *EnvAWSFingerprint) linkSpeed(ec2meta *ec2metadata.EC2Metadata) int { return netSpeed } -func ec2MetaClient(endpoint string, timeout time.Duration) (*ec2metadata.EC2Metadata, error) { +func (f *EnvAWSFingerprint) imdsClient(ctx context.Context) (*imds.Client, error) { client := &http.Client{ - Timeout: timeout, Transport: cleanhttp.DefaultTransport(), } + cfg, err := config.LoadDefaultConfig(ctx, + config.WithHTTPClient(client), + config.WithRetryMaxAttempts(0), + ) + if err != nil { + return nil, err + } - c := aws.NewConfig().WithHTTPClient(client).WithMaxRetries(0) - if endpoint != "" { - c = c.WithEndpoint(endpoint) + imdsClient := imds.NewFromConfig(cfg, func(o *imds.Options) { + // endpoint should only be overridden for testing + if f.endpoint != "" { + o.Endpoint = f.endpoint + } + }) + return imdsClient, nil +} + +func isAWS(ctx context.Context, client *imds.Client) bool { + resp, err := client.GetMetadata(ctx, &imds.GetMetadataInput{ + Path: "ami-id", + }) + if err != nil { + return false } - sess, err := session.NewSession(c) + s, err := readMetadataResponse(resp) if err != nil { - return nil, err + return false } - return ec2metadata.New(sess, c), nil + + return s != "" } -func isAWS(ec2meta *ec2metadata.EC2Metadata) bool { - v, err := ec2meta.GetMetadata("ami-id") - v = strings.TrimSpace(v) - return err == nil && v != "" +// readImdsResponse reads and formats the IMDS response +// and most importantly, closes the io.ReadCloser +func readMetadataResponse(resp *imds.GetMetadataOutput) (string, error) { + defer resp.Content.Close() + + b, err := io.ReadAll(resp.Content) + if err != nil { + return "", err + } + return strings.TrimSpace(string(b)), nil } diff --git a/client/fingerprint/env_aws_test.go b/client/fingerprint/env_aws_test.go index 6131b95281b..f2fd416d334 100644 --- a/client/fingerprint/env_aws_test.go +++ b/client/fingerprint/env_aws_test.go @@ -9,10 +9,13 @@ import ( "net/http/httptest" "testing" + "github.com/aws/smithy-go" + smithyHttp "github.com/aws/smithy-go/transport/http" "github.com/hashicorp/nomad/ci" "github.com/hashicorp/nomad/client/config" "github.com/hashicorp/nomad/helper/testlog" "github.com/hashicorp/nomad/nomad/structs" + "github.com/shoenig/test/must" "github.com/stretchr/testify/require" ) @@ -77,6 +80,45 @@ func TestEnvAWSFingerprint_aws(t *testing.T) { } } +func TestEnvAWSFingerprint_handleImdsError(t *testing.T) { + ci.Parallel(t) + + f := NewEnvAWSFingerprint(testlog.HCLogger(t)) + + cases := []struct { + name string + err error + exp error + }{ + { + name: "random errors return error", + err: fmt.Errorf("not http error"), + exp: fmt.Errorf("not http error"), + }, + { + name: "other smithy errors return error", + err: &smithy.OperationError{}, + exp: &smithy.OperationError{}, + }, + { + name: "http response errors correctly handled", + err: &smithyHttp.ResponseError{ + Response: &smithyHttp.Response{ + Response: &http.Response{ + StatusCode: 404, + }, + }, + }, + exp: nil, + }, + } + + for _, c := range cases { + err := f.(*EnvAWSFingerprint).handleImdsError(c.err, "some attribute") + must.Eq(t, c.exp, err) + } +} + func TestNetworkFingerprint_AWS(t *testing.T) { ci.Parallel(t) @@ -192,6 +234,9 @@ func TestNetworkFingerprint_AWS_NoNetwork(t *testing.T) { require.Equal(t, "ami-1234", response.Attributes["platform.aws.ami-id"]) + // assert the key is not present in the Attributes map if the return value was empty + require.NotContains(t, response.Attributes, "unique.platform.aws.local-ipv4") + require.Nil(t, response.NodeResources.Networks) } diff --git a/e2e/remotetasks/remotetasks.go b/e2e/remotetasks/remotetasks.go index e46d10f1167..a71eafcb7af 100644 --- a/e2e/remotetasks/remotetasks.go +++ b/e2e/remotetasks/remotetasks.go @@ -4,14 +4,15 @@ package remotetasks import ( + "context" "fmt" "os" "testing" "time" - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/session" - "github.com/aws/aws-sdk-go/service/ecs" + "github.com/aws/aws-sdk-go-v2/aws" + "github.com/aws/aws-sdk-go-v2/config" + "github.com/aws/aws-sdk-go-v2/service/ecs" "github.com/hashicorp/nomad/api" "github.com/hashicorp/nomad/e2e/e2eutil" "github.com/hashicorp/nomad/e2e/framework" @@ -78,7 +79,9 @@ func (tc *RemoteTasksTest) AfterEach(f *framework.F) { func (tc *RemoteTasksTest) TestECSJob(f *framework.F) { t := f.T() - ecsClient := ecsOrSkip(t, tc.Nomad()) + ctx := context.Background() + + ecsClient := ecsOrSkip(ctx, t, tc.Nomad()) jobID := "ecsjob-" + uuid.Generate()[0:8] tc.jobIDs = append(tc.jobIDs, jobID) @@ -92,7 +95,7 @@ func (tc *RemoteTasksTest) TestECSJob(f *framework.F) { arn := arnForAlloc(t, tc.Nomad().Allocations(), allocID) // Use ARN to lookup status of ECS task in AWS - ensureECSRunning(t, ecsClient, arn) + ensureECSRunning(ctx, t, ecsClient, arn) t.Logf("Task %s is running!", arn) @@ -102,10 +105,10 @@ func (tc *RemoteTasksTest) TestECSJob(f *framework.F) { // Ensure it is stopped in ECS input := ecs.DescribeTasksInput{ Cluster: aws.String("nomad-rtd-e2e"), - Tasks: []*string{aws.String(arn)}, + Tasks: []string{arn}, } testutil.WaitForResult(func() (bool, error) { - resp, err := ecsClient.DescribeTasks(&input) + resp, err := ecsClient.DescribeTasks(ctx, &input) if err != nil { return false, err } @@ -121,7 +124,9 @@ func (tc *RemoteTasksTest) TestECSJob(f *framework.F) { func (tc *RemoteTasksTest) TestECSDrain(f *framework.F) { t := f.T() - ecsClient := ecsOrSkip(t, tc.Nomad()) + ctx := context.Background() + + ecsClient := ecsOrSkip(ctx, t, tc.Nomad()) jobID := "ecsjob-" + uuid.Generate()[0:8] tc.jobIDs = append(tc.jobIDs, jobID) @@ -132,7 +137,7 @@ func (tc *RemoteTasksTest) TestECSDrain(f *framework.F) { e2eutil.WaitForAllocsRunning(t, tc.Nomad(), []string{origAlloc}) arn := arnForAlloc(t, tc.Nomad().Allocations(), origAlloc) - ensureECSRunning(t, ecsClient, arn) + ensureECSRunning(ctx, t, ecsClient, arn) t.Logf("Task %s is running! Now to drain the node.", arn) @@ -197,7 +202,9 @@ func (tc *RemoteTasksTest) TestECSDrain(f *framework.F) { func (tc *RemoteTasksTest) TestECSDeployment(f *framework.F) { t := f.T() - ecsClient := ecsOrSkip(t, tc.Nomad()) + ctx := context.Background() + + ecsClient := ecsOrSkip(ctx, t, tc.Nomad()) jobID := "ecsjob-" + uuid.Generate()[0:8] tc.jobIDs = append(tc.jobIDs, jobID) @@ -211,7 +218,7 @@ func (tc *RemoteTasksTest) TestECSDeployment(f *framework.F) { origARN := arnForAlloc(t, tc.Nomad().Allocations(), origAllocID) // Use ARN to lookup status of ECS task in AWS - ensureECSRunning(t, ecsClient, origARN) + ensureECSRunning(ctx, t, ecsClient, origARN) t.Logf("Task %s is running! Updating...", origARN) @@ -271,10 +278,10 @@ func (tc *RemoteTasksTest) TestECSDeployment(f *framework.F) { // Ensure original ARN is stopped in ECS input := ecs.DescribeTasksInput{ Cluster: aws.String("nomad-rtd-e2e"), - Tasks: []*string{aws.String(origARN)}, + Tasks: []string{origARN}, } testutil.WaitForResult(func() (bool, error) { - resp, err := ecsClient.DescribeTasks(&input) + resp, err := ecsClient.DescribeTasks(ctx, &input) if err != nil { return false, err } @@ -287,12 +294,13 @@ func (tc *RemoteTasksTest) TestECSDeployment(f *framework.F) { // ecsOrSkip returns an AWS ECS client or skips the test if ECS is unreachable // by the test runner or the ECS remote task driver isn't healthy. -func ecsOrSkip(t *testing.T, nomadClient *api.Client) *ecs.ECS { - awsSession := session.Must(session.NewSession()) +func ecsOrSkip(ctx context.Context, t *testing.T, nomadClient *api.Client) *ecs.Client { + cfg, err := config.LoadDefaultConfig(ctx, config.WithRegion("us-east-1")) + require.NoError(t, err) - ecsClient := ecs.New(awsSession, aws.NewConfig().WithRegion("us-east-1")) + ecsClient := ecs.NewFromConfig(cfg) - _, err := ecsClient.ListClusters(&ecs.ListClustersInput{}) + _, err = ecsClient.ListClusters(ctx, &ecs.ListClustersInput{}) if err != nil { t.Skipf("Skipping ECS Remote Task Driver Task. Error querying AWS ECS API: %v", err) } @@ -378,14 +386,14 @@ func arnForAlloc(t *testing.T, allocAPI *api.Allocations, allocID string) string } // ensureECSRunning asserts that the given ARN is a running ECS task. -func ensureECSRunning(t *testing.T, ecsClient *ecs.ECS, arn string) { +func ensureECSRunning(ctx context.Context, t *testing.T, ecsClient *ecs.Client, arn string) { t.Logf("Ensuring ARN=%s is running", arn) input := ecs.DescribeTasksInput{ Cluster: aws.String("nomad-rtd-e2e"), - Tasks: []*string{aws.String(arn)}, + Tasks: []string{arn}, } testutil.WaitForResult(func() (bool, error) { - resp, err := ecsClient.DescribeTasks(&input) + resp, err := ecsClient.DescribeTasks(ctx, &input) if err != nil { return false, err } diff --git a/go.mod b/go.mod index f0791392ec0..eb868253a50 100644 --- a/go.mod +++ b/go.mod @@ -18,7 +18,11 @@ require ( github.com/Microsoft/go-winio v0.6.1 github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e github.com/armon/go-metrics v0.5.3 - github.com/aws/aws-sdk-go v1.55.5 + github.com/aws/aws-sdk-go-v2 v1.32.7 + github.com/aws/aws-sdk-go-v2/config v1.28.6 + github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.21 + github.com/aws/aws-sdk-go-v2/service/ecs v1.53.0 + github.com/aws/smithy-go v1.22.1 github.com/brianvoe/gofakeit/v6 v6.20.1 github.com/container-storage-interface/spec v1.10.0 github.com/containerd/go-cni v1.1.11 @@ -176,6 +180,16 @@ require ( github.com/apparentlymart/go-cidr v1.0.1 // indirect github.com/apparentlymart/go-textseg/v15 v15.0.0 // indirect github.com/armon/go-radix v1.0.0 // indirect + github.com/aws/aws-sdk-go v1.55.5 // indirect + github.com/aws/aws-sdk-go-v2/credentials v1.17.47 // indirect + github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26 // indirect + github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26 // indirect + github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1 // indirect + github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.6 // indirect + github.com/aws/aws-sdk-go-v2/service/sso v1.24.7 // indirect + github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.6 // indirect + github.com/aws/aws-sdk-go-v2/service/sts v1.33.2 // indirect github.com/beorn7/perks v1.0.1 // indirect github.com/bgentry/go-netrc v0.0.0-20140422174119-9fd32a8b3d3d // indirect github.com/bgentry/speakeasy v0.1.0 // indirect diff --git a/go.sum b/go.sum index 36a3e403c95..07c417f7a36 100644 --- a/go.sum +++ b/go.sum @@ -294,6 +294,34 @@ github.com/aws/aws-sdk-go v1.30.27/go.mod h1:5zCpMtNQVjRREroY7sYe8lOMRSxkhG6MZve github.com/aws/aws-sdk-go v1.44.122/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX4oIKwKHZo= github.com/aws/aws-sdk-go v1.55.5 h1:KKUZBfBoyqy5d3swXyiC7Q76ic40rYcbqH7qjh59kzU= github.com/aws/aws-sdk-go v1.55.5/go.mod h1:eRwEWoyTWFMVYVQzKMNHWP5/RV4xIUGMQfXQHfHkpNU= +github.com/aws/aws-sdk-go-v2 v1.32.7 h1:ky5o35oENWi0JYWUZkB7WYvVPP+bcRF5/Iq7JWSb5Rw= +github.com/aws/aws-sdk-go-v2 v1.32.7/go.mod h1:P5WJBrYqqbWVaOxgH0X/FYYD47/nooaPOZPlQdmiN2U= +github.com/aws/aws-sdk-go-v2/config v1.28.6 h1:D89IKtGrs/I3QXOLNTH93NJYtDhm8SYa9Q5CsPShmyo= +github.com/aws/aws-sdk-go-v2/config v1.28.6/go.mod h1:GDzxJ5wyyFSCoLkS+UhGB0dArhb9mI+Co4dHtoTxbko= +github.com/aws/aws-sdk-go-v2/credentials v1.17.47 h1:48bA+3/fCdi2yAwVt+3COvmatZ6jUDNkDTIsqDiMUdw= +github.com/aws/aws-sdk-go-v2/credentials v1.17.47/go.mod h1:+KdckOejLW3Ks3b0E3b5rHsr2f9yuORBum0WPnE5o5w= +github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.21 h1:AmoU1pziydclFT/xRV+xXE/Vb8fttJCLRPv8oAkprc0= +github.com/aws/aws-sdk-go-v2/feature/ec2/imds v1.16.21/go.mod h1:AjUdLYe4Tgs6kpH4Bv7uMZo7pottoyHMn4eTcIcneaY= +github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26 h1:I/5wmGMffY4happ8NOCuIUEWGUvvFp5NSeQcXl9RHcI= +github.com/aws/aws-sdk-go-v2/internal/configsources v1.3.26/go.mod h1:FR8f4turZtNy6baO0KJ5FJUmXH/cSkI9fOngs0yl6mA= +github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26 h1:zXFLuEuMMUOvEARXFUVJdfqZ4bvvSgdGRq/ATcrQxzM= +github.com/aws/aws-sdk-go-v2/internal/endpoints/v2 v2.6.26/go.mod h1:3o2Wpy0bogG1kyOPrgkXA8pgIfEEv0+m19O9D5+W8y8= +github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1 h1:VaRN3TlFdd6KxX1x3ILT5ynH6HvKgqdiXoTxAF4HQcQ= +github.com/aws/aws-sdk-go-v2/internal/ini v1.8.1/go.mod h1:FbtygfRFze9usAadmnGJNc8KsP346kEe+y2/oyhGAGc= +github.com/aws/aws-sdk-go-v2/service/ecs v1.53.0 h1:TCQZX4ztlcWXAcZouKh9qJMcVaH/qTidFTfsvJwUI30= +github.com/aws/aws-sdk-go-v2/service/ecs v1.53.0/go.mod h1:Ghi1OWUv4+VMEULWiHsKH2gNA3KAcMoLWsvU0eRXvIA= +github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1 h1:iXtILhvDxB6kPvEXgsDhGaZCSC6LQET5ZHSdJozeI0Y= +github.com/aws/aws-sdk-go-v2/service/internal/accept-encoding v1.12.1/go.mod h1:9nu0fVANtYiAePIBh2/pFUSwtJ402hLnp854CNoDOeE= +github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.6 h1:50+XsN70RS7dwJ2CkVNXzj7U2L1HKP8nqTd3XWEXBN4= +github.com/aws/aws-sdk-go-v2/service/internal/presigned-url v1.12.6/go.mod h1:WqgLmwY7so32kG01zD8CPTJWVWM+TzJoOVHwTg4aPug= +github.com/aws/aws-sdk-go-v2/service/sso v1.24.7 h1:rLnYAfXQ3YAccocshIH5mzNNwZBkBo+bP6EhIxak6Hw= +github.com/aws/aws-sdk-go-v2/service/sso v1.24.7/go.mod h1:ZHtuQJ6t9A/+YDuxOLnbryAmITtr8UysSny3qcyvJTc= +github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.6 h1:JnhTZR3PiYDNKlXy50/pNeix9aGMo6lLpXwJ1mw8MD4= +github.com/aws/aws-sdk-go-v2/service/ssooidc v1.28.6/go.mod h1:URronUEGfXZN1VpdktPSD1EkAL9mfrV+2F4sjH38qOY= +github.com/aws/aws-sdk-go-v2/service/sts v1.33.2 h1:s4074ZO1Hk8qv65GqNXqDjmkf4HSQqJukaLuuW0TpDA= +github.com/aws/aws-sdk-go-v2/service/sts v1.33.2/go.mod h1:mVggCnIWoM09jP71Wh+ea7+5gAp53q+49wDFs1SW5z8= +github.com/aws/smithy-go v1.22.1 h1:/HPHZQ0g7f4eUeK6HKglFz8uwVfZKgoI25rb/J+dnro= +github.com/aws/smithy-go v1.22.1/go.mod h1:irrKGvNn1InZwb2d7fkIRNucdfwR8R+Ts3wxYa/cJHg= github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=