node_group_checks.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. """
  2. Ansible action plugin to ensure inventory variables are set
  3. appropriately related to openshift_node_group_name
  4. """
  5. from ansible.plugins.action import ActionBase
  6. from ansible import errors
  7. # Runs on first master
  8. # Checks each openshift_node_group_name is found in openshift_node_groups
  9. # Checks that master label is present in one of those groups
  10. # Checks that node label is present in one of those groups
  11. def get_or_fail(group, key):
  12. """Find a key in a group dictionary or fail"""
  13. res = group.get(key)
  14. if res is None:
  15. msg = "Each group in openshift_node_groups must have {} key".format(key)
  16. raise errors.AnsibleModuleError(msg)
  17. return res
  18. def validate_labels(labels_found):
  19. """Ensure mandatory_labels are found in the labels we found, labels_found"""
  20. mandatory_labels = ('node-role.kubernetes.io/master=true',
  21. 'node-role.kubernetes.io/infra=true')
  22. for item in mandatory_labels:
  23. if item not in labels_found:
  24. msg = ("At least one group in openshift_node_groups requires the"
  25. " {} label").format(item)
  26. raise errors.AnsibleModuleError(msg)
  27. def process_group(group, groups_found, labels_found):
  28. """Validate format of each group in openshift_node_groups"""
  29. name = get_or_fail(group, 'name')
  30. if name in groups_found:
  31. msg = ("Duplicate definition of group {} in"
  32. " openshift_node_groups").format(name)
  33. raise errors.AnsibleModuleError(msg)
  34. groups_found.add(name)
  35. labels = get_or_fail(group, 'labels')
  36. if not issubclass(type(labels), list):
  37. msg = "labels value of each group in openshift_node_groups must be a list"
  38. raise errors.AnsibleModuleError(msg)
  39. labels_found.update(labels)
  40. class ActionModule(ActionBase):
  41. """Action plugin to execute node_group_checks."""
  42. def template_var(self, hostvars, host, varname):
  43. """Retrieve a variable from hostvars and template it.
  44. If undefined, return None type."""
  45. # We will set the current host and variable checked for easy debugging
  46. # if there are any unhandled exceptions.
  47. # pylint: disable=W0201
  48. self.last_checked_var = varname
  49. # pylint: disable=W0201
  50. self.last_checked_host = host
  51. res = hostvars[host].get(varname)
  52. if res is None:
  53. return None
  54. return self._templar.template(res)
  55. def get_node_group_name(self, hostvars, host):
  56. """Ensure openshift_node_group_name is defined for nodes"""
  57. group_name = self.template_var(hostvars, host, 'openshift_node_group_name')
  58. if not group_name:
  59. msg = "openshift_node_group_name must be defined for all nodes"
  60. raise errors.AnsibleModuleError(msg)
  61. return group_name
  62. def run_check(self, hostvars, host, groups_found):
  63. """Run the check for each host"""
  64. group_name = self.get_node_group_name(hostvars, host)
  65. if group_name not in groups_found:
  66. msg = "Group: {} not found in openshift_node_groups".format(group_name)
  67. raise errors.AnsibleModuleError(msg)
  68. def run(self, tmp=None, task_vars=None):
  69. """Run node_group_checks action plugin"""
  70. result = super(ActionModule, self).run(tmp, task_vars)
  71. result["changed"] = False
  72. result["failed"] = False
  73. result["msg"] = "Node group checks passed"
  74. # self.task_vars holds all in-scope variables.
  75. # Ignore settting self.task_vars outside of init.
  76. # pylint: disable=W0201
  77. self.task_vars = task_vars or {}
  78. # pylint: disable=W0201
  79. self.last_checked_host = "none"
  80. # pylint: disable=W0201
  81. self.last_checked_var = "none"
  82. # check_hosts is hard-set to oo_nodes_to_config
  83. check_hosts = self.task_vars['groups'].get('oo_nodes_to_config')
  84. if not check_hosts:
  85. result["msg"] = "skipping; oo_nodes_to_config is required for this check"
  86. return result
  87. # We need to access each host's variables
  88. hostvars = self.task_vars.get('hostvars')
  89. if not hostvars:
  90. msg = hostvars
  91. raise errors.AnsibleModuleError(msg)
  92. openshift_node_groups = self.task_vars.get('openshift_node_groups')
  93. if not openshift_node_groups:
  94. msg = "openshift_node_groups undefined"
  95. raise errors.AnsibleModuleError(msg)
  96. openshift_node_groups = self._templar.template(openshift_node_groups)
  97. groups_found = set()
  98. labels_found = set()
  99. # gather the groups and labels we believe should be present.
  100. for group in openshift_node_groups:
  101. process_group(group, groups_found, labels_found)
  102. if len(groups_found) == 0:
  103. msg = "No groups found in openshift_node_groups"
  104. raise errors.AnsibleModuleError(msg)
  105. validate_labels(labels_found)
  106. # We loop through each host in the provided list check_hosts
  107. for host in check_hosts:
  108. try:
  109. self.run_check(hostvars, host, groups_found)
  110. except Exception as uncaught_e:
  111. msg = "last_checked_host: {}, last_checked_var: {};"
  112. msg = msg.format(self.last_checked_host, self.last_checked_var)
  113. msg += str(uncaught_e)
  114. raise errors.AnsibleModuleError(msg)
  115. return result