From b152515fcb7cf759c267fc9f33ad87ce2d39d2ea Mon Sep 17 00:00:00 2001 From: Sloane Hertel Date: Thu, 23 Aug 2018 19:42:32 -0400 Subject: [PATCH] RDS inventory plugin (#41919) Comments out uses of rds_instance in the integration tests and replace with AWS CLI until rds_instance has been merged --- .../aws_rds_inventory_plugin_added.yaml | 3 + lib/ansible/plugins/inventory/aws_rds.py | 351 ++++++++++++++++++ .../targets/inventory_aws_rds/aliases | 2 + .../playbooks/create_inventory_config.yml | 11 + .../playbooks/empty_inventory_config.yml | 9 + .../playbooks/populate_cache.yml | 76 ++++ .../test_invalid_aws_rds_inventory_config.yml | 9 + .../playbooks/test_inventory_cache.yml | 18 + .../playbooks/test_populating_inventory.yml | 106 ++++++ ..._populating_inventory_with_constructed.yml | 76 ++++ .../playbooks/test_refresh_inventory.yml | 94 +++++ .../targets/inventory_aws_rds/runme.sh | 35 ++ .../inventory_aws_rds/templates/inventory.yml | 6 + .../templates/inventory_with_cache.yml | 9 + .../templates/inventory_with_constructed.yml | 13 + .../inventory_aws_rds/test.aws_rds.yml | 0 16 files changed, 818 insertions(+) create mode 100644 changelogs/fragments/aws_rds_inventory_plugin_added.yaml create mode 100644 lib/ansible/plugins/inventory/aws_rds.py create mode 100644 test/integration/targets/inventory_aws_rds/aliases create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/create_inventory_config.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/empty_inventory_config.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/populate_cache.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/test_invalid_aws_rds_inventory_config.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/test_inventory_cache.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory_with_constructed.yml create mode 100644 test/integration/targets/inventory_aws_rds/playbooks/test_refresh_inventory.yml create mode 100755 test/integration/targets/inventory_aws_rds/runme.sh create mode 100644 test/integration/targets/inventory_aws_rds/templates/inventory.yml create mode 100644 test/integration/targets/inventory_aws_rds/templates/inventory_with_cache.yml create mode 100644 test/integration/targets/inventory_aws_rds/templates/inventory_with_constructed.yml create mode 100644 test/integration/targets/inventory_aws_rds/test.aws_rds.yml diff --git a/changelogs/fragments/aws_rds_inventory_plugin_added.yaml b/changelogs/fragments/aws_rds_inventory_plugin_added.yaml new file mode 100644 index 0000000000..d3a9a155fa --- /dev/null +++ b/changelogs/fragments/aws_rds_inventory_plugin_added.yaml @@ -0,0 +1,3 @@ +--- +minor_changes: + - aws_rds - Add new inventory plugin for RDS instances and clusters to match behavior in the ec2 inventory script. diff --git a/lib/ansible/plugins/inventory/aws_rds.py b/lib/ansible/plugins/inventory/aws_rds.py new file mode 100644 index 0000000000..8c76ba6d80 --- /dev/null +++ b/lib/ansible/plugins/inventory/aws_rds.py @@ -0,0 +1,351 @@ +# Copyright (c) 2018 Ansible Project +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +DOCUMENTATION = ''' + name: aws_rds + plugin_type: inventory + short_description: rds instance source + description: + - Get instances and clusters from Amazon Web Services RDS. + - Uses a .aws_rds.yaml (or .aws_rds.yml) YAML configuration file. + options: + boto_profile: + description: The boto profile to use. The plugin will look for an instance role if no credentials + are provided. + env: + - name: AWS_PROFILE + - name: AWS_DEFAULT_PROFILE + aws_access_key_id: + description: The AWS access key to use. If you have specified a profile, you don't need to provide + an access key/secret key/session token. + env: + - name: AWS_ACCESS_KEY_ID + - name: AWS_ACCESS_KEY + - name: EC2_ACCESS_KEY + aws_secret_access_key: + description: The AWS secret key that corresponds to the access key. If you have specified a profile, + you don't need to provide an access key/secret key/session token. + env: + - name: AWS_SECRET_ACCESS_KEY + - name: AWS_SECRET_KEY + - name: EC2_SECRET_KEY + aws_security_token: + description: The AWS security token if using temporary access and secret keys. + env: + - name: AWS_SECURITY_TOKEN + - name: AWS_SESSION_TOKEN + - name: EC2_SECURITY_TOKEN + regions: + description: A list of regions in which to describe RDS instances and clusters. Available regions are listed here + U(https://docs.aws.amazon.com/AmazonRDS/latest/UserGuide/Concepts.RegionsAndAvailabilityZones.html) + default: [] + filters: + description: A dictionary of filter value pairs. Available filters are listed here + U(https://docs.aws.amazon.com/cli/latest/reference/rds/describe-db-instances.html#options). If you filter by + db-cluster-id and I(include_clusters) is True it will apply to clusters as well. + default: {} + strict_permissions: + description: By default if an AccessDenied exception is encountered this plugin will fail. You can set strict_permissions to + False in the inventory config file which will allow the restrictions to be gracefully skipped. + type: bool + default: True + include_clusters: + description: Whether or not to query for Aurora clusters as well as instances + type: bool + default: False + statuses: + description: A list of desired states for instances/clusters to be added to inventory. Set to ['all'] as a shorthand to find everything. + type: list + default: + - creating + - available + extends_documentation_fragment: + - inventory_cache + - constructed + requirements: + - boto3 + - botocore + author: Sloane Hertel (@s-hertel) +''' + +EXAMPLES = ''' +plugin: aws_rds +regions: + - us-east-1 + - ca-central-1 +keyed_groups: + - key: 'db_parameter_groups|json_query("[].db_parameter_group_name")' + prefix: rds_parameter_group + - key: engine + prefix: rds + - key: tags + - key: region +''' + +from ansible.errors import AnsibleError +from ansible.module_utils._text import to_native +from ansible.module_utils.aws.core import is_boto3_error_code +from ansible.module_utils.ec2 import ansible_dict_to_boto3_filter_list, boto3_tag_list_to_ansible_dict +from ansible.module_utils.ec2 import camel_dict_to_snake_dict +from ansible.plugins.inventory import BaseInventoryPlugin, Constructable, Cacheable + +try: + import boto3 + import botocore +except ImportError: + raise AnsibleError('The RDS dynamic inventory plugin requires boto3 and botocore.') + + +class InventoryModule(BaseInventoryPlugin, Constructable, Cacheable): + + NAME = 'aws_rds' + + def __init__(self): + super(InventoryModule, self).__init__() + self.credentials = {} + self.boto_profile = None + + def _boto3_conn(self, regions): + ''' + :param regions: A list of regions to create a boto3 client + + Generator that yields a boto3 client and the region + ''' + for region in regions: + try: + connection = boto3.session.Session(profile_name=self.boto_profile).client('rds', region, **self.credentials) + except (botocore.exceptions.ProfileNotFound, botocore.exceptions.PartialCredentialsError) as e: + if self.boto_profile: + try: + connection = boto3.session.Session(profile_name=self.boto_profile).client('rds', region) + except (botocore.exceptions.ProfileNotFound, botocore.exceptions.PartialCredentialsError) as e: + raise AnsibleError("Insufficient credentials found: %s" % to_native(e)) + else: + raise AnsibleError("Insufficient credentials found: %s" % to_native(e)) + yield connection, region + + def _get_hosts_by_region(self, connection, filters, strict): + + def _add_tags_for_hosts(connection, hosts, strict): + for host in hosts: + if 'DBInstanceArn' in host: + resource_arn = host['DBInstanceArn'] + else: + resource_arn = host['DBClusterArn'] + + try: + tags = connection.list_tags_for_resource(ResourceName=resource_arn)['TagList'] + except is_boto3_error_code('AccessDenied') as e: + if not strict: + tags = [] + else: + raise e + host['Tags'] = tags + + def wrapper(f, *args, **kwargs): + try: + results = f(*args, **kwargs) + if 'DBInstances' in results: + results = results['DBInstances'] + else: + results = results['DBClusters'] + _add_tags_for_hosts(connection, results, strict) + except is_boto3_error_code('AccessDenied') as e: # pylint: disable=duplicate-except + if not strict: + results = [] + else: + raise AnsibleError("Failed to query RDS: {0}".format(to_native(e))) + except (botocore.exceptions.BotoCoreError, botocore.exceptions.ClientError) as e: # pylint: disable=duplicate-except + raise AnsibleError("Failed to query RDS: {0}".format(to_native(e))) + return results + return wrapper + + def _get_all_hosts(self, regions, instance_filters, cluster_filters, strict, statuses, gather_clusters=False): + ''' + :param regions: a list of regions in which to describe hosts + :param instance_filters: a list of boto3 filter dicionaries + :param cluster_filters: a list of boto3 filter dicionaries + :param strict: a boolean determining whether to fail or ignore 403 error codes + :param statuses: a list of statuses that the returned hosts should match + :return A list of host dictionaries + ''' + all_instances = [] + all_clusters = [] + for connection, region in self._boto3_conn(regions): + paginator = connection.get_paginator('describe_db_instances') + all_instances.extend( + self._get_hosts_by_region(connection, instance_filters, strict) + (paginator.paginate(Filters=instance_filters).build_full_result) + ) + if gather_clusters: + all_clusters.extend( + self._get_hosts_by_region(connection, cluster_filters, strict) + (connection.describe_db_clusters, **{'Filters': cluster_filters}) + ) + sorted_hosts = list( + sorted(all_instances, key=lambda x: x['DBInstanceIdentifier']) + + sorted(all_clusters, key=lambda x: x['DBClusterIdentifier']) + ) + return self.find_hosts_with_valid_statuses(sorted_hosts, statuses) + + def find_hosts_with_valid_statuses(self, hosts, statuses): + if 'all' in statuses: + return hosts + valid_hosts = [] + for host in hosts: + if host.get('DBInstanceStatus') in statuses: + valid_hosts.append(host) + elif host.get('Status') in statuses: + valid_hosts.append(host) + return valid_hosts + + def _populate(self, hosts): + group = 'aws_rds' + self.inventory.add_group(group) + if hosts: + self._add_hosts(hosts=hosts, group=group) + self.inventory.add_child('all', group) + + def _populate_from_source(self, source_data): + hostvars = source_data.pop('_meta', {}).get('hostvars', {}) + for group in source_data: + if group == 'all': + continue + else: + self.inventory.add_group(group) + hosts = source_data[group].get('hosts', []) + for host in hosts: + self._populate_host_vars([host], hostvars.get(host, {}), group) + self.inventory.add_child('all', group) + + def _get_hostname(self, host): + if host.get('DBInstanceIdentifier'): + return host['DBInstanceIdentifier'] + else: + return host['DBClusterIdentifier'] + + def _format_inventory(self, hosts): + results = {'_meta': {'hostvars': {}}} + group = 'aws_rds' + results[group] = {'hosts': []} + for host in hosts: + hostname = self._get_hostname(host) + results[group]['hosts'].append(hostname) + h = self.inventory.get_host(hostname) + results['_meta']['hostvars'][h.name] = h.vars + return results + + def _add_hosts(self, hosts, group): + ''' + :param hosts: a list of hosts to be added to a group + :param group: the name of the group to which the hosts belong + ''' + for host in hosts: + hostname = self._get_hostname(host) + host = camel_dict_to_snake_dict(host, ignore_list=['Tags']) + host['tags'] = boto3_tag_list_to_ansible_dict(host.get('tags', [])) + + # Allow easier grouping by region + if 'availability_zone' in host: + host['region'] = host['availability_zone'][:-1] + elif 'availability_zones' in host: + host['region'] = host['availability_zones'][0][:-1] + + self.inventory.add_host(hostname, group=group) + for hostvar, hostval in host.items(): + self.inventory.set_variable(hostname, hostvar, hostval) + + # Use constructed if applicable + strict = self.get_option('strict') + # Composed variables + self._set_composite_vars(self.get_option('compose'), host, hostname, strict=strict) + # Complex groups based on jinaj2 conditionals, hosts that meet the conditional are added to group + self._add_host_to_composed_groups(self.get_option('groups'), host, hostname, strict=strict) + # Create groups based on variable values and add the corresponding hosts to it + self._add_host_to_keyed_groups(self.get_option('keyed_groups'), host, hostname, strict=strict) + + def _set_credentials(self): + ''' + :param config_data: contents of the inventory config file + ''' + self.boto_profile = self.get_option('boto_profile') + aws_access_key_id = self.get_option('aws_access_key_id') + aws_secret_access_key = self.get_option('aws_secret_access_key') + aws_security_token = self.get_option('aws_security_token') + + if not self.boto_profile and not (aws_access_key_id and aws_secret_access_key): + session = botocore.session.get_session() + if session.get_credentials() is not None: + aws_access_key_id = session.get_credentials().access_key + aws_secret_access_key = session.get_credentials().secret_key + aws_security_token = session.get_credentials().token + + if not self.boto_profile and not (aws_access_key_id and aws_secret_access_key): + raise AnsibleError("Insufficient boto credentials found. Please provide them in your " + "inventory configuration file or set them as environment variables.") + + if aws_access_key_id: + self.credentials['aws_access_key_id'] = aws_access_key_id + if aws_secret_access_key: + self.credentials['aws_secret_access_key'] = aws_secret_access_key + if aws_security_token: + self.credentials['aws_session_token'] = aws_security_token + + def verify_file(self, path): + ''' + :param loader: an ansible.parsing.dataloader.DataLoader object + :param path: the path to the inventory config file + :return the contents of the config file + ''' + if super(InventoryModule, self).verify_file(path): + if path.endswith('.aws_rds.yml') or path.endswith('.aws_rds.yaml'): + return True + return False + + def parse(self, inventory, loader, path, cache=True): + super(InventoryModule, self).parse(inventory, loader, path) + + config_data = self._read_config_data(path) + self._set_credentials() + + # get user specifications + regions = self.get_option('regions') + filters = self.get_option('filters') + strict_permissions = self.get_option('strict_permissions') + statuses = self.get_option('statuses') + include_clusters = self.get_option('include_clusters') + instance_filters = ansible_dict_to_boto3_filter_list(filters) + cluster_filters = [] + if 'db-cluster-id' in filters and include_clusters: + cluster_filters = ansible_dict_to_boto3_filter_list({'db-cluster-id': filters['db-cluster-id']}) + + cache_key = self.get_cache_key(path) + # false when refresh_cache or --flush-cache is used + if cache: + # get the user-specified directive + cache = self.get_option('cache') + + # Generate inventory + formatted_inventory = {} + cache_needs_update = False + if cache: + try: + results = self.cache.get(cache_key) + except KeyError: + # if cache expires or cache file doesn't exist + cache_needs_update = True + else: + self._populate_from_source(results) + + if not cache or cache_needs_update: + results = self._get_all_hosts(regions, instance_filters, cluster_filters, strict_permissions, statuses, include_clusters) + self._populate(results) + formatted_inventory = self._format_inventory(results) + + # If the cache has expired/doesn't exist or if refresh_inventory/flush cache is used + # when the user is using caching, update the cached inventory + if cache_needs_update or (not cache and self.get_option('cache')): + self.cache.set(cache_key, formatted_inventory) diff --git a/test/integration/targets/inventory_aws_rds/aliases b/test/integration/targets/inventory_aws_rds/aliases new file mode 100644 index 0000000000..5692719518 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/aliases @@ -0,0 +1,2 @@ +cloud/aws +unsupported diff --git a/test/integration/targets/inventory_aws_rds/playbooks/create_inventory_config.yml b/test/integration/targets/inventory_aws_rds/playbooks/create_inventory_config.yml new file mode 100644 index 0000000000..e7dbe2ce4d --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/create_inventory_config.yml @@ -0,0 +1,11 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + vars: + template_name: "../templates/{{ template | default('inventory.yml') }}" + tasks: + - name: write inventory config file + copy: + dest: ../test.aws_rds.yml + content: "{{ lookup('template', template_name) }}" diff --git a/test/integration/targets/inventory_aws_rds/playbooks/empty_inventory_config.yml b/test/integration/targets/inventory_aws_rds/playbooks/empty_inventory_config.yml new file mode 100644 index 0000000000..d7e2cda3a7 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/empty_inventory_config.yml @@ -0,0 +1,9 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + - name: write inventory config file + copy: + dest: ../test.aws_rds.yml + content: "" diff --git a/test/integration/targets/inventory_aws_rds/playbooks/populate_cache.yml b/test/integration/targets/inventory_aws_rds/playbooks/populate_cache.yml new file mode 100644 index 0000000000..8698e0a996 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/populate_cache.yml @@ -0,0 +1,76 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + + - block: + - set_fact: + instance_id: 'rds-mariadb-{{ resource_prefix }}' + + - name: assert group was populated with inventory but is empty + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" + + # Create new host, add it to inventory and then terminate it without updating the cache + + # TODO: Uncomment once rds_instance has been added + #- name: set connection information for all tasks + # 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: yes + + - name: Use AWS CLI to create an RDS DB instance + command: "aws rds create-db-instance --db-instance-identifier '{{ instance_id }}' --engine 'mariadb' --db-instance-class 'db.t2.micro' --allocated-storage 20 --master-user-password '{{ resource_prefix }}' --master-username 'ansibletestuser'" + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: create minimal mariadb instance in default VPC and default subnet group + # rds_instance: + # state: present + # engine: mariadb + # instance_class: db.t2.micro + # storage: 20 + # instance_id: 'rds-mariadb-{{ resource_prefix }}' + # master_username: 'ansible-test-user' + # master_password: 'password-{{ resource_prefix }}' + # <<: *aws_connection_info + # register: setup_instance + + - meta: refresh_inventory + + - assert: + that: + - groups.aws_rds + + always: + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + # ignore_errors: yes + # when: setup_instance is defined diff --git a/test/integration/targets/inventory_aws_rds/playbooks/test_invalid_aws_rds_inventory_config.yml b/test/integration/targets/inventory_aws_rds/playbooks/test_invalid_aws_rds_inventory_config.yml new file mode 100644 index 0000000000..499513570b --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/test_invalid_aws_rds_inventory_config.yml @@ -0,0 +1,9 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + - name: assert inventory was not populated by aws_rds inventory plugin + assert: + that: + - "'aws_rds' not in groups" diff --git a/test/integration/targets/inventory_aws_rds/playbooks/test_inventory_cache.yml b/test/integration/targets/inventory_aws_rds/playbooks/test_inventory_cache.yml new file mode 100644 index 0000000000..7eadbad853 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/test_inventory_cache.yml @@ -0,0 +1,18 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + - name: assert cache was used to populate inventory + assert: + that: + - "'aws_rds' in groups" + - "groups.aws_rds | length == 1" + + - meta: refresh_inventory + + - name: assert refresh_inventory updated the cache + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" diff --git a/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory.yml b/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory.yml new file mode 100644 index 0000000000..0b1ba33080 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory.yml @@ -0,0 +1,106 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + + - block: + + - set_fact: + instance_id: "{{ resource_prefix }}-mariadb" + + - debug: var=groups + - name: assert group was populated with inventory but is empty + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" + + # Create new host, refresh inventory, remove host, refresh inventory + + #- name: set connection information for all tasks + # 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: yes + + - name: Use AWS CLI to create an RDS DB instance + command: "aws rds create-db-instance --db-instance-identifier '{{ instance_id }}' --engine 'mariadb' --db-instance-class 'db.t2.micro' --allocated-storage 20 --master-user-password '{{ resource_prefix }}' --master-username 'ansibletestuser'" + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: create minimal mariadb instance in default VPC and default subnet group + # rds_instance: + # state: present + # engine: mariadb + # instance_class: db.t2.micro + # storage: 20 + # instance_id: 'rds-mariadb-{{ resource_prefix }}' + # master_username: 'ansible-test-user' + # master_password: 'password-{{ resource_prefix }}' + # <<: *aws_connection_info + # register: setup_instance + + - meta: refresh_inventory + + - name: assert group was populated with inventory and is no longer empty + assert: + that: + - "'aws_rds' in groups" + - "groups.aws_rds | length == 1" + - "groups.aws_rds.0 == '{{ instance_id }}'" + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + + - meta: refresh_inventory + + - name: assert group was populated with inventory but is empty + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" + + always: + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + # ignore_errors: yes + # when: setup_instance is defined diff --git a/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory_with_constructed.yml b/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory_with_constructed.yml new file mode 100644 index 0000000000..c3ce5c008e --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/test_populating_inventory_with_constructed.yml @@ -0,0 +1,76 @@ +--- +- hosts: 127.0.0.1 + connection: local + gather_facts: no + tasks: + + - block: + + - set_fact: + instance_id: "{{ resource_prefix }}constructed" + + # Create new host, refresh inventory + + #- name: set connection information for all tasks + # 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: yes + + # TODO: Uncomment once rds_instance has been added + #- name: create minimal mariadb instance in default VPC and default subnet group + # rds_instance: + # state: present + # engine: mariadb + # instance_class: db.t2.micro + # storage: 20 + # instance_id: 'rds-mariadb-{{ resource_prefix }}' + # master_username: 'ansible-test-user' + # master_password: 'password-{{ resource_prefix }}' + # <<: *aws_connection_info + # register: setup_instance + + - name: Use AWS CLI to create an RDS DB instance + command: "aws rds create-db-instance --db-instance-identifier '{{ instance_id }}' --engine 'mariadb' --db-instance-class 'db.t2.micro' --allocated-storage 20 --master-user-password '{{ resource_prefix }}' --master-username 'ansibletestuser' --tags Key='workload_type',Value='other'" + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + - meta: refresh_inventory + - debug: var=groups + + - name: assert the keyed groups from constructed config were added to inventory + assert: + that: + # There are 6 groups: all, ungrouped, aws_rds, tag keyed group, engine keyed group, parameter group keyed group + - "groups | length == 6" + - groups.tag_workload_type_other + - groups.rds_mariadb + - groups.rds_parameter_group_default_mariadb10_0 + + always: + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + # TODO: Uncomment once rds_instance has been added + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + # ignore_errors: yes + # when: setup_instance is defined diff --git a/test/integration/targets/inventory_aws_rds/playbooks/test_refresh_inventory.yml b/test/integration/targets/inventory_aws_rds/playbooks/test_refresh_inventory.yml new file mode 100644 index 0000000000..de73edf373 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/playbooks/test_refresh_inventory.yml @@ -0,0 +1,94 @@ +- name: test updating inventory + block: + - set_fact: + instance_id: "{{ resource_prefix }}update" + + - name: assert group was populated with inventory but is empty + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" + + #- name: set connection information for all tasks + # 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: yes + + #- name: create minimal mariadb instance in default VPC and default subnet group + # rds_instance: + # state: present + # engine: mariadb + # instance_class: db.t2.micro + # storage: 20 + # instance_id: 'rds-mariadb-{{ resource_prefix }}' + # master_username: 'ansible-test-user' + # master_password: 'password-{{ resource_prefix }}' + # <<: *aws_connection_info + # register: setup_instance + + - name: Use AWS CLI to create an RDS DB instance + command: "aws rds create-db-instance --db-instance-identifier '{{ instance_id }}' --engine 'mariadb' --db-instance-class 'db.t2.micro' --allocated-storage 20 --master-user-password '{{ resource_prefix }}' --master-username 'ansibletestuser'" + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + - meta: refresh_inventory + + - name: assert group was populated with inventory and is no longer empty + assert: + that: + - "'aws_rds' in groups" + - "groups.aws_rds | length == 1" + - "groups.aws_rds.0 == '{{ resource_prefix }}'" + + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + - meta: refresh_inventory + + - name: assert group was populated with inventory but is empty + assert: + that: + - "'aws_rds' in groups" + - "not groups.aws_rds" + + always: + + - name: Use AWS CLI to delete the DB instance + command: "aws rds delete-db-instance --db-instance-identifier '{{ instance_id }}' --skip-final-snapshot" + ignore_errors: True + environment: + AWS_ACCESS_KEY_ID: "{{ aws_access_key }}" + AWS_SECRET_ACCESS_KEY: "{{ aws_secret_key }}" + AWS_SESSION_TOKEN: "{{ security_token }}" + AWS_DEFAULT_REGION: "{{ aws_region }}" + + #- name: remove mariadb instance + # rds_instance: + # state: absent + # engine: mariadb + # skip_final_snapshot: yes + # instance_id: ansible-rds-mariadb-example + # <<: *aws_connection_info + # ignore_errors: yes + # when: setup_instance is defined diff --git a/test/integration/targets/inventory_aws_rds/runme.sh b/test/integration/targets/inventory_aws_rds/runme.sh new file mode 100755 index 0000000000..3c968c3f79 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/runme.sh @@ -0,0 +1,35 @@ +#!/usr/bin/env bash + +set -eux + +# ensure test config is empty +ansible-playbook playbooks/empty_inventory_config.yml "$@" + +export ANSIBLE_INVENTORY_ENABLED=aws_rds + +# test with default inventory file +ansible-playbook playbooks/test_invalid_aws_rds_inventory_config.yml "$@" + +export ANSIBLE_INVENTORY=test.aws_rds.yml + +# test empty inventory config +ansible-playbook playbooks/test_invalid_aws_rds_inventory_config.yml "$@" + +# generate inventory config and test using it +ansible-playbook playbooks/create_inventory_config.yml -e @../../integration_config.yml "$@" +ansible-playbook playbooks/test_populating_inventory.yml -e @../../integration_config.yml "$@" + +# generate inventory config with caching and test using it +ansible-playbook playbooks/create_inventory_config.yml -e "template='inventory_with_cache.yml' @../../integration_config.yml" "$@" +ansible-playbook playbooks/populate_cache.yml -e @../../integration_config.yml "$@" +ansible-playbook playbooks/test_inventory_cache.yml "$@" + +# remove inventory cache +rm -r aws_rds_cache_dir/ + +# generate inventory config with constructed features and test using it +ansible-playbook playbooks/create_inventory_config.yml -e "template='inventory_with_constructed.yml' @../../integration_config.yml" "$@" +ansible-playbook playbooks/test_populating_inventory_with_constructed.yml -e @../../integration_config.yml "$@" + +# cleanup inventory config +ansible-playbook playbooks/empty_inventory_config.yml "$@" diff --git a/test/integration/targets/inventory_aws_rds/templates/inventory.yml b/test/integration/targets/inventory_aws_rds/templates/inventory.yml new file mode 100644 index 0000000000..1da89481c0 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/templates/inventory.yml @@ -0,0 +1,6 @@ +plugin: aws_rds +aws_access_key_id: '{{ aws_access_key }}' +aws_secret_access_key: '{{ aws_secret_key }}' +aws_security_token: '{{ security_token }}' +regions: + - '{{ aws_region }}' diff --git a/test/integration/targets/inventory_aws_rds/templates/inventory_with_cache.yml b/test/integration/targets/inventory_aws_rds/templates/inventory_with_cache.yml new file mode 100644 index 0000000000..609f760b98 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/templates/inventory_with_cache.yml @@ -0,0 +1,9 @@ +plugin: aws_rds +cache: True +cache_plugin: jsonfile +cache_connection: aws_rds_cache_dir +aws_access_key_id: '{{ aws_access_key }}' +aws_secret_access_key: '{{ aws_secret_key }}' +aws_security_token: '{{ security_token }}' +regions: + - '{{ aws_region }}' diff --git a/test/integration/targets/inventory_aws_rds/templates/inventory_with_constructed.yml b/test/integration/targets/inventory_aws_rds/templates/inventory_with_constructed.yml new file mode 100644 index 0000000000..2362c75d85 --- /dev/null +++ b/test/integration/targets/inventory_aws_rds/templates/inventory_with_constructed.yml @@ -0,0 +1,13 @@ +plugin: aws_rds +aws_access_key_id: '{{ aws_access_key }}' +aws_secret_access_key: '{{ aws_secret_key }}' +aws_security_token: '{{ security_token }}' +regions: + - '{{ aws_region }}' +keyed_groups: + - key: 'db_parameter_groups|json_query("[].db_parameter_group_name")' + prefix: rds_parameter_group + - key: tags + prefix: tag + - key: engine + prefix: rds diff --git a/test/integration/targets/inventory_aws_rds/test.aws_rds.yml b/test/integration/targets/inventory_aws_rds/test.aws_rds.yml new file mode 100644 index 0000000000..e69de29bb2