docker_image_availability.py 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. # pylint: disable=missing-docstring
  2. from openshift_checks import OpenShiftCheck, get_var
  3. from openshift_checks.mixins import DockerHostMixin
  4. class DockerImageAvailability(DockerHostMixin, OpenShiftCheck):
  5. """Check that required Docker images are available.
  6. This check attempts to ensure that required docker images are
  7. either present locally, or able to be pulled down from available
  8. registries defined in a host machine.
  9. """
  10. name = "docker_image_availability"
  11. tags = ["preflight"]
  12. dependencies = ["skopeo", "python-docker-py"]
  13. deployment_image_info = {
  14. "origin": {
  15. "namespace": "openshift",
  16. "name": "origin",
  17. },
  18. "openshift-enterprise": {
  19. "namespace": "openshift3",
  20. "name": "ose",
  21. },
  22. }
  23. @classmethod
  24. def is_active(cls, task_vars):
  25. """Skip hosts with unsupported deployment types."""
  26. deployment_type = get_var(task_vars, "openshift_deployment_type")
  27. has_valid_deployment_type = deployment_type in cls.deployment_image_info
  28. return super(DockerImageAvailability, cls).is_active(task_vars) and has_valid_deployment_type
  29. def run(self, tmp, task_vars):
  30. msg, failed, changed = self.ensure_dependencies(task_vars)
  31. if failed:
  32. return {
  33. "failed": True,
  34. "changed": changed,
  35. "msg": "Some dependencies are required in order to check Docker image availability.\n" + msg
  36. }
  37. required_images = self.required_images(task_vars)
  38. missing_images = set(required_images) - set(self.local_images(required_images, task_vars))
  39. # exit early if all images were found locally
  40. if not missing_images:
  41. return {"changed": changed}
  42. registries = self.known_docker_registries(task_vars)
  43. if not registries:
  44. return {"failed": True, "msg": "Unable to retrieve any docker registries.", "changed": changed}
  45. available_images = self.available_images(missing_images, registries, task_vars)
  46. unavailable_images = set(missing_images) - set(available_images)
  47. if unavailable_images:
  48. return {
  49. "failed": True,
  50. "msg": (
  51. "One or more required Docker images are not available:\n {}\n"
  52. "Configured registries: {}"
  53. ).format(",\n ".join(sorted(unavailable_images)), ", ".join(registries)),
  54. "changed": changed,
  55. }
  56. return {"changed": changed}
  57. def required_images(self, task_vars):
  58. deployment_type = get_var(task_vars, "openshift_deployment_type")
  59. image_info = self.deployment_image_info[deployment_type]
  60. openshift_release = get_var(task_vars, "openshift_release", default="latest")
  61. openshift_image_tag = get_var(task_vars, "openshift_image_tag")
  62. is_containerized = get_var(task_vars, "openshift", "common", "is_containerized")
  63. images = set(self.required_docker_images(
  64. image_info["namespace"],
  65. image_info["name"],
  66. ["registry-console"] if "enterprise" in deployment_type else [], # include enterprise-only image names
  67. openshift_release,
  68. is_containerized,
  69. ))
  70. # append images with qualified image tags to our list of required images.
  71. # these are images with a (v0.0.0.0) tag, rather than a standard release
  72. # format tag (v0.0). We want to check this set in both containerized and
  73. # non-containerized installations.
  74. images.update(
  75. self.required_qualified_docker_images(
  76. image_info["namespace"],
  77. image_info["name"],
  78. openshift_image_tag,
  79. ),
  80. )
  81. return images
  82. @staticmethod
  83. def required_docker_images(namespace, name, additional_image_names, version, is_containerized):
  84. if is_containerized:
  85. return ["{}/{}:{}".format(namespace, name, version)] if name else []
  86. # include additional non-containerized images specific to the current deployment type
  87. return ["{}/{}:{}".format(namespace, img_name, version) for img_name in additional_image_names]
  88. @staticmethod
  89. def required_qualified_docker_images(namespace, name, version):
  90. # pylint: disable=invalid-name
  91. return [
  92. "{}/{}-{}:{}".format(namespace, name, suffix, version)
  93. for suffix in ["haproxy-router", "docker-registry", "deployer", "pod"]
  94. ]
  95. def local_images(self, images, task_vars):
  96. """Filter a list of images and return those available locally."""
  97. return [
  98. image for image in images
  99. if self.is_image_local(image, task_vars)
  100. ]
  101. def is_image_local(self, image, task_vars):
  102. result = self.module_executor("docker_image_facts", {"name": image}, task_vars)
  103. if result.get("failed", False):
  104. return False
  105. return bool(result.get("images", []))
  106. @staticmethod
  107. def known_docker_registries(task_vars):
  108. docker_facts = get_var(task_vars, "openshift", "docker")
  109. regs = set(docker_facts["additional_registries"])
  110. deployment_type = get_var(task_vars, "openshift_deployment_type")
  111. if deployment_type == "origin":
  112. regs.update(["docker.io"])
  113. elif "enterprise" in deployment_type:
  114. regs.update(["registry.access.redhat.com"])
  115. return list(regs)
  116. def available_images(self, images, registries, task_vars):
  117. """Inspect existing images using Skopeo and return all images successfully inspected."""
  118. return [
  119. image for image in images
  120. if any(self.is_available_skopeo_image(image, registry, task_vars) for registry in registries)
  121. ]
  122. def is_available_skopeo_image(self, image, registry, task_vars):
  123. """Uses Skopeo to determine if required image exists in a given registry."""
  124. cmd_str = "skopeo inspect docker://{registry}/{image}".format(
  125. registry=registry,
  126. image=image,
  127. )
  128. args = {"_raw_params": cmd_str}
  129. result = self.module_executor("command", args, task_vars)
  130. return not result.get("failed", False) and result.get("rc", 0) == 0