ec2_ami_find.py 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. #!/usr/bin/python
  2. #pylint: skip-file
  3. # flake8: noqa
  4. #
  5. # This file is part of Ansible
  6. #
  7. # Ansible is free software: you can redistribute it and/or modify
  8. # it under the terms of the GNU General Public License as published by
  9. # the Free Software Foundation, either version 3 of the License, or
  10. # (at your option) any later version.
  11. #
  12. # Ansible is distributed in the hope that it will be useful,
  13. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. # GNU General Public License for more details.
  16. #
  17. # You should have received a copy of the GNU General Public License
  18. # along with Ansible. If not, see <http://www.gnu.org/licenses/>.
  19. DOCUMENTATION = '''
  20. ---
  21. module: ec2_ami_find
  22. version_added: 2.0
  23. short_description: Searches for AMIs to obtain the AMI ID and other information
  24. description:
  25. - Returns list of matching AMIs with AMI ID, along with other useful information
  26. - Can search AMIs with different owners
  27. - Can search by matching tag(s), by AMI name and/or other criteria
  28. - Results can be sorted and sliced
  29. author: Tom Bamford
  30. notes:
  31. - This module is not backwards compatible with the previous version of the ec2_search_ami module which worked only for Ubuntu AMIs listed on cloud-images.ubuntu.com.
  32. - See the example below for a suggestion of how to search by distro/release.
  33. options:
  34. region:
  35. description:
  36. - The AWS region to use.
  37. required: true
  38. aliases: [ 'aws_region', 'ec2_region' ]
  39. owner:
  40. description:
  41. - Search AMIs owned by the specified owner
  42. - Can specify an AWS account ID, or one of the special IDs 'self', 'amazon' or 'aws-marketplace'
  43. - If not specified, all EC2 AMIs in the specified region will be searched.
  44. - You can include wildcards in many of the search options. An asterisk (*) matches zero or more characters, and a question mark (?) matches exactly one character. You can escape special characters using a backslash (\) before the character. For example, a value of \*amazon\?\\ searches for the literal string *amazon?\.
  45. required: false
  46. default: null
  47. ami_id:
  48. description:
  49. - An AMI ID to match.
  50. default: null
  51. required: false
  52. ami_tags:
  53. description:
  54. - A hash/dictionary of tags to match for the AMI.
  55. default: null
  56. required: false
  57. architecture:
  58. description:
  59. - An architecture type to match (e.g. x86_64).
  60. default: null
  61. required: false
  62. hypervisor:
  63. description:
  64. - A hypervisor type type to match (e.g. xen).
  65. default: null
  66. required: false
  67. is_public:
  68. description:
  69. - Whether or not the image(s) are public.
  70. choices: ['yes', 'no']
  71. default: null
  72. required: false
  73. name:
  74. description:
  75. - An AMI name to match.
  76. default: null
  77. required: false
  78. platform:
  79. description:
  80. - Platform type to match.
  81. default: null
  82. required: false
  83. sort:
  84. description:
  85. - Optional attribute which with to sort the results.
  86. - If specifying 'tag', the 'tag_name' parameter is required.
  87. choices: ['name', 'description', 'tag']
  88. default: null
  89. required: false
  90. sort_tag:
  91. description:
  92. - Tag name with which to sort results.
  93. - Required when specifying 'sort=tag'.
  94. default: null
  95. required: false
  96. sort_order:
  97. description:
  98. - Order in which to sort results.
  99. - Only used when the 'sort' parameter is specified.
  100. choices: ['ascending', 'descending']
  101. default: 'ascending'
  102. required: false
  103. sort_start:
  104. description:
  105. - Which result to start with (when sorting).
  106. - Corresponds to Python slice notation.
  107. default: null
  108. required: false
  109. sort_end:
  110. description:
  111. - Which result to end with (when sorting).
  112. - Corresponds to Python slice notation.
  113. default: null
  114. required: false
  115. state:
  116. description:
  117. - AMI state to match.
  118. default: 'available'
  119. required: false
  120. virtualization_type:
  121. description:
  122. - Virtualization type to match (e.g. hvm).
  123. default: null
  124. required: false
  125. no_result_action:
  126. description:
  127. - What to do when no results are found.
  128. - "'success' reports success and returns an empty array"
  129. - "'fail' causes the module to report failure"
  130. choices: ['success', 'fail']
  131. default: 'success'
  132. required: false
  133. requirements:
  134. - boto
  135. '''
  136. EXAMPLES = '''
  137. # Note: These examples do not set authentication details, see the AWS Guide for details.
  138. # Search for the AMI tagged "project:website"
  139. - ec2_ami_find:
  140. owner: self
  141. tags:
  142. project: website
  143. no_result_action: fail
  144. register: ami_find
  145. # Search for the latest Ubuntu 14.04 AMI
  146. - ec2_ami_find:
  147. name: "ubuntu/images/ebs/ubuntu-trusty-14.04-amd64-server-*"
  148. owner: 099720109477
  149. sort: name
  150. sort_order: descending
  151. sort_end: 1
  152. register: ami_find
  153. # Launch an EC2 instance
  154. - ec2:
  155. image: "{{ ami_search.results[0].ami_id }}"
  156. instance_type: m4.medium
  157. key_name: mykey
  158. wait: yes
  159. '''
  160. try:
  161. import boto.ec2
  162. HAS_BOTO=True
  163. except ImportError:
  164. HAS_BOTO=False
  165. import json
  166. def main():
  167. argument_spec = ec2_argument_spec()
  168. argument_spec.update(dict(
  169. region = dict(required=True,
  170. aliases = ['aws_region', 'ec2_region']),
  171. owner = dict(required=False, default=None),
  172. ami_id = dict(required=False),
  173. ami_tags = dict(required=False, type='dict',
  174. aliases = ['search_tags', 'image_tags']),
  175. architecture = dict(required=False),
  176. hypervisor = dict(required=False),
  177. is_public = dict(required=False),
  178. name = dict(required=False),
  179. platform = dict(required=False),
  180. sort = dict(required=False, default=None,
  181. choices=['name', 'description', 'tag']),
  182. sort_tag = dict(required=False),
  183. sort_order = dict(required=False, default='ascending',
  184. choices=['ascending', 'descending']),
  185. sort_start = dict(required=False),
  186. sort_end = dict(required=False),
  187. state = dict(required=False, default='available'),
  188. virtualization_type = dict(required=False),
  189. no_result_action = dict(required=False, default='success',
  190. choices = ['success', 'fail']),
  191. )
  192. )
  193. module = AnsibleModule(
  194. argument_spec=argument_spec,
  195. )
  196. if not HAS_BOTO:
  197. module.fail_json(msg='boto required for this module, install via pip or your package manager')
  198. ami_id = module.params.get('ami_id')
  199. ami_tags = module.params.get('ami_tags')
  200. architecture = module.params.get('architecture')
  201. hypervisor = module.params.get('hypervisor')
  202. is_public = module.params.get('is_public')
  203. name = module.params.get('name')
  204. owner = module.params.get('owner')
  205. platform = module.params.get('platform')
  206. sort = module.params.get('sort')
  207. sort_tag = module.params.get('sort_tag')
  208. sort_order = module.params.get('sort_order')
  209. sort_start = module.params.get('sort_start')
  210. sort_end = module.params.get('sort_end')
  211. state = module.params.get('state')
  212. virtualization_type = module.params.get('virtualization_type')
  213. no_result_action = module.params.get('no_result_action')
  214. filter = {'state': state}
  215. if ami_id:
  216. filter['image_id'] = ami_id
  217. if ami_tags:
  218. for tag in ami_tags:
  219. filter['tag:'+tag] = ami_tags[tag]
  220. if architecture:
  221. filter['architecture'] = architecture
  222. if hypervisor:
  223. filter['hypervisor'] = hypervisor
  224. if is_public:
  225. filter['is_public'] = is_public
  226. if name:
  227. filter['name'] = name
  228. if platform:
  229. filter['platform'] = platform
  230. if virtualization_type:
  231. filter['virtualization_type'] = virtualization_type
  232. ec2 = ec2_connect(module)
  233. images_result = ec2.get_all_images(owners=owner, filters=filter)
  234. if no_result_action == 'fail' and len(images_result) == 0:
  235. module.fail_json(msg="No AMIs matched the attributes: %s" % json.dumps(filter))
  236. results = []
  237. for image in images_result:
  238. data = {
  239. 'ami_id': image.id,
  240. 'architecture': image.architecture,
  241. 'description': image.description,
  242. 'is_public': image.is_public,
  243. 'name': image.name,
  244. 'owner_id': image.owner_id,
  245. 'platform': image.platform,
  246. 'root_device_name': image.root_device_name,
  247. 'root_device_type': image.root_device_type,
  248. 'state': image.state,
  249. 'tags': image.tags,
  250. 'virtualization_type': image.virtualization_type,
  251. }
  252. if image.kernel_id:
  253. data['kernel_id'] = image.kernel_id
  254. if image.ramdisk_id:
  255. data['ramdisk_id'] = image.ramdisk_id
  256. results.append(data)
  257. if sort == 'tag':
  258. if not sort_tag:
  259. module.fail_json(msg="'sort_tag' option must be given with 'sort=tag'")
  260. results.sort(key=lambda e: e['tags'][sort_tag], reverse=(sort_order=='descending'))
  261. elif sort:
  262. results.sort(key=lambda e: e[sort], reverse=(sort_order=='descending'))
  263. try:
  264. if sort and sort_start and sort_end:
  265. results = results[int(sort_start):int(sort_end)]
  266. elif sort and sort_start:
  267. results = results[int(sort_start):]
  268. elif sort and sort_end:
  269. results = results[:int(sort_end)]
  270. except TypeError:
  271. module.fail_json(msg="Please supply numeric values for sort_start and/or sort_end")
  272. module.exit_json(results=results)
  273. # import module snippets
  274. from ansible.module_utils.basic import *
  275. from ansible.module_utils.ec2 import *
  276. if __name__ == '__main__':
  277. main()