openshift_health_check.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. """
  2. Ansible action plugin to execute health checks in OpenShift clusters.
  3. """
  4. # pylint: disable=wrong-import-position,missing-docstring,invalid-name
  5. import sys
  6. import os
  7. from collections import defaultdict
  8. try:
  9. from __main__ import display
  10. except ImportError:
  11. from ansible.utils.display import Display
  12. display = Display()
  13. from ansible.plugins.action import ActionBase
  14. # Augment sys.path so that we can import checks from a directory relative to
  15. # this callback plugin.
  16. sys.path.insert(1, os.path.dirname(os.path.dirname(__file__)))
  17. from openshift_checks import OpenShiftCheck, OpenShiftCheckException, load_checks # noqa: E402
  18. class ActionModule(ActionBase):
  19. def run(self, tmp=None, task_vars=None):
  20. result = super(ActionModule, self).run(tmp, task_vars)
  21. task_vars = task_vars or {}
  22. # vars are not supportably available in the callback plugin,
  23. # so record any it will need in the result.
  24. result['playbook_context'] = task_vars.get('r_openshift_health_checker_playbook_context')
  25. if "openshift" not in task_vars:
  26. result["failed"] = True
  27. result["msg"] = "'openshift' is undefined, did 'openshift_facts' run?"
  28. return result
  29. try:
  30. known_checks = self.load_known_checks()
  31. args = self._task.args
  32. resolved_checks = resolve_checks(args.get("checks", []), known_checks.values())
  33. except OpenShiftCheckException as e:
  34. result["failed"] = True
  35. result["msg"] = str(e)
  36. return result
  37. result["checks"] = check_results = {}
  38. user_disabled_checks = [
  39. check.strip()
  40. for check in task_vars.get("openshift_disable_check", "").split(",")
  41. ]
  42. for check_name in resolved_checks:
  43. display.banner("CHECK [{} : {}]".format(check_name, task_vars["ansible_host"]))
  44. check = known_checks[check_name]
  45. if not check.is_active(task_vars):
  46. r = dict(skipped=True, skipped_reason="Not active for this host")
  47. elif check_name in user_disabled_checks:
  48. r = dict(skipped=True, skipped_reason="Disabled by user request")
  49. else:
  50. try:
  51. r = check.run(tmp, task_vars)
  52. except OpenShiftCheckException as e:
  53. r = dict(
  54. failed=True,
  55. msg=str(e),
  56. )
  57. check_results[check_name] = r
  58. if r.get("failed", False):
  59. result["failed"] = True
  60. result["msg"] = "One or more checks failed"
  61. result["changed"] = any(r.get("changed", False) for r in check_results.values())
  62. return result
  63. def load_known_checks(self):
  64. load_checks()
  65. known_checks = {}
  66. for cls in OpenShiftCheck.subclasses():
  67. check_name = cls.name
  68. if check_name in known_checks:
  69. other_cls = known_checks[check_name].__class__
  70. raise OpenShiftCheckException(
  71. "non-unique check name '{}' in: '{}.{}' and '{}.{}'".format(
  72. check_name,
  73. cls.__module__, cls.__name__,
  74. other_cls.__module__, other_cls.__name__))
  75. known_checks[check_name] = cls(execute_module=self._execute_module)
  76. return known_checks
  77. def resolve_checks(names, all_checks):
  78. """Returns a set of resolved check names.
  79. Resolving a check name expands tag references (e.g., "@tag") to all the
  80. checks that contain the given tag. OpenShiftCheckException is raised if
  81. names contains an unknown check or tag name.
  82. names should be a sequence of strings.
  83. all_checks should be a sequence of check classes/instances.
  84. """
  85. known_check_names = set(check.name for check in all_checks)
  86. known_tag_names = set(name for check in all_checks for name in check.tags)
  87. check_names = set(name for name in names if not name.startswith('@'))
  88. tag_names = set(name[1:] for name in names if name.startswith('@'))
  89. unknown_check_names = check_names - known_check_names
  90. unknown_tag_names = tag_names - known_tag_names
  91. if unknown_check_names or unknown_tag_names:
  92. msg = []
  93. if unknown_check_names:
  94. msg.append('Unknown check names: {}.'.format(', '.join(sorted(unknown_check_names))))
  95. if unknown_tag_names:
  96. msg.append('Unknown tag names: {}.'.format(', '.join(sorted(unknown_tag_names))))
  97. msg.append('Make sure there is no typo in the playbook and no files are missing.')
  98. raise OpenShiftCheckException('\n'.join(msg))
  99. tag_to_checks = defaultdict(set)
  100. for check in all_checks:
  101. for tag in check.tags:
  102. tag_to_checks[tag].add(check.name)
  103. resolved = check_names.copy()
  104. for tag in tag_names:
  105. resolved.update(tag_to_checks[tag])
  106. return resolved