service_test.py 19 KB

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