testcases.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. import functools
  2. import os
  3. import pytest
  4. from docker.errors import APIError
  5. from docker.utils import version_lt
  6. from .. import unittest
  7. from ..helpers import BUSYBOX_IMAGE_WITH_TAG
  8. from compose.cli.docker_client import docker_client
  9. from compose.config.config import resolve_environment
  10. from compose.config.environment import Environment
  11. from compose.const import API_VERSIONS
  12. from compose.const import COMPOSE_SPEC as VERSION
  13. from compose.const import COMPOSEFILE_V1 as V1
  14. from compose.const import LABEL_PROJECT
  15. from compose.progress_stream import stream_output
  16. from compose.service import Service
  17. SWARM_SKIP_CONTAINERS_ALL = os.environ.get('SWARM_SKIP_CONTAINERS_ALL', '0') != '0'
  18. SWARM_SKIP_CPU_SHARES = os.environ.get('SWARM_SKIP_CPU_SHARES', '0') != '0'
  19. SWARM_SKIP_RM_VOLUMES = os.environ.get('SWARM_SKIP_RM_VOLUMES', '0') != '0'
  20. SWARM_ASSUME_MULTINODE = os.environ.get('SWARM_ASSUME_MULTINODE', '0') != '0'
  21. def pull_busybox(client):
  22. client.pull(BUSYBOX_IMAGE_WITH_TAG, stream=False)
  23. def get_links(container):
  24. links = container.get('HostConfig.Links') or []
  25. def format_link(link):
  26. _, alias = link.split(':')
  27. return alias.split('/')[-1]
  28. return [format_link(link) for link in links]
  29. def engine_max_version():
  30. if 'DOCKER_VERSION' not in os.environ:
  31. return VERSION
  32. version = os.environ['DOCKER_VERSION'].partition('-')[0]
  33. if version_lt(version, '1.10'):
  34. return V1
  35. return VERSION
  36. def min_version_skip(version):
  37. return pytest.mark.skipif(
  38. engine_max_version() < version,
  39. reason="Engine version %s is too low" % version
  40. )
  41. class DockerClientTestCase(unittest.TestCase):
  42. @classmethod
  43. def setUpClass(cls):
  44. version = API_VERSIONS[engine_max_version()]
  45. cls.client = docker_client(Environment(), version)
  46. @classmethod
  47. def tearDownClass(cls):
  48. cls.client.close()
  49. del cls.client
  50. def tearDown(self):
  51. for c in self.client.containers(
  52. all=True,
  53. filters={'label': '%s=composetest' % LABEL_PROJECT}):
  54. self.client.remove_container(c['Id'], force=True)
  55. for i in self.client.images(
  56. filters={'label': 'com.docker.compose.test_image'}):
  57. try:
  58. self.client.remove_image(i, force=True)
  59. except APIError as e:
  60. if e.is_server_error():
  61. pass
  62. volumes = self.client.volumes().get('Volumes') or []
  63. for v in volumes:
  64. if 'composetest_' in v['Name']:
  65. self.client.remove_volume(v['Name'])
  66. networks = self.client.networks()
  67. for n in networks:
  68. if 'composetest_' in n['Name']:
  69. self.client.remove_network(n['Name'])
  70. def create_service(self, name, **kwargs):
  71. if 'image' not in kwargs and 'build' not in kwargs:
  72. kwargs['image'] = BUSYBOX_IMAGE_WITH_TAG
  73. if 'command' not in kwargs:
  74. kwargs['command'] = ["top"]
  75. kwargs['environment'] = resolve_environment(
  76. kwargs, Environment.from_env_file(None)
  77. )
  78. labels = dict(kwargs.setdefault('labels', {}))
  79. labels['com.docker.compose.test-name'] = self.id()
  80. return Service(name, client=self.client, project='composetest', **kwargs)
  81. def check_build(self, *args, **kwargs):
  82. kwargs.setdefault('rm', True)
  83. build_output = self.client.build(*args, **kwargs)
  84. with open(os.devnull, 'w') as devnull:
  85. for event in stream_output(build_output, devnull):
  86. pass
  87. def require_api_version(self, minimum):
  88. api_version = self.client.version()['ApiVersion']
  89. if version_lt(api_version, minimum):
  90. pytest.skip("API version is too low ({} < {})".format(api_version, minimum))
  91. def get_volume_data(self, volume_name):
  92. if not is_cluster(self.client):
  93. return self.client.inspect_volume(volume_name)
  94. volumes = self.client.volumes(filters={'name': volume_name})['Volumes']
  95. assert len(volumes) > 0
  96. return self.client.inspect_volume(volumes[0]['Name'])
  97. def if_runtime_available(runtime):
  98. def decorator(f):
  99. @functools.wraps(f)
  100. def wrapper(self, *args, **kwargs):
  101. if runtime not in self.client.info().get('Runtimes', {}):
  102. return pytest.skip("This daemon does not support the '{}'' runtime".format(runtime))
  103. return f(self, *args, **kwargs)
  104. return wrapper
  105. return decorator
  106. def is_cluster(client):
  107. if SWARM_ASSUME_MULTINODE:
  108. return True
  109. def get_nodes_number():
  110. try:
  111. return len(client.nodes())
  112. except APIError:
  113. # If the Engine is not part of a Swarm, the SDK will raise
  114. # an APIError
  115. return 0
  116. if not hasattr(is_cluster, 'nodes') or is_cluster.nodes is None:
  117. # Only make the API call if the value hasn't been cached yet
  118. is_cluster.nodes = get_nodes_number()
  119. return is_cluster.nodes > 1
  120. def no_cluster(reason):
  121. def decorator(f):
  122. @functools.wraps(f)
  123. def wrapper(self, *args, **kwargs):
  124. if is_cluster(self.client):
  125. pytest.skip("Test will not be run in cluster mode: %s" % reason)
  126. return
  127. return f(self, *args, **kwargs)
  128. return wrapper
  129. return decorator