inventory.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. #!/usr/bin/env python
  2. """
  3. This is an Ansible dynamic inventory for OpenStack.
  4. It requires your OpenStack credentials to be set in clouds.yaml or your shell
  5. environment.
  6. """
  7. from __future__ import print_function
  8. from collections import Mapping
  9. import json
  10. import os
  11. from keystoneauth1.exceptions.catalog import EndpointNotFound
  12. import shade
  13. def base_openshift_inventory(cluster_hosts):
  14. '''Set the base openshift inventory.'''
  15. inventory = {}
  16. masters = [server.name for server in cluster_hosts
  17. if server.metadata['host-type'] == 'master']
  18. etcd = [server.name for server in cluster_hosts
  19. if server.metadata['host-type'] == 'etcd']
  20. if not etcd:
  21. etcd = masters
  22. infra_hosts = [server.name for server in cluster_hosts
  23. if server.metadata['host-type'] == 'node' and
  24. server.metadata['sub-host-type'] == 'infra']
  25. app = [server.name for server in cluster_hosts
  26. if server.metadata['host-type'] == 'node' and
  27. server.metadata['sub-host-type'] == 'app']
  28. cns = [server.name for server in cluster_hosts
  29. if server.metadata['host-type'] == 'cns']
  30. nodes = list(set(masters + infra_hosts + app + cns))
  31. dns = [server.name for server in cluster_hosts
  32. if server.metadata['host-type'] == 'dns']
  33. load_balancers = [server.name for server in cluster_hosts
  34. if server.metadata['host-type'] == 'lb']
  35. osev3 = list(set(nodes + etcd + load_balancers))
  36. inventory['cluster_hosts'] = {'hosts': [s.name for s in cluster_hosts]}
  37. inventory['OSEv3'] = {'hosts': osev3, 'vars': {}}
  38. inventory['masters'] = {'hosts': masters}
  39. inventory['etcd'] = {'hosts': etcd}
  40. inventory['nodes'] = {'hosts': nodes}
  41. inventory['infra_hosts'] = {'hosts': infra_hosts}
  42. inventory['app'] = {'hosts': app}
  43. inventory['glusterfs'] = {'hosts': cns}
  44. inventory['dns'] = {'hosts': dns}
  45. inventory['lb'] = {'hosts': load_balancers}
  46. inventory['localhost'] = {'ansible_connection': 'local'}
  47. return inventory
  48. def get_docker_storage_mountpoints(volumes):
  49. '''Check volumes to see if they're being used for docker storage'''
  50. docker_storage_mountpoints = {}
  51. for volume in volumes:
  52. if volume.metadata.get('purpose') == "openshift_docker_storage":
  53. for attachment in volume.attachments:
  54. if attachment.server_id in docker_storage_mountpoints:
  55. docker_storage_mountpoints[attachment.server_id].append(attachment.device)
  56. else:
  57. docker_storage_mountpoints[attachment.server_id] = [attachment.device]
  58. return docker_storage_mountpoints
  59. def _get_hostvars(server, docker_storage_mountpoints):
  60. ssh_ip_address = server.public_v4 or server.private_v4
  61. hostvars = {
  62. 'ansible_host': ssh_ip_address
  63. }
  64. public_v4 = server.public_v4 or server.private_v4
  65. if public_v4:
  66. hostvars['public_v4'] = server.public_v4
  67. hostvars['openshift_public_ip'] = server.public_v4
  68. # TODO(shadower): what about multiple networks?
  69. if server.private_v4:
  70. hostvars['private_v4'] = server.private_v4
  71. hostvars['openshift_ip'] = server.private_v4
  72. # NOTE(shadower): Yes, we set both hostname and IP to the private
  73. # IP address for each node. OpenStack doesn't resolve nodes by
  74. # name at all, so using a hostname here would require an internal
  75. # DNS which would complicate the setup and potentially introduce
  76. # performance issues.
  77. hostvars['openshift_hostname'] = server.metadata.get(
  78. 'openshift_hostname', server.private_v4)
  79. hostvars['openshift_public_hostname'] = server.name
  80. if server.metadata['host-type'] == 'cns':
  81. hostvars['glusterfs_devices'] = ['/dev/nvme0n1']
  82. node_labels = server.metadata.get('node_labels')
  83. # NOTE(shadower): the node_labels value must be a dict not string
  84. if not isinstance(node_labels, Mapping):
  85. node_labels = json.loads(node_labels)
  86. if node_labels:
  87. hostvars['openshift_node_labels'] = node_labels
  88. # check for attached docker storage volumes
  89. if 'os-extended-volumes:volumes_attached' in server:
  90. if server.id in docker_storage_mountpoints:
  91. hostvars['docker_storage_mountpoints'] = ' '.join(
  92. docker_storage_mountpoints[server.id])
  93. return hostvars
  94. def build_inventory():
  95. '''Build the dynamic inventory.'''
  96. cloud = shade.openstack_cloud()
  97. # TODO(shadower): filter the servers based on the `OPENSHIFT_CLUSTER`
  98. # environment variable.
  99. cluster_hosts = [
  100. server for server in cloud.list_servers()
  101. if 'metadata' in server and 'clusterid' in server.metadata]
  102. inventory = base_openshift_inventory(cluster_hosts)
  103. for server in cluster_hosts:
  104. if 'group' in server.metadata:
  105. group = server.metadata.get('group')
  106. if group not in inventory:
  107. inventory[group] = {'hosts': []}
  108. inventory[group]['hosts'].append(server.name)
  109. inventory['_meta'] = {'hostvars': {}}
  110. # Some clouds don't have Cinder. That's okay:
  111. try:
  112. volumes = cloud.list_volumes()
  113. except EndpointNotFound:
  114. volumes = []
  115. # cinder volumes used for docker storage
  116. docker_storage_mountpoints = get_docker_storage_mountpoints(volumes)
  117. for server in cluster_hosts:
  118. inventory['_meta']['hostvars'][server.name] = _get_hostvars(
  119. server,
  120. docker_storage_mountpoints)
  121. stout = _get_stack_outputs(cloud)
  122. if stout is not None:
  123. try:
  124. inventory['localhost'].update({
  125. 'openshift_openstack_api_lb_provider':
  126. stout['api_lb_provider'],
  127. 'openshift_openstack_api_lb_port_id':
  128. stout['api_lb_vip_port_id'],
  129. 'openshift_openstack_api_lb_sg_id':
  130. stout['api_lb_sg_id']})
  131. except KeyError:
  132. pass # Not an API load balanced deployment
  133. try:
  134. inventory['OSEv3']['vars'][
  135. 'openshift_master_cluster_hostname'] = stout['private_api_ip']
  136. except KeyError:
  137. pass # Internal LB not specified
  138. inventory['localhost']['openshift_openstack_public_api_ip'] = \
  139. stout.get('public_api_ip')
  140. inventory['localhost']['openshift_openstack_public_router_ip'] = \
  141. stout.get('public_router_ip')
  142. try:
  143. inventory['OSEv3']['vars'] = _get_kuryr_vars(cloud, stout)
  144. except KeyError:
  145. pass # Not a kuryr deployment
  146. return inventory
  147. def _get_stack_outputs(cloud_client):
  148. """Returns a dictionary with the stack outputs"""
  149. cluster_name = os.getenv('OPENSHIFT_CLUSTER', 'openshift-cluster')
  150. stack = cloud_client.get_stack(cluster_name)
  151. if stack is None or stack['stack_status'] not in (
  152. 'CREATE_COMPLETE', 'UPDATE_COMPLETE'):
  153. return None
  154. data = {}
  155. for output in stack['outputs']:
  156. data[output['output_key']] = output['output_value']
  157. return data
  158. def _get_kuryr_vars(cloud_client, data):
  159. """Returns a dictionary of Kuryr variables resulting of heat stacking"""
  160. settings = {}
  161. settings['kuryr_openstack_pod_subnet_id'] = data['pod_subnet']
  162. settings['kuryr_openstack_worker_nodes_subnet_id'] = data['vm_subnet']
  163. settings['kuryr_openstack_service_subnet_id'] = data['service_subnet']
  164. settings['kuryr_openstack_pod_sg_id'] = data['pod_access_sg_id']
  165. settings['kuryr_openstack_pod_project_id'] = (
  166. cloud_client.current_project_id)
  167. settings['kuryr_openstack_api_lb_ip'] = data['private_api_ip']
  168. settings['kuryr_openstack_auth_url'] = cloud_client.auth['auth_url']
  169. settings['kuryr_openstack_username'] = cloud_client.auth['username']
  170. settings['kuryr_openstack_password'] = cloud_client.auth['password']
  171. if 'user_domain_id' in cloud_client.auth:
  172. settings['kuryr_openstack_user_domain_name'] = (
  173. cloud_client.auth['user_domain_id'])
  174. else:
  175. settings['kuryr_openstack_user_domain_name'] = (
  176. cloud_client.auth['user_domain_name'])
  177. # FIXME(apuimedo): consolidate kuryr controller credentials into the same
  178. # vars the openstack playbook uses.
  179. settings['kuryr_openstack_project_id'] = cloud_client.current_project_id
  180. if 'project_domain_id' in cloud_client.auth:
  181. settings['kuryr_openstack_project_domain_name'] = (
  182. cloud_client.auth['project_domain_id'])
  183. else:
  184. settings['kuryr_openstack_project_domain_name'] = (
  185. cloud_client.auth['project_domain_name'])
  186. return settings
  187. if __name__ == '__main__':
  188. print(json.dumps(build_inventory(), indent=4, sort_keys=True))