openstack.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. #!/usr/bin/env python
  2. # pylint: skip-file
  3. # Copyright (c) 2012, Marco Vito Moscaritolo <marco@agavee.com>
  4. # Copyright (c) 2013, Jesse Keating <jesse.keating@rackspace.com>
  5. # Copyright (c) 2015, Hewlett-Packard Development Company, L.P.
  6. # Copyright (c) 2016, Rackspace Australia
  7. #
  8. # This module is free software: you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License as published by
  10. # the Free Software Foundation, either version 3 of the License, or
  11. # (at your option) any later version.
  12. #
  13. # This software is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU General Public License
  19. # along with this software. If not, see <http://www.gnu.org/licenses/>.
  20. # The OpenStack Inventory module uses os-client-config for configuration.
  21. # https://github.com/stackforge/os-client-config
  22. # This means it will either:
  23. # - Respect normal OS_* environment variables like other OpenStack tools
  24. # - Read values from a clouds.yaml file.
  25. # If you want to configure via clouds.yaml, you can put the file in:
  26. # - Current directory
  27. # - ~/.config/openstack/clouds.yaml
  28. # - /etc/openstack/clouds.yaml
  29. # - /etc/ansible/openstack.yml
  30. # The clouds.yaml file can contain entries for multiple clouds and multiple
  31. # regions of those clouds. If it does, this inventory module will connect to
  32. # all of them and present them as one contiguous inventory.
  33. #
  34. # See the adjacent openstack.yml file for an example config file
  35. # There are two ansible inventory specific options that can be set in
  36. # the inventory section.
  37. # expand_hostvars controls whether or not the inventory will make extra API
  38. # calls to fill out additional information about each server
  39. # use_hostnames changes the behavior from registering every host with its UUID
  40. # and making a group of its hostname to only doing this if the
  41. # hostname in question has more than one server
  42. # fail_on_errors causes the inventory to fail and return no hosts if one cloud
  43. # has failed (for example, bad credentials or being offline).
  44. # When set to False, the inventory will return hosts from
  45. # whichever other clouds it can contact. (Default: True)
  46. import argparse
  47. import collections
  48. import os
  49. import sys
  50. import time
  51. from distutils.version import StrictVersion
  52. try:
  53. import json
  54. except:
  55. import simplejson as json
  56. import os_client_config
  57. import shade
  58. import shade.inventory
  59. CONFIG_FILES = ['/etc/ansible/openstack.yaml', '/etc/ansible/openstack.yml']
  60. def get_groups_from_server(server_vars, namegroup=True):
  61. groups = []
  62. region = server_vars['region']
  63. cloud = server_vars['cloud']
  64. metadata = server_vars.get('metadata', {})
  65. # Create a group for the cloud
  66. groups.append(cloud)
  67. # Create a group on region
  68. groups.append(region)
  69. # And one by cloud_region
  70. groups.append("%s_%s" % (cloud, region))
  71. # Check if group metadata key in servers' metadata
  72. if 'group' in metadata:
  73. groups.append(metadata['group'])
  74. for extra_group in metadata.get('groups', '').split(','):
  75. if extra_group:
  76. groups.append(extra_group.strip())
  77. groups.append('instance-%s' % server_vars['id'])
  78. if namegroup:
  79. groups.append(server_vars['name'])
  80. for key in ('flavor', 'image'):
  81. if 'name' in server_vars[key]:
  82. groups.append('%s-%s' % (key, server_vars[key]['name']))
  83. for key, value in iter(metadata.items()):
  84. groups.append('meta-%s_%s' % (key, value))
  85. az = server_vars.get('az', None)
  86. if az:
  87. # Make groups for az, region_az and cloud_region_az
  88. groups.append(az)
  89. groups.append('%s_%s' % (region, az))
  90. groups.append('%s_%s_%s' % (cloud, region, az))
  91. return groups
  92. def get_host_groups(inventory, refresh=False):
  93. (cache_file, cache_expiration_time) = get_cache_settings()
  94. if is_cache_stale(cache_file, cache_expiration_time, refresh=refresh):
  95. groups = to_json(get_host_groups_from_cloud(inventory))
  96. open(cache_file, 'w').write(groups)
  97. else:
  98. groups = open(cache_file, 'r').read()
  99. return groups
  100. def append_hostvars(hostvars, groups, key, server, namegroup=False):
  101. hostvars[key] = dict(
  102. ansible_ssh_host=server['interface_ip'],
  103. openstack=server)
  104. for group in get_groups_from_server(server, namegroup=namegroup):
  105. groups[group].append(key)
  106. def get_host_groups_from_cloud(inventory):
  107. groups = collections.defaultdict(list)
  108. firstpass = collections.defaultdict(list)
  109. hostvars = {}
  110. list_args = {}
  111. if hasattr(inventory, 'extra_config'):
  112. use_hostnames = inventory.extra_config['use_hostnames']
  113. list_args['expand'] = inventory.extra_config['expand_hostvars']
  114. if StrictVersion(shade.__version__) >= StrictVersion("1.6.0"):
  115. list_args['fail_on_cloud_config'] = \
  116. inventory.extra_config['fail_on_errors']
  117. else:
  118. use_hostnames = False
  119. for server in inventory.list_hosts(**list_args):
  120. if 'interface_ip' not in server:
  121. continue
  122. firstpass[server['name']].append(server)
  123. for name, servers in firstpass.items():
  124. if len(servers) == 1 and use_hostnames:
  125. append_hostvars(hostvars, groups, name, servers[0])
  126. else:
  127. server_ids = set()
  128. # Trap for duplicate results
  129. for server in servers:
  130. server_ids.add(server['id'])
  131. if len(server_ids) == 1 and use_hostnames:
  132. append_hostvars(hostvars, groups, name, servers[0])
  133. else:
  134. for server in servers:
  135. append_hostvars(
  136. hostvars, groups, server['id'], server,
  137. namegroup=True)
  138. groups['_meta'] = {'hostvars': hostvars}
  139. return groups
  140. def is_cache_stale(cache_file, cache_expiration_time, refresh=False):
  141. ''' Determines if cache file has expired, or if it is still valid '''
  142. if refresh:
  143. return True
  144. if os.path.isfile(cache_file) and os.path.getsize(cache_file) > 0:
  145. mod_time = os.path.getmtime(cache_file)
  146. current_time = time.time()
  147. if (mod_time + cache_expiration_time) > current_time:
  148. return False
  149. return True
  150. def get_cache_settings():
  151. config = os_client_config.config.OpenStackConfig(
  152. config_files=os_client_config.config.CONFIG_FILES + CONFIG_FILES)
  153. # For inventory-wide caching
  154. cache_expiration_time = config.get_cache_expiration_time()
  155. cache_path = config.get_cache_path()
  156. if not os.path.exists(cache_path):
  157. os.makedirs(cache_path)
  158. cache_file = os.path.join(cache_path, 'ansible-inventory.cache')
  159. return (cache_file, cache_expiration_time)
  160. def to_json(in_dict):
  161. return json.dumps(in_dict, sort_keys=True, indent=2)
  162. def parse_args():
  163. parser = argparse.ArgumentParser(description='OpenStack Inventory Module')
  164. parser.add_argument('--private',
  165. action='store_true',
  166. help='Use private address for ansible host')
  167. parser.add_argument('--refresh', action='store_true',
  168. help='Refresh cached information')
  169. parser.add_argument('--debug', action='store_true', default=False,
  170. help='Enable debug output')
  171. group = parser.add_mutually_exclusive_group(required=True)
  172. group.add_argument('--list', action='store_true',
  173. help='List active servers')
  174. group.add_argument('--host', help='List details about the specific host')
  175. return parser.parse_args()
  176. def main():
  177. args = parse_args()
  178. try:
  179. config_files = os_client_config.config.CONFIG_FILES + CONFIG_FILES
  180. shade.simple_logging(debug=args.debug)
  181. inventory_args = dict(
  182. refresh=args.refresh,
  183. config_files=config_files,
  184. private=args.private,
  185. )
  186. if hasattr(shade.inventory.OpenStackInventory, 'extra_config'):
  187. inventory_args.update(dict(
  188. config_key='ansible',
  189. config_defaults={
  190. 'use_hostnames': False,
  191. 'expand_hostvars': True,
  192. 'fail_on_errors': True,
  193. }
  194. ))
  195. inventory = shade.inventory.OpenStackInventory(**inventory_args)
  196. if args.list:
  197. output = get_host_groups(inventory, refresh=args.refresh)
  198. elif args.host:
  199. output = to_json(inventory.get_host(args.host))
  200. print(output)
  201. except shade.OpenStackCloudException as e:
  202. sys.stderr.write('%s\n' % e.message)
  203. sys.exit(1)
  204. sys.exit(0)
  205. if __name__ == '__main__':
  206. main()