Skip to content

Commit

Permalink
Various ARN handling fixes (#1848) (#1858)
Browse files Browse the repository at this point in the history
[PR #1848/25a636ce backport][stable-6] Various ARN handling fixes

This is a backport of PR #1848 as merged into main (25a636c).
Depends-On: ansible-collections/amazon.aws#1619
SUMMARY
fixes: #1846
Various modules had hard-coded ARN handling which assumed the use of the main  partition.  This causes problems for folks using Gov Cloud (and aws-cn)
ISSUE TYPE

Bugfix Pull Request

COMPONENT NAME
plugins/modules/batch_compute_environment.py
plugins/modules/ec2_launch_template.py
plugins/modules/elasticache_info.py
plugins/modules/iam_group.py
plugins/modules/iam_role.py
plugins/modules/msk_config.py
plugins/modules/redshift.py
plugins/modules/sns_topic.py
ADDITIONAL INFORMATION

Reviewed-by: Mark Chappell
  • Loading branch information
patchback[bot] authored Jun 27, 2023
1 parent 4eb66d6 commit 5fb6c15
Show file tree
Hide file tree
Showing 9 changed files with 35 additions and 29 deletions.
11 changes: 11 additions & 0 deletions changelogs/fragments/1846-arn.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
bugfixes:
- iam_role - fixed incorrect rejection of Gov Cloud ARNs in ``boundary`` parameter (https://github.com/ansible-collections/community.aws/issues/1846).
- batch_compute_environment - fixed incorrect handling of Gov Cloud ARNs in ``compute_environment_name`` parameter (https://github.com/ansible-collections/community.aws/issues/1846).
- ec2_launch_template - fixed incorrect handling of Gov Cloud ARNs in ``compute_environment_name`` parameter (https://github.com/ansible-collections/community.aws/issues/1846).
- elasticache_info - remove hard coded use of ``aws`` partition (https://github.com/ansible-collections/community.aws/issues/1846).
- msk_cluster - remove hard coded use of ``aws`` partition (https://github.com/ansible-collections/community.aws/issues/1846).
- redshift - fixed hard coded use of ``aws`` partition (https://github.com/ansible-collections/community.aws/issues/1846).
minor_changes:
- sns_topic - refactored ARN validation handling (https://github.com/ansible-collections/community.aws/pull/1848).
- iam_role - refactored ARN validation handling (https://github.com/ansible-collections/community.aws/pull/1848).
- iam_group - refactored ARN validation handling (https://github.com/ansible-collections/community.aws/pull/1848).
3 changes: 2 additions & 1 deletion plugins/modules/batch_compute_environment.py
Original file line number Diff line number Diff line change
Expand Up @@ -233,6 +233,7 @@
from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict
from ansible.module_utils.common.dict_transformations import snake_dict_to_camel_dict

from ansible_collections.amazon.aws.plugins.module_utils.arn import validate_aws_arn
from ansible_collections.community.aws.plugins.module_utils.modules import AnsibleCommunityAWSModule as AnsibleAWSModule


Expand Down Expand Up @@ -270,7 +271,7 @@ def validate_params(module):
module.fail_json(
msg=f"Function compute_environment_name {compute_environment_name} is invalid. Names must contain only alphanumeric characters and underscores."
)
if not compute_environment_name.startswith("arn:aws:batch:"):
if not validate_aws_arn(compute_environment_name, service="batch"):
if len(compute_environment_name) > 128:
module.fail_json(msg=f'compute_environment_name "{compute_environment_name}" exceeds 128 character limit')

Expand Down
4 changes: 2 additions & 2 deletions plugins/modules/ec2_launch_template.py
Original file line number Diff line number Diff line change
Expand Up @@ -422,7 +422,6 @@
type: int
"""

import re
from uuid import uuid4

try:
Expand All @@ -436,6 +435,7 @@
from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict
from ansible.module_utils.common.dict_transformations import snake_dict_to_camel_dict

from ansible_collections.amazon.aws.plugins.module_utils.arn import validate_aws_arn
from ansible_collections.amazon.aws.plugins.module_utils.botocore import is_boto3_error_code
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import ansible_dict_to_boto3_tag_list
Expand All @@ -446,7 +446,7 @@


def determine_iam_role(module, name_or_arn):
if re.match(r"^arn:aws:iam::\d+:instance-profile/[\w+=/,.@-]+$", name_or_arn):
if validate_aws_arn(name_or_arn, service="iam", resource_type="instance-profile"):
return {"arn": name_or_arn}
iam = module.client("iam", retry_decorator=AWSRetry.jittered_backoff())
try:
Expand Down
17 changes: 3 additions & 14 deletions plugins/modules/elasticache_info.py
Original file line number Diff line number Diff line change
Expand Up @@ -410,6 +410,7 @@
from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.botocore import is_boto3_error_code
from ansible_collections.amazon.aws.plugins.module_utils.iam import get_aws_account_info
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

Expand Down Expand Up @@ -444,30 +445,18 @@ def get_elasticache_tags_with_backoff(client, cluster_id):
return client.list_tags_for_resource(ResourceName=cluster_id)["TagList"]


def get_aws_account_id(module):
try:
client = module.client("sts")
except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e:
module.fail_json_aws(e, msg="Can't authorize connection")

try:
return client.get_caller_identity()["Account"]
except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e:
module.fail_json_aws(e, msg="Couldn't obtain AWS account id")


def get_elasticache_clusters(client, module):
region = module.region
try:
clusters = describe_cache_clusters_with_backoff(client, cluster_id=module.params.get("name"))
except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e:
module.fail_json_aws(e, msg="Couldn't obtain cache cluster info")

account_id = get_aws_account_id(module)
account_id, partition = get_aws_account_info(module)
results = []
for cluster in clusters:
cluster = camel_dict_to_snake_dict(cluster)
arn = f"arn:aws:elasticache:{region}:{account_id}:cluster:{cluster['cache_cluster_id']}"
arn = f"arn:{partition}:elasticache:{region}:{account_id}:cluster:{cluster['cache_cluster_id']}"
try:
tags = get_elasticache_tags_with_backoff(client, arn)
except is_boto3_error_code("CacheClusterNotFound"):
Expand Down
5 changes: 3 additions & 2 deletions plugins/modules/iam_group.py
Original file line number Diff line number Diff line change
Expand Up @@ -171,6 +171,7 @@

from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.arn import validate_aws_arn
from ansible_collections.amazon.aws.plugins.module_utils.botocore import is_boto3_error_code
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry

Expand Down Expand Up @@ -203,7 +204,7 @@ def compare_group_members(current_group_members, new_group_members):


def convert_friendly_names_to_arns(connection, module, policy_names):
if not any(not policy.startswith("arn:") for policy in policy_names if policy is not None):
if all(validate_aws_arn(policy, service="iam") for policy in policy_names if policy is not None):
return policy_names
allpolicies = {}
paginator = connection.get_paginator("list_policies")
Expand All @@ -213,7 +214,7 @@ def convert_friendly_names_to_arns(connection, module, policy_names):
allpolicies[policy["PolicyName"]] = policy["Arn"]
allpolicies[policy["Arn"]] = policy["Arn"]
try:
return [allpolicies[policy] for policy in policy_names]
return [allpolicies[policy] for policy in policy_names if policy is not None]
except KeyError as e:
module.fail_json(msg="Couldn't find policy: " + str(e))

Expand Down
7 changes: 4 additions & 3 deletions plugins/modules/iam_role.py
Original file line number Diff line number Diff line change
Expand Up @@ -224,6 +224,7 @@

from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.arn import validate_aws_arn
from ansible_collections.amazon.aws.plugins.module_utils.botocore import is_boto3_error_code
from ansible_collections.amazon.aws.plugins.module_utils.policy import compare_policies
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
Expand Down Expand Up @@ -265,7 +266,7 @@ def wait_iam_exists(module, client):


def convert_friendly_names_to_arns(module, client, policy_names):
if not any(not policy.startswith("arn:") for policy in policy_names):
if all(validate_aws_arn(policy, service="iam") for policy in policy_names if policy is not None):
return policy_names

allpolicies = {}
Expand All @@ -275,7 +276,7 @@ def convert_friendly_names_to_arns(module, client, policy_names):
allpolicies[policy["PolicyName"]] = policy["Arn"]
allpolicies[policy["Arn"]] = policy["Arn"]
try:
return [allpolicies[policy] for policy in policy_names]
return [allpolicies[policy] for policy in policy_names if policy is not None]
except KeyError as e:
module.fail_json_aws(e, msg="Couldn't find policy")

Expand Down Expand Up @@ -746,7 +747,7 @@ def main():
if module.params.get("boundary"):
if module.params.get("create_instance_profile"):
module.fail_json(msg="When using a boundary policy, `create_instance_profile` must be set to `false`.")
if not module.params.get("boundary").startswith("arn:aws:iam"):
if not validate_aws_arn(module.params.get("boundary"), service="iam"):
module.fail_json(msg="Boundary policy must be an ARN")
if module.params.get("max_session_duration"):
max_session_duration = module.params.get("max_session_duration")
Expand Down
4 changes: 3 additions & 1 deletion plugins/modules/msk_config.py
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,7 @@

from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict

from ansible_collections.amazon.aws.plugins.module_utils.iam import get_aws_account_info
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry

from ansible_collections.community.aws.plugins.module_utils.modules import AnsibleCommunityAWSModule as AnsibleAWSModule
Expand Down Expand Up @@ -283,7 +284,8 @@ def main():
# return some useless staff in check mode if configuration doesn't exists
# can be useful when these options are referenced by other modules during check mode run
if module.check_mode and not response.get("Arn"):
arn = "arn:aws:kafka:region:account:configuration/name/id"
account_id, partition = get_aws_account_info(module)
arn = f"arn:{partition}:kafka:{module.region}:{account_id}:configuration/{module.params['name']}/id"
revision = 1
server_properties = ""
else:
Expand Down
8 changes: 4 additions & 4 deletions plugins/modules/redshift.py
Original file line number Diff line number Diff line change
Expand Up @@ -263,7 +263,7 @@
from ansible.module_utils.common.dict_transformations import snake_dict_to_camel_dict

from ansible_collections.amazon.aws.plugins.module_utils.botocore import is_boto3_error_code
from ansible_collections.amazon.aws.plugins.module_utils.iam import get_aws_account_id
from ansible_collections.amazon.aws.plugins.module_utils.iam import get_aws_account_info
from ansible_collections.amazon.aws.plugins.module_utils.retries import AWSRetry
from ansible_collections.amazon.aws.plugins.module_utils.tagging import ansible_dict_to_boto3_tag_list
from ansible_collections.amazon.aws.plugins.module_utils.tagging import boto3_tag_list_to_ansible_dict
Expand All @@ -275,9 +275,9 @@
def _ensure_tags(redshift, identifier, existing_tags, module):
"""Compares and update resource tags"""

account_id = get_aws_account_id(module)
region = module.params.get("region")
resource_arn = f"arn:aws:redshift:{region}:{account_id}:cluster:{identifier}"
account_id, partition = get_aws_account_info(module)
region = module.region
resource_arn = f"arn:{partition}:redshift:{region}:{account_id}:cluster:{identifier}"
tags = module.params.get("tags")
purge_tags = module.params.get("purge_tags")

Expand Down
5 changes: 3 additions & 2 deletions plugins/modules/sns_topic.py
Original file line number Diff line number Diff line change
Expand Up @@ -338,9 +338,10 @@
except ImportError:
pass # handled by AnsibleAWSModule

from ansible_collections.amazon.aws.plugins.module_utils.transformation import scrub_none_parameters
from ansible_collections.amazon.aws.plugins.module_utils.arn import parse_aws_arn
from ansible_collections.amazon.aws.plugins.module_utils.policy import compare_policies
from ansible_collections.amazon.aws.plugins.module_utils.tagging import ansible_dict_to_boto3_tag_list
from ansible_collections.amazon.aws.plugins.module_utils.transformation import scrub_none_parameters

from ansible_collections.community.aws.plugins.module_utils.modules import AnsibleCommunityAWSModule as AnsibleAWSModule
from ansible_collections.community.aws.plugins.module_utils.sns import list_topics
Expand Down Expand Up @@ -579,7 +580,7 @@ def _delete_topic(self):
return True

def _name_is_arn(self):
return self.name.startswith("arn:")
return bool(parse_aws_arn(self.name))

def ensure_ok(self):
changed = False
Expand Down

0 comments on commit 5fb6c15

Please sign in to comment.