openshift_container_binary_sync.py 7.2 KB

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