Server IP : 85.214.239.14 / Your IP : 18.117.91.170 Web Server : Apache/2.4.62 (Debian) System : Linux h2886529.stratoserver.net 4.9.0 #1 SMP Tue Jan 9 19:45:01 MSK 2024 x86_64 User : www-data ( 33) PHP Version : 7.4.18 Disable Function : pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare, MySQL : OFF | cURL : OFF | WGET : ON | Perl : ON | Python : ON | Sudo : ON | Pkexec : OFF Directory : /lib/python3/dist-packages/ansible_collections/amazon/aws/plugins/modules/ |
Upload File : |
#!/usr/bin/python # Copyright (c) 2017 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 = r''' --- module: iam_user version_added: 5.0.0 short_description: Manage AWS IAM users description: - A module to manage AWS IAM users. - The module does not manage groups that users belong to, groups memberships can be managed using M(community.aws.iam_group). - This module was originally added to C(community.aws) in release 1.0.0. author: - Josh Souza (@joshsouza) options: name: description: - The name of the user to create. required: true type: str password: description: - The password to apply to the user. required: false type: str version_added: 2.2.0 version_added_collection: community.aws password_reset_required: description: - Defines if the user is required to set a new password after login. required: false type: bool default: false version_added: 3.1.0 version_added_collection: community.aws update_password: default: always choices: ['always', 'on_create'] description: - When to update user passwords. - I(update_password=always) will ensure the password is set to I(password). - I(update_password=on_create) will only set the password for newly created users. type: str version_added: 2.2.0 version_added_collection: community.aws remove_password: description: - Option to delete user login passwords. - This field is mutually exclusive to I(password). type: 'bool' version_added: 2.2.0 version_added_collection: community.aws managed_policies: description: - A list of managed policy ARNs or friendly names to attach to the user. - To embed an inline policy, use M(community.aws.iam_policy). required: false type: list elements: str default: [] aliases: ['managed_policy'] state: description: - Create or remove the IAM user. required: true choices: [ 'present', 'absent' ] type: str purge_policies: description: - When I(purge_policies=true) any managed policies not listed in I(managed_policies) will be detached. required: false default: false type: bool aliases: ['purge_policy', 'purge_managed_policies'] wait: description: - When I(wait=True) the module will wait for up to I(wait_timeout) seconds for IAM user creation before returning. default: True type: bool version_added: 2.2.0 version_added_collection: community.aws wait_timeout: description: - How long (in seconds) to wait for creation / updates to complete. default: 120 type: int version_added: 2.2.0 version_added_collection: community.aws notes: - Support for I(tags) and I(purge_tags) was added in release 2.1.0. extends_documentation_fragment: - amazon.aws.aws - amazon.aws.ec2 - amazon.aws.tags - amazon.aws.boto3 ''' EXAMPLES = r''' # Note: These examples do not set authentication details, see the AWS Guide for details. # Note: This module does not allow management of groups that users belong to. # Groups should manage their membership directly using community.aws.iam_group, # as users belong to them. - name: Create a user amazon.aws.iam_user: name: testuser1 state: present - name: Create a user with a password amazon.aws.iam_user: name: testuser1 password: SomeSecurePassword state: present - name: Create a user and attach a managed policy using its ARN amazon.aws.iam_user: name: testuser1 managed_policies: - arn:aws:iam::aws:policy/AmazonSNSFullAccess state: present - name: Remove all managed policies from an existing user with an empty list amazon.aws.iam_user: name: testuser1 state: present purge_policies: true - name: Create user with tags amazon.aws.iam_user: name: testuser1 state: present tags: Env: Prod - name: Delete the user amazon.aws.iam_user: name: testuser1 state: absent ''' RETURN = r''' user: description: dictionary containing all the user information returned: success type: complex contains: arn: description: the Amazon Resource Name (ARN) specifying the user type: str sample: "arn:aws:iam::123456789012:user/testuser1" create_date: description: the date and time, in ISO 8601 date-time format, when the user was created type: str sample: "2017-02-08T04:36:28+00:00" user_id: description: the stable and unique string identifying the user type: str sample: "AGPA12345EXAMPLE54321" user_name: description: the friendly name that identifies the user type: str sample: "testuser1" path: description: the path to the user type: str sample: "/" tags: description: user tags type: dict returned: always sample: {"Env": "Prod"} ''' try: import botocore except ImportError: pass # caught by AnsibleAWSModule from ansible.module_utils.common.dict_transformations import camel_dict_to_snake_dict from ansible_collections.amazon.aws.plugins.module_utils.core import AnsibleAWSModule from ansible_collections.amazon.aws.plugins.module_utils.core import is_boto3_error_code from ansible_collections.amazon.aws.plugins.module_utils.ec2 import ansible_dict_to_boto3_tag_list from ansible_collections.amazon.aws.plugins.module_utils.ec2 import boto3_tag_list_to_ansible_dict from ansible_collections.amazon.aws.plugins.module_utils.ec2 import compare_aws_tags def compare_attached_policies(current_attached_policies, new_attached_policies): # If new_attached_policies is None it means we want to remove all policies if len(current_attached_policies) > 0 and new_attached_policies is None: return False current_attached_policies_arn_list = [] for policy in current_attached_policies: current_attached_policies_arn_list.append(policy['PolicyArn']) if not set(current_attached_policies_arn_list).symmetric_difference(set(new_attached_policies)): return True else: return False def convert_friendly_names_to_arns(connection, module, policy_names): # List comprehension that looks for any policy in the 'policy_names' list # that does not begin with 'arn'. If there aren't any, short circuit. # If there are, translate friendly name to the full arn if not any(not policy.startswith('arn:') for policy in policy_names if policy is not None): return policy_names allpolicies = {} paginator = connection.get_paginator('list_policies') policies = paginator.paginate().build_full_result()['Policies'] for policy in policies: allpolicies[policy['PolicyName']] = policy['Arn'] allpolicies[policy['Arn']] = policy['Arn'] try: return [allpolicies[policy] for policy in policy_names] except KeyError as e: module.fail_json(msg="Couldn't find policy: " + str(e)) def wait_iam_exists(connection, module): user_name = module.params.get('name') wait_timeout = module.params.get('wait_timeout') delay = min(wait_timeout, 5) max_attempts = wait_timeout // delay try: waiter = connection.get_waiter('user_exists') waiter.wait( WaiterConfig={'Delay': delay, 'MaxAttempts': max_attempts}, UserName=user_name, ) except botocore.exceptions.WaiterError as e: module.fail_json_aws(e, msg='Timeout while waiting on IAM user creation') except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg='Failed while waiting on IAM user creation') def create_or_update_login_profile(connection, module): # Apply new password / update password for the user user_params = dict() user_params['UserName'] = module.params.get('name') user_params['Password'] = module.params.get('password') user_params['PasswordResetRequired'] = module.params.get('password_reset_required') retval = {} try: retval = connection.update_login_profile(**user_params) except is_boto3_error_code('NoSuchEntity'): # Login profile does not yet exist - create it try: retval = connection.create_login_profile(**user_params) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to create user login profile") except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: # pylint: disable=duplicate-except module.fail_json_aws(e, msg="Unable to update user login profile") return True, retval def delete_login_profile(connection, module): ''' Deletes a users login profile. Parameters: connection: IAM client module: AWSModule Returns: (bool): True if login profile deleted, False if no login profile found to delete ''' user_params = dict() user_params['UserName'] = module.params.get('name') # User does not have login profile - nothing to delete if not user_has_login_profile(connection, module, user_params['UserName']): return False if not module.check_mode: try: connection.delete_login_profile(**user_params) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: # pylint: disable=duplicate-except module.fail_json_aws(e, msg="Unable to delete user login profile") return True def create_or_update_user(connection, module): params = dict() params['UserName'] = module.params.get('name') managed_policies = module.params.get('managed_policies') purge_policies = module.params.get('purge_policies') if module.params.get('tags') is not None: params["Tags"] = ansible_dict_to_boto3_tag_list(module.params.get('tags')) changed = False if managed_policies: managed_policies = convert_friendly_names_to_arns(connection, module, managed_policies) # Get user user = get_user(connection, module, params['UserName']) # If user is None, create it new_login_profile = False if user is None: # Check mode means we would create the user if module.check_mode: module.exit_json(changed=True) try: connection.create_user(**params) changed = True except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to create user") # Wait for user to be fully available before continuing if module.params.get('wait'): wait_iam_exists(connection, module) if module.params.get('password') is not None: login_profile_result, login_profile_data = create_or_update_login_profile(connection, module) if login_profile_data.get('LoginProfile', {}).get('PasswordResetRequired', False): new_login_profile = True else: login_profile_result = None update_result = update_user_tags(connection, module, params, user) if module.params['update_password'] == "always" and module.params.get('password') is not None: # Can't compare passwords, so just return changed on check mode runs if module.check_mode: module.exit_json(changed=True) login_profile_result, login_profile_data = create_or_update_login_profile(connection, module) if login_profile_data.get('LoginProfile', {}).get('PasswordResetRequired', False): new_login_profile = True elif module.params.get('remove_password'): login_profile_result = delete_login_profile(connection, module) changed = bool(update_result) or bool(login_profile_result) # Manage managed policies current_attached_policies = get_attached_policy_list(connection, module, params['UserName']) if not compare_attached_policies(current_attached_policies, managed_policies): current_attached_policies_arn_list = [] for policy in current_attached_policies: current_attached_policies_arn_list.append(policy['PolicyArn']) # If managed_policies has a single empty element we want to remove all attached policies if purge_policies: # Detach policies not present for policy_arn in list(set(current_attached_policies_arn_list) - set(managed_policies)): changed = True if not module.check_mode: try: connection.detach_user_policy(UserName=params['UserName'], PolicyArn=policy_arn) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to detach policy {0} from user {1}".format( policy_arn, params['UserName'])) # If there are policies to adjust that aren't in the current list, then things have changed # Otherwise the only changes were in purging above if set(managed_policies).difference(set(current_attached_policies_arn_list)): changed = True # If there are policies in managed_policies attach each policy if managed_policies != [None] and not module.check_mode: for policy_arn in managed_policies: try: connection.attach_user_policy(UserName=params['UserName'], PolicyArn=policy_arn) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to attach policy {0} to user {1}".format( policy_arn, params['UserName'])) if module.check_mode: module.exit_json(changed=changed) # Get the user again user = get_user(connection, module, params['UserName']) if changed and new_login_profile: # `LoginProfile` is only returned on `create_login_profile` method user['user']['password_reset_required'] = login_profile_data.get('LoginProfile', {}).get('PasswordResetRequired', False) module.exit_json(changed=changed, iam_user=user, user=user['user']) def destroy_user(connection, module): user_name = module.params.get('name') user = get_user(connection, module, user_name) # User is not present if not user: module.exit_json(changed=False) # Check mode means we would remove this user if module.check_mode: module.exit_json(changed=True) # Remove any attached policies otherwise deletion fails try: for policy in get_attached_policy_list(connection, module, user_name): connection.detach_user_policy(UserName=user_name, PolicyArn=policy['PolicyArn']) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to delete user {0}".format(user_name)) try: # Remove user's access keys access_keys = connection.list_access_keys(UserName=user_name)["AccessKeyMetadata"] for access_key in access_keys: connection.delete_access_key(UserName=user_name, AccessKeyId=access_key["AccessKeyId"]) # Remove user's login profile (console password) delete_login_profile(connection, module) # Remove user's ssh public keys ssh_public_keys = connection.list_ssh_public_keys(UserName=user_name)["SSHPublicKeys"] for ssh_public_key in ssh_public_keys: connection.delete_ssh_public_key(UserName=user_name, SSHPublicKeyId=ssh_public_key["SSHPublicKeyId"]) # Remove user's service specific credentials service_credentials = connection.list_service_specific_credentials(UserName=user_name)["ServiceSpecificCredentials"] for service_specific_credential in service_credentials: connection.delete_service_specific_credential( UserName=user_name, ServiceSpecificCredentialId=service_specific_credential["ServiceSpecificCredentialId"] ) # Remove user's signing certificates signing_certificates = connection.list_signing_certificates(UserName=user_name)["Certificates"] for signing_certificate in signing_certificates: connection.delete_signing_certificate( UserName=user_name, CertificateId=signing_certificate["CertificateId"] ) # Remove user's MFA devices mfa_devices = connection.list_mfa_devices(UserName=user_name)["MFADevices"] for mfa_device in mfa_devices: connection.deactivate_mfa_device(UserName=user_name, SerialNumber=mfa_device["SerialNumber"]) # Remove user's inline policies inline_policies = connection.list_user_policies(UserName=user_name)["PolicyNames"] for policy_name in inline_policies: connection.delete_user_policy(UserName=user_name, PolicyName=policy_name) # Remove user's group membership user_groups = connection.list_groups_for_user(UserName=user_name)["Groups"] for group in user_groups: connection.remove_user_from_group(UserName=user_name, GroupName=group["GroupName"]) connection.delete_user(UserName=user_name) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg="Unable to delete user {0}".format(user_name)) module.exit_json(changed=True) def get_user(connection, module, name): params = dict() params['UserName'] = name try: user = connection.get_user(**params) except is_boto3_error_code('NoSuchEntity'): return None except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: # pylint: disable=duplicate-except module.fail_json_aws(e, msg="Unable to get user {0}".format(name)) tags = boto3_tag_list_to_ansible_dict(user['User'].pop('Tags', [])) user = camel_dict_to_snake_dict(user) user['user']['tags'] = tags return user def get_attached_policy_list(connection, module, name): try: return connection.list_attached_user_policies(UserName=name)['AttachedPolicies'] except is_boto3_error_code('NoSuchEntity'): return None except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: # pylint: disable=duplicate-except module.fail_json_aws(e, msg="Unable to get policies for user {0}".format(name)) def user_has_login_profile(connection, module, name): ''' Returns whether or not given user has a login profile. Parameters: connection: IAM client module: AWSModule name (str): Username of user Returns: (bool): True if user had login profile, False if not ''' try: connection.get_login_profile(UserName=name) except is_boto3_error_code('NoSuchEntity'): return False except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: # pylint: disable=duplicate-except module.fail_json_aws(e, msg="Unable to get login profile for user {0}".format(name)) return True def update_user_tags(connection, module, params, user): user_name = params['UserName'] existing_tags = user['user']['tags'] new_tags = params.get('Tags') if new_tags is None: return False new_tags = boto3_tag_list_to_ansible_dict(new_tags) purge_tags = module.params.get('purge_tags') tags_to_add, tags_to_remove = compare_aws_tags(existing_tags, new_tags, purge_tags=purge_tags) if not module.check_mode: try: if tags_to_remove: connection.untag_user(UserName=user_name, TagKeys=tags_to_remove) if tags_to_add: connection.tag_user(UserName=user_name, Tags=ansible_dict_to_boto3_tag_list(tags_to_add)) except (botocore.exceptions.ClientError, botocore.exceptions.BotoCoreError) as e: module.fail_json_aws(e, msg='Unable to set tags for user %s' % user_name) changed = bool(tags_to_add) or bool(tags_to_remove) return changed def main(): argument_spec = dict( name=dict(required=True, type='str'), password=dict(type='str', no_log=True), password_reset_required=dict(type='bool', default=False, no_log=False), update_password=dict(default='always', choices=['always', 'on_create'], no_log=False), remove_password=dict(type='bool', no_log=False), managed_policies=dict(default=[], type='list', aliases=['managed_policy'], elements='str'), state=dict(choices=['present', 'absent'], required=True), purge_policies=dict(default=False, type='bool', aliases=['purge_policy', 'purge_managed_policies']), tags=dict(type='dict', aliases=['resource_tags']), purge_tags=dict(type='bool', default=True), wait=dict(type='bool', default=True), wait_timeout=dict(default=120, type='int'), ) module = AnsibleAWSModule( argument_spec=argument_spec, supports_check_mode=True, mutually_exclusive=[['password', 'remove_password']], ) module.deprecate("The 'iam_user' return key is deprecated and will be replaced by 'user'. Both values are returned for now.", date='2024-05-01', collection_name='amazon.aws') connection = module.client('iam') state = module.params.get("state") if state == 'present': create_or_update_user(connection, module) else: destroy_user(connection, module) if __name__ == '__main__': main()