2014-06-24 04:01:57 +02:00
|
|
|
package aws
|
|
|
|
|
|
|
|
import (
|
2014-07-30 00:22:37 +02:00
|
|
|
"fmt"
|
2014-11-21 17:58:34 +01:00
|
|
|
"log"
|
2015-12-10 22:43:13 +01:00
|
|
|
"net/http"
|
2015-04-20 00:54:42 +02:00
|
|
|
"strings"
|
2014-06-24 04:01:57 +02:00
|
|
|
|
2015-10-22 20:03:25 +02:00
|
|
|
"github.com/hashicorp/go-cleanhttp"
|
2015-09-28 03:58:48 +02:00
|
|
|
"github.com/hashicorp/go-multierror"
|
2016-03-14 17:17:05 +01:00
|
|
|
"github.com/hashicorp/terraform/helper/logging"
|
2016-03-14 17:06:22 +01:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
2015-02-12 17:48:48 +01:00
|
|
|
|
2016-02-07 22:40:51 +01:00
|
|
|
"crypto/tls"
|
|
|
|
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
2015-07-14 23:39:50 +02:00
|
|
|
"github.com/aws/aws-sdk-go/aws/awserr"
|
2016-03-14 17:06:22 +01:00
|
|
|
"github.com/aws/aws-sdk-go/aws/request"
|
2015-10-30 00:52:10 +01:00
|
|
|
"github.com/aws/aws-sdk-go/aws/session"
|
2016-03-05 23:12:19 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/apigateway"
|
2016-07-15 13:54:36 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/applicationautoscaling"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/autoscaling"
|
2015-07-07 09:00:05 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudformation"
|
2016-04-14 21:55:11 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudfront"
|
2015-08-28 00:11:56 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudtrail"
|
2015-06-07 11:23:32 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudwatch"
|
2016-01-29 20:13:52 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudwatchevents"
|
2015-09-16 23:02:28 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/cloudwatchlogs"
|
2015-10-30 22:35:16 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/codecommit"
|
2015-07-19 06:09:00 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/codedeploy"
|
2015-09-13 18:57:58 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/directoryservice"
|
2015-06-04 02:12:41 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/dynamodb"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/ec2"
|
2015-12-22 16:31:30 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/ecr"
|
2015-05-04 23:48:09 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/ecs"
|
2015-06-02 22:19:50 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/efs"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/elasticache"
|
2016-03-07 21:42:30 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/elasticbeanstalk"
|
2015-10-02 00:12:46 +02:00
|
|
|
elasticsearch "github.com/aws/aws-sdk-go/service/elasticsearchservice"
|
2016-05-27 00:29:42 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/elastictranscoder"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/elb"
|
2016-05-05 12:34:18 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/emr"
|
2015-11-09 23:26:55 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/firehose"
|
2015-09-03 23:57:56 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/glacier"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/iam"
|
|
|
|
"github.com/aws/aws-sdk-go/service/kinesis"
|
2015-09-09 17:58:24 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/kms"
|
2015-06-09 21:12:47 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/lambda"
|
2015-05-11 05:20:17 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/opsworks"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/rds"
|
2015-11-11 21:51:46 +01:00
|
|
|
"github.com/aws/aws-sdk-go/service/redshift"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/route53"
|
|
|
|
"github.com/aws/aws-sdk-go/service/s3"
|
2016-06-26 23:07:14 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/ses"
|
2016-07-12 13:55:58 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/simpledb"
|
2015-06-03 20:36:57 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/sns"
|
|
|
|
"github.com/aws/aws-sdk-go/service/sqs"
|
2016-04-28 03:49:42 +02:00
|
|
|
"github.com/aws/aws-sdk-go/service/sts"
|
2014-06-24 04:01:57 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type Config struct {
|
2016-01-11 16:22:09 +01:00
|
|
|
AccessKey string
|
|
|
|
SecretKey string
|
|
|
|
CredsFilename string
|
|
|
|
Profile string
|
|
|
|
Token string
|
|
|
|
Region string
|
|
|
|
MaxRetries int
|
2015-04-20 00:54:42 +02:00
|
|
|
|
|
|
|
AllowedAccountIds []interface{}
|
|
|
|
ForbiddenAccountIds []interface{}
|
2015-07-22 23:57:29 +02:00
|
|
|
|
2016-08-10 16:10:34 +02:00
|
|
|
DynamoDBEndpoint string
|
|
|
|
KinesisEndpoint string
|
|
|
|
Ec2Endpoint string
|
|
|
|
IamEndpoint string
|
|
|
|
ElbEndpoint string
|
|
|
|
S3Endpoint string
|
|
|
|
Insecure bool
|
|
|
|
SkipIamCredsValidation bool
|
|
|
|
SkipIamAccountId bool
|
|
|
|
SkipMetadataApiCheck bool
|
2014-11-21 17:58:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type AWSClient struct {
|
2016-05-27 00:29:42 +02:00
|
|
|
cfconn *cloudformation.CloudFormation
|
|
|
|
cloudfrontconn *cloudfront.CloudFront
|
|
|
|
cloudtrailconn *cloudtrail.CloudTrail
|
|
|
|
cloudwatchconn *cloudwatch.CloudWatch
|
|
|
|
cloudwatchlogsconn *cloudwatchlogs.CloudWatchLogs
|
|
|
|
cloudwatcheventsconn *cloudwatchevents.CloudWatchEvents
|
|
|
|
dsconn *directoryservice.DirectoryService
|
|
|
|
dynamodbconn *dynamodb.DynamoDB
|
|
|
|
ec2conn *ec2.EC2
|
|
|
|
ecrconn *ecr.ECR
|
|
|
|
ecsconn *ecs.ECS
|
|
|
|
efsconn *efs.EFS
|
|
|
|
elbconn *elb.ELB
|
|
|
|
emrconn *emr.EMR
|
|
|
|
esconn *elasticsearch.ElasticsearchService
|
|
|
|
apigateway *apigateway.APIGateway
|
2016-07-15 13:54:36 +02:00
|
|
|
appautoscalingconn *applicationautoscaling.ApplicationAutoScaling
|
2016-05-27 00:29:42 +02:00
|
|
|
autoscalingconn *autoscaling.AutoScaling
|
|
|
|
s3conn *s3.S3
|
2016-06-26 23:07:14 +02:00
|
|
|
sesConn *ses.SES
|
2016-07-12 13:55:58 +02:00
|
|
|
simpledbconn *simpledb.SimpleDB
|
2016-05-27 00:29:42 +02:00
|
|
|
sqsconn *sqs.SQS
|
|
|
|
snsconn *sns.SNS
|
|
|
|
stsconn *sts.STS
|
|
|
|
redshiftconn *redshift.Redshift
|
|
|
|
r53conn *route53.Route53
|
|
|
|
accountid string
|
|
|
|
region string
|
|
|
|
rdsconn *rds.RDS
|
|
|
|
iamconn *iam.IAM
|
|
|
|
kinesisconn *kinesis.Kinesis
|
|
|
|
kmsconn *kms.KMS
|
|
|
|
firehoseconn *firehose.Firehose
|
|
|
|
elasticacheconn *elasticache.ElastiCache
|
|
|
|
elasticbeanstalkconn *elasticbeanstalk.ElasticBeanstalk
|
|
|
|
elastictranscoderconn *elastictranscoder.ElasticTranscoder
|
|
|
|
lambdaconn *lambda.Lambda
|
|
|
|
opsworksconn *opsworks.OpsWorks
|
|
|
|
glacierconn *glacier.Glacier
|
|
|
|
codedeployconn *codedeploy.CodeDeploy
|
|
|
|
codecommitconn *codecommit.CodeCommit
|
2014-11-21 17:58:34 +01:00
|
|
|
}
|
|
|
|
|
2015-07-22 23:57:29 +02:00
|
|
|
// Client configures and returns a fully initialized AWSClient
|
2014-11-21 17:58:34 +01:00
|
|
|
func (c *Config) Client() (interface{}, error) {
|
|
|
|
// Get the auth and region. This can fail if keys/regions were not
|
|
|
|
// specified and we're attempting to use the environment.
|
|
|
|
var errs []error
|
|
|
|
|
|
|
|
log.Println("[INFO] Building AWS region structure")
|
2015-03-13 15:42:50 +01:00
|
|
|
err := c.ValidateRegion()
|
2014-11-21 17:58:34 +01:00
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, err)
|
|
|
|
}
|
|
|
|
|
2016-03-05 23:12:19 +01:00
|
|
|
var client AWSClient
|
2014-11-21 17:58:34 +01:00
|
|
|
if len(errs) == 0 {
|
2015-02-19 22:38:56 +01:00
|
|
|
// store AWS region in client struct, for region specific operations such as
|
|
|
|
// bucket storage in S3
|
|
|
|
client.region = c.Region
|
2015-02-20 15:55:54 +01:00
|
|
|
|
2015-03-13 15:42:50 +01:00
|
|
|
log.Println("[INFO] Building AWS auth structure")
|
2016-08-10 16:10:34 +02:00
|
|
|
creds := GetCredentials(c)
|
2015-12-10 22:43:13 +01:00
|
|
|
// Call Get to check for credential provider. If nothing found, we'll get an
|
|
|
|
// error, and we can present it nicely to the user
|
2015-12-12 23:58:19 +01:00
|
|
|
cp, err := creds.Get()
|
2015-12-10 22:43:13 +01:00
|
|
|
if err != nil {
|
2016-01-27 23:30:03 +01:00
|
|
|
if awsErr, ok := err.(awserr.Error); ok && awsErr.Code() == "NoCredentialProviders" {
|
2016-04-14 21:55:11 +02:00
|
|
|
errs = append(errs, fmt.Errorf(`No valid credential sources found for AWS Provider.
|
|
|
|
Please see https://terraform.io/docs/providers/aws/index.html for more information on
|
2016-01-27 23:30:03 +01:00
|
|
|
providing credentials for the AWS Provider`))
|
|
|
|
} else {
|
|
|
|
errs = append(errs, fmt.Errorf("Error loading credentials for AWS Provider: %s", err))
|
|
|
|
}
|
2015-12-10 22:43:13 +01:00
|
|
|
return nil, &multierror.Error{Errors: errs}
|
|
|
|
}
|
2015-12-12 23:58:19 +01:00
|
|
|
|
|
|
|
log.Printf("[INFO] AWS Auth provider used: %q", cp.ProviderName)
|
|
|
|
|
2015-04-16 22:02:04 +02:00
|
|
|
awsConfig := &aws.Config{
|
|
|
|
Credentials: creds,
|
2015-07-28 22:29:46 +02:00
|
|
|
Region: aws.String(c.Region),
|
|
|
|
MaxRetries: aws.Int(c.MaxRetries),
|
2015-10-22 20:03:25 +02:00
|
|
|
HTTPClient: cleanhttp.DefaultClient(),
|
2015-04-13 21:01:41 +02:00
|
|
|
}
|
|
|
|
|
2016-03-14 17:17:05 +01:00
|
|
|
if logging.IsDebugOrHigher() {
|
|
|
|
awsConfig.LogLevel = aws.LogLevel(aws.LogDebugWithHTTPBody)
|
|
|
|
awsConfig.Logger = awsLogger{}
|
|
|
|
}
|
|
|
|
|
2016-02-07 22:40:51 +01:00
|
|
|
if c.Insecure {
|
|
|
|
transport := awsConfig.HTTPClient.Transport.(*http.Transport)
|
|
|
|
transport.TLSClientConfig = &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
}
|
|
|
|
}
|
2015-12-11 16:27:49 +01:00
|
|
|
|
2016-03-14 17:06:22 +01:00
|
|
|
// Set up base session
|
2015-10-30 00:52:10 +01:00
|
|
|
sess := session.New(awsConfig)
|
2016-03-14 17:06:22 +01:00
|
|
|
sess.Handlers.Build.PushFrontNamed(addTerraformVersionToUserAgent)
|
2015-12-11 16:27:49 +01:00
|
|
|
|
2016-07-21 00:49:57 +02:00
|
|
|
// Some services exist only in us-east-1, e.g. because they manage
|
|
|
|
// resources that can span across multiple regions, or because
|
|
|
|
// signature format v4 requires region to be us-east-1 for global
|
|
|
|
// endpoints:
|
|
|
|
// http://docs.aws.amazon.com/general/latest/gr/sigv4_changes.html
|
|
|
|
usEast1Sess := sess.Copy(&aws.Config{Region: aws.String("us-east-1")})
|
|
|
|
|
|
|
|
// Some services have user-configurable endpoints
|
|
|
|
awsEc2Sess := sess.Copy(&aws.Config{Endpoint: aws.String(c.Ec2Endpoint)})
|
|
|
|
awsElbSess := sess.Copy(&aws.Config{Endpoint: aws.String(c.ElbEndpoint)})
|
2016-03-14 17:40:36 +01:00
|
|
|
awsIamSess := sess.Copy(&aws.Config{Endpoint: aws.String(c.IamEndpoint)})
|
2016-07-21 00:49:57 +02:00
|
|
|
dynamoSess := sess.Copy(&aws.Config{Endpoint: aws.String(c.DynamoDBEndpoint)})
|
|
|
|
kinesisSess := sess.Copy(&aws.Config{Endpoint: aws.String(c.KinesisEndpoint)})
|
2015-07-14 23:39:50 +02:00
|
|
|
|
2016-07-21 01:38:14 +02:00
|
|
|
// These two services need to be set up early so we can check on AccountID
|
|
|
|
client.iamconn = iam.New(awsIamSess)
|
2016-04-28 03:49:42 +02:00
|
|
|
client.stsconn = sts.New(sess)
|
2016-07-21 01:38:14 +02:00
|
|
|
|
2016-08-10 16:10:34 +02:00
|
|
|
if c.SkipIamCredsValidation == false {
|
|
|
|
err = c.ValidateCredentials(client.stsconn)
|
|
|
|
if err != nil {
|
|
|
|
errs = append(errs, err)
|
|
|
|
return nil, &multierror.Error{Errors: errs}
|
|
|
|
}
|
2016-04-28 03:49:42 +02:00
|
|
|
}
|
|
|
|
|
2016-08-10 16:10:34 +02:00
|
|
|
if c.SkipIamAccountId == false {
|
|
|
|
accountId, err := GetAccountId(client.iamconn, client.stsconn, cp.ProviderName)
|
|
|
|
if err == nil {
|
|
|
|
client.accountid = accountId
|
|
|
|
}
|
|
|
|
|
|
|
|
authErr := c.ValidateAccountId(client.accountid)
|
|
|
|
if authErr != nil {
|
|
|
|
errs = append(errs, authErr)
|
|
|
|
}
|
2015-04-20 00:54:42 +02:00
|
|
|
}
|
|
|
|
|
2016-07-21 00:49:57 +02:00
|
|
|
client.apigateway = apigateway.New(sess)
|
2016-07-15 13:54:36 +02:00
|
|
|
client.appautoscalingconn = applicationautoscaling.New(sess)
|
2015-10-30 00:52:10 +01:00
|
|
|
client.autoscalingconn = autoscaling.New(sess)
|
2016-07-21 00:49:57 +02:00
|
|
|
client.cfconn = cloudformation.New(sess)
|
|
|
|
client.cloudfrontconn = cloudfront.New(sess)
|
|
|
|
client.cloudtrailconn = cloudtrail.New(sess)
|
|
|
|
client.cloudwatchconn = cloudwatch.New(sess)
|
|
|
|
client.cloudwatcheventsconn = cloudwatchevents.New(sess)
|
|
|
|
client.cloudwatchlogsconn = cloudwatchlogs.New(sess)
|
|
|
|
client.codecommitconn = codecommit.New(usEast1Sess)
|
|
|
|
client.codedeployconn = codedeploy.New(sess)
|
|
|
|
client.dsconn = directoryservice.New(sess)
|
|
|
|
client.dynamodbconn = dynamodb.New(dynamoSess)
|
2016-02-07 22:40:51 +01:00
|
|
|
client.ec2conn = ec2.New(awsEc2Sess)
|
2015-12-22 16:31:30 +01:00
|
|
|
client.ecrconn = ecr.New(sess)
|
2015-10-30 00:52:10 +01:00
|
|
|
client.ecsconn = ecs.New(sess)
|
|
|
|
client.efsconn = efs.New(sess)
|
|
|
|
client.elasticacheconn = elasticache.New(sess)
|
2016-07-21 00:49:57 +02:00
|
|
|
client.elasticbeanstalkconn = elasticbeanstalk.New(sess)
|
|
|
|
client.elastictranscoderconn = elastictranscoder.New(sess)
|
|
|
|
client.elbconn = elb.New(awsElbSess)
|
|
|
|
client.emrconn = emr.New(sess)
|
|
|
|
client.esconn = elasticsearch.New(sess)
|
|
|
|
client.firehoseconn = firehose.New(sess)
|
|
|
|
client.glacierconn = glacier.New(sess)
|
|
|
|
client.kinesisconn = kinesis.New(kinesisSess)
|
|
|
|
client.kmsconn = kms.New(sess)
|
2015-10-30 00:52:10 +01:00
|
|
|
client.lambdaconn = lambda.New(sess)
|
|
|
|
client.opsworksconn = opsworks.New(usEast1Sess)
|
2016-07-21 00:49:57 +02:00
|
|
|
client.r53conn = route53.New(usEast1Sess)
|
|
|
|
client.rdsconn = rds.New(sess)
|
2015-11-11 21:51:46 +01:00
|
|
|
client.redshiftconn = redshift.New(sess)
|
2016-07-22 17:32:51 +02:00
|
|
|
client.simpledbconn = simpledb.New(sess)
|
2016-07-21 00:49:57 +02:00
|
|
|
client.s3conn = s3.New(sess)
|
|
|
|
client.sesConn = ses.New(sess)
|
|
|
|
client.snsconn = sns.New(sess)
|
|
|
|
client.sqsconn = sqs.New(sess)
|
2014-11-21 17:58:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(errs) > 0 {
|
|
|
|
return nil, &multierror.Error{Errors: errs}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &client, nil
|
2014-06-24 04:01:57 +02:00
|
|
|
}
|
|
|
|
|
2015-04-27 20:06:49 +02:00
|
|
|
// ValidateRegion returns an error if the configured region is not a
|
|
|
|
// valid aws region and nil otherwise.
|
2015-03-13 15:42:50 +01:00
|
|
|
func (c *Config) ValidateRegion() error {
|
2016-06-28 13:18:36 +02:00
|
|
|
var regions = [13]string{
|
|
|
|
"ap-northeast-1",
|
|
|
|
"ap-northeast-2",
|
|
|
|
"ap-south-1",
|
|
|
|
"ap-southeast-1",
|
|
|
|
"ap-southeast-2",
|
|
|
|
"cn-north-1",
|
|
|
|
"eu-central-1",
|
|
|
|
"eu-west-1",
|
|
|
|
"sa-east-1",
|
|
|
|
"us-east-1",
|
|
|
|
"us-gov-west-1",
|
|
|
|
"us-west-1",
|
|
|
|
"us-west-2",
|
|
|
|
}
|
2014-07-30 00:22:37 +02:00
|
|
|
|
|
|
|
for _, valid := range regions {
|
|
|
|
if c.Region == valid {
|
2015-03-13 15:42:50 +01:00
|
|
|
return nil
|
2014-07-30 00:22:37 +02:00
|
|
|
}
|
|
|
|
}
|
2015-03-13 15:42:50 +01:00
|
|
|
return fmt.Errorf("Not a valid region: %s", c.Region)
|
2014-06-24 04:01:57 +02:00
|
|
|
}
|
2015-04-20 00:54:42 +02:00
|
|
|
|
2015-08-07 16:49:59 +02:00
|
|
|
// Validate credentials early and fail before we do any graph walking.
|
2016-06-21 00:14:07 +02:00
|
|
|
func (c *Config) ValidateCredentials(stsconn *sts.STS) error {
|
|
|
|
_, err := stsconn.GetCallerIdentity(&sts.GetCallerIdentityInput{})
|
2015-07-21 15:57:59 +02:00
|
|
|
return err
|
2015-07-14 23:39:50 +02:00
|
|
|
}
|
|
|
|
|
2015-04-27 20:06:49 +02:00
|
|
|
// ValidateAccountId returns a context-specific error if the configured account
|
|
|
|
// id is explicitly forbidden or not authorised; and nil if it is authorised.
|
2016-04-28 03:49:42 +02:00
|
|
|
func (c *Config) ValidateAccountId(accountId string) error {
|
2015-04-20 00:54:42 +02:00
|
|
|
if c.AllowedAccountIds == nil && c.ForbiddenAccountIds == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("[INFO] Validating account ID")
|
|
|
|
|
|
|
|
if c.ForbiddenAccountIds != nil {
|
|
|
|
for _, id := range c.ForbiddenAccountIds {
|
2016-04-28 03:49:42 +02:00
|
|
|
if id == accountId {
|
2015-04-20 00:54:42 +02:00
|
|
|
return fmt.Errorf("Forbidden account ID (%s)", id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.AllowedAccountIds != nil {
|
|
|
|
for _, id := range c.AllowedAccountIds {
|
2016-04-28 03:49:42 +02:00
|
|
|
if id == accountId {
|
2015-04-20 00:54:42 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2016-04-28 03:49:42 +02:00
|
|
|
return fmt.Errorf("Account ID not allowed (%s)", accountId)
|
2015-04-20 00:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2015-12-10 22:43:13 +01:00
|
|
|
|
2016-03-14 17:06:22 +01:00
|
|
|
// addTerraformVersionToUserAgent is a named handler that will add Terraform's
|
|
|
|
// version information to requests made by the AWS SDK.
|
|
|
|
var addTerraformVersionToUserAgent = request.NamedHandler{
|
|
|
|
Name: "terraform.TerraformVersionUserAgentHandler",
|
|
|
|
Fn: request.MakeAddToUserAgentHandler(
|
2016-07-21 20:07:16 +02:00
|
|
|
"terraform", terraform.VersionString()),
|
2016-03-14 17:06:22 +01:00
|
|
|
}
|
2016-03-14 17:17:05 +01:00
|
|
|
|
|
|
|
type awsLogger struct{}
|
|
|
|
|
|
|
|
func (l awsLogger) Log(args ...interface{}) {
|
|
|
|
tokens := make([]string, 0, len(args))
|
|
|
|
for _, arg := range args {
|
|
|
|
if token, ok := arg.(string); ok {
|
|
|
|
tokens = append(tokens, token)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
log.Printf("[DEBUG] [aws-sdk-go] %s", strings.Join(tokens, " "))
|
|
|
|
}
|