diff --git a/cloud/amazon/ec2_elb_lb.py b/cloud/amazon/ec2_elb_lb.py index 4c8fdddd423..d98b27587a2 100644 --- a/cloud/amazon/ec2_elb_lb.py +++ b/cloud/amazon/ec2_elb_lb.py @@ -107,7 +107,6 @@ options: description: - Wait a specified timeout allowing connections to drain before terminating an instance required: false - default: "None" aliases: [] version_added: "1.8" idle_timeout: @@ -128,6 +127,21 @@ options: - An associative array of stickness policy settings. Policy will be applied to all listeners ( see example ) required: false version_added: "2.0" + wait: + description: + - When specified, Ansible will check the status of the load balancer to ensure it has been successfully + removed from AWS. + required: false + default: no + choices: ["yes", "no"] + version_added: 2.0 + wait_timeout: + description: + - Used in conjunction with wait. Number of seconds to wait for the elb to be terminated. + A maximum of 600 seconds (10 minutes) is allowed. + required: false + default: 60 + version_added: 2.0 extends_documentation_fragment: - aws @@ -203,6 +217,21 @@ EXAMPLES = """ name: "test-please-delete" state: absent +# Ensure ELB is gone and wait for check (for default timeout) +- local_action: + module: ec2_elb_lb + name: "test-please-delete" + state: absent + wait: yes + +# Ensure ELB is gone and wait for check with timeout value +- local_action: + module: ec2_elb_lb + name: "test-please-delete" + state: absent + wait: yes + wait_timeout: 600 + # Normally, this module will purge any listeners that exist on the ELB # but aren't specified in the listeners parameter. If purge_listeners is # false it leaves them alone @@ -325,7 +354,7 @@ class ElbManager(object): scheme="internet-facing", connection_draining_timeout=None, idle_timeout=None, cross_az_load_balancing=None, access_logs=None, - stickiness=None, region=None, **aws_connect_params): + stickiness=None, wait=None, wait_timeout=None, region=None, **aws_connect_params): self.module = module self.name = name @@ -343,6 +372,8 @@ class ElbManager(object): self.cross_az_load_balancing = cross_az_load_balancing self.access_logs = access_logs self.stickiness = stickiness + self.wait = wait + self.wait_timeout = wait_timeout self.aws_connect_params = aws_connect_params self.region = region @@ -351,6 +382,7 @@ class ElbManager(object): self.status = 'gone' self.elb_conn = self._get_elb_connection() self.elb = self._get_elb() + self.ec2_conn = self._get_ec2_connection() def ensure_ok(self): """Create the ELB""" @@ -381,6 +413,14 @@ class ElbManager(object): """Destroy the ELB""" if self.elb: self._delete_elb() + if self.wait: + elb_removed = self._wait_for_elb_removed() + # Unfortunately even though the ELB itself is removed quickly + # the interfaces take longer so reliant security groups cannot + # be deleted until the interface has registered as removed. + elb_interface_removed = self._wait_for_elb_interface_removed() + if not (elb_removed and elb_interface_removed): + self.module.fail_json(msg='Timed out waiting for removal of load balancer.') def get_info(self): try: @@ -481,6 +521,50 @@ class ElbManager(object): return info + def _wait_for_elb_removed(self): + polling_increment_secs = 15 + max_retries = (self.wait_timeout / polling_increment_secs) + status_achieved = False + + for x in range(0, max_retries): + try: + result = self.elb_conn.get_all_lb_attributes(self.name) + except (boto.exception.BotoServerError, StandardError), e: + if "LoadBalancerNotFound" in e.code: + status_achieved = True + break + else: + time.sleep(polling_increment_secs) + + return status_achieved + + def _wait_for_elb_interface_removed(self): + polling_increment_secs = 15 + max_retries = (self.wait_timeout / polling_increment_secs) + status_achieved = False + + elb_interfaces = self.ec2_conn.get_all_network_interfaces( + filters={'attachment.instance-owner-id': 'amazon-elb', + 'description': 'ELB {0}'.format(self.name) }) + + for x in range(0, max_retries): + for interface in elb_interfaces: + try: + result = self.ec2_conn.get_all_network_interfaces(interface.id) + if result == []: + status_achieved = True + break + else: + time.sleep(polling_increment_secs) + except (boto.exception.BotoServerError, StandardError), e: + if 'InvalidNetworkInterfaceID' in e.code: + status_achieved = True + break + else: + self.module.fail_json(msg=str(e)) + + return status_achieved + def _get_elb(self): elbs = self.elb_conn.get_all_load_balancers() for elb in elbs: @@ -495,6 +579,13 @@ class ElbManager(object): except (boto.exception.NoAuthHandlerFound, AnsibleAWSError), e: self.module.fail_json(msg=str(e)) + def _get_ec2_connection(self): + try: + return connect_to_aws(boto.ec2, self.region, + **self.aws_connect_params) + except (boto.exception.NoAuthHandlerFound, StandardError), e: + self.module.fail_json(msg=str(e)) + def _delete_elb(self): # True if succeeds, exception raised if not result = self.elb_conn.delete_load_balancer(name=self.name) @@ -896,7 +987,9 @@ def main(): idle_timeout={'default': None, 'required': False}, cross_az_load_balancing={'default': None, 'required': False}, stickiness={'default': None, 'required': False, 'type': 'dict'}, - access_logs={'default': None, 'required': False, 'type': 'dict'} + access_logs={'default': None, 'required': False, 'type': 'dict'}, + wait={'default': False, 'type': 'bool', 'required': False}, + wait_timeout={'default': 60, 'type': 'int', 'required': False} ) ) @@ -929,6 +1022,8 @@ def main(): idle_timeout = module.params['idle_timeout'] cross_az_load_balancing = module.params['cross_az_load_balancing'] stickiness = module.params['stickiness'] + wait = module.params['wait'] + wait_timeout = module.params['wait_timeout'] if state == 'present' and not listeners: module.fail_json(msg="At least one port is required for ELB creation") @@ -936,6 +1031,9 @@ def main(): if state == 'present' and not (zones or subnets): module.fail_json(msg="At least one availability zone or subnet is required for ELB creation") + if wait_timeout > 600: + module.fail_json(msg='wait_timeout maximum is 600 seconds') + if security_group_names: security_group_ids = [] try: @@ -956,7 +1054,7 @@ def main(): subnets, purge_subnets, scheme, connection_draining_timeout, idle_timeout, cross_az_load_balancing, - access_logs, stickiness, + access_logs, stickiness, wait, wait_timeout, region=region, **aws_connect_params) # check for unsupported attributes for this version of boto