sanity_checks.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365
  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. STORAGE_KIND_TUPLE = (
  31. 'openshift_hosted_registry_storage_kind',
  32. 'openshift_loggingops_storage_kind',
  33. 'openshift_logging_storage_kind',
  34. 'openshift_metrics_storage_kind',
  35. 'openshift_prometheus_alertbuffer_storage_kind',
  36. 'openshift_prometheus_alertmanager_storage_kind',
  37. 'openshift_prometheus_storage_kind')
  38. REMOVED_VARIABLES = (
  39. # TODO(michaelgugino): Remove these in 3.11
  40. ('openshift_metrics_image_prefix', 'openshift_metrics_<component>_image'),
  41. ('openshift_metrics_image_version', 'openshift_metrics_<component>_image'),
  42. ('openshift_grafana_proxy_image_prefix', 'openshift_grafana_proxy_image'),
  43. ('openshift_grafana_proxy_image_version', 'openshift_grafana_proxy_image'),
  44. ('openshift_logging_image_prefix', 'openshift_logging_image'),
  45. ('openshift_logging_image_verion', 'openshift_logging_image'),
  46. ('openshift_logging_curator_image_prefix', 'openshift_logging_curator_image'),
  47. ('openshift_logging_curator_image_version', 'openshift_logging_curator_image'),
  48. ('openshift_logging_elasticsearch_image_prefix', 'openshift_logging_elasticsearch_image'),
  49. ('openshift_logging_elasticsearch_image_version', 'openshift_logging_elasticsearch_image'),
  50. ('openshift_logging_elasticsearch_proxy_image_prefix', 'openshift_logging_elasticsearch_proxy_image'),
  51. ('openshift_logging_elasticsearch_proxy_image_version', 'openshift_logging_elasticsearch_proxy_image'),
  52. ('openshift_logging_fluentd_image_prefix', 'openshift_logging_fluentd_image'),
  53. ('openshift_logging_fluentd_image_version', 'openshift_logging_fluentd_image'),
  54. ('openshift_logging_kibana_image_prefix', 'openshift_logging_kibana_image'),
  55. ('openshift_logging_kibana_image_version', 'openshift_logging_kibana_image'),
  56. ('openshift_logging_kibana_proxy_image_prefix', 'openshift_logging_kibana_proxy_image'),
  57. ('openshift_logging_kibana_proxy_image_version', 'openshift_logging_kibana_proxy_image'),
  58. ('openshift_logging_mux_image_prefix', 'openshift_logging_mux_image'),
  59. ('openshift_logging_mux_image_version', 'openshift_logging_mux_image'),
  60. ('openshift_prometheus_image_prefix', 'openshift_prometheus_image'),
  61. ('openshift_prometheus_image_version', 'openshift_prometheus_image'),
  62. ('openshift_prometheus_proxy_image_prefix', 'openshift_prometheus_proxy_image'),
  63. ('openshift_prometheus_proxy_image_version', 'openshift_prometheus_proxy_image'),
  64. ('openshift_prometheus_altermanager_image_prefix', 'openshift_prometheus_alertmanager_image'),
  65. # A typo was introduced at some point, need to warn for this older version.
  66. ('openshift_prometheus_altermanager_image_prefix', 'openshift_prometheus_alertmanager_image'),
  67. ('openshift_prometheus_alertmanager_image_version', 'openshift_prometheus_alertmanager_image'),
  68. ('openshift_prometheus_alertbuffer_image_prefix', 'openshift_prometheus_alertbuffer_image'),
  69. ('openshift_prometheus_alertbuffer_image_version', 'openshift_prometheus_alertbuffer_image'),
  70. ('openshift_prometheus_node_exporter_image_prefix', 'openshift_prometheus_node_exporter_image'),
  71. ('openshift_prometheus_node_exporter_image_version', 'openshift_prometheus_node_exporter_image'),
  72. ('openshift_descheduler_image_prefix', 'openshift_descheduler_image'),
  73. ('openshift_descheduler_image_version', 'openshift_descheduler_image'),
  74. ('openshift_docker_gc_version', 'openshift_docker_gc_image'),
  75. ('openshift_web_console_prefix', 'openshift_web_console_image'),
  76. ('openshift_web_console_version', 'openshift_web_console_image'),
  77. ('openshift_web_console_image_name', 'openshift_web_console_image'),
  78. ('openshift_storage_glusterfs_version', 'openshift_storage_glusterfs_image'),
  79. ('openshift_storage_glusterfs_block_version', 'openshift_storage_glusterfs_block_image'),
  80. ('openshift_storage_glusterfs_s3_version', 'openshift_storage_glusterfs_s3_image'),
  81. ('openshift_storage_glusterfs_heketi_version', 'openshift_storage_glusterfs_heketi_image'),
  82. ('openshift_storage_glusterfs_registry_version', 'openshift_storage_glusterfs_registry_image'),
  83. ('openshift_storage_glusterfs_registry_block_version', 'openshift_storage_glusterfs_registry_block_image'),
  84. ('openshift_storage_glusterfs_registry_s3_version', 'openshift_storage_glusterfs_registry_s3_image'),
  85. ('openshift_storage_glusterfs_registry_heketi_version', 'openshift_storage_glusterfs_registry_heketi_image'),
  86. )
  87. # TODO(michaelgugino): Remove in 3.11
  88. CHANGED_IMAGE_VARS = (
  89. 'openshift_storage_glusterfs_image',
  90. 'openshift_storage_glusterfs_block_image',
  91. 'openshift_storage_glusterfs_s3_image',
  92. 'openshift_storage_glusterfs_heketi_image',
  93. 'openshift_storage_glusterfs_registry_image',
  94. 'openshift_storage_glusterfs_registry_block_image',
  95. 'openshift_storage_glusterfs_registry_s3_image',
  96. 'openshift_storage_glusterfs_registry_heketi_image',
  97. )
  98. def to_bool(var_to_check):
  99. """Determine a boolean value given the multiple
  100. ways bools can be specified in ansible."""
  101. # http://yaml.org/type/bool.html
  102. yes_list = (True, 1, "True", "1", "true", "TRUE",
  103. "Yes", "yes", "Y", "y", "YES",
  104. "on", "ON", "On")
  105. return var_to_check in yes_list
  106. def check_for_removed_vars(hostvars, host):
  107. """Fails if removed variables are found"""
  108. found_removed = []
  109. for item in REMOVED_VARIABLES:
  110. if item in hostvars[host]:
  111. found_removed.append(item)
  112. if found_removed:
  113. msg = "Found removed variables: "
  114. for item in found_removed:
  115. msg += "{} is replaced by {}; ".format(item[0], item[1])
  116. raise errors.AnsibleModuleError(msg)
  117. return None
  118. class ActionModule(ActionBase):
  119. """Action plugin to execute sanity checks."""
  120. def template_var(self, hostvars, host, varname):
  121. """Retrieve a variable from hostvars and template it.
  122. If undefined, return None type."""
  123. # We will set the current host and variable checked for easy debugging
  124. # if there are any unhandled exceptions.
  125. # pylint: disable=W0201
  126. self.last_checked_var = varname
  127. # pylint: disable=W0201
  128. self.last_checked_host = host
  129. res = hostvars[host].get(varname)
  130. if res is None:
  131. return None
  132. return self._templar.template(res)
  133. def check_openshift_deployment_type(self, hostvars, host):
  134. """Ensure a valid openshift_deployment_type is set"""
  135. openshift_deployment_type = self.template_var(hostvars, host,
  136. 'openshift_deployment_type')
  137. if openshift_deployment_type not in VALID_DEPLOYMENT_TYPES:
  138. type_strings = ", ".join(VALID_DEPLOYMENT_TYPES)
  139. msg = "openshift_deployment_type must be defined and one of {}".format(type_strings)
  140. raise errors.AnsibleModuleError(msg)
  141. return openshift_deployment_type
  142. def check_python_version(self, hostvars, host, distro):
  143. """Ensure python version is 3 for Fedora and python 2 for others"""
  144. ansible_python = self.template_var(hostvars, host, 'ansible_python')
  145. if distro == "Fedora":
  146. if ansible_python['version']['major'] != 3:
  147. msg = "openshift-ansible requires Python 3 for {};".format(distro)
  148. msg += " For information on enabling Python 3 with Ansible,"
  149. msg += " see https://docs.ansible.com/ansible/python_3_support.html"
  150. raise errors.AnsibleModuleError(msg)
  151. else:
  152. if ansible_python['version']['major'] != 2:
  153. msg = "openshift-ansible requires Python 2 for {};".format(distro)
  154. def check_image_tag_format(self, hostvars, host, openshift_deployment_type):
  155. """Ensure openshift_image_tag is formatted correctly"""
  156. openshift_image_tag = self.template_var(hostvars, host, 'openshift_image_tag')
  157. if not openshift_image_tag or openshift_image_tag == 'latest':
  158. return None
  159. regex_to_match = IMAGE_TAG_REGEX[openshift_deployment_type]['re']
  160. res = re.match(regex_to_match, str(openshift_image_tag))
  161. if res is None:
  162. msg = IMAGE_TAG_REGEX[openshift_deployment_type]['error_msg']
  163. msg = msg.format(str(openshift_image_tag))
  164. raise errors.AnsibleModuleError(msg)
  165. def network_plugin_check(self, hostvars, host):
  166. """Ensure only one type of network plugin is enabled"""
  167. res = []
  168. # Loop through each possible network plugin boolean, determine the
  169. # actual boolean value, and append results into a list.
  170. for plugin, default_val in NET_PLUGIN_LIST:
  171. res_temp = self.template_var(hostvars, host, plugin)
  172. if res_temp is None:
  173. res_temp = default_val
  174. res.append(to_bool(res_temp))
  175. if sum(res) not in (0, 1):
  176. plugin_str = list(zip([x[0] for x in NET_PLUGIN_LIST], res))
  177. msg = "Host Checked: {} Only one of must be true. Found: {}".format(host, plugin_str)
  178. raise errors.AnsibleModuleError(msg)
  179. def check_hostname_vars(self, hostvars, host):
  180. """Checks to ensure openshift_hostname
  181. and openshift_public_hostname
  182. conform to the proper length of 63 characters or less"""
  183. for varname in ('openshift_public_hostname', 'openshift_hostname'):
  184. var_value = self.template_var(hostvars, host, varname)
  185. if var_value and len(var_value) > 63:
  186. msg = '{} must be 63 characters or less'.format(varname)
  187. raise errors.AnsibleModuleError(msg)
  188. def check_session_auth_secrets(self, hostvars, host):
  189. """Checks session_auth_secrets is correctly formatted"""
  190. sas = self.template_var(hostvars, host,
  191. 'openshift_master_session_auth_secrets')
  192. ses = self.template_var(hostvars, host,
  193. 'openshift_master_session_encryption_secrets')
  194. # This variable isn't mandatory, only check if set.
  195. if sas is None and ses is None:
  196. return None
  197. if not (
  198. issubclass(type(sas), list) and issubclass(type(ses), list)
  199. ) or len(sas) != len(ses):
  200. raise errors.AnsibleModuleError(
  201. 'Expects openshift_master_session_auth_secrets and '
  202. 'openshift_master_session_encryption_secrets are equal length lists')
  203. for secret in sas:
  204. if len(secret) < 32:
  205. raise errors.AnsibleModuleError(
  206. 'Invalid secret in openshift_master_session_auth_secrets. '
  207. 'Secrets must be at least 32 characters in length.')
  208. for secret in ses:
  209. if len(secret) not in [16, 24, 32]:
  210. raise errors.AnsibleModuleError(
  211. 'Invalid secret in openshift_master_session_encryption_secrets. '
  212. 'Secrets must be 16, 24, or 32 characters in length.')
  213. return None
  214. def check_unsupported_nfs_configs(self, hostvars, host):
  215. """Fails if nfs storage is in use for any components. This check is
  216. ignored if openshift_enable_unsupported_configurations=True"""
  217. enable_unsupported = self.template_var(
  218. hostvars, host, 'openshift_enable_unsupported_configurations')
  219. if to_bool(enable_unsupported):
  220. return None
  221. for storage in STORAGE_KIND_TUPLE:
  222. kind = self.template_var(hostvars, host, storage)
  223. if kind == 'nfs':
  224. raise errors.AnsibleModuleError(
  225. 'nfs is an unsupported type for {}. '
  226. 'openshift_enable_unsupported_configurations=True must'
  227. 'be specified to continue with this configuration.'
  228. ''.format(storage))
  229. return None
  230. def check_htpasswd_provider(self, hostvars, host):
  231. """Fails if openshift_master_identity_providers contains an entry of
  232. kind HTPasswdPasswordIdentityProvider and
  233. openshift_master_manage_htpasswd is False"""
  234. idps = self.template_var(
  235. hostvars, host, 'openshift_master_identity_providers')
  236. if not idps:
  237. # If we don't find any identity_providers, nothing for us to do.
  238. return None
  239. manage_pass = self.template_var(
  240. hostvars, host, 'openshift_master_manage_htpasswd')
  241. if to_bool(manage_pass):
  242. # If we manage the file, we can just generate in the new path.
  243. return None
  244. old_keys = ('file', 'fileName', 'file_name', 'filename')
  245. for idp in idps:
  246. if idp['kind'] == 'HTPasswdPasswordIdentityProvider':
  247. for old_key in old_keys:
  248. if old_key in idp is not None:
  249. raise errors.AnsibleModuleError(
  250. 'openshift_master_identity_providers contains a '
  251. 'provider of kind==HTPasswdPasswordIdentityProvider '
  252. 'and {} is set. Please migrate your htpasswd '
  253. 'files to /etc/origin/master/htpasswd and update your '
  254. 'existing master configs, and remove the {} key'
  255. 'before proceeding.'.format(old_key, old_key))
  256. def check_contains_version(self, hostvars, host):
  257. """Fails if variable has old format not containing image version"""
  258. found_incorrect = []
  259. for img_var in CHANGED_IMAGE_VARS:
  260. img_string = self.template_var(hostvars, host, img_var)
  261. if not img_string:
  262. return None
  263. # split the image string by '/' to account for something like docker://
  264. img_string_parts = img_string.split('/')
  265. if ':' not in img_string_parts[-1]:
  266. found_incorrect.append((img_var, img_string))
  267. if found_incorrect:
  268. msg = ("Found image variables without version. Please ensure any image"
  269. "defined contains ':someversion'; ")
  270. for item in found_incorrect:
  271. msg += "{} was: {}; ".format(item[0], item[1])
  272. raise errors.AnsibleModuleError(msg)
  273. return None
  274. def run_checks(self, hostvars, host):
  275. """Execute the hostvars validations against host"""
  276. distro = self.template_var(hostvars, host, 'ansible_distribution')
  277. odt = self.check_openshift_deployment_type(hostvars, host)
  278. self.check_python_version(hostvars, host, distro)
  279. self.check_image_tag_format(hostvars, host, odt)
  280. self.network_plugin_check(hostvars, host)
  281. self.check_hostname_vars(hostvars, host)
  282. self.check_session_auth_secrets(hostvars, host)
  283. self.check_unsupported_nfs_configs(hostvars, host)
  284. self.check_htpasswd_provider(hostvars, host)
  285. check_for_removed_vars(hostvars, host)
  286. self.check_contains_version(hostvars, host)
  287. def run(self, tmp=None, task_vars=None):
  288. result = super(ActionModule, self).run(tmp, task_vars)
  289. # self.task_vars holds all in-scope variables.
  290. # Ignore settting self.task_vars outside of init.
  291. # pylint: disable=W0201
  292. self.task_vars = task_vars or {}
  293. # pylint: disable=W0201
  294. self.last_checked_host = "none"
  295. # pylint: disable=W0201
  296. self.last_checked_var = "none"
  297. # self._task.args holds task parameters.
  298. # check_hosts is a parameter to this plugin, and should provide
  299. # a list of hosts.
  300. check_hosts = self._task.args.get('check_hosts')
  301. if not check_hosts:
  302. msg = "check_hosts is required"
  303. raise errors.AnsibleModuleError(msg)
  304. # We need to access each host's variables
  305. hostvars = self.task_vars.get('hostvars')
  306. if not hostvars:
  307. msg = hostvars
  308. raise errors.AnsibleModuleError(msg)
  309. # We loop through each host in the provided list check_hosts
  310. for host in check_hosts:
  311. try:
  312. self.run_checks(hostvars, host)
  313. except Exception as uncaught_e:
  314. msg = "last_checked_host: {}, last_checked_var: {};"
  315. msg = msg.format(self.last_checked_host, self.last_checked_var)
  316. msg += str(uncaught_e)
  317. raise errors.AnsibleModuleError(msg)
  318. result["changed"] = False
  319. result["failed"] = False
  320. result["msg"] = "Sanity Checks passed"
  321. return result