mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 05:23:58 -07:00 
			
		
		
		
	ec2_vol_facts: moved to boto3 (#43348)
* ec2_vol_facts: moved to boto3 * vol_facts: formatting fixes * vol_facts: formatting fixes * vol_facts: added integration tests * vol_facts: improved integration tests * vol_facts: integration tests, fixed ami * vol_facts: integration tests, fixed ami * vol_facts: refactor, post-review update * vol_facts: implemented pagination * vol_facts: pep8 style fix * CI IAM policy requirements fix * Tests fixed, added to unsupported, removed empty files * removed shippable alias
This commit is contained in:
		
					parent
					
						
							
								d71670655b
							
						
					
				
			
			
				commit
				
					
						4f70eb3e26
					
				
			
		
					 4 changed files with 168 additions and 40 deletions
				
			
		|  | @ -18,6 +18,7 @@ short_description: Gather facts about ec2 volumes in AWS | |||
| description: | ||||
|     - Gather facts about ec2 volumes in AWS | ||||
| version_added: "2.1" | ||||
| requirements: [ boto3 ] | ||||
| author: "Rob White (@wimnat)" | ||||
| options: | ||||
|   filters: | ||||
|  | @ -59,57 +60,67 @@ RETURN = '''# ''' | |||
| import traceback | ||||
| 
 | ||||
| try: | ||||
|     import boto.ec2 | ||||
|     from boto.exception import BotoServerError | ||||
|     HAS_BOTO = True | ||||
|     from botocore.exceptions import ClientError | ||||
| except ImportError: | ||||
|     HAS_BOTO = False | ||||
|     pass  # caught by imported HAS_BOTO3 | ||||
| 
 | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| from ansible.module_utils.ec2 import connect_to_aws, ec2_argument_spec, get_aws_connection_info | ||||
| from ansible.module_utils.ec2 import connect_to_aws, ec2_argument_spec, get_aws_connection_info, boto3_conn, HAS_BOTO3, boto3_tag_list_to_ansible_dict | ||||
| from ansible.module_utils.ec2 import ansible_dict_to_boto3_filter_list, camel_dict_to_snake_dict | ||||
| from ansible.module_utils._text import to_native | ||||
| 
 | ||||
| 
 | ||||
| def get_volume_info(volume): | ||||
| def get_volume_info(volume, region): | ||||
| 
 | ||||
|     attachment = volume.attach_data | ||||
|     attachment = volume["attachments"] | ||||
| 
 | ||||
|     volume_info = { | ||||
|         'create_time': volume.create_time, | ||||
|         'id': volume.id, | ||||
|         'encrypted': volume.encrypted, | ||||
|         'iops': volume.iops, | ||||
|         'size': volume.size, | ||||
|         'snapshot_id': volume.snapshot_id, | ||||
|         'status': volume.status, | ||||
|         'type': volume.type, | ||||
|         'zone': volume.zone, | ||||
|         'region': volume.region.name, | ||||
|         'create_time': volume["create_time"], | ||||
|         'id': volume["volume_id"], | ||||
|         'encrypted': volume["encrypted"], | ||||
|         'iops': volume["iops"] if "iops" in volume else None, | ||||
|         'size': volume["size"], | ||||
|         'snapshot_id': volume["snapshot_id"], | ||||
|         'status': volume["state"], | ||||
|         'type': volume["volume_type"], | ||||
|         'zone': volume["availability_zone"], | ||||
|         'region': region, | ||||
|         'attachment_set': { | ||||
|             'attach_time': attachment.attach_time, | ||||
|             'device': attachment.device, | ||||
|             'instance_id': attachment.instance_id, | ||||
|             'status': attachment.status | ||||
|             'attach_time': attachment[0]["attach_time"] if len(attachment) > 0 else None, | ||||
|             'device': attachment[0]["device"] if len(attachment) > 0 else None, | ||||
|             'instance_id': attachment[0]["instance_id"] if len(attachment) > 0 else None, | ||||
|             'status': attachment[0]["state"] if len(attachment) > 0 else None, | ||||
|             'delete_on_termination': attachment[0]["delete_on_termination"] if len(attachment) > 0 else None | ||||
|         }, | ||||
|         'tags': volume.tags | ||||
|         'tags': boto3_tag_list_to_ansible_dict(volume['tags']) | ||||
|     } | ||||
| 
 | ||||
|     return volume_info | ||||
| 
 | ||||
| 
 | ||||
| def list_ec2_volumes(connection, module): | ||||
| def describe_volumes_with_backoff(connection, filters): | ||||
|     paginator = connection.get_paginator('describe_volumes') | ||||
|     return paginator.paginate(Filters=filters).build_full_result() | ||||
| 
 | ||||
|     filters = module.params.get("filters") | ||||
| 
 | ||||
| def list_ec2_volumes(connection, module, region): | ||||
| 
 | ||||
|     # Replace filter key underscores with dashes, for compatibility, except if we're dealing with tags | ||||
|     sanitized_filters = module.params.get("filters") | ||||
|     for key in sanitized_filters: | ||||
|         if not key.startswith("tag:"): | ||||
|             sanitized_filters[key.replace("_", "-")] = sanitized_filters.pop(key) | ||||
|     volume_dict_array = [] | ||||
| 
 | ||||
|     try: | ||||
|         all_volumes = connection.get_all_volumes(filters=filters) | ||||
|     except BotoServerError as e: | ||||
|         module.fail_json(msg=e.message) | ||||
|         all_volumes = describe_volumes_with_backoff(connection, ansible_dict_to_boto3_filter_list(sanitized_filters)) | ||||
| 
 | ||||
|     for volume in all_volumes: | ||||
|         volume_dict_array.append(get_volume_info(volume)) | ||||
|     except ClientError as e: | ||||
|         module.fail_json(msg=e.response, exception=traceback.format_exc()) | ||||
| 
 | ||||
|     for volume in all_volumes["Volumes"]: | ||||
|         volume = camel_dict_to_snake_dict(volume, ignore_list=['Tags']) | ||||
|         volume_dict_array.append(get_volume_info(volume, region)) | ||||
|     module.exit_json(volumes=volume_dict_array) | ||||
| 
 | ||||
| 
 | ||||
|  | @ -123,20 +134,21 @@ def main(): | |||
| 
 | ||||
|     module = AnsibleModule(argument_spec=argument_spec) | ||||
| 
 | ||||
|     if not HAS_BOTO: | ||||
|         module.fail_json(msg='boto required for this module') | ||||
|     if not HAS_BOTO3: | ||||
|         module.fail_json(msg='boto3 required for this module') | ||||
| 
 | ||||
|     region, ec2_url, aws_connect_params = get_aws_connection_info(module) | ||||
|     region, ec2_url, aws_connect_params = get_aws_connection_info(module, boto3=True) | ||||
| 
 | ||||
|     if region: | ||||
|         try: | ||||
|             connection = connect_to_aws(boto.ec2, region, **aws_connect_params) | ||||
|         except (boto.exception.NoAuthHandlerFound, Exception) as e: | ||||
|             module.fail_json(msg=to_native(e), exception=traceback.format_exc()) | ||||
|     else: | ||||
|         module.fail_json(msg="region must be specified") | ||||
|     connection = boto3_conn( | ||||
|         module, | ||||
|         conn_type='client', | ||||
|         resource='ec2', | ||||
|         region=region, | ||||
|         endpoint=ec2_url, | ||||
|         **aws_connect_params | ||||
|     ) | ||||
| 
 | ||||
|     list_ec2_volumes(connection, module) | ||||
|     list_ec2_volumes(connection, module, region) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|  |  | |||
							
								
								
									
										2
									
								
								test/integration/targets/ec2_vol_facts/aliases
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								test/integration/targets/ec2_vol_facts/aliases
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| cloud/aws | ||||
| unsupported | ||||
							
								
								
									
										3
									
								
								test/integration/targets/ec2_vol_facts/meta/main.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								test/integration/targets/ec2_vol_facts/meta/main.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | |||
| dependencies: | ||||
|   - prepare_tests | ||||
|   - setup_ec2 | ||||
							
								
								
									
										111
									
								
								test/integration/targets/ec2_vol_facts/tasks/main.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								test/integration/targets/ec2_vol_facts/tasks/main.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,111 @@ | |||
| --- | ||||
| # tasks file for test_ec2_vol_facts | ||||
| - name: Set up AWS connection info | ||||
|   set_fact: | ||||
|     aws_connection_info: &aws_connection_info | ||||
|       aws_access_key: "{{ aws_access_key }}" | ||||
|       aws_secret_key: "{{ aws_secret_key }}" | ||||
|       security_token: "{{ security_token }}" | ||||
|       region: "{{ aws_region }}" | ||||
|   no_log: true | ||||
| 
 | ||||
| - block: | ||||
|     - ec2_ami_facts: | ||||
|         <<: *aws_connection_info | ||||
|         filters: | ||||
|           architecture: x86_64 | ||||
|           virtualization-type: hvm | ||||
|           root-device-type: ebs | ||||
|           name: "amzn-ami-hvm*" | ||||
|       register: amis | ||||
| 
 | ||||
|     - name: Create test instance | ||||
|       ec2_instance: | ||||
|           name: "{{ resource_prefix }}_ansible_ec2_vol_facts_test" | ||||
|           instance_type: t2.nano | ||||
|           image_id: "{{ (amis.images | sort(attribute='creation_date') | last).image_id }}" | ||||
|           wait: yes | ||||
|           tags: | ||||
|             Environment: test | ||||
|           <<: *aws_connection_info | ||||
|       register: instance | ||||
| 
 | ||||
|     - name: Ensure there's only one matching instance | ||||
|       assert: | ||||
|         that: | ||||
|             - "instance.instance_ids|length == 1" | ||||
|             - "instance.instances|length == 1" | ||||
| 
 | ||||
|     - name: Create test volume | ||||
|       ec2_vol: | ||||
|         instance: "{{ instance.instance_ids[0] }}" | ||||
|         volume_size: 4 | ||||
|         name: "{{ resource_prefix }}_ansible_ec2_vol_facts_test.db" | ||||
|         device_name: /dev/xvdf | ||||
|         iops: 100 | ||||
|         tags: | ||||
|             Tag Name with Space-and-dash: Tag Value with Space-and-dash | ||||
|         <<: *aws_connection_info | ||||
|         delete_on_termination: yes | ||||
|       register: volume | ||||
| 
 | ||||
|     - name: Gather volume info | ||||
|       ec2_vol_facts: | ||||
|           <<: *aws_connection_info | ||||
|           filters: | ||||
|             "tag:Name": "{{ resource_prefix }}_ansible_ec2_vol_facts_test.db" | ||||
|       register: volume_facts | ||||
|       check_mode: no | ||||
| 
 | ||||
|     - name: Format check | ||||
|       assert: | ||||
|         that: | ||||
|             - "volume_facts.volumes|length == 1" | ||||
|             - "v.attachment_set.attach_time is defined" | ||||
|             - "v.attachment_set.device      is defined and v.attachment_set.device      == volume.device" | ||||
|             - "v.attachment_set.instance_id is defined and v.attachment_set.instance_id == instance.instance_ids[0]" | ||||
|             - "v.attachment_set.status      is defined and v.attachment_set.status      == 'attached'" | ||||
|             - "v.create_time                is defined" | ||||
|             - "v.encrypted                  is defined and v.encrypted                  == false" | ||||
|             - "v.id                         is defined and v.id                         == volume.volume_id" | ||||
|             - "v.iops                       is defined and v.iops                       == 100" | ||||
|             - "v.region                     is defined and v.region                     == aws_region" | ||||
|             - "v.size                       is defined and v.size                       == 4" | ||||
|             - "v.snapshot_id                is defined and v.snapshot_id                == ''" | ||||
|             - "v.status                     is defined and v.status                     == 'in-use'" | ||||
|             - "v.tags.Name                  is defined and v.tags.Name                  == resource_prefix + '_ansible_ec2_vol_facts_test.db'" | ||||
|             - "v.tags['Tag Name with Space-and-dash']                                   == 'Tag Value with Space-and-dash'" | ||||
|             - "v.type                       is defined and v.type                       == 'io1'" | ||||
|             - "v.zone                       is defined and v.zone                       == instance.instances[0].placement.availability_zone" | ||||
|       vars: | ||||
|           v: "{{ volume_facts.volumes[0] }}" | ||||
| 
 | ||||
|     - name: New format check | ||||
|       assert: | ||||
|         that: | ||||
|             - "v.attachment_set.delete_on_termination is defined" | ||||
|       vars: | ||||
|           v: "{{ volume_facts.volumes[0] }}" | ||||
|       when: ansible_version.full is version('2.7', '>=') | ||||
| 
 | ||||
|   always: | ||||
|     - name: Remove the instance | ||||
|       ec2_instance: | ||||
|           state: absent | ||||
|           filters: | ||||
|               "tag:Name": "{{ resource_prefix }}_ansible_ec2_vol_facts_test" | ||||
|           <<: *aws_connection_info | ||||
|       register: result | ||||
|       until: result is not failed | ||||
|       ignore_errors: yes | ||||
|       retries: 10 | ||||
| 
 | ||||
|     - name: Remove the volume | ||||
|       ec2_vol: | ||||
|         id: "{{ volume.volume_id }}" | ||||
|         state: absent | ||||
|         <<: *aws_connection_info | ||||
|       register: result | ||||
|       until: result is not failed | ||||
|       ignore_errors: yes | ||||
|       retries: 10 | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue