2021-10-17 17:15:44 +00:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go-v2/aws"
|
|
|
|
awshttp "github.com/aws/aws-sdk-go-v2/aws/transport/http"
|
2022-06-09 10:30:53 +00:00
|
|
|
"github.com/aws/aws-sdk-go-v2/feature/ec2/imds"
|
2021-10-17 17:15:44 +00:00
|
|
|
"github.com/aws/smithy-go/logging"
|
|
|
|
)
|
|
|
|
|
|
|
|
// resolveDefaultAWSConfig will write default configuration values into the cfg
|
|
|
|
// value. It will write the default values, overwriting any previous value.
|
|
|
|
//
|
|
|
|
// This should be used as the first resolver in the slice of resolvers when
|
|
|
|
// resolving external configuration.
|
|
|
|
func resolveDefaultAWSConfig(ctx context.Context, cfg *aws.Config, cfgs configs) error {
|
2022-06-09 10:30:53 +00:00
|
|
|
var sources []interface{}
|
|
|
|
for _, s := range cfgs {
|
|
|
|
sources = append(sources, s)
|
|
|
|
}
|
|
|
|
|
2021-10-17 17:15:44 +00:00
|
|
|
*cfg = aws.Config{
|
2022-06-09 10:30:53 +00:00
|
|
|
Credentials: aws.AnonymousCredentials{},
|
|
|
|
Logger: logging.NewStandardLogger(os.Stderr),
|
|
|
|
ConfigSources: sources,
|
2021-10-17 17:15:44 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveCustomCABundle extracts the first instance of a custom CA bundle filename
|
|
|
|
// from the external configurations. It will update the HTTP Client's builder
|
|
|
|
// to be configured with the custom CA bundle.
|
|
|
|
//
|
|
|
|
// Config provider used:
|
|
|
|
// * customCABundleProvider
|
|
|
|
func resolveCustomCABundle(ctx context.Context, cfg *aws.Config, cfgs configs) error {
|
|
|
|
pemCerts, found, err := getCustomCABundle(ctx, cfgs)
|
|
|
|
if err != nil {
|
|
|
|
// TODO error handling, What is the best way to handle this?
|
|
|
|
// capture previous errors continue. error out if all errors
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.HTTPClient == nil {
|
|
|
|
cfg.HTTPClient = awshttp.NewBuildableClient()
|
|
|
|
}
|
|
|
|
|
|
|
|
trOpts, ok := cfg.HTTPClient.(*awshttp.BuildableClient)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("unable to add custom RootCAs HTTPClient, "+
|
|
|
|
"has no WithTransportOptions, %T", cfg.HTTPClient)
|
|
|
|
}
|
|
|
|
|
|
|
|
var appendErr error
|
|
|
|
client := trOpts.WithTransportOptions(func(tr *http.Transport) {
|
|
|
|
if tr.TLSClientConfig == nil {
|
|
|
|
tr.TLSClientConfig = &tls.Config{}
|
|
|
|
}
|
|
|
|
if tr.TLSClientConfig.RootCAs == nil {
|
|
|
|
tr.TLSClientConfig.RootCAs = x509.NewCertPool()
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := ioutil.ReadAll(pemCerts)
|
|
|
|
if err != nil {
|
|
|
|
appendErr = fmt.Errorf("failed to read custom CA bundle PEM file")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !tr.TLSClientConfig.RootCAs.AppendCertsFromPEM(b) {
|
|
|
|
appendErr = fmt.Errorf("failed to load custom CA bundle PEM file")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if appendErr != nil {
|
|
|
|
return appendErr
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.HTTPClient = client
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveRegion extracts the first instance of a Region from the configs slice.
|
|
|
|
//
|
|
|
|
// Config providers used:
|
|
|
|
// * regionProvider
|
|
|
|
func resolveRegion(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
v, found, err := getRegion(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
// TODO error handling, What is the best way to handle this?
|
|
|
|
// capture previous errors continue. error out if all errors
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Region = v
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveDefaultRegion extracts the first instance of a default region and sets `aws.Config.Region` to the default
|
|
|
|
// region if region had not been resolved from other sources.
|
|
|
|
func resolveDefaultRegion(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
if len(cfg.Region) > 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
v, found, err := getDefaultRegion(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Region = v
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveHTTPClient extracts the first instance of a HTTPClient and sets `aws.Config.HTTPClient` to the HTTPClient instance
|
|
|
|
// if one has not been resolved from other sources.
|
|
|
|
func resolveHTTPClient(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
c, found, err := getHTTPClient(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.HTTPClient = c
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveAPIOptions extracts the first instance of APIOptions and sets `aws.Config.APIOptions` to the resolved API options
|
|
|
|
// if one has not been resolved from other sources.
|
|
|
|
func resolveAPIOptions(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
o, found, err := getAPIOptions(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.APIOptions = o
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// resolveEndpointResolver extracts the first instance of a EndpointResolverFunc from the config slice
|
|
|
|
// and sets the functions result on the aws.Config.EndpointResolver
|
|
|
|
func resolveEndpointResolver(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
endpointResolver, found, err := getEndpointResolver(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.EndpointResolver = endpointResolver
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-09 10:30:53 +00:00
|
|
|
// resolveEndpointResolver extracts the first instance of a EndpointResolverFunc from the config slice
|
|
|
|
// and sets the functions result on the aws.Config.EndpointResolver
|
|
|
|
func resolveEndpointResolverWithOptions(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
endpointResolver, found, err := getEndpointResolverWithOptions(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.EndpointResolverWithOptions = endpointResolver
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-17 17:15:44 +00:00
|
|
|
func resolveLogger(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
logger, found, err := getLogger(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Logger = logger
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveClientLogMode(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
mode, found, err := getClientLogMode(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.ClientLogMode = mode
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveRetryer(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
retryer, found, err := getRetryer(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-06-09 10:30:53 +00:00
|
|
|
|
|
|
|
if found {
|
|
|
|
cfg.Retryer = retryer
|
2021-10-17 17:15:44 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-09 10:30:53 +00:00
|
|
|
// Only load the retry options if a custom retryer has not be specified.
|
|
|
|
if err = resolveRetryMaxAttempts(ctx, cfg, configs); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return resolveRetryMode(ctx, cfg, configs)
|
2021-10-17 17:15:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func resolveEC2IMDSRegion(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
if len(cfg.Region) > 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
region, found, err := getEC2IMDSRegion(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Region = region
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2022-06-09 10:30:53 +00:00
|
|
|
|
|
|
|
func resolveDefaultsModeOptions(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
defaultsMode, found, err := getDefaultsMode(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
defaultsMode = aws.DefaultsModeLegacy
|
|
|
|
}
|
|
|
|
|
|
|
|
var environment aws.RuntimeEnvironment
|
|
|
|
if defaultsMode == aws.DefaultsModeAuto {
|
|
|
|
envConfig, _, _ := getAWSConfigSources(configs)
|
|
|
|
|
|
|
|
client, found, err := getDefaultsModeIMDSClient(ctx, configs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
client = imds.NewFromConfig(*cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
environment, err = resolveDefaultsModeRuntimeEnvironment(ctx, envConfig, client)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.DefaultsMode = defaultsMode
|
|
|
|
cfg.RuntimeEnvironment = environment
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveRetryMaxAttempts(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
maxAttempts, found, err := getRetryMaxAttempts(ctx, configs)
|
|
|
|
if err != nil || !found {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.RetryMaxAttempts = maxAttempts
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveRetryMode(ctx context.Context, cfg *aws.Config, configs configs) error {
|
|
|
|
retryMode, found, err := getRetryMode(ctx, configs)
|
|
|
|
if err != nil || !found {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.RetryMode = retryMode
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|