testcases.py 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import functools
  4. import os
  5. import pytest
  6. import six
  7. from docker.errors import APIError
  8. from docker.utils import version_lt
  9. from .. import unittest
  10. from compose.cli.docker_client import docker_client
  11. from compose.config.config import resolve_environment
  12. from compose.config.environment import Environment
  13. from compose.const import API_VERSIONS
  14. from compose.const import COMPOSEFILE_V1 as V1
  15. from compose.const import COMPOSEFILE_V2_0 as V2_0
  16. from compose.const import COMPOSEFILE_V2_0 as V2_1
  17. from compose.const import COMPOSEFILE_V2_2 as V2_2
  18. from compose.const import COMPOSEFILE_V2_3 as V2_3
  19. from compose.const import COMPOSEFILE_V3_0 as V3_0
  20. from compose.const import COMPOSEFILE_V3_2 as V3_2
  21. from compose.const import COMPOSEFILE_V3_5 as V3_5
  22. from compose.const import LABEL_PROJECT
  23. from compose.progress_stream import stream_output
  24. from compose.service import Service
  25. SWARM_SKIP_CONTAINERS_ALL = os.environ.get('SWARM_SKIP_CONTAINERS_ALL', '0') != '0'
  26. SWARM_SKIP_CPU_SHARES = os.environ.get('SWARM_SKIP_CPU_SHARES', '0') != '0'
  27. SWARM_SKIP_RM_VOLUMES = os.environ.get('SWARM_SKIP_RM_VOLUMES', '0') != '0'
  28. SWARM_ASSUME_MULTINODE = os.environ.get('SWARM_ASSUME_MULTINODE', '0') != '0'
  29. def pull_busybox(client):
  30. client.pull('busybox:latest', stream=False)
  31. def get_links(container):
  32. links = container.get('HostConfig.Links') or []
  33. def format_link(link):
  34. _, alias = link.split(':')
  35. return alias.split('/')[-1]
  36. return [format_link(link) for link in links]
  37. def engine_max_version():
  38. if 'DOCKER_VERSION' not in os.environ:
  39. return V3_5
  40. version = os.environ['DOCKER_VERSION'].partition('-')[0]
  41. if version_lt(version, '1.10'):
  42. return V1
  43. if version_lt(version, '1.12'):
  44. return V2_0
  45. if version_lt(version, '1.13'):
  46. return V2_1
  47. if version_lt(version, '17.06'):
  48. return V3_2
  49. return V3_5
  50. def min_version_skip(version):
  51. return pytest.mark.skipif(
  52. engine_max_version() < version,
  53. reason="Engine version %s is too low" % version
  54. )
  55. def v2_only():
  56. return min_version_skip(V2_0)
  57. def v2_1_only():
  58. return min_version_skip(V2_1)
  59. def v2_2_only():
  60. return min_version_skip(V2_2)
  61. def v2_3_only():
  62. return min_version_skip(V2_3)
  63. def v3_only():
  64. return min_version_skip(V3_0)
  65. class DockerClientTestCase(unittest.TestCase):
  66. @classmethod
  67. def setUpClass(cls):
  68. version = API_VERSIONS[engine_max_version()]
  69. cls.client = docker_client(Environment(), version)
  70. @classmethod
  71. def tearDownClass(cls):
  72. del cls.client
  73. def tearDown(self):
  74. for c in self.client.containers(
  75. all=True,
  76. filters={'label': '%s=composetest' % LABEL_PROJECT}):
  77. self.client.remove_container(c['Id'], force=True)
  78. for i in self.client.images(
  79. filters={'label': 'com.docker.compose.test_image'}):
  80. try:
  81. self.client.remove_image(i, force=True)
  82. except APIError as e:
  83. if e.is_server_error():
  84. pass
  85. volumes = self.client.volumes().get('Volumes') or []
  86. for v in volumes:
  87. if 'composetest_' in v['Name']:
  88. self.client.remove_volume(v['Name'])
  89. networks = self.client.networks()
  90. for n in networks:
  91. if 'composetest_' in n['Name']:
  92. self.client.remove_network(n['Name'])
  93. def create_service(self, name, **kwargs):
  94. if 'image' not in kwargs and 'build' not in kwargs:
  95. kwargs['image'] = 'busybox:latest'
  96. if 'command' not in kwargs:
  97. kwargs['command'] = ["top"]
  98. kwargs['environment'] = resolve_environment(
  99. kwargs, Environment.from_env_file(None)
  100. )
  101. labels = dict(kwargs.setdefault('labels', {}))
  102. labels['com.docker.compose.test-name'] = self.id()
  103. return Service(name, client=self.client, project='composetest', **kwargs)
  104. def check_build(self, *args, **kwargs):
  105. kwargs.setdefault('rm', True)
  106. build_output = self.client.build(*args, **kwargs)
  107. stream_output(build_output, open('/dev/null', 'w'))
  108. def require_api_version(self, minimum):
  109. api_version = self.client.version()['ApiVersion']
  110. if version_lt(api_version, minimum):
  111. pytest.skip("API version is too low ({} < {})".format(api_version, minimum))
  112. def get_volume_data(self, volume_name):
  113. if not is_cluster(self.client):
  114. return self.client.inspect_volume(volume_name)
  115. volumes = self.client.volumes(filters={'name': volume_name})['Volumes']
  116. assert len(volumes) > 0
  117. return self.client.inspect_volume(volumes[0]['Name'])
  118. def if_runtime_available(runtime):
  119. if runtime == 'nvidia':
  120. command = 'nvidia-container-runtime'
  121. if six.PY3:
  122. import shutil
  123. return pytest.mark.skipif(
  124. shutil.which(command) is None,
  125. reason="Nvida runtime not exists"
  126. )
  127. return pytest.mark.skipif(
  128. any(
  129. os.access(os.path.join(path, command), os.X_OK)
  130. for path in os.environ["PATH"].split(os.pathsep)
  131. ) is False,
  132. reason="Nvida runtime not exists"
  133. )
  134. return pytest.skip("Runtime %s not exists", runtime)
  135. def is_cluster(client):
  136. if SWARM_ASSUME_MULTINODE:
  137. return True
  138. def get_nodes_number():
  139. try:
  140. return len(client.nodes())
  141. except APIError:
  142. # If the Engine is not part of a Swarm, the SDK will raise
  143. # an APIError
  144. return 0
  145. if not hasattr(is_cluster, 'nodes') or is_cluster.nodes is None:
  146. # Only make the API call if the value hasn't been cached yet
  147. is_cluster.nodes = get_nodes_number()
  148. return is_cluster.nodes > 1
  149. def no_cluster(reason):
  150. def decorator(f):
  151. @functools.wraps(f)
  152. def wrapper(self, *args, **kwargs):
  153. if is_cluster(self.client):
  154. pytest.skip("Test will not be run in cluster mode: %s" % reason)
  155. return
  156. return f(self, *args, **kwargs)
  157. return wrapper
  158. return decorator