mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2024-09-14 20:13:21 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			316 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			316 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
#!/usr/bin/python
 | 
						|
# This file is part of Ansible
 | 
						|
#
 | 
						|
# Ansible is free software: you can redistribute it and/or modify
 | 
						|
# it under the terms of the GNU General Public License as published by
 | 
						|
# the Free Software Foundation, either version 3 of the License, or
 | 
						|
# (at your option) any later version.
 | 
						|
#
 | 
						|
# Ansible is distributed in the hope that it will be useful,
 | 
						|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
# GNU General Public License for more details.
 | 
						|
#
 | 
						|
# You should have received a copy of the GNU General Public License
 | 
						|
# along with Ansible.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
DOCUMENTATION = """
 | 
						|
---
 | 
						|
module: ec2_asg
 | 
						|
short_description: Create or delete AWS Autoscaling Groups
 | 
						|
description:
 | 
						|
  - Can create or delete AWS Autoscaling Groups
 | 
						|
  - Works with the ec2_lc module to manage Launch Configurations
 | 
						|
version_added: "1.6"
 | 
						|
author: Gareth Rushgrove
 | 
						|
options:
 | 
						|
  state:
 | 
						|
    description:
 | 
						|
      - register or deregister the instance
 | 
						|
    required: true
 | 
						|
    choices: ['present', 'absent']
 | 
						|
  name:
 | 
						|
    description:
 | 
						|
      - Unique name for group to be created or deleted
 | 
						|
    required: true
 | 
						|
  load_balancers:
 | 
						|
    description:
 | 
						|
      - List of ELB names to use for the group
 | 
						|
    required: false
 | 
						|
  availability_zones:
 | 
						|
    description:
 | 
						|
      - List of availability zone names in which to create the group.  Defaults to all the availability zones in the region if vpc_zone_identifier is not set.
 | 
						|
    required: false
 | 
						|
  launch_config_name:
 | 
						|
    description:
 | 
						|
      - Name of the Launch configuration to use for the group. See the ec2_lc module for managing these.
 | 
						|
    required: false
 | 
						|
  min_size:
 | 
						|
    description:
 | 
						|
      - Minimum number of instances in group
 | 
						|
    required: false
 | 
						|
  max_size:
 | 
						|
    description:
 | 
						|
      - Maximum number of instances in group
 | 
						|
    required: false
 | 
						|
  desired_capacity:
 | 
						|
    description:
 | 
						|
      - Desired number of instances in group
 | 
						|
    required: false
 | 
						|
  region:
 | 
						|
    description:
 | 
						|
      - The AWS region to use. If not specified then the value of the EC2_REGION environment variable, if any, is used.
 | 
						|
    required: false
 | 
						|
    aliases: ['aws_region', 'ec2_region']
 | 
						|
  vpc_zone_identifier:
 | 
						|
    description:
 | 
						|
      - List of VPC subnets to use
 | 
						|
    required: false
 | 
						|
    default: None
 | 
						|
  tags:
 | 
						|
    description:
 | 
						|
      - List of tag dictionaries to use. Required keys are 'key', 'value'. Optional key is 'propagate_at_launch', which defaults to true.
 | 
						|
    required: false
 | 
						|
    default: None
 | 
						|
    version_added: "1.7"
 | 
						|
  health_check_period:
 | 
						|
    description:
 | 
						|
      - Length of time in seconds after a new EC2 instance comes into service that Auto Scaling starts checking its health.
 | 
						|
    required: false
 | 
						|
    default: 500 seconds
 | 
						|
    version_added: "1.7"
 | 
						|
  health_check_type:
 | 
						|
    description:
 | 
						|
      - The service you want the health status from, Amazon EC2 or Elastic Load Balancer.
 | 
						|
    required: false
 | 
						|
    default: EC2
 | 
						|
    version_added: "1.7"
 | 
						|
    choices: ['EC2', 'ELB']
 | 
						|
extends_documentation_fragment: aws
 | 
						|
"""
 | 
						|
 | 
						|
EXAMPLES = '''
 | 
						|
- ec2_asg:
 | 
						|
    name: special
 | 
						|
    load_balancers: 'lb1,lb2'
 | 
						|
    availability_zones: 'eu-west-1a,eu-west-1b'
 | 
						|
    launch_config_name: 'lc-1'
 | 
						|
    min_size: 1
 | 
						|
    max_size: 10
 | 
						|
    desired_capacity: 5
 | 
						|
    vpc_zone_identifier: 'subnet-abcd1234,subnet-1a2b3c4d'
 | 
						|
    tags:
 | 
						|
      - key: environment
 | 
						|
        value: production
 | 
						|
        propagate_at_launch: no
 | 
						|
 | 
						|
'''
 | 
						|
 | 
						|
import sys
 | 
						|
import time
 | 
						|
 | 
						|
from ansible.module_utils.basic import *
 | 
						|
from ansible.module_utils.ec2 import *
 | 
						|
 | 
						|
try:
 | 
						|
    import boto.ec2.autoscale
 | 
						|
    from boto.ec2.autoscale import AutoScaleConnection, AutoScalingGroup, Tag
 | 
						|
    from boto.exception import BotoServerError
 | 
						|
except ImportError:
 | 
						|
    print "failed=True msg='boto required for this module'"
 | 
						|
    sys.exit(1)
 | 
						|
 | 
						|
ASG_ATTRIBUTES = ('launch_config_name', 'max_size', 'min_size', 'desired_capacity',
 | 
						|
                     'vpc_zone_identifier', 'availability_zones',
 | 
						|
                     'health_check_period', 'health_check_type')
 | 
						|
 | 
						|
def enforce_required_arguments(module):
 | 
						|
    ''' As many arguments are not required for autoscale group deletion
 | 
						|
        they cannot be mandatory arguments for the module, so we enforce
 | 
						|
        them here '''
 | 
						|
    missing_args = []
 | 
						|
    for arg in ('min_size', 'max_size', 'launch_config_name'):
 | 
						|
        if module.params[arg] is None:
 | 
						|
            missing_args.append(arg)
 | 
						|
    if missing_args:
 | 
						|
        module.fail_json(msg="Missing required arguments for autoscaling group create/update: %s" % ",".join(missing_args))
 | 
						|
 | 
						|
 | 
						|
def get_properties(autoscaling_group):
 | 
						|
    properties = dict((attr, getattr(autoscaling_group, attr)) for attr in ASG_ATTRIBUTES)
 | 
						|
    if autoscaling_group.instances:
 | 
						|
        properties['instances'] = [i.instance_id for i in autoscaling_group.instances]
 | 
						|
    properties['load_balancers'] = autoscaling_group.load_balancers
 | 
						|
    return properties
 | 
						|
 | 
						|
 | 
						|
def create_autoscaling_group(connection, module):
 | 
						|
 | 
						|
    group_name = module.params.get('name')
 | 
						|
    load_balancers = module.params['load_balancers']
 | 
						|
    availability_zones = module.params['availability_zones']
 | 
						|
    launch_config_name = module.params.get('launch_config_name')
 | 
						|
    min_size = module.params['min_size']
 | 
						|
    max_size = module.params['max_size']
 | 
						|
    desired_capacity = module.params.get('desired_capacity')
 | 
						|
    vpc_zone_identifier = module.params.get('vpc_zone_identifier')
 | 
						|
    set_tags = module.params.get('tags')
 | 
						|
    health_check_period = module.params.get('health_check_period')
 | 
						|
    health_check_type = module.params.get('health_check_type')
 | 
						|
    
 | 
						|
    as_groups = connection.get_all_groups(names=[group_name])
 | 
						|
 | 
						|
    if not vpc_zone_identifier and not availability_zones:
 | 
						|
        region, ec2_url, aws_connect_params = get_aws_connection_info(module)
 | 
						|
        try:
 | 
						|
            ec2_connection = connect_to_aws(boto.ec2, region, **aws_connect_params)
 | 
						|
        except boto.exception.NoAuthHandlerFound, e:
 | 
						|
            module.fail_json(msg=str(e))
 | 
						|
 | 
						|
    asg_tags = []
 | 
						|
    for tag in set_tags:
 | 
						|
        asg_tags.append(Tag(key=tag.get('key'),
 | 
						|
             value=tag.get('value'),
 | 
						|
             propagate_at_launch=bool(tag.get('propagate_at_launch', True)),
 | 
						|
             resource_id=group_name))
 | 
						|
 | 
						|
    if not as_groups:
 | 
						|
        if not vpc_zone_identifier and not availability_zones:
 | 
						|
            availability_zones = module.params['availability_zones'] = [zone.name for zone in ec2_connection.get_all_zones()]
 | 
						|
        enforce_required_arguments(module)
 | 
						|
        launch_configs = connection.get_all_launch_configurations(names=[launch_config_name])
 | 
						|
        ag = AutoScalingGroup(
 | 
						|
                 group_name=group_name,
 | 
						|
                 load_balancers=load_balancers,
 | 
						|
                 availability_zones=availability_zones,
 | 
						|
                 launch_config=launch_configs[0],
 | 
						|
                 min_size=min_size,
 | 
						|
                 max_size=max_size,
 | 
						|
                 desired_capacity=desired_capacity,
 | 
						|
                 vpc_zone_identifier=vpc_zone_identifier,
 | 
						|
                 connection=connection,
 | 
						|
                 tags=asg_tags,
 | 
						|
                 health_check_period=health_check_period,
 | 
						|
                 health_check_type=health_check_type)
 | 
						|
 | 
						|
        try:
 | 
						|
            connection.create_auto_scaling_group(ag)
 | 
						|
            asg_properties = get_properties(ag)
 | 
						|
            module.exit_json(changed=True, **asg_properties)
 | 
						|
        except BotoServerError, e:
 | 
						|
            module.fail_json(msg=str(e))
 | 
						|
    else:
 | 
						|
        as_group = as_groups[0]
 | 
						|
        changed = False
 | 
						|
        for attr in ASG_ATTRIBUTES:
 | 
						|
            if module.params.get(attr) and getattr(as_group, attr) != module.params.get(attr):
 | 
						|
                changed = True
 | 
						|
                setattr(as_group, attr, module.params.get(attr))
 | 
						|
 | 
						|
        if len(set_tags) > 0:
 | 
						|
            existing_tags = as_group.tags
 | 
						|
            existing_tag_map = dict((tag.key, tag) for tag in existing_tags)
 | 
						|
            for tag in set_tags:
 | 
						|
                if 'key' not in tag:
 | 
						|
                    continue
 | 
						|
                if ( not tag['key'] in existing_tag_map or
 | 
						|
                     existing_tag_map[tag['key']].value != tag['value'] or
 | 
						|
                     ('propagate_at_launch' in tag and
 | 
						|
                     existing_tag_map[tag['key']].propagate_at_launch != tag['propagate_at_launch']) ):
 | 
						|
 | 
						|
                    changed = True
 | 
						|
                    continue
 | 
						|
            if changed:
 | 
						|
                connection.create_or_update_tags(asg_tags)
 | 
						|
 | 
						|
        # handle loadbalancers separately because None != []
 | 
						|
        load_balancers = module.params.get('load_balancers') or []
 | 
						|
        if load_balancers and as_group.load_balancers != load_balancers:
 | 
						|
            changed = True
 | 
						|
            as_group.load_balancers = module.params.get('load_balancers')
 | 
						|
 | 
						|
        try:
 | 
						|
            if changed:
 | 
						|
                as_group.update()
 | 
						|
            asg_properties = get_properties(as_group)
 | 
						|
            module.exit_json(changed=changed, **asg_properties)
 | 
						|
        except BotoServerError, e:
 | 
						|
            module.fail_json(msg=str(e))
 | 
						|
 | 
						|
    result = as_groups[0]
 | 
						|
    module.exit_json(changed=changed, name=result.name,
 | 
						|
        autoscaling_group_arn=result.autoscaling_group_arn,
 | 
						|
        availability_zones=result.availability_zones,
 | 
						|
        created_time=str(result.created_time),
 | 
						|
        default_cooldown=result.default_cooldown,
 | 
						|
        health_check_period=result.health_check_period,
 | 
						|
        health_check_type=result.health_check_type,
 | 
						|
        instance_id=result.instance_id,
 | 
						|
        instances=[instance.instance_id for instance in result.instances],
 | 
						|
        launch_config_name=result.launch_config_name,
 | 
						|
        load_balancers=result.load_balancers,
 | 
						|
        min_size=result.min_size, max_size=result.max_size,
 | 
						|
        placement_group=result.placement_group,
 | 
						|
        tags=result.tags,
 | 
						|
        termination_policies=result.termination_policies,
 | 
						|
        vpc_zone_identifier=result.vpc_zone_identifier)
 | 
						|
 | 
						|
 | 
						|
def delete_autoscaling_group(connection, module):
 | 
						|
    group_name = module.params.get('name')
 | 
						|
    groups = connection.get_all_groups(names=[group_name])
 | 
						|
    if groups:
 | 
						|
        group = groups[0]
 | 
						|
        group.shutdown_instances()
 | 
						|
 | 
						|
        instances = True
 | 
						|
        while instances:
 | 
						|
            groups = connection.get_all_groups()
 | 
						|
            for group in groups:
 | 
						|
                if group.name == group_name:
 | 
						|
                    if not group.instances:
 | 
						|
                        instances = False
 | 
						|
            time.sleep(10)
 | 
						|
 | 
						|
        group.delete()
 | 
						|
        module.exit_json(changed=True)
 | 
						|
    else:
 | 
						|
        module.exit_json(changed=False)
 | 
						|
 | 
						|
 | 
						|
def main():
 | 
						|
    argument_spec = ec2_argument_spec()
 | 
						|
    argument_spec.update(
 | 
						|
        dict(
 | 
						|
            name=dict(required=True, type='str'),
 | 
						|
            load_balancers=dict(type='list'),
 | 
						|
            availability_zones=dict(type='list'),
 | 
						|
            launch_config_name=dict(type='str'),
 | 
						|
            min_size=dict(type='int'),
 | 
						|
            max_size=dict(type='int'),
 | 
						|
            desired_capacity=dict(type='int'),
 | 
						|
            vpc_zone_identifier=dict(type='str'),
 | 
						|
            state=dict(default='present', choices=['present', 'absent']),
 | 
						|
            tags=dict(type='list', default=[]),
 | 
						|
            health_check_period=dict(type='int', default=300),
 | 
						|
            health_check_type=dict(default='EC2', choices=['EC2', 'ELB']),
 | 
						|
        )
 | 
						|
    )
 | 
						|
    module = AnsibleModule(argument_spec=argument_spec)
 | 
						|
 | 
						|
    state = module.params.get('state')
 | 
						|
 | 
						|
    region, ec2_url, aws_connect_params = get_aws_connection_info(module)
 | 
						|
    try:
 | 
						|
        connection = connect_to_aws(boto.ec2.autoscale, region, **aws_connect_params)
 | 
						|
        if not connection:
 | 
						|
            module.fail_json(msg="failed to connect to AWS for the given region: %s" % str(region))
 | 
						|
    except boto.exception.NoAuthHandlerFound, e:
 | 
						|
        module.fail_json(msg=str(e))
 | 
						|
 | 
						|
    if state == 'present':
 | 
						|
        create_autoscaling_group(connection, module)
 | 
						|
    elif state == 'absent':
 | 
						|
        delete_autoscaling_group(connection, module)
 | 
						|
 | 
						|
main()
 |