openshift_container_binary_sync.py 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. # pylint: disable=missing-docstring,invalid-name
  4. import random
  5. import tempfile
  6. import shutil
  7. import os.path
  8. # pylint: disable=redefined-builtin,wildcard-import,unused-wildcard-import
  9. from ansible.module_utils.basic import * # noqa: F403
  10. DOCUMENTATION = '''
  11. ---
  12. module: openshift_container_binary_sync
  13. short_description: Copies OpenShift binaries out of the given image tag to host system.
  14. '''
  15. class BinarySyncError(Exception):
  16. def __init__(self, msg):
  17. super(BinarySyncError, self).__init__(msg)
  18. self.msg = msg
  19. # pylint: disable=too-few-public-methods,too-many-instance-attributes
  20. class BinarySyncer(object):
  21. """
  22. Syncs the openshift, oc, and kubectl binaries/symlinks out of
  23. a container onto the host system.
  24. """
  25. def __init__(self, module, image, backend):
  26. self.module = module
  27. self.changed = False
  28. self.output = []
  29. self.bin_dir = '/usr/local/bin'
  30. self._image = image
  31. self.backend = backend
  32. self.temp_dir = None # TBD
  33. def sync(self):
  34. if self.backend == 'atomic':
  35. return self._sync_atomic()
  36. return self._sync_docker()
  37. def _sync_atomic(self):
  38. self.temp_dir = tempfile.mkdtemp()
  39. temp_dir_mount = tempfile.mkdtemp()
  40. try:
  41. image_spec = self.image
  42. rc, stdout, stderr = self.module.run_command(['atomic', 'mount',
  43. '--storage', "ostree",
  44. image_spec, temp_dir_mount])
  45. if rc:
  46. raise BinarySyncError("Error mounting image. stdout=%s, stderr=%s" %
  47. (stdout, stderr))
  48. for i in ["openshift", "oc"]:
  49. src_file = os.path.join(temp_dir_mount, "usr/bin", i)
  50. shutil.copy(src_file, self.temp_dir)
  51. self._sync_binaries()
  52. finally:
  53. self.module.run_command(['atomic', 'umount', temp_dir_mount])
  54. shutil.rmtree(temp_dir_mount)
  55. shutil.rmtree(self.temp_dir)
  56. def _sync_docker(self):
  57. container_name = "openshift-cli-%s" % random.randint(1, 100000)
  58. rc, stdout, stderr = self.module.run_command(['docker', 'create', '--name',
  59. container_name, self.image])
  60. if rc:
  61. raise BinarySyncError("Error creating temporary docker container. stdout=%s, stderr=%s" %
  62. (stdout, stderr))
  63. self.output.append(stdout)
  64. try:
  65. self.temp_dir = tempfile.mkdtemp()
  66. self.output.append("Using temp dir: %s" % self.temp_dir)
  67. rc, stdout, stderr = self.module.run_command(['docker', 'cp', "%s:/usr/bin/openshift" % container_name,
  68. self.temp_dir])
  69. if rc:
  70. raise BinarySyncError("Error copying file from docker container: stdout=%s, stderr=%s" %
  71. (stdout, stderr))
  72. rc, stdout, stderr = self.module.run_command(['docker', 'cp', "%s:/usr/bin/oc" % container_name,
  73. self.temp_dir])
  74. if rc:
  75. raise BinarySyncError("Error copying file from docker container: stdout=%s, stderr=%s" %
  76. (stdout, stderr))
  77. self._sync_binaries()
  78. finally:
  79. shutil.rmtree(self.temp_dir)
  80. self.module.run_command(['docker', 'rm', container_name])
  81. def _sync_binaries(self):
  82. self._sync_binary('openshift')
  83. # In older versions, oc was a symlink to openshift:
  84. if os.path.islink(os.path.join(self.temp_dir, 'oc')):
  85. self._sync_symlink('oc', 'openshift')
  86. else:
  87. self._sync_binary('oc')
  88. # Ensure correct symlinks created:
  89. self._sync_symlink('kubectl', 'oc')
  90. # Remove old oadm binary
  91. if os.path.exists(os.path.join(self.bin_dir, 'oadm')):
  92. os.remove(os.path.join(self.bin_dir, 'oadm'))
  93. def _sync_symlink(self, binary_name, link_to):
  94. """ Ensure the given binary name exists and links to the expected binary. """
  95. # The symlink we are creating:
  96. link_path = os.path.join(self.bin_dir, binary_name)
  97. # The expected file we should be linking to:
  98. link_dest = os.path.join(self.bin_dir, link_to)
  99. if not os.path.exists(link_path) or \
  100. not os.path.islink(link_path) or \
  101. os.path.realpath(link_path) != os.path.realpath(link_dest):
  102. if os.path.exists(link_path):
  103. os.remove(link_path)
  104. os.symlink(link_to, os.path.join(self.bin_dir, binary_name))
  105. self.output.append("Symlinked %s to %s." % (link_path, link_dest))
  106. self.changed = True
  107. def _sync_binary(self, binary_name):
  108. src_path = os.path.join(self.temp_dir, binary_name)
  109. dest_path = os.path.join(self.bin_dir, binary_name)
  110. incoming_checksum = self.module.run_command(['sha256sum', src_path])[1]
  111. if not os.path.exists(dest_path) or self.module.run_command(['sha256sum', dest_path])[1] != incoming_checksum:
  112. # See: https://github.com/openshift/openshift-ansible/issues/4965
  113. if os.path.islink(dest_path):
  114. os.unlink(dest_path)
  115. self.output.append('Removed old symlink {} before copying binary.'.format(dest_path))
  116. shutil.move(src_path, dest_path)
  117. self.output.append("Moved %s to %s." % (src_path, dest_path))
  118. self.changed = True
  119. @property
  120. def raw_image(self):
  121. """
  122. Returns the image as it was originally passed in to the instance.
  123. .. note::
  124. This image string will only work directly with the atomic command.
  125. :returns: The original image passed in.
  126. :rtype: str
  127. """
  128. return self._image
  129. @property
  130. def image(self):
  131. """
  132. Returns the image without atomic prefixes used to map to skopeo args.
  133. :returns: The image string without prefixes
  134. :rtype: str
  135. """
  136. image = self._image
  137. for remove in ('oci:', 'http:', 'https:'):
  138. if image.startswith(remove):
  139. image = image.replace(remove, '')
  140. return image
  141. def main():
  142. module = AnsibleModule( # noqa: F405
  143. argument_spec=dict(
  144. image=dict(required=True),
  145. backend=dict(required=True),
  146. ),
  147. supports_check_mode=True
  148. )
  149. image = module.params['image']
  150. backend = module.params['backend']
  151. if backend not in ["docker", "atomic"]:
  152. module.fail_json(msg="unknown backend")
  153. binary_syncer = BinarySyncer(module, image, backend)
  154. try:
  155. binary_syncer.sync()
  156. except BinarySyncError as ex:
  157. module.fail_json(msg=ex.msg)
  158. return module.exit_json(changed=binary_syncer.changed,
  159. output=binary_syncer.output)
  160. if __name__ == '__main__':
  161. main()