__init__.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. """
  2. Health checks for OpenShift clusters.
  3. """
  4. import operator
  5. import os
  6. from abc import ABCMeta, abstractmethod, abstractproperty
  7. from importlib import import_module
  8. from ansible.module_utils import six
  9. from ansible.module_utils.six.moves import reduce # pylint: disable=import-error,redefined-builtin
  10. class OpenShiftCheckException(Exception):
  11. """Raised when a check encounters a failure condition."""
  12. def __init__(self, name, msg=None):
  13. # msg is for the message the user will see when this is raised.
  14. # name is for test code to identify the error without looking at msg text.
  15. if msg is None: # for parameter backward compatibility
  16. msg = name
  17. name = self.__class__.__name__
  18. self.name = name
  19. super(OpenShiftCheckException, self).__init__(msg)
  20. class OpenShiftCheckExceptionList(OpenShiftCheckException):
  21. """A container for multiple logging errors that may be detected in one check."""
  22. def __init__(self, errors):
  23. self.errors = errors
  24. super(OpenShiftCheckExceptionList, self).__init__(
  25. 'OpenShiftCheckExceptionList',
  26. '\n'.join(str(msg) for msg in errors)
  27. )
  28. # make iterable
  29. def __getitem__(self, index):
  30. return self.errors[index]
  31. @six.add_metaclass(ABCMeta)
  32. class OpenShiftCheck(object):
  33. """
  34. A base class for defining checks for an OpenShift cluster environment.
  35. Expect optional params: method execute_module, dict task_vars, and string tmp.
  36. execute_module is expected to have a signature compatible with _execute_module
  37. from ansible plugins/action/__init__.py, e.g.:
  38. def execute_module(module_name=None, module_args=None, tmp=None, task_vars=None, *args):
  39. This is stored so that it can be invoked in subclasses via check.execute_module("name", args)
  40. which provides the check's stored task_vars and tmp.
  41. """
  42. def __init__(self, execute_module=None, task_vars=None, tmp=None):
  43. self._execute_module = execute_module
  44. self.task_vars = task_vars or {}
  45. self.tmp = tmp
  46. # set to True when the check changes the host, for accurate total "changed" count
  47. self.changed = False
  48. @abstractproperty
  49. def name(self):
  50. """The name of this check, usually derived from the class name."""
  51. return "openshift_check"
  52. @property
  53. def tags(self):
  54. """A list of tags that this check satisfy.
  55. Tags are used to reference multiple checks with a single '@tagname'
  56. special check name.
  57. """
  58. return []
  59. @staticmethod
  60. def is_active():
  61. """Returns true if this check applies to the ansible-playbook run."""
  62. return True
  63. @abstractmethod
  64. def run(self):
  65. """Executes a check, normally implemented as a module."""
  66. return {}
  67. @classmethod
  68. def subclasses(cls):
  69. """Returns a generator of subclasses of this class and its subclasses."""
  70. # AUDIT: no-member makes sense due to this having a metaclass
  71. for subclass in cls.__subclasses__(): # pylint: disable=no-member
  72. yield subclass
  73. for subclass in subclass.subclasses():
  74. yield subclass
  75. def execute_module(self, module_name=None, module_args=None):
  76. """Invoke an Ansible module from a check.
  77. Invoke stored _execute_module, normally copied from the action
  78. plugin, with its params and the task_vars and tmp given at
  79. check initialization. No positional parameters beyond these
  80. are specified. If it's necessary to specify any of the other
  81. parameters to _execute_module then that should just be invoked
  82. directly (with awareness of changes in method signature per
  83. Ansible version).
  84. So e.g. check.execute_module("foo", dict(arg1=...))
  85. Return: result hash from module execution.
  86. """
  87. if self._execute_module is None:
  88. raise NotImplementedError(
  89. self.__class__.__name__ +
  90. " invoked execute_module without providing the method at initialization."
  91. )
  92. return self._execute_module(module_name, module_args, self.tmp, self.task_vars)
  93. def get_var(self, *keys, **kwargs):
  94. """Get deeply nested values from task_vars.
  95. Ansible task_vars structures are Python dicts, often mapping strings to
  96. other dicts. This helper makes it easier to get a nested value, raising
  97. OpenShiftCheckException when a key is not found or returning a default value
  98. provided as a keyword argument.
  99. """
  100. try:
  101. value = reduce(operator.getitem, keys, self.task_vars)
  102. except (KeyError, TypeError):
  103. if "default" in kwargs:
  104. return kwargs["default"]
  105. raise OpenShiftCheckException("'{}' is undefined".format(".".join(map(str, keys))))
  106. return value
  107. @staticmethod
  108. def get_major_minor_version(openshift_image_tag):
  109. """Parse and return the deployed version of OpenShift as a tuple."""
  110. if openshift_image_tag and openshift_image_tag[0] == 'v':
  111. openshift_image_tag = openshift_image_tag[1:]
  112. # map major release versions across releases
  113. # to a common major version
  114. openshift_major_release_version = {
  115. "1": "3",
  116. }
  117. components = openshift_image_tag.split(".")
  118. if not components or len(components) < 2:
  119. msg = "An invalid version of OpenShift was found for this host: {}"
  120. raise OpenShiftCheckException(msg.format(openshift_image_tag))
  121. if components[0] in openshift_major_release_version:
  122. components[0] = openshift_major_release_version[components[0]]
  123. components = tuple(int(x) for x in components[:2])
  124. return components
  125. def find_ansible_mount(self, path):
  126. """Return the mount point for path from ansible_mounts."""
  127. # reorganize list of mounts into dict by path
  128. mount_for_path = {
  129. mount['mount']: mount
  130. for mount
  131. in self.get_var('ansible_mounts')
  132. }
  133. # NOTE: including base cases '/' and '' to ensure the loop ends
  134. mount_targets = set(mount_for_path.keys()) | {'/', ''}
  135. mount_point = path
  136. while mount_point not in mount_targets:
  137. mount_point = os.path.dirname(mount_point)
  138. try:
  139. return mount_for_path[mount_point]
  140. except KeyError:
  141. known_mounts = ', '.join('"{}"'.format(mount) for mount in sorted(mount_for_path))
  142. raise OpenShiftCheckException(
  143. 'Unable to determine mount point for path "{}".\n'
  144. 'Known mount points: {}.'.format(path, known_mounts or 'none')
  145. )
  146. LOADER_EXCLUDES = (
  147. "__init__.py",
  148. "mixins.py",
  149. "logging.py",
  150. )
  151. def load_checks(path=None, subpkg=""):
  152. """Dynamically import all check modules for the side effect of registering checks."""
  153. if path is None:
  154. path = os.path.dirname(__file__)
  155. modules = []
  156. for name in os.listdir(path):
  157. if os.path.isdir(os.path.join(path, name)):
  158. modules = modules + load_checks(os.path.join(path, name), subpkg + "." + name)
  159. continue
  160. if name.endswith(".py") and name not in LOADER_EXCLUDES:
  161. modules.append(import_module(__package__ + subpkg + "." + name[:-3]))
  162. return modules