188 lines
6.4 KiB
Go
188 lines
6.4 KiB
Go
/*
|
|
Copyright 2016 The Kubernetes Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package aws
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
"github.com/aws/aws-sdk-go/service/autoscaling"
|
|
"github.com/golang/glog"
|
|
)
|
|
|
|
// autoScaling is the interface represents a specific aspect of the auto-scaling service provided by AWS SDK for use in CA
|
|
type autoScaling interface {
|
|
DescribeAutoScalingGroups(input *autoscaling.DescribeAutoScalingGroupsInput) (*autoscaling.DescribeAutoScalingGroupsOutput, error)
|
|
DescribeLaunchConfigurations(*autoscaling.DescribeLaunchConfigurationsInput) (*autoscaling.DescribeLaunchConfigurationsOutput, error)
|
|
DescribeTags(input *autoscaling.DescribeTagsInput) (*autoscaling.DescribeTagsOutput, error)
|
|
SetDesiredCapacity(input *autoscaling.SetDesiredCapacityInput) (*autoscaling.SetDesiredCapacityOutput, error)
|
|
TerminateInstanceInAutoScalingGroup(input *autoscaling.TerminateInstanceInAutoScalingGroupInput) (*autoscaling.TerminateInstanceInAutoScalingGroupOutput, error)
|
|
}
|
|
|
|
// autoScalingWrapper provides several utility methods over the auto-scaling service provided by AWS SDK
|
|
type autoScalingWrapper struct {
|
|
autoScaling
|
|
}
|
|
|
|
func (m autoScalingWrapper) getInstanceTypeByLCName(name string) (string, error) {
|
|
params := &autoscaling.DescribeLaunchConfigurationsInput{
|
|
LaunchConfigurationNames: []*string{aws.String(name)},
|
|
MaxRecords: aws.Int64(1),
|
|
}
|
|
launchConfigurations, err := m.DescribeLaunchConfigurations(params)
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed LaunchConfiguration info request for %s: %v", name, err)
|
|
return "", err
|
|
}
|
|
if len(launchConfigurations.LaunchConfigurations) < 1 {
|
|
return "", fmt.Errorf("Unable to get first LaunchConfiguration for %s", name)
|
|
}
|
|
|
|
return *launchConfigurations.LaunchConfigurations[0].InstanceType, nil
|
|
}
|
|
|
|
func (m autoScalingWrapper) getAutoscalingGroupByName(name string) (*autoscaling.Group, error) {
|
|
params := &autoscaling.DescribeAutoScalingGroupsInput{
|
|
AutoScalingGroupNames: []*string{aws.String(name)},
|
|
MaxRecords: aws.Int64(1),
|
|
}
|
|
groups, err := m.DescribeAutoScalingGroups(params)
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed ASG info request for %s: %v", name, err)
|
|
return nil, err
|
|
}
|
|
if len(groups.AutoScalingGroups) < 1 {
|
|
return nil, fmt.Errorf("Unable to get first autoscaling.Group for %s", name)
|
|
}
|
|
return groups.AutoScalingGroups[0], nil
|
|
}
|
|
|
|
func (m *autoScalingWrapper) getAutoscalingGroupsByNames(names []string) ([]*autoscaling.Group, error) {
|
|
glog.V(6).Infof("Starting getAutoscalingGroupsByNames with names=%v", names)
|
|
|
|
if len(names) < 1 {
|
|
glog.V(4).Info("Failed to describe ASGs: Must specify at least one ASG name.")
|
|
return nil, fmt.Errorf("List of ASG names was empty. Must specify at least one ASG name")
|
|
}
|
|
|
|
nameRefs := []*string{}
|
|
for _, n := range names {
|
|
nameRefs = append(nameRefs, aws.String(n))
|
|
}
|
|
params := &autoscaling.DescribeAutoScalingGroupsInput{
|
|
AutoScalingGroupNames: nameRefs,
|
|
MaxRecords: aws.Int64(maxRecordsReturnedByAPI),
|
|
}
|
|
description, err := m.DescribeAutoScalingGroups(params)
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed to describe ASGs : %v", err)
|
|
return nil, err
|
|
}
|
|
if len(description.AutoScalingGroups) < 1 {
|
|
return nil, errors.New("No ASGs found")
|
|
}
|
|
|
|
asgs := description.AutoScalingGroups
|
|
for description.NextToken != nil {
|
|
description, err = m.DescribeAutoScalingGroups(&autoscaling.DescribeAutoScalingGroupsInput{
|
|
NextToken: description.NextToken,
|
|
MaxRecords: aws.Int64(maxRecordsReturnedByAPI),
|
|
})
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed to describe ASGs : %v", err)
|
|
return nil, err
|
|
}
|
|
asgs = append(asgs, description.AutoScalingGroups...)
|
|
}
|
|
|
|
glog.V(6).Infof("Finishing getAutoscalingGroupsByNames asgs=%v", asgs)
|
|
|
|
return asgs, nil
|
|
}
|
|
|
|
func (m *autoScalingWrapper) getAutoscalingGroupsByTags(keys []string) ([]*autoscaling.Group, error) {
|
|
glog.V(6).Infof("Starting getAutoscalingGroupsByTag with keys=%v", keys)
|
|
|
|
numKeys := len(keys)
|
|
|
|
// DescribeTags does an OR query when multiple filters on different tags are specified.
|
|
// In other words, DescribeTags returns [asg1, asg1] for keys [t1, t2] when there's only one asg tagged both t1 and t2.
|
|
filters := []*autoscaling.Filter{}
|
|
for _, key := range keys {
|
|
filter := &autoscaling.Filter{
|
|
Name: aws.String("key"),
|
|
Values: []*string{aws.String(key)},
|
|
}
|
|
filters = append(filters, filter)
|
|
}
|
|
description, err := m.DescribeTags(&autoscaling.DescribeTagsInput{
|
|
Filters: filters,
|
|
MaxRecords: aws.Int64(maxRecordsReturnedByAPI),
|
|
})
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed to describe ASG tags for keys %v : %v", keys, err)
|
|
return nil, err
|
|
}
|
|
if len(description.Tags) < 1 {
|
|
return nil, fmt.Errorf("Unable to find ASGs for tag keys %v", keys)
|
|
}
|
|
tags := []*autoscaling.TagDescription{}
|
|
tags = append(tags, description.Tags...)
|
|
|
|
for description.NextToken != nil {
|
|
description, err = m.DescribeTags(&autoscaling.DescribeTagsInput{
|
|
NextToken: description.NextToken,
|
|
MaxRecords: aws.Int64(maxRecordsReturnedByAPI),
|
|
})
|
|
if err != nil {
|
|
glog.V(4).Infof("Failed to describe ASG tags for key %v: %v", keys, err)
|
|
return nil, err
|
|
}
|
|
tags = append(tags, description.Tags...)
|
|
}
|
|
|
|
// De-duplicate asg names
|
|
asgNameOccurrences := map[string]int{}
|
|
for _, t := range tags {
|
|
asgName := *(t.ResourceId)
|
|
if n, ok := asgNameOccurrences[asgName]; ok {
|
|
asgNameOccurrences[asgName] = n + 1
|
|
} else {
|
|
asgNameOccurrences[asgName] = 1
|
|
}
|
|
}
|
|
// Accordingly to how DescribeTags API works, the result contains ASGs which not all but only subset of tags are associated.
|
|
// Explicitly select ASGs to which all the tags are associated so that we won't end up calling DescribeAutoScalingGroups API
|
|
// multiple times on an ASG
|
|
asgNames := []string{}
|
|
for asgName, n := range asgNameOccurrences {
|
|
if n == numKeys {
|
|
asgNames = append(asgNames, asgName)
|
|
}
|
|
}
|
|
|
|
asgs, err := m.getAutoscalingGroupsByNames(asgNames)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
glog.V(6).Infof("Finishing getAutoscalingGroupsByTag with asgs=%v", asgs)
|
|
|
|
return asgs, nil
|
|
}
|