service_test.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. from __future__ import unicode_literals
  2. from __future__ import absolute_import
  3. from .. import unittest
  4. import mock
  5. import docker
  6. from docker.utils import LogConfig
  7. from compose.service import Service
  8. from compose.container import Container
  9. from compose.const import LABEL_SERVICE, LABEL_PROJECT, LABEL_ONE_OFF
  10. from compose.service import (
  11. ConfigError,
  12. NeedsBuildError,
  13. NoSuchImageError,
  14. build_volume_binding,
  15. get_container_data_volumes,
  16. merge_volume_bindings,
  17. parse_repository_tag,
  18. parse_volume_spec,
  19. )
  20. class ServiceTest(unittest.TestCase):
  21. def setUp(self):
  22. self.mock_client = mock.create_autospec(docker.Client)
  23. def test_name_validations(self):
  24. self.assertRaises(ConfigError, lambda: Service(name='', image='foo'))
  25. self.assertRaises(ConfigError, lambda: Service(name=' ', image='foo'))
  26. self.assertRaises(ConfigError, lambda: Service(name='/', image='foo'))
  27. self.assertRaises(ConfigError, lambda: Service(name='!', image='foo'))
  28. self.assertRaises(ConfigError, lambda: Service(name='\xe2', image='foo'))
  29. Service('a', image='foo')
  30. Service('foo', image='foo')
  31. Service('foo-bar', image='foo')
  32. Service('foo.bar', image='foo')
  33. Service('foo_bar', image='foo')
  34. Service('_', image='foo')
  35. Service('___', image='foo')
  36. Service('-', image='foo')
  37. Service('--', image='foo')
  38. Service('.__.', image='foo')
  39. def test_project_validation(self):
  40. self.assertRaises(ConfigError, lambda: Service('bar'))
  41. self.assertRaises(ConfigError, lambda: Service(name='foo', project='>', image='foo'))
  42. Service(name='foo', project='bar.bar__', image='foo')
  43. def test_containers(self):
  44. service = Service('db', self.mock_client, 'myproject', image='foo')
  45. self.mock_client.containers.return_value = []
  46. self.assertEqual(service.containers(), [])
  47. def test_containers_with_containers(self):
  48. self.mock_client.containers.return_value = [
  49. dict(Name=str(i), Image='foo', Id=i) for i in range(3)
  50. ]
  51. service = Service('db', self.mock_client, 'myproject', image='foo')
  52. self.assertEqual([c.id for c in service.containers()], range(3))
  53. expected_labels = [
  54. '{0}=myproject'.format(LABEL_PROJECT),
  55. '{0}=db'.format(LABEL_SERVICE),
  56. '{0}=False'.format(LABEL_ONE_OFF),
  57. ]
  58. self.mock_client.containers.assert_called_once_with(
  59. all=False,
  60. filters={'label': expected_labels})
  61. def test_get_volumes_from_container(self):
  62. container_id = 'aabbccddee'
  63. service = Service(
  64. 'test',
  65. image='foo',
  66. volumes_from=[mock.Mock(id=container_id, spec=Container)])
  67. self.assertEqual(service._get_volumes_from(), [container_id])
  68. def test_get_volumes_from_service_container_exists(self):
  69. container_ids = ['aabbccddee', '12345']
  70. from_service = mock.create_autospec(Service)
  71. from_service.containers.return_value = [
  72. mock.Mock(id=container_id, spec=Container)
  73. for container_id in container_ids
  74. ]
  75. service = Service('test', volumes_from=[from_service], image='foo')
  76. self.assertEqual(service._get_volumes_from(), container_ids)
  77. def test_get_volumes_from_service_no_container(self):
  78. container_id = 'abababab'
  79. from_service = mock.create_autospec(Service)
  80. from_service.containers.return_value = []
  81. from_service.create_container.return_value = mock.Mock(
  82. id=container_id,
  83. spec=Container)
  84. service = Service('test', image='foo', volumes_from=[from_service])
  85. self.assertEqual(service._get_volumes_from(), [container_id])
  86. from_service.create_container.assert_called_once_with()
  87. def test_split_domainname_none(self):
  88. service = Service('foo', image='foo', hostname='name', client=self.mock_client)
  89. self.mock_client.containers.return_value = []
  90. opts = service._get_container_create_options({'image': 'foo'}, 1)
  91. self.assertEqual(opts['hostname'], 'name', 'hostname')
  92. self.assertFalse('domainname' in opts, 'domainname')
  93. def test_memory_swap_limit(self):
  94. service = Service(name='foo', image='foo', hostname='name', client=self.mock_client, mem_limit=1000000000, memswap_limit=2000000000)
  95. self.mock_client.containers.return_value = []
  96. opts = service._get_container_create_options({'some': 'overrides'}, 1)
  97. self.assertEqual(opts['memswap_limit'], 2000000000)
  98. self.assertEqual(opts['mem_limit'], 1000000000)
  99. def test_log_opt(self):
  100. log_opt = {'address': 'tcp://192.168.0.42:123'}
  101. service = Service(name='foo', image='foo', hostname='name', client=self.mock_client, log_driver='syslog', log_opt=log_opt)
  102. self.mock_client.containers.return_value = []
  103. opts = service._get_container_create_options({'some': 'overrides'}, 1)
  104. self.assertIsInstance(opts['host_config']['LogConfig'], LogConfig)
  105. self.assertEqual(opts['host_config']['LogConfig'].type, 'syslog')
  106. self.assertEqual(opts['host_config']['LogConfig'].config, log_opt)
  107. def test_split_domainname_fqdn(self):
  108. service = Service(
  109. 'foo',
  110. hostname='name.domain.tld',
  111. image='foo',
  112. client=self.mock_client)
  113. self.mock_client.containers.return_value = []
  114. opts = service._get_container_create_options({'image': 'foo'}, 1)
  115. self.assertEqual(opts['hostname'], 'name', 'hostname')
  116. self.assertEqual(opts['domainname'], 'domain.tld', 'domainname')
  117. def test_split_domainname_both(self):
  118. service = Service(
  119. 'foo',
  120. hostname='name',
  121. image='foo',
  122. domainname='domain.tld',
  123. client=self.mock_client)
  124. self.mock_client.containers.return_value = []
  125. opts = service._get_container_create_options({'image': 'foo'}, 1)
  126. self.assertEqual(opts['hostname'], 'name', 'hostname')
  127. self.assertEqual(opts['domainname'], 'domain.tld', 'domainname')
  128. def test_split_domainname_weird(self):
  129. service = Service(
  130. 'foo',
  131. hostname='name.sub',
  132. domainname='domain.tld',
  133. image='foo',
  134. client=self.mock_client)
  135. self.mock_client.containers.return_value = []
  136. opts = service._get_container_create_options({'image': 'foo'}, 1)
  137. self.assertEqual(opts['hostname'], 'name.sub', 'hostname')
  138. self.assertEqual(opts['domainname'], 'domain.tld', 'domainname')
  139. def test_get_container_not_found(self):
  140. self.mock_client.containers.return_value = []
  141. service = Service('foo', client=self.mock_client, image='foo')
  142. self.assertRaises(ValueError, service.get_container)
  143. @mock.patch('compose.service.Container', autospec=True)
  144. def test_get_container(self, mock_container_class):
  145. container_dict = dict(Name='default_foo_2')
  146. self.mock_client.containers.return_value = [container_dict]
  147. service = Service('foo', image='foo', client=self.mock_client)
  148. container = service.get_container(number=2)
  149. self.assertEqual(container, mock_container_class.from_ps.return_value)
  150. mock_container_class.from_ps.assert_called_once_with(
  151. self.mock_client, container_dict)
  152. @mock.patch('compose.service.log', autospec=True)
  153. def test_pull_image(self, mock_log):
  154. service = Service('foo', client=self.mock_client, image='someimage:sometag')
  155. service.pull()
  156. self.mock_client.pull.assert_called_once_with(
  157. 'someimage',
  158. tag='sometag',
  159. stream=True)
  160. mock_log.info.assert_called_once_with('Pulling foo (someimage:sometag)...')
  161. def test_pull_image_no_tag(self):
  162. service = Service('foo', client=self.mock_client, image='ababab')
  163. service.pull()
  164. self.mock_client.pull.assert_called_once_with(
  165. 'ababab',
  166. tag='latest',
  167. stream=True)
  168. @mock.patch('compose.service.Container', autospec=True)
  169. def test_recreate_container(self, _):
  170. mock_container = mock.create_autospec(Container)
  171. service = Service('foo', client=self.mock_client, image='someimage')
  172. service.image = lambda: {'Id': 'abc123'}
  173. new_container = service.recreate_container(mock_container)
  174. mock_container.stop.assert_called_once_with(timeout=10)
  175. self.mock_client.rename.assert_called_once_with(
  176. mock_container.id,
  177. '%s_%s' % (mock_container.short_id, mock_container.name))
  178. new_container.start.assert_called_once_with()
  179. mock_container.remove.assert_called_once_with()
  180. @mock.patch('compose.service.Container', autospec=True)
  181. def test_recreate_container_with_timeout(self, _):
  182. mock_container = mock.create_autospec(Container)
  183. self.mock_client.inspect_image.return_value = {'Id': 'abc123'}
  184. service = Service('foo', client=self.mock_client, image='someimage')
  185. service.recreate_container(mock_container, timeout=1)
  186. mock_container.stop.assert_called_once_with(timeout=1)
  187. def test_parse_repository_tag(self):
  188. self.assertEqual(parse_repository_tag("root"), ("root", ""))
  189. self.assertEqual(parse_repository_tag("root:tag"), ("root", "tag"))
  190. self.assertEqual(parse_repository_tag("user/repo"), ("user/repo", ""))
  191. self.assertEqual(parse_repository_tag("user/repo:tag"), ("user/repo", "tag"))
  192. self.assertEqual(parse_repository_tag("url:5000/repo"), ("url:5000/repo", ""))
  193. self.assertEqual(parse_repository_tag("url:5000/repo:tag"), ("url:5000/repo", "tag"))
  194. @mock.patch('compose.service.Container', autospec=True)
  195. def test_create_container_latest_is_used_when_no_tag_specified(self, mock_container):
  196. service = Service('foo', client=self.mock_client, image='someimage')
  197. images = []
  198. def pull(repo, tag=None, **kwargs):
  199. self.assertEqual('someimage', repo)
  200. self.assertEqual('latest', tag)
  201. images.append({'Id': 'abc123'})
  202. return []
  203. service.image = lambda *args, **kwargs: mock_get_image(images)
  204. self.mock_client.pull = pull
  205. service.create_container()
  206. self.assertEqual(1, len(images))
  207. def test_create_container_with_build(self):
  208. service = Service('foo', client=self.mock_client, build='.')
  209. images = []
  210. service.image = lambda *args, **kwargs: mock_get_image(images)
  211. service.build = lambda: images.append({'Id': 'abc123'})
  212. service.create_container(do_build=True)
  213. self.assertEqual(1, len(images))
  214. def test_create_container_no_build(self):
  215. service = Service('foo', client=self.mock_client, build='.')
  216. service.image = lambda: {'Id': 'abc123'}
  217. service.create_container(do_build=False)
  218. self.assertFalse(self.mock_client.build.called)
  219. def test_create_container_no_build_but_needs_build(self):
  220. service = Service('foo', client=self.mock_client, build='.')
  221. service.image = lambda *args, **kwargs: mock_get_image([])
  222. with self.assertRaises(NeedsBuildError):
  223. service.create_container(do_build=False)
  224. def test_build_does_not_pull(self):
  225. self.mock_client.build.return_value = [
  226. '{"stream": "Successfully built 12345"}',
  227. ]
  228. service = Service('foo', client=self.mock_client, build='.')
  229. service.build()
  230. self.assertEqual(self.mock_client.build.call_count, 1)
  231. self.assertFalse(self.mock_client.build.call_args[1]['pull'])
  232. def mock_get_image(images):
  233. if images:
  234. return images[0]
  235. else:
  236. raise NoSuchImageError()
  237. class ServiceVolumesTest(unittest.TestCase):
  238. def setUp(self):
  239. self.mock_client = mock.create_autospec(docker.Client)
  240. def test_parse_volume_spec_only_one_path(self):
  241. spec = parse_volume_spec('/the/volume')
  242. self.assertEqual(spec, (None, '/the/volume', 'rw'))
  243. def test_parse_volume_spec_internal_and_external(self):
  244. spec = parse_volume_spec('external:interval')
  245. self.assertEqual(spec, ('external', 'interval', 'rw'))
  246. def test_parse_volume_spec_with_mode(self):
  247. spec = parse_volume_spec('external:interval:ro')
  248. self.assertEqual(spec, ('external', 'interval', 'ro'))
  249. spec = parse_volume_spec('external:interval:z')
  250. self.assertEqual(spec, ('external', 'interval', 'z'))
  251. def test_parse_volume_spec_too_many_parts(self):
  252. with self.assertRaises(ConfigError):
  253. parse_volume_spec('one:two:three:four')
  254. def test_build_volume_binding(self):
  255. binding = build_volume_binding(parse_volume_spec('/outside:/inside'))
  256. self.assertEqual(binding, ('/inside', '/outside:/inside:rw'))
  257. def test_get_container_data_volumes(self):
  258. options = [
  259. '/host/volume:/host/volume:ro',
  260. '/new/volume',
  261. '/existing/volume',
  262. ]
  263. self.mock_client.inspect_image.return_value = {
  264. 'ContainerConfig': {
  265. 'Volumes': {
  266. '/mnt/image/data': {},
  267. }
  268. }
  269. }
  270. container = Container(self.mock_client, {
  271. 'Image': 'ababab',
  272. 'Volumes': {
  273. '/host/volume': '/host/volume',
  274. '/existing/volume': '/var/lib/docker/aaaaaaaa',
  275. '/removed/volume': '/var/lib/docker/bbbbbbbb',
  276. '/mnt/image/data': '/var/lib/docker/cccccccc',
  277. },
  278. }, has_been_inspected=True)
  279. expected = {
  280. '/existing/volume': '/var/lib/docker/aaaaaaaa:/existing/volume:rw',
  281. '/mnt/image/data': '/var/lib/docker/cccccccc:/mnt/image/data:rw',
  282. }
  283. binds = get_container_data_volumes(container, options)
  284. self.assertEqual(binds, expected)
  285. def test_merge_volume_bindings(self):
  286. options = [
  287. '/host/volume:/host/volume:ro',
  288. '/host/rw/volume:/host/rw/volume',
  289. '/new/volume',
  290. '/existing/volume',
  291. ]
  292. self.mock_client.inspect_image.return_value = {
  293. 'ContainerConfig': {'Volumes': {}}
  294. }
  295. intermediate_container = Container(self.mock_client, {
  296. 'Image': 'ababab',
  297. 'Volumes': {'/existing/volume': '/var/lib/docker/aaaaaaaa'},
  298. }, has_been_inspected=True)
  299. expected = [
  300. '/host/volume:/host/volume:ro',
  301. '/host/rw/volume:/host/rw/volume:rw',
  302. '/var/lib/docker/aaaaaaaa:/existing/volume:rw',
  303. ]
  304. binds = merge_volume_bindings(options, intermediate_container)
  305. self.assertEqual(set(binds), set(expected))
  306. def test_mount_same_host_path_to_two_volumes(self):
  307. service = Service(
  308. 'web',
  309. image='busybox',
  310. volumes=[
  311. '/host/path:/data1',
  312. '/host/path:/data2',
  313. ],
  314. client=self.mock_client,
  315. )
  316. self.mock_client.inspect_image.return_value = {
  317. 'Id': 'ababab',
  318. 'ContainerConfig': {
  319. 'Volumes': {}
  320. }
  321. }
  322. create_options = service._get_container_create_options(
  323. override_options={},
  324. number=1,
  325. )
  326. self.assertEqual(
  327. set(create_options['host_config']['Binds']),
  328. set([
  329. '/host/path:/data1:rw',
  330. '/host/path:/data2:rw',
  331. ]),
  332. )
  333. def test_different_host_path_in_container_json(self):
  334. service = Service(
  335. 'web',
  336. image='busybox',
  337. volumes=['/host/path:/data'],
  338. client=self.mock_client,
  339. )
  340. self.mock_client.inspect_image.return_value = {
  341. 'Id': 'ababab',
  342. 'ContainerConfig': {
  343. 'Volumes': {
  344. '/data': {},
  345. }
  346. }
  347. }
  348. self.mock_client.inspect_container.return_value = {
  349. 'Id': '123123123',
  350. 'Image': 'ababab',
  351. 'Volumes': {
  352. '/data': '/mnt/sda1/host/path',
  353. },
  354. }
  355. create_options = service._get_container_create_options(
  356. override_options={},
  357. number=1,
  358. previous_container=Container(self.mock_client, {'Id': '123123123'}),
  359. )
  360. self.assertEqual(
  361. create_options['host_config']['Binds'],
  362. ['/mnt/sda1/host/path:/data:rw'],
  363. )
  364. def test_create_with_special_volume_mode(self):
  365. self.mock_client.inspect_image.return_value = {'Id': 'imageid'}
  366. create_calls = []
  367. def create_container(*args, **kwargs):
  368. create_calls.append((args, kwargs))
  369. return {'Id': 'containerid'}
  370. self.mock_client.create_container = create_container
  371. volumes = ['/tmp:/foo:z']
  372. Service(
  373. 'web',
  374. client=self.mock_client,
  375. image='busybox',
  376. volumes=volumes,
  377. ).create_container()
  378. self.assertEqual(len(create_calls), 1)
  379. self.assertEqual(create_calls[0][1]['host_config']['Binds'], volumes)