sanity_checks.py 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. """
  2. Ansible action plugin to ensure inventory variables are set
  3. appropriately and no conflicting options have been provided.
  4. """
  5. import re
  6. from ansible.plugins.action import ActionBase
  7. from ansible import errors
  8. # Valid values for openshift_deployment_type
  9. VALID_DEPLOYMENT_TYPES = ('origin', 'openshift-enterprise')
  10. # Tuple of variable names and default values if undefined.
  11. NET_PLUGIN_LIST = (('openshift_use_openshift_sdn', True),
  12. ('openshift_use_flannel', False),
  13. ('openshift_use_nuage', False),
  14. ('openshift_use_contiv', False),
  15. ('openshift_use_calico', False),
  16. ('openshift_use_kuryr', False))
  17. ENTERPRISE_TAG_REGEX_ERROR = """openshift_image_tag must be in the format
  18. v#.#[.#[.#]]. Examples: v1.2, v3.4.1, v3.5.1.3,
  19. v3.5.1.3.4, v1.2-1, v1.2.3-4, v1.2.3-4.5, v1.2.3-4.5.6
  20. You specified openshift_image_tag={}"""
  21. ORIGIN_TAG_REGEX_ERROR = """openshift_image_tag must be in the format
  22. v#.#[.#-optional.#]. Examples: v1.2.3, v3.5.1-alpha.1
  23. You specified openshift_image_tag={}"""
  24. ORIGIN_TAG_REGEX = {'re': '(^v?\\d+\\.\\d+.*)',
  25. 'error_msg': ORIGIN_TAG_REGEX_ERROR}
  26. ENTERPRISE_TAG_REGEX = {'re': '(^v\\d+\\.\\d+(\\.\\d+)*(-\\d+(\\.\\d+)*)?$)',
  27. 'error_msg': ENTERPRISE_TAG_REGEX_ERROR}
  28. IMAGE_TAG_REGEX = {'origin': ORIGIN_TAG_REGEX,
  29. 'openshift-enterprise': ENTERPRISE_TAG_REGEX}
  30. UNSUPPORTED_OCP_VERSIONS = {
  31. '^3.8.*$': 'OCP 3.8 is not supported and cannot be installed'
  32. }
  33. CONTAINERIZED_NO_TAG_ERROR_MSG = """To install a containerized Origin release,
  34. you must set openshift_release or openshift_image_tag in your inventory to
  35. specify which version of the OpenShift component images to use.
  36. (Suggestion: add openshift_release="x.y" to inventory.)"""
  37. def to_bool(var_to_check):
  38. """Determine a boolean value given the multiple
  39. ways bools can be specified in ansible."""
  40. # http://yaml.org/type/bool.html
  41. yes_list = (True, 1, "True", "1", "true", "TRUE",
  42. "Yes", "yes", "Y", "y", "YES",
  43. "on", "ON", "On")
  44. return var_to_check in yes_list
  45. class ActionModule(ActionBase):
  46. """Action plugin to execute sanity checks."""
  47. def template_var(self, hostvars, host, varname):
  48. """Retrieve a variable from hostvars and template it.
  49. If undefined, return None type."""
  50. # We will set the current host and variable checked for easy debugging
  51. # if there are any unhandled exceptions.
  52. # pylint: disable=W0201
  53. self.last_checked_var = varname
  54. # pylint: disable=W0201
  55. self.last_checked_host = host
  56. res = hostvars[host].get(varname)
  57. if res is None:
  58. return None
  59. return self._templar.template(res)
  60. def check_openshift_deployment_type(self, hostvars, host):
  61. """Ensure a valid openshift_deployment_type is set"""
  62. openshift_deployment_type = self.template_var(hostvars, host,
  63. 'openshift_deployment_type')
  64. if openshift_deployment_type not in VALID_DEPLOYMENT_TYPES:
  65. type_strings = ", ".join(VALID_DEPLOYMENT_TYPES)
  66. msg = "openshift_deployment_type must be defined and one of {}".format(type_strings)
  67. raise errors.AnsibleModuleError(msg)
  68. return openshift_deployment_type
  69. def check_python_version(self, hostvars, host, distro):
  70. """Ensure python version is 3 for Fedora and python 2 for others"""
  71. ansible_python = self.template_var(hostvars, host, 'ansible_python')
  72. if distro == "Fedora":
  73. if ansible_python['version']['major'] != 3:
  74. msg = "openshift-ansible requires Python 3 for {};".format(distro)
  75. msg += " For information on enabling Python 3 with Ansible,"
  76. msg += " see https://docs.ansible.com/ansible/python_3_support.html"
  77. raise errors.AnsibleModuleError(msg)
  78. else:
  79. if ansible_python['version']['major'] != 2:
  80. msg = "openshift-ansible requires Python 2 for {};".format(distro)
  81. def check_image_tag_format(self, hostvars, host, openshift_deployment_type):
  82. """Ensure openshift_image_tag is formatted correctly"""
  83. openshift_image_tag = self.template_var(hostvars, host, 'openshift_image_tag')
  84. if not openshift_image_tag or openshift_image_tag == 'latest':
  85. return None
  86. regex_to_match = IMAGE_TAG_REGEX[openshift_deployment_type]['re']
  87. res = re.match(regex_to_match, str(openshift_image_tag))
  88. if res is None:
  89. msg = IMAGE_TAG_REGEX[openshift_deployment_type]['error_msg']
  90. msg = msg.format(str(openshift_image_tag))
  91. raise errors.AnsibleModuleError(msg)
  92. def no_origin_image_version(self, hostvars, host, openshift_deployment_type):
  93. """Ensure we can determine what image version to use with origin
  94. fail when:
  95. - openshift_is_containerized
  96. - openshift_deployment_type == 'origin'
  97. - openshift_release is not defined
  98. - openshift_image_tag is not defined"""
  99. if not openshift_deployment_type == 'origin':
  100. return None
  101. oic = self.template_var(hostvars, host, 'openshift_is_containerized')
  102. if not to_bool(oic):
  103. return None
  104. orelease = self.template_var(hostvars, host, 'openshift_release')
  105. oitag = self.template_var(hostvars, host, 'openshift_image_tag')
  106. if not orelease and not oitag:
  107. raise errors.AnsibleModuleError(CONTAINERIZED_NO_TAG_ERROR_MSG)
  108. def network_plugin_check(self, hostvars, host):
  109. """Ensure only one type of network plugin is enabled"""
  110. res = []
  111. # Loop through each possible network plugin boolean, determine the
  112. # actual boolean value, and append results into a list.
  113. for plugin, default_val in NET_PLUGIN_LIST:
  114. res_temp = self.template_var(hostvars, host, plugin)
  115. if res_temp is None:
  116. res_temp = default_val
  117. res.append(to_bool(res_temp))
  118. if sum(res) != 1:
  119. plugin_str = list(zip([x[0] for x in NET_PLUGIN_LIST], res))
  120. msg = "Host Checked: {} Only one of must be true. Found: {}".format(host, plugin_str)
  121. raise errors.AnsibleModuleError(msg)
  122. def check_hostname_vars(self, hostvars, host):
  123. """Checks to ensure openshift_hostname
  124. and openshift_public_hostname
  125. conform to the proper length of 63 characters or less"""
  126. for varname in ('openshift_public_hostname', 'openshift_hostname'):
  127. var_value = self.template_var(hostvars, host, varname)
  128. if var_value and len(var_value) > 63:
  129. msg = '{} must be 63 characters or less'.format(varname)
  130. raise errors.AnsibleModuleError(msg)
  131. def check_supported_ocp_version(self, hostvars, host, openshift_deployment_type):
  132. """Checks that the OCP version supported"""
  133. if openshift_deployment_type == 'origin':
  134. return None
  135. openshift_version = self.template_var(hostvars, host, 'openshift_version')
  136. for regex_to_match, error_msg in UNSUPPORTED_OCP_VERSIONS.items():
  137. res = re.match(regex_to_match, str(openshift_version))
  138. if res is not None:
  139. raise errors.AnsibleModuleError(error_msg)
  140. return None
  141. def run_checks(self, hostvars, host):
  142. """Execute the hostvars validations against host"""
  143. distro = self.template_var(hostvars, host, 'ansible_distribution')
  144. odt = self.check_openshift_deployment_type(hostvars, host)
  145. self.check_python_version(hostvars, host, distro)
  146. self.check_image_tag_format(hostvars, host, odt)
  147. self.no_origin_image_version(hostvars, host, odt)
  148. self.network_plugin_check(hostvars, host)
  149. self.check_hostname_vars(hostvars, host)
  150. self.check_supported_ocp_version(hostvars, host, odt)
  151. def run(self, tmp=None, task_vars=None):
  152. result = super(ActionModule, self).run(tmp, task_vars)
  153. # self.task_vars holds all in-scope variables.
  154. # Ignore settting self.task_vars outside of init.
  155. # pylint: disable=W0201
  156. self.task_vars = task_vars or {}
  157. # pylint: disable=W0201
  158. self.last_checked_host = "none"
  159. # pylint: disable=W0201
  160. self.last_checked_var = "none"
  161. # self._task.args holds task parameters.
  162. # check_hosts is a parameter to this plugin, and should provide
  163. # a list of hosts.
  164. check_hosts = self._task.args.get('check_hosts')
  165. if not check_hosts:
  166. msg = "check_hosts is required"
  167. raise errors.AnsibleModuleError(msg)
  168. # We need to access each host's variables
  169. hostvars = self.task_vars.get('hostvars')
  170. if not hostvars:
  171. msg = hostvars
  172. raise errors.AnsibleModuleError(msg)
  173. # We loop through each host in the provided list check_hosts
  174. for host in check_hosts:
  175. try:
  176. self.run_checks(hostvars, host)
  177. except Exception as uncaught_e:
  178. msg = "last_checked_host: {}, last_checked_var: {};"
  179. msg = msg.format(self.last_checked_host, self.last_checked_var)
  180. msg += str(uncaught_e)
  181. raise errors.AnsibleModuleError(msg)
  182. result["changed"] = False
  183. result["failed"] = False
  184. result["msg"] = "Sanity Checks passed"
  185. return result