Skip to content

Commit

Permalink
Use TagSpecifications parameter when creating EC2 resources (ansible-…
Browse files Browse the repository at this point in the history
…collections#1876)

Use TagSpecifications parameter when creating EC2 resources

SUMMARY
For the last couple of years Amazon's supported tagging EC2 resources as part of the creation actions.  Switch the last amazon.aws EC2 modules over to using TagSpecifications during creation to support folks using Tagging requirements as part of their IAM/SCP policies
ISSUE TYPE

Feature Pull Request

COMPONENT NAME
ec2_vpc_route_table
ec2_vpc_igw
ec2_vpc_subnet
ec2_eip
ADDITIONAL INFORMATION
fixes: ansible-collections#1843

Reviewed-by: Alina Buzachis
  • Loading branch information
tremble authored Nov 28, 2023
1 parent e02fb78 commit 1b55a66
Show file tree
Hide file tree
Showing 14 changed files with 178 additions and 22 deletions.
2 changes: 2 additions & 0 deletions changelogs/fragments/1843-ec2_eip.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
minor_changes:
- ec2_eip - use ``ResourceTags`` to set initial tags upon creation (https://github.com/ansible-collections/amazon.aws/issues/1843)
2 changes: 2 additions & 0 deletions changelogs/fragments/1843-ec2_vpc_igw.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
minor_changes:
- ec2_vpc_igw - use ``ResourceTags`` to set initial tags upon creation (https://github.com/ansible-collections/amazon.aws/issues/1843)
2 changes: 2 additions & 0 deletions changelogs/fragments/1843-ec2_vpc_route_table.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
minor_changes:
- ec2_vpc_route_table - use ``ResourceTags`` to set initial tags upon creation (https://github.com/ansible-collections/amazon.aws/issues/1843)
4 changes: 4 additions & 0 deletions changelogs/fragments/1843-ec2_vpc_subnet.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
minor_changes:
- ec2_vpc_subnet - use ``ResourceTags`` to set initial tags upon creation (https://github.com/ansible-collections/amazon.aws/issues/1843)
- ec2_vpc_subnet - the default value for ``tags`` has been changed from ``{}`` to ``None``, to remove tags from a subnet an empty map must
be explicitly passed to the module (https://github.com/ansible-collections/amazon.aws/pull/1876).
34 changes: 34 additions & 0 deletions plugins/module_utils/tagging.py
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,12 @@ def boto3_tag_list_to_ansible_dict(tags_list, tag_name_key_name=None, tag_value_

def ansible_dict_to_boto3_tag_list(tags_dict, tag_name_key_name="Key", tag_value_key_name="Value"):
"""Convert a flat dict of key:value pairs representing AWS resource tags to a boto3 list of dicts
Note: booleans are converted to their Capitalized text form ("True" and "False"), this is
different to ansible_dict_to_boto3_filter_list because historically we've used "to_text()" and
AWS stores tags as strings, whereas for things which are actually booleans in AWS are returned
as lowercase strings in filters.
Args:
tags_dict (dict): Dict representing AWS resource tags.
tag_name_key_name (str): Value to use as the key for all tag keys (useful because boto3 doesn't always use "Key")
Expand Down Expand Up @@ -101,6 +107,34 @@ def ansible_dict_to_boto3_tag_list(tags_dict, tag_name_key_name="Key", tag_value
return tags_list


def _tag_name_to_filter_key(tag_name):
return f"tag:{tag_name}"


def ansible_dict_to_tag_filter_dict(tags_dict):
"""Prepends "tag:" to all of the keys (not the values) in a dict
This is useful when you're then going to build a filter including the tags.
Note: booleans are converted to their Capitalized text form ("True" and "False"), this is
different to ansible_dict_to_boto3_filter_list because historically we've used "to_text()" and
AWS stores tags as strings, whereas for things which are actually booleans in AWS are returned
as lowercase strings in filters.
Args:
tags_dict (dict): Dict representing AWS resource tags.
Basic Usage:
>>> filters = ansible_dict_to_boto3_filter_list(ansible_dict_to_tag_filter_dict(tags))
Returns:
Dict: A dictionary suitable for passing to ansible_dict_to_boto3_filter_list which can
also be combined with other common filter parameters.
"""
if not tags_dict:
return {}
return {_tag_name_to_filter_key(k): to_native(v) for k, v in tags_dict.items()}


def boto3_tag_specifications(tags_dict, types=None):
"""Converts a list of resource types and a flat dictionary of key:value pairs representing AWS
resource tags to a TagSpecification object.
Expand Down
68 changes: 59 additions & 9 deletions plugins/modules/ec2_eip.py
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,7 @@
from ansible_collections.amazon.aws.plugins.module_utils.ec2 import ensure_ec2_tags
from ansible_collections.amazon.aws.plugins.module_utils.modules import AnsibleAWSModule
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_specifications
from ansible_collections.amazon.aws.plugins.module_utils.transformation import ansible_dict_to_boto3_filter_list


Expand Down Expand Up @@ -342,7 +343,16 @@ def address_is_associated_with_device(ec2, module, address, device_id, is_instan
return False


def allocate_address(ec2, module, domain, reuse_existing_ip_allowed, check_mode, tag_dict=None, public_ipv4_pool=None):
def allocate_address(
ec2,
module,
domain,
reuse_existing_ip_allowed,
check_mode,
tags,
search_tags=None,
public_ipv4_pool=None,
):
"""Allocate a new elastic IP address (when needed) and return it"""
if not domain:
domain = "standard"
Expand All @@ -351,8 +361,8 @@ def allocate_address(ec2, module, domain, reuse_existing_ip_allowed, check_mode,
filters = []
filters.append({"Name": "domain", "Values": [domain]})

if tag_dict is not None:
filters += ansible_dict_to_boto3_filter_list(tag_dict)
if search_tags is not None:
filters += ansible_dict_to_boto3_filter_list(search_tags)

try:
all_addresses = ec2.describe_addresses(Filters=filters, aws_retry=True)
Expand All @@ -369,12 +379,26 @@ def allocate_address(ec2, module, domain, reuse_existing_ip_allowed, check_mode,
return unassociated_addresses[0], False

if public_ipv4_pool:
return allocate_address_from_pool(ec2, module, domain, check_mode, public_ipv4_pool), True
return (
allocate_address_from_pool(
ec2,
module,
domain,
check_mode,
public_ipv4_pool,
tags,
),
True,
)

params = {"Domain": domain}
if tags:
params["TagSpecifications"] = boto3_tag_specifications(tags, types="elastic-ip")

try:
if check_mode:
return None, True
result = ec2.allocate_address(Domain=domain, aws_retry=True), True
result = ec2.allocate_address(aws_retry=True, **params), True
except (botocore.exceptions.BotoCoreError, botocore.exceptions.ClientError) as e:
module.fail_json_aws(e, msg="Couldn't allocate Elastic IP address")
return result
Expand Down Expand Up @@ -434,6 +458,7 @@ def ensure_present(
reuse_existing_ip_allowed,
allow_reassociation,
check_mode,
tags,
is_instance=True,
):
changed = False
Expand All @@ -443,7 +468,14 @@ def ensure_present(
if check_mode:
return {"changed": True}

address, changed = allocate_address(ec2, module, domain, reuse_existing_ip_allowed, check_mode)
address, changed = allocate_address(
ec2,
module,
domain,
reuse_existing_ip_allowed,
check_mode,
tags,
)

if device_id:
# Allocate an IP for instance since no public_ip was provided
Expand Down Expand Up @@ -485,7 +517,14 @@ def ensure_absent(ec2, module, address, device_id, check_mode, is_instance=True)
return release_address(ec2, module, address, check_mode)


def allocate_address_from_pool(ec2, module, domain, check_mode, public_ipv4_pool):
def allocate_address_from_pool(
ec2,
module,
domain,
check_mode,
public_ipv4_pool,
tags,
):
# type: (EC2Connection, AnsibleAWSModule, str, bool, str) -> Address
"""Overrides botocore's allocate_address function to support BYOIP"""
if check_mode:
Expand All @@ -499,6 +538,9 @@ def allocate_address_from_pool(ec2, module, domain, check_mode, public_ipv4_pool
if public_ipv4_pool is not None:
params["PublicIpv4Pool"] = public_ipv4_pool

if tags:
params["TagSpecifications"] = boto3_tag_specifications(tags, types="elastic-ip")

try:
result = ec2.allocate_address(aws_retry=True, **params)
except (botocore.exceptions.BotoCoreError, botocore.exceptions.ClientError) as e:
Expand Down Expand Up @@ -583,7 +625,7 @@ def main():
module.fail_json(msg=str(e))

# Tags for *searching* for an EIP.
tag_dict = generate_tag_dict(module, tag_name, tag_value)
search_tags = generate_tag_dict(module, tag_name, tag_value)

try:
if device_id:
Expand All @@ -603,6 +645,7 @@ def main():
reuse_existing_ip_allowed,
allow_reassociation,
module.check_mode,
tags,
is_instance=is_instance,
)
if "allocation_id" not in result:
Expand All @@ -617,7 +660,14 @@ def main():
}
else:
address, changed = allocate_address(
ec2, module, domain, reuse_existing_ip_allowed, module.check_mode, tag_dict, public_ipv4_pool
ec2,
module,
domain,
reuse_existing_ip_allowed,
module.check_mode,
tags,
search_tags,
public_ipv4_pool,
)
if address:
result = {
Expand Down
6 changes: 5 additions & 1 deletion plugins/modules/ec2_vpc_igw.py
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,7 @@
from ansible_collections.amazon.aws.plugins.module_utils.modules import AnsibleAWSModule
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_list_to_ansible_dict
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_specifications
from ansible_collections.amazon.aws.plugins.module_utils.transformation import ansible_dict_to_boto3_filter_list
from ansible_collections.amazon.aws.plugins.module_utils.waiters import get_waiter

Expand Down Expand Up @@ -311,7 +312,10 @@ def ensure_igw_present(self, igw_id, vpc_id, tags, purge_tags, force_attach, det
self.get_matching_vpc(vpc_id)

try:
response = self._connection.create_internet_gateway(aws_retry=True)
create_params = {}
if tags:
create_params["TagSpecifications"] = boto3_tag_specifications(tags, types="internet-gateway")
response = self._connection.create_internet_gateway(aws_retry=True, **create_params)

# Ensure the gateway exists before trying to attach it or add tags
waiter = get_waiter(self._connection, "internet_gateway_exists")
Expand Down
6 changes: 5 additions & 1 deletion plugins/modules/ec2_vpc_route_table.py
Original file line number Diff line number Diff line change
Expand Up @@ -303,6 +303,7 @@
from ansible_collections.amazon.aws.plugins.module_utils.ec2 import ensure_ec2_tags
from ansible_collections.amazon.aws.plugins.module_utils.modules import AnsibleAWSModule
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_specifications
from ansible_collections.amazon.aws.plugins.module_utils.transformation import ansible_dict_to_boto3_filter_list
from ansible_collections.amazon.aws.plugins.module_utils.waiters import get_waiter

Expand Down Expand Up @@ -778,7 +779,10 @@ def ensure_route_table_present(connection, module):
changed = True
if not module.check_mode:
try:
route_table = connection.create_route_table(aws_retry=True, VpcId=vpc_id)["RouteTable"]
create_params = {"VpcId": vpc_id}
if tags:
create_params["TagSpecifications"] = boto3_tag_specifications(tags, types="route-table")
route_table = connection.create_route_table(aws_retry=True, **create_params)["RouteTable"]
# try to wait for route table to be present before moving on
get_waiter(connection, "route_table_exists").wait(
RouteTableIds=[route_table["RouteTableId"]],
Expand Down
24 changes: 14 additions & 10 deletions plugins/modules/ec2_vpc_subnet.py
Original file line number Diff line number Diff line change
Expand Up @@ -72,8 +72,6 @@
- Ignored unless I(wait=True).
default: 300
type: int
tags:
default: {}
extends_documentation_fragment:
- amazon.aws.common.modules
- amazon.aws.region.modules
Expand Down Expand Up @@ -212,14 +210,15 @@
except ImportError:
pass # caught by AnsibleAWSModule

from ansible.module_utils._text import to_text
from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.arn import is_outpost_arn
from ansible_collections.amazon.aws.plugins.module_utils.ec2 import ensure_ec2_tags
from ansible_collections.amazon.aws.plugins.module_utils.modules import AnsibleAWSModule
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import ansible_dict_to_tag_filter_dict
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_list_to_ansible_dict
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_specifications
from ansible_collections.amazon.aws.plugins.module_utils.transformation import ansible_dict_to_boto3_filter_list
from ansible_collections.amazon.aws.plugins.module_utils.waiters import get_waiter

Expand Down Expand Up @@ -268,7 +267,7 @@ def handle_waiter(conn, module, waiter_name, params, start_time):
module.fail_json_aws(e, "An exception happened while trying to wait for updates")


def create_subnet(conn, module, vpc_id, cidr, ipv6_cidr=None, outpost_arn=None, az=None, start_time=None):
def create_subnet(conn, module, vpc_id, cidr, tags, ipv6_cidr=None, outpost_arn=None, az=None, start_time=None):
wait = module.params["wait"]

params = dict(VpcId=vpc_id, CidrBlock=cidr)
Expand All @@ -279,6 +278,9 @@ def create_subnet(conn, module, vpc_id, cidr, ipv6_cidr=None, outpost_arn=None,
if az:
params["AvailabilityZone"] = az

if tags:
params["TagSpecifications"] = boto3_tag_specifications(tags, types="subnet")

if outpost_arn:
if is_outpost_arn(outpost_arn):
params["OutpostArn"] = outpost_arn
Expand Down Expand Up @@ -312,9 +314,12 @@ def ensure_tags(conn, module, subnet, tags, purge_tags, start_time):
retry_codes=["InvalidSubnetID.NotFound"],
)

if not changed:
return changed

if module.params["wait"] and not module.check_mode:
# Wait for tags to be updated
filters = [{"Name": f"tag:{k}", "Values": [v]} for k, v in tags.items()]
filters = ansible_dict_to_boto3_filter_list(ansible_dict_to_tag_filter_dict(tags))
handle_waiter(conn, module, "subnet_exists", {"SubnetIds": [subnet["id"]], "Filters": filters}, start_time)

return changed
Expand Down Expand Up @@ -452,6 +457,7 @@ def ensure_subnet_present(conn, module):
module,
module.params["vpc_id"],
module.params["cidr"],
module.params["tags"],
ipv6_cidr=module.params["ipv6_cidr"],
outpost_arn=module.params["outpost_arn"],
az=module.params["az"],
Expand All @@ -478,10 +484,8 @@ def ensure_subnet_present(conn, module):
)
changed = True

if module.params["tags"] != subnet["tags"]:
stringified_tags_dict = dict((to_text(k), to_text(v)) for k, v in module.params["tags"].items())
if ensure_tags(conn, module, subnet, stringified_tags_dict, module.params["purge_tags"], start_time):
changed = True
if ensure_tags(conn, module, subnet, module.params["tags"], module.params["purge_tags"], start_time):
changed = True

subnet = get_matching_subnet(conn, module, module.params["vpc_id"], module.params["cidr"])
if not module.check_mode and module.params["wait"]:
Expand Down Expand Up @@ -549,7 +553,7 @@ def main():
ipv6_cidr=dict(default="", required=False),
outpost_arn=dict(default="", type="str", required=False),
state=dict(default="present", choices=["present", "absent"]),
tags=dict(default={}, required=False, type="dict", aliases=["resource_tags"]),
tags=dict(required=False, type="dict", aliases=["resource_tags"]),
vpc_id=dict(required=True),
map_public=dict(default=False, required=False, type="bool"),
assign_instances_ipv6=dict(default=False, required=False, type="bool"),
Expand Down
2 changes: 2 additions & 0 deletions tests/integration/targets/ec2_eip/tasks/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,8 @@
- assert:
that:
- eip is changed
- "'ec2:CreateTags' not in eip.resource_actions"
- "'ec2:DeleteTags' not in eip.resource_actions"
- eip.public_ip is defined and ( eip.public_ip | ansible.utils.ipaddr )
- eip.allocation_id is defined and eip.allocation_id.startswith("eipalloc-")
- ( eip_info_start.addresses | length ) + 1 == ( eip_info.addresses | length
Expand Down
4 changes: 3 additions & 1 deletion tests/integration/targets/ec2_vpc_igw/tasks/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@
tags:
Description: Created by ansible-test
register: vpc_2_result

# ============================================================
- name: Search for internet gateway by VPC - no matches
ec2_vpc_igw_info:
Expand Down Expand Up @@ -91,6 +91,8 @@
- name: Assert creation happened (expected changed=true)
assert:
that:
- "'ec2:CreateTags' not in vpc_igw_create.resource_actions"
- "'ec2:DeleteTags' not in vpc_igw_create.resource_actions"
- vpc_igw_create is changed
- vpc_igw_create.gateway_id.startswith("igw-")
- vpc_igw_create.vpc_id == vpc_result.vpc.id
Expand Down
2 changes: 2 additions & 0 deletions tests/integration/targets/ec2_vpc_route_table/tasks/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,8 @@
assert:
that:
- create_public_table.changed
- "'ec2:CreateTags' not in create_public_table.resource_actions"
- "'ec2:DeleteTags' not in create_public_table.resource_actions"
- create_public_table.route_table.id.startswith('rtb-')
- "'Public' in create_public_table.route_table.tags"
- create_public_table.route_table.tags['Public'] == 'true'
Expand Down
2 changes: 2 additions & 0 deletions tests/integration/targets/ec2_vpc_subnet/tasks/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,8 @@
assert:
that:
- 'vpc_subnet_create'
- "'ec2:CreateTags' not in vpc_subnet_create.resource_actions"
- "'ec2:DeleteTags' not in vpc_subnet_create.resource_actions"
- 'vpc_subnet_create.subnet.id.startswith("subnet-")'
- '"Name" in vpc_subnet_create.subnet.tags and vpc_subnet_create.subnet.tags["Name"] == ec2_vpc_subnet_name'
- '"Description" in vpc_subnet_create.subnet.tags and vpc_subnet_create.subnet.tags["Description"] == ec2_vpc_subnet_description'
Expand Down
Loading

0 comments on commit 1b55a66

Please sign in to comment.