Merge pull request #2421 from robparrott/ec2-idempotency
Make ec2 module idempotent with Amazon EC2 via client-token attribute.
This commit is contained in:
commit
24f619030f
1 changed files with 69 additions and 39 deletions
108
ec2
108
ec2
|
@ -1,4 +1,4 @@
|
||||||
#!/usr/bin/python -tt
|
#!/usr/local/bin/python -tt
|
||||||
# This file is part of Ansible
|
# This file is part of Ansible
|
||||||
#
|
#
|
||||||
# Ansible is free software: you can redistribute it and/or modify
|
# Ansible is free software: you can redistribute it and/or modify
|
||||||
|
@ -28,6 +28,12 @@ options:
|
||||||
required: true
|
required: true
|
||||||
default: null
|
default: null
|
||||||
aliases: ['keypair']
|
aliases: ['keypair']
|
||||||
|
id:
|
||||||
|
description:
|
||||||
|
- identifier for this instance or set of instances, so that the module will be idempotent with respect to EC2 instances.
|
||||||
|
required: false
|
||||||
|
default: null
|
||||||
|
aliases: []
|
||||||
group:
|
group:
|
||||||
description:
|
description:
|
||||||
- security group to use with the instance
|
- security group to use with the instance
|
||||||
|
@ -149,6 +155,7 @@ def main():
|
||||||
module = AnsibleModule(
|
module = AnsibleModule(
|
||||||
argument_spec = dict(
|
argument_spec = dict(
|
||||||
key_name = dict(required=True, aliases = ['keypair']),
|
key_name = dict(required=True, aliases = ['keypair']),
|
||||||
|
id = dict(),
|
||||||
group = dict(),
|
group = dict(),
|
||||||
group_id = dict(),
|
group_id = dict(),
|
||||||
instance_type = dict(aliases=['type']),
|
instance_type = dict(aliases=['type']),
|
||||||
|
@ -169,6 +176,7 @@ def main():
|
||||||
)
|
)
|
||||||
|
|
||||||
key_name = module.params.get('key_name')
|
key_name = module.params.get('key_name')
|
||||||
|
id = module.params.get('id')
|
||||||
group_name = module.params.get('group')
|
group_name = module.params.get('group')
|
||||||
group_id = module.params.get('group_id')
|
group_id = module.params.get('group_id')
|
||||||
instance_type = module.params.get('instance_type')
|
instance_type = module.params.get('instance_type')
|
||||||
|
@ -212,12 +220,30 @@ def main():
|
||||||
except boto.exception.NoAuthHandlerFound, e:
|
except boto.exception.NoAuthHandlerFound, e:
|
||||||
module.fail_json(msg = str(e))
|
module.fail_json(msg = str(e))
|
||||||
|
|
||||||
|
# Lookup any instances that much our run id.
|
||||||
|
|
||||||
|
running_instances = []
|
||||||
|
count_remaining = int(count)
|
||||||
|
|
||||||
|
if id != None:
|
||||||
|
filter_dict = {'client-token':id, 'instance-state-name' : 'running'}
|
||||||
|
previous_reservations = ec2.get_all_instances(None, filter_dict )
|
||||||
|
for res in previous_reservations:
|
||||||
|
for prev_instance in res.instances:
|
||||||
|
running_instances.append(prev_instance)
|
||||||
|
count_remaining = count_remaining - len(running_instances)
|
||||||
|
# module.fail_json(msg = "known running instances: %s" % (running_instances))
|
||||||
|
|
||||||
|
|
||||||
# Both min_count and max_count equal count parameter. This means the launch request is explicit (we want count, or fail) in how many instances we want.
|
# Both min_count and max_count equal count parameter. This means the launch request is explicit (we want count, or fail) in how many instances we want.
|
||||||
|
|
||||||
try:
|
|
||||||
res = ec2.run_instances(image, key_name = key_name,
|
if count_remaining > 0:
|
||||||
min_count = count,
|
try:
|
||||||
max_count = count,
|
res = ec2.run_instances(image, key_name = key_name,
|
||||||
|
client_token=id,
|
||||||
|
min_count = count_remaining,
|
||||||
|
max_count = count_remaining,
|
||||||
monitoring_enabled = monitoring,
|
monitoring_enabled = monitoring,
|
||||||
security_groups = [group_name],
|
security_groups = [group_name],
|
||||||
instance_type = instance_type,
|
instance_type = instance_type,
|
||||||
|
@ -225,50 +251,54 @@ def main():
|
||||||
ramdisk_id = ramdisk,
|
ramdisk_id = ramdisk,
|
||||||
subnet_id = vpc_subnet_id,
|
subnet_id = vpc_subnet_id,
|
||||||
user_data = user_data)
|
user_data = user_data)
|
||||||
except boto.exception.BotoServerError, e:
|
except boto.exception.BotoServerError, e:
|
||||||
module.fail_json(msg = "%s: %s" % (e.error_code, e.error_message))
|
|
||||||
|
|
||||||
instids = [ i.id for i in res.instances ]
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
res.connection.get_all_instances(instids)
|
|
||||||
break
|
|
||||||
except boto.exception.EC2ResponseError as e:
|
|
||||||
if "<Code>InvalidInstanceID.NotFound</Code>" in str(e):
|
|
||||||
# there's a race between start and get an instance
|
|
||||||
continue
|
|
||||||
else:
|
|
||||||
module.fail_json(msg = str(e))
|
|
||||||
|
|
||||||
if instance_tags:
|
|
||||||
try:
|
|
||||||
ec2.create_tags(instids, module.from_json(instance_tags))
|
|
||||||
except boto.exception.EC2ResponseError as e:
|
|
||||||
module.fail_json(msg = "%s: %s" % (e.error_code, e.error_message))
|
module.fail_json(msg = "%s: %s" % (e.error_code, e.error_message))
|
||||||
|
|
||||||
# wait here until the instances are up
|
instids = [ i.id for i in res.instances ]
|
||||||
res_list = res.connection.get_all_instances(instids)
|
while True:
|
||||||
this_res = res_list[0]
|
try:
|
||||||
num_running = 0
|
res.connection.get_all_instances(instids)
|
||||||
wait_timeout = time.time() + wait_timeout
|
break
|
||||||
while wait and wait_timeout > time.time() and num_running < len(instids):
|
except boto.exception.EC2ResponseError as e:
|
||||||
|
if "<Code>InvalidInstanceID.NotFound</Code>" in str(e):
|
||||||
|
# there's a race between start and get an instance
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
module.fail_json(msg = str(e))
|
||||||
|
|
||||||
|
if instance_tags:
|
||||||
|
try:
|
||||||
|
ec2.create_tags(instids, module.from_json(instance_tags))
|
||||||
|
except boto.exception.EC2ResponseError as e:
|
||||||
|
module.fail_json(msg = "%s: %s" % (e.error_code, e.error_message))
|
||||||
|
|
||||||
|
# wait here until the instances are up
|
||||||
res_list = res.connection.get_all_instances(instids)
|
res_list = res.connection.get_all_instances(instids)
|
||||||
this_res = res_list[0]
|
this_res = res_list[0]
|
||||||
num_running = len([ i for i in this_res.instances if i.state=='running' ])
|
num_running = 0
|
||||||
time.sleep(5)
|
wait_timeout = time.time() + wait_timeout
|
||||||
if wait and wait_timeout <= time.time():
|
while wait and wait_timeout > time.time() and num_running < len(instids):
|
||||||
# waiting took too long
|
res_list = res.connection.get_all_instances(instids)
|
||||||
module.fail_json(msg = "wait for instances running timeout on %s" % time.asctime())
|
this_res = res_list[0]
|
||||||
instances = []
|
num_running = len([ i for i in this_res.instances if i.state=='running' ])
|
||||||
for inst in this_res.instances:
|
time.sleep(5)
|
||||||
|
if wait and wait_timeout <= time.time():
|
||||||
|
# waiting took too long
|
||||||
|
module.fail_json(msg = "wait for instances running timeout on %s" % time.asctime())
|
||||||
|
|
||||||
|
for inst in this_res.instances:
|
||||||
|
running_instances.append(inst)
|
||||||
|
|
||||||
|
instance_dict_array = []
|
||||||
|
for inst in running_instances:
|
||||||
d = {
|
d = {
|
||||||
'id': inst.id,
|
'id': inst.id,
|
||||||
'public_ip': inst.ip_address,
|
'public_ip': inst.ip_address,
|
||||||
'public_dns_name': inst.public_dns_name
|
'public_dns_name': inst.public_dns_name
|
||||||
}
|
}
|
||||||
instances.append(d)
|
instance_dict_array.append(d)
|
||||||
|
|
||||||
module.exit_json(changed=True, instances=instances)
|
module.exit_json(changed=True, instances=instance_dict_array)
|
||||||
|
|
||||||
# this is magic, see lib/ansible/module_common.py
|
# this is magic, see lib/ansible/module_common.py
|
||||||
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
||||||
|
|
Loading…
Reference in a new issue