docker_image_availability.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. # pylint: disable=missing-docstring
  2. from openshift_checks import OpenShiftCheck, get_var
  3. class DockerImageAvailability(OpenShiftCheck):
  4. """Check that required Docker images are available.
  5. This check attempts to ensure that required docker images are
  6. either present locally, or able to be pulled down from available
  7. registries defined in a host machine.
  8. """
  9. name = "docker_image_availability"
  10. tags = ["preflight"]
  11. skopeo_image = "openshift/openshift-ansible"
  12. deployment_image_info = {
  13. "origin": {
  14. "namespace": "openshift",
  15. "name": "origin",
  16. },
  17. "openshift-enterprise": {
  18. "namespace": "openshift3",
  19. "name": "ose",
  20. },
  21. }
  22. @classmethod
  23. def is_active(cls, task_vars):
  24. """Skip hosts with unsupported deployment types."""
  25. deployment_type = get_var(task_vars, "openshift_deployment_type")
  26. has_valid_deployment_type = deployment_type in cls.deployment_image_info
  27. return super(DockerImageAvailability, cls).is_active(task_vars) and has_valid_deployment_type
  28. def run(self, tmp, task_vars):
  29. required_images = self.required_images(task_vars)
  30. missing_images = set(required_images) - set(self.local_images(required_images, task_vars))
  31. # exit early if all images were found locally
  32. if not missing_images:
  33. return {"changed": False}
  34. msg, failed, changed = self.update_skopeo_image(task_vars)
  35. # exit early if Skopeo update fails
  36. if failed:
  37. if "Error connecting: Error while fetching server API version" in msg:
  38. msg = (
  39. "It appears Docker is not running.\n"
  40. "Please start Docker on this host before running this check.\n"
  41. "The full error reported was:\n " + msg
  42. )
  43. elif "Failed to import docker-py" in msg:
  44. msg = (
  45. "The required Python docker-py module is not installed.\n"
  46. "Suggestion: install the python-docker-py package on this host."
  47. )
  48. else:
  49. msg = "The full message reported by the docker_image module was:\n" + msg
  50. return {
  51. "failed": True,
  52. "changed": changed,
  53. "msg": (
  54. "Unable to update the {img_name} image on this host;\n"
  55. "This is required in order to check Docker image availability.\n"
  56. "{msg}"
  57. ).format(img_name=self.skopeo_image, msg=msg),
  58. }
  59. registries = self.known_docker_registries(task_vars)
  60. if not registries:
  61. return {"failed": True, "msg": "Unable to retrieve any docker registries."}
  62. available_images = self.available_images(missing_images, registries, task_vars)
  63. unavailable_images = set(missing_images) - set(available_images)
  64. if unavailable_images:
  65. return {
  66. "failed": True,
  67. "msg": (
  68. "One or more required images are not available: {}.\n"
  69. "Configured registries: {}"
  70. ).format(", ".join(sorted(unavailable_images)), ", ".join(registries)),
  71. "changed": changed,
  72. }
  73. return {"changed": changed}
  74. def required_images(self, task_vars):
  75. deployment_type = get_var(task_vars, "openshift_deployment_type")
  76. image_info = self.deployment_image_info[deployment_type]
  77. openshift_release = get_var(task_vars, "openshift_release", default="latest")
  78. openshift_image_tag = get_var(task_vars, "openshift_image_tag", default=openshift_release)
  79. if openshift_image_tag and openshift_image_tag[0] != 'v':
  80. openshift_image_tag = 'v' + openshift_image_tag
  81. is_containerized = get_var(task_vars, "openshift", "common", "is_containerized")
  82. images = set(self.non_qualified_docker_images(image_info["namespace"], image_info["name"], openshift_release,
  83. is_containerized))
  84. # append images with qualified image tags to our list of required images.
  85. # these are images with a (v0.0.0.0) tag, rather than a standard release
  86. # format tag (v0.0). We want to check this set in both containerized and
  87. # non-containerized installations.
  88. images.update(
  89. self.qualified_docker_images(image_info["namespace"], image_info["name"], openshift_image_tag)
  90. )
  91. return images
  92. def local_images(self, images, task_vars):
  93. """Filter a list of images and return those available locally."""
  94. return [
  95. image for image in images
  96. if self.is_image_local(image, task_vars)
  97. ]
  98. def is_image_local(self, image, task_vars):
  99. result = self.module_executor("docker_image_facts", {"name": image}, task_vars)
  100. if result.get("failed", False):
  101. return False
  102. return bool(result.get("images", []))
  103. def known_docker_registries(self, task_vars):
  104. result = self.module_executor("docker_info", {}, task_vars)
  105. if result.get("failed", False):
  106. return False
  107. docker_info = result.get("info", {})
  108. return [registry.get("Name", "") for registry in docker_info.get("Registries", {})]
  109. def available_images(self, images, registries, task_vars):
  110. """Inspect existing images using Skopeo and return all images successfully inspected."""
  111. return [
  112. image for image in images
  113. if self.is_image_available(image, registries, task_vars)
  114. ]
  115. def is_image_available(self, image, registries, task_vars):
  116. for registry in registries:
  117. if self.is_available_skopeo_image(image, registry, task_vars):
  118. return True
  119. return False
  120. def is_available_skopeo_image(self, image, registry, task_vars):
  121. """Uses Skopeo to determine if required image exists in a given registry."""
  122. cmd_str = "skopeo inspect docker://{registry}/{image}".format(
  123. registry=registry,
  124. image=image,
  125. )
  126. args = {
  127. "name": "skopeo_inspect",
  128. "image": self.skopeo_image,
  129. "command": cmd_str,
  130. "detach": False,
  131. "cleanup": True,
  132. }
  133. result = self.module_executor("docker_container", args, task_vars)
  134. return not result.get("failed", False)
  135. @staticmethod
  136. def non_qualified_docker_images(namespace, name, version, is_containerized):
  137. if is_containerized:
  138. return ["{}/{}:{}".format(namespace, name, version)] if name else []
  139. return ["{}/{}:{}".format(namespace, name, version)] if name else []
  140. @staticmethod
  141. def qualified_docker_images(namespace, name, version):
  142. return [
  143. "{}/{}-{}:{}".format(namespace, name, suffix, version)
  144. for suffix in ["haproxy-router", "docker-registry", "deployer", "pod"]
  145. ]
  146. # ensures that the skopeo docker image exists, and updates it
  147. # with latest if image was already present locally.
  148. def update_skopeo_image(self, task_vars):
  149. result = self.module_executor("docker_image", {"name": self.skopeo_image}, task_vars)
  150. return result.get("msg", ""), result.get("failed", False), result.get("changed", False)