oo_iam_kms.py 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. #!/usr/bin/env python
  2. '''
  3. ansible module for creating AWS IAM KMS keys
  4. '''
  5. # vim: expandtab:tabstop=4:shiftwidth=4
  6. #
  7. # AWS IAM KMS ansible module
  8. #
  9. #
  10. # Copyright 2016 Red Hat Inc.
  11. #
  12. # Licensed under the Apache License, Version 2.0 (the "License");
  13. # you may not use this file except in compliance with the License.
  14. # You may obtain a copy of the License at
  15. #
  16. # http://www.apache.org/licenses/LICENSE-2.0
  17. #
  18. # Unless required by applicable law or agreed to in writing, software
  19. # distributed under the License is distributed on an "AS IS" BASIS,
  20. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  21. # See the License for the specific language governing permissions and
  22. # limitations under the License.
  23. #
  24. # Jenkins environment doesn't have all the required libraries
  25. # pylint: disable=import-error
  26. import time
  27. import boto3
  28. # Ansible modules need this wildcard import
  29. # pylint: disable=unused-wildcard-import, wildcard-import, redefined-builtin
  30. from ansible.module_utils.basic import AnsibleModule
  31. AWS_ALIAS_URL = "http://docs.aws.amazon.com/kms/latest/developerguide/programming-aliases.html"
  32. class AwsIamKms(object):
  33. '''
  34. ansible module for AWS IAM KMS
  35. '''
  36. def __init__(self):
  37. ''' constructor '''
  38. self.module = None
  39. self.kms_client = None
  40. self.aliases = None
  41. @staticmethod
  42. def valid_alias_name(user_alias):
  43. ''' AWS KMS aliases must start with 'alias/' '''
  44. valid_start = 'alias/'
  45. if user_alias.startswith(valid_start):
  46. return True
  47. return False
  48. def get_all_kms_info(self):
  49. '''fetch all kms info and return them
  50. list_keys doesn't have information regarding aliases
  51. list_aliases doesn't have the full kms arn
  52. fetch both and join them on the targetKeyId
  53. '''
  54. aliases = self.kms_client.list_aliases()['Aliases']
  55. keys = self.kms_client.list_keys()['Keys']
  56. for alias in aliases:
  57. for key in keys:
  58. if 'TargetKeyId' in alias and 'KeyId' in key:
  59. if alias['TargetKeyId'] == key['KeyId']:
  60. alias.update(key)
  61. return aliases
  62. def get_kms_entry(self, user_alias, alias_list):
  63. ''' return single alias details from list of aliases '''
  64. for alias in alias_list:
  65. if user_alias == alias.get('AliasName', False):
  66. return alias
  67. msg = "Did not find alias {}".format(user_alias)
  68. self.module.exit_json(failed=True, results=msg)
  69. @staticmethod
  70. def exists(user_alias, alias_list):
  71. ''' Check if KMS alias already exists '''
  72. for alias in alias_list:
  73. if user_alias == alias.get('AliasName'):
  74. return True
  75. return False
  76. def main(self):
  77. ''' entry point for module '''
  78. self.module = AnsibleModule(
  79. argument_spec=dict(
  80. state=dict(default='list', choices=['list', 'present'], type='str'),
  81. region=dict(default=None, required=True, type='str'),
  82. alias=dict(default=None, type='str'),
  83. # description default cannot be None
  84. description=dict(default='', type='str'),
  85. aws_access_key=dict(default=None, type='str'),
  86. aws_secret_key=dict(default=None, type='str'),
  87. ),
  88. )
  89. state = self.module.params['state']
  90. aws_access_key = self.module.params['aws_access_key']
  91. aws_secret_key = self.module.params['aws_secret_key']
  92. if aws_access_key and aws_secret_key:
  93. boto3.setup_default_session(aws_access_key_id=aws_access_key,
  94. aws_secret_access_key=aws_secret_key,
  95. region_name=self.module.params['region'])
  96. else:
  97. boto3.setup_default_session(region_name=self.module.params['region'])
  98. self.kms_client = boto3.client('kms')
  99. aliases = self.get_all_kms_info()
  100. if state == 'list':
  101. if self.module.params['alias'] is not None:
  102. user_kms = self.get_kms_entry(self.module.params['alias'],
  103. aliases)
  104. self.module.exit_json(changed=False, results=user_kms,
  105. state="list")
  106. else:
  107. self.module.exit_json(changed=False, results=aliases,
  108. state="list")
  109. if state == 'present':
  110. # early sanity check to make sure the alias name conforms with
  111. # AWS alias name requirements
  112. if not self.valid_alias_name(self.module.params['alias']):
  113. self.module.exit_json(failed=True, changed=False,
  114. results="Alias must start with the prefix " +
  115. "'alias/'. Please see " + AWS_ALIAS_URL,
  116. state='present')
  117. if not self.exists(self.module.params['alias'], aliases):
  118. # if we didn't find it, create it
  119. response = self.kms_client.create_key(KeyUsage='ENCRYPT_DECRYPT',
  120. Description=self.module.params['description'])
  121. kid = response['KeyMetadata']['KeyId']
  122. response = self.kms_client.create_alias(AliasName=self.module.params['alias'],
  123. TargetKeyId=kid)
  124. # sleep for a bit so that the KMS data can be queried
  125. time.sleep(10)
  126. # get details for newly created KMS entry
  127. new_alias_list = self.kms_client.list_aliases()['Aliases']
  128. user_kms = self.get_kms_entry(self.module.params['alias'],
  129. new_alias_list)
  130. self.module.exit_json(changed=True, results=user_kms,
  131. state='present')
  132. # already exists, normally we would check whether we need to update it
  133. # but this module isn't written to allow changing the alias name
  134. # or changing whether the key is enabled/disabled
  135. user_kms = self.get_kms_entry(self.module.params['alias'], aliases)
  136. self.module.exit_json(changed=False, results=user_kms,
  137. state="present")
  138. self.module.exit_json(failed=True,
  139. changed=False,
  140. results='Unknown state passed. %s' % state,
  141. state="unknown")
  142. if __name__ == '__main__':
  143. AwsIamKms().main()