openshift_health_check.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  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. try:
  8. from __main__ import display
  9. except ImportError:
  10. from ansible.utils.display import Display
  11. display = Display()
  12. from ansible.plugins.action import ActionBase
  13. # Augment sys.path so that we can import checks from a directory relative to
  14. # this callback plugin.
  15. sys.path.insert(1, os.path.dirname(os.path.dirname(__file__)))
  16. from openshift_checks import OpenShiftCheck, OpenShiftCheckException # noqa: E402
  17. class ActionModule(ActionBase):
  18. def run(self, tmp=None, task_vars=None):
  19. result = super(ActionModule, self).run(tmp, task_vars)
  20. if task_vars is None:
  21. task_vars = {}
  22. if "openshift" not in task_vars:
  23. result["failed"] = True
  24. result["msg"] = "'openshift' is undefined, did 'openshift_facts' run?"
  25. return result
  26. try:
  27. known_checks = self.load_known_checks()
  28. except OpenShiftCheckException as e:
  29. result["failed"] = True
  30. result["msg"] = str(e)
  31. return result
  32. args = self._task.args
  33. requested_checks = resolve_checks(args.get("checks", []), known_checks.values())
  34. unknown_checks = requested_checks - set(known_checks)
  35. if unknown_checks:
  36. result["failed"] = True
  37. result["msg"] = (
  38. "One or more checks are unknown: {}. "
  39. "Make sure there is no typo in the playbook and no files are missing."
  40. ).format(", ".join(unknown_checks))
  41. return result
  42. result["checks"] = check_results = {}
  43. for check_name in requested_checks & set(known_checks):
  44. display.banner("CHECK [{} : {}]".format(check_name, task_vars["ansible_host"]))
  45. check = known_checks[check_name]
  46. if check.is_active(task_vars):
  47. try:
  48. r = check.run(tmp, task_vars)
  49. except OpenShiftCheckException as e:
  50. r = {}
  51. r["failed"] = True
  52. r["msg"] = str(e)
  53. else:
  54. r = {"skipped": True}
  55. check_results[check_name] = r
  56. if r.get("failed", False):
  57. result["failed"] = True
  58. result["msg"] = "One or more checks failed"
  59. return result
  60. def load_known_checks(self):
  61. known_checks = {}
  62. known_check_classes = set(cls for cls in OpenShiftCheck.subclasses())
  63. for cls in known_check_classes:
  64. check_name = cls.name
  65. if check_name in known_checks:
  66. other_cls = known_checks[check_name].__class__
  67. raise OpenShiftCheckException(
  68. "non-unique check name '{}' in: '{}.{}' and '{}.{}'".format(
  69. check_name,
  70. cls.__module__, cls.__name__,
  71. other_cls.__module__, other_cls.__name__))
  72. known_checks[check_name] = cls(module_executor=self._execute_module)
  73. return known_checks
  74. def resolve_checks(names, all_checks):
  75. """Returns a set of resolved check names.
  76. Resolving a check name involves expanding tag references (e.g., '@tag') with
  77. all the checks that contain the given tag.
  78. names should be a sequence of strings.
  79. all_checks should be a sequence of check classes/instances.
  80. """
  81. resolved = set()
  82. for name in names:
  83. if name.startswith("@"):
  84. for check in all_checks:
  85. if name[1:] in check.tags:
  86. resolved.add(check.name)
  87. else:
  88. resolved.add(name)
  89. return resolved