service_test.py 48 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import docker
  4. import pytest
  5. from docker.constants import DEFAULT_DOCKER_API_VERSION
  6. from docker.errors import APIError
  7. from .. import mock
  8. from .. import unittest
  9. from compose.config.errors import DependencyError
  10. from compose.config.types import ServicePort
  11. from compose.config.types import ServiceSecret
  12. from compose.config.types import VolumeFromSpec
  13. from compose.config.types import VolumeSpec
  14. from compose.const import API_VERSIONS
  15. from compose.const import LABEL_CONFIG_HASH
  16. from compose.const import LABEL_ONE_OFF
  17. from compose.const import LABEL_PROJECT
  18. from compose.const import LABEL_SERVICE
  19. from compose.const import SECRETS_PATH
  20. from compose.container import Container
  21. from compose.parallel import ParallelStreamWriter
  22. from compose.project import OneOffFilter
  23. from compose.service import build_ulimits
  24. from compose.service import build_volume_binding
  25. from compose.service import BuildAction
  26. from compose.service import ContainerNetworkMode
  27. from compose.service import format_environment
  28. from compose.service import formatted_ports
  29. from compose.service import get_container_data_volumes
  30. from compose.service import ImageType
  31. from compose.service import merge_volume_bindings
  32. from compose.service import NeedsBuildError
  33. from compose.service import NetworkMode
  34. from compose.service import NoSuchImageError
  35. from compose.service import parse_repository_tag
  36. from compose.service import Service
  37. from compose.service import ServiceNetworkMode
  38. from compose.service import warn_on_masked_volume
  39. class ServiceTest(unittest.TestCase):
  40. def setUp(self):
  41. self.mock_client = mock.create_autospec(docker.APIClient)
  42. self.mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  43. self.mock_client._general_configs = {}
  44. def test_containers(self):
  45. service = Service('db', self.mock_client, 'myproject', image='foo')
  46. self.mock_client.containers.return_value = []
  47. assert list(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. assert [c.id for c in service.containers()] == list(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_container_without_name(self):
  63. self.mock_client.containers.return_value = [
  64. {'Image': 'foo', 'Id': '1', 'Name': '1'},
  65. {'Image': 'foo', 'Id': '2', 'Name': None},
  66. {'Image': 'foo', 'Id': '3'},
  67. ]
  68. service = Service('db', self.mock_client, 'myproject', image='foo')
  69. assert [c.id for c in service.containers()] == ['1']
  70. assert service._next_container_number() == 2
  71. assert service.get_container(1).id == '1'
  72. def test_get_volumes_from_container(self):
  73. container_id = 'aabbccddee'
  74. service = Service(
  75. 'test',
  76. image='foo',
  77. volumes_from=[
  78. VolumeFromSpec(
  79. mock.Mock(id=container_id, spec=Container),
  80. 'rw',
  81. 'container')])
  82. assert service._get_volumes_from() == [container_id + ':rw']
  83. def test_get_volumes_from_container_read_only(self):
  84. container_id = 'aabbccddee'
  85. service = Service(
  86. 'test',
  87. image='foo',
  88. volumes_from=[
  89. VolumeFromSpec(
  90. mock.Mock(id=container_id, spec=Container),
  91. 'ro',
  92. 'container')])
  93. assert service._get_volumes_from() == [container_id + ':ro']
  94. def test_get_volumes_from_service_container_exists(self):
  95. container_ids = ['aabbccddee', '12345']
  96. from_service = mock.create_autospec(Service)
  97. from_service.containers.return_value = [
  98. mock.Mock(id=container_id, spec=Container)
  99. for container_id in container_ids
  100. ]
  101. service = Service(
  102. 'test',
  103. volumes_from=[VolumeFromSpec(from_service, 'rw', 'service')],
  104. image='foo')
  105. assert service._get_volumes_from() == [container_ids[0] + ":rw"]
  106. def test_get_volumes_from_service_container_exists_with_flags(self):
  107. for mode in ['ro', 'rw', 'z', 'rw,z', 'z,rw']:
  108. container_ids = ['aabbccddee:' + mode, '12345:' + mode]
  109. from_service = mock.create_autospec(Service)
  110. from_service.containers.return_value = [
  111. mock.Mock(id=container_id.split(':')[0], spec=Container)
  112. for container_id in container_ids
  113. ]
  114. service = Service(
  115. 'test',
  116. volumes_from=[VolumeFromSpec(from_service, mode, 'service')],
  117. image='foo')
  118. assert service._get_volumes_from() == [container_ids[0]]
  119. def test_get_volumes_from_service_no_container(self):
  120. container_id = 'abababab'
  121. from_service = mock.create_autospec(Service)
  122. from_service.containers.return_value = []
  123. from_service.create_container.return_value = mock.Mock(
  124. id=container_id,
  125. spec=Container)
  126. service = Service(
  127. 'test',
  128. image='foo',
  129. volumes_from=[VolumeFromSpec(from_service, 'rw', 'service')])
  130. assert service._get_volumes_from() == [container_id + ':rw']
  131. from_service.create_container.assert_called_once_with()
  132. def test_memory_swap_limit(self):
  133. self.mock_client.create_host_config.return_value = {}
  134. service = Service(
  135. name='foo',
  136. image='foo',
  137. hostname='name',
  138. client=self.mock_client,
  139. mem_limit=1000000000,
  140. memswap_limit=2000000000)
  141. service._get_container_create_options({'some': 'overrides'}, 1)
  142. assert self.mock_client.create_host_config.called
  143. assert self.mock_client.create_host_config.call_args[1]['mem_limit'] == 1000000000
  144. assert self.mock_client.create_host_config.call_args[1]['memswap_limit'] == 2000000000
  145. def test_self_reference_external_link(self):
  146. service = Service(
  147. name='foo',
  148. external_links=['default_foo_1']
  149. )
  150. with pytest.raises(DependencyError):
  151. service.get_container_name('foo', 1)
  152. def test_mem_reservation(self):
  153. self.mock_client.create_host_config.return_value = {}
  154. service = Service(
  155. name='foo',
  156. image='foo',
  157. hostname='name',
  158. client=self.mock_client,
  159. mem_reservation='512m'
  160. )
  161. service._get_container_create_options({'some': 'overrides'}, 1)
  162. assert self.mock_client.create_host_config.called is True
  163. assert self.mock_client.create_host_config.call_args[1]['mem_reservation'] == '512m'
  164. def test_cgroup_parent(self):
  165. self.mock_client.create_host_config.return_value = {}
  166. service = Service(
  167. name='foo',
  168. image='foo',
  169. hostname='name',
  170. client=self.mock_client,
  171. cgroup_parent='test')
  172. service._get_container_create_options({'some': 'overrides'}, 1)
  173. assert self.mock_client.create_host_config.called
  174. assert self.mock_client.create_host_config.call_args[1]['cgroup_parent'] == 'test'
  175. def test_log_opt(self):
  176. self.mock_client.create_host_config.return_value = {}
  177. log_opt = {'syslog-address': 'tcp://192.168.0.42:123'}
  178. logging = {'driver': 'syslog', 'options': log_opt}
  179. service = Service(
  180. name='foo',
  181. image='foo',
  182. hostname='name',
  183. client=self.mock_client,
  184. log_driver='syslog',
  185. logging=logging)
  186. service._get_container_create_options({'some': 'overrides'}, 1)
  187. assert self.mock_client.create_host_config.called
  188. assert self.mock_client.create_host_config.call_args[1]['log_config'] == {
  189. 'Type': 'syslog', 'Config': {'syslog-address': 'tcp://192.168.0.42:123'}
  190. }
  191. def test_stop_grace_period(self):
  192. self.mock_client.api_version = '1.25'
  193. self.mock_client.create_host_config.return_value = {}
  194. service = Service(
  195. 'foo',
  196. image='foo',
  197. client=self.mock_client,
  198. stop_grace_period="1m35s")
  199. opts = service._get_container_create_options({'image': 'foo'}, 1)
  200. assert opts['stop_timeout'] == 95
  201. def test_split_domainname_none(self):
  202. service = Service(
  203. 'foo',
  204. image='foo',
  205. hostname='name.domain.tld',
  206. client=self.mock_client)
  207. opts = service._get_container_create_options({'image': 'foo'}, 1)
  208. assert opts['hostname'] == 'name.domain.tld', 'hostname'
  209. assert not ('domainname' in opts), 'domainname'
  210. def test_split_domainname_fqdn(self):
  211. self.mock_client.api_version = '1.22'
  212. service = Service(
  213. 'foo',
  214. hostname='name.domain.tld',
  215. image='foo',
  216. client=self.mock_client)
  217. opts = service._get_container_create_options({'image': 'foo'}, 1)
  218. assert opts['hostname'] == 'name', 'hostname'
  219. assert opts['domainname'] == 'domain.tld', 'domainname'
  220. def test_split_domainname_both(self):
  221. self.mock_client.api_version = '1.22'
  222. service = Service(
  223. 'foo',
  224. hostname='name',
  225. image='foo',
  226. domainname='domain.tld',
  227. client=self.mock_client)
  228. opts = service._get_container_create_options({'image': 'foo'}, 1)
  229. assert opts['hostname'] == 'name', 'hostname'
  230. assert opts['domainname'] == 'domain.tld', 'domainname'
  231. def test_split_domainname_weird(self):
  232. self.mock_client.api_version = '1.22'
  233. service = Service(
  234. 'foo',
  235. hostname='name.sub',
  236. domainname='domain.tld',
  237. image='foo',
  238. client=self.mock_client)
  239. opts = service._get_container_create_options({'image': 'foo'}, 1)
  240. assert opts['hostname'] == 'name.sub', 'hostname'
  241. assert opts['domainname'] == 'domain.tld', 'domainname'
  242. def test_no_default_hostname_when_not_using_networking(self):
  243. service = Service(
  244. 'foo',
  245. image='foo',
  246. use_networking=False,
  247. client=self.mock_client,
  248. )
  249. opts = service._get_container_create_options({'image': 'foo'}, 1)
  250. assert opts.get('hostname') is None
  251. def test_get_container_create_options_with_name_option(self):
  252. service = Service(
  253. 'foo',
  254. image='foo',
  255. client=self.mock_client,
  256. container_name='foo1')
  257. name = 'the_new_name'
  258. opts = service._get_container_create_options(
  259. {'name': name},
  260. 1,
  261. one_off=OneOffFilter.only)
  262. assert opts['name'] == name
  263. def test_get_container_create_options_does_not_mutate_options(self):
  264. labels = {'thing': 'real'}
  265. environment = {'also': 'real'}
  266. service = Service(
  267. 'foo',
  268. image='foo',
  269. labels=dict(labels),
  270. client=self.mock_client,
  271. environment=dict(environment),
  272. )
  273. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  274. prev_container = mock.Mock(
  275. id='ababab',
  276. image_config={'ContainerConfig': {}})
  277. prev_container.get.return_value = None
  278. opts = service._get_container_create_options(
  279. {},
  280. 1,
  281. previous_container=prev_container)
  282. assert service.options['labels'] == labels
  283. assert service.options['environment'] == environment
  284. assert opts['labels'][LABEL_CONFIG_HASH] == \
  285. '2524a06fcb3d781aa2c981fc40bcfa08013bb318e4273bfa388df22023e6f2aa'
  286. assert opts['environment'] == ['also=real']
  287. def test_get_container_create_options_sets_affinity_with_binds(self):
  288. service = Service(
  289. 'foo',
  290. image='foo',
  291. client=self.mock_client,
  292. )
  293. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  294. prev_container = mock.Mock(
  295. id='ababab',
  296. image_config={'ContainerConfig': {'Volumes': ['/data']}})
  297. def container_get(key):
  298. return {
  299. 'Mounts': [
  300. {
  301. 'Destination': '/data',
  302. 'Source': '/some/path',
  303. 'Name': 'abab1234',
  304. },
  305. ]
  306. }.get(key, None)
  307. prev_container.get.side_effect = container_get
  308. opts = service._get_container_create_options(
  309. {},
  310. 1,
  311. previous_container=prev_container)
  312. assert opts['environment'] == ['affinity:container==ababab']
  313. def test_get_container_create_options_no_affinity_without_binds(self):
  314. service = Service('foo', image='foo', client=self.mock_client)
  315. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  316. prev_container = mock.Mock(
  317. id='ababab',
  318. image_config={'ContainerConfig': {}})
  319. prev_container.get.return_value = None
  320. opts = service._get_container_create_options(
  321. {},
  322. 1,
  323. previous_container=prev_container)
  324. assert opts['environment'] == []
  325. def test_get_container_not_found(self):
  326. self.mock_client.containers.return_value = []
  327. service = Service('foo', client=self.mock_client, image='foo')
  328. with pytest.raises(ValueError):
  329. service.get_container()
  330. @mock.patch('compose.service.Container', autospec=True)
  331. def test_get_container(self, mock_container_class):
  332. container_dict = dict(Name='default_foo_2')
  333. self.mock_client.containers.return_value = [container_dict]
  334. service = Service('foo', image='foo', client=self.mock_client)
  335. container = service.get_container(number=2)
  336. assert container == mock_container_class.from_ps.return_value
  337. mock_container_class.from_ps.assert_called_once_with(
  338. self.mock_client, container_dict)
  339. @mock.patch('compose.service.log', autospec=True)
  340. def test_pull_image(self, mock_log):
  341. service = Service('foo', client=self.mock_client, image='someimage:sometag')
  342. service.pull()
  343. self.mock_client.pull.assert_called_once_with(
  344. 'someimage',
  345. tag='sometag',
  346. stream=True)
  347. mock_log.info.assert_called_once_with('Pulling foo (someimage:sometag)...')
  348. def test_pull_image_no_tag(self):
  349. service = Service('foo', client=self.mock_client, image='ababab')
  350. service.pull()
  351. self.mock_client.pull.assert_called_once_with(
  352. 'ababab',
  353. tag='latest',
  354. stream=True)
  355. @mock.patch('compose.service.log', autospec=True)
  356. def test_pull_image_digest(self, mock_log):
  357. service = Service('foo', client=self.mock_client, image='someimage@sha256:1234')
  358. service.pull()
  359. self.mock_client.pull.assert_called_once_with(
  360. 'someimage',
  361. tag='sha256:1234',
  362. stream=True)
  363. mock_log.info.assert_called_once_with('Pulling foo (someimage@sha256:1234)...')
  364. @mock.patch('compose.service.Container', autospec=True)
  365. def test_recreate_container(self, _):
  366. mock_container = mock.create_autospec(Container)
  367. service = Service('foo', client=self.mock_client, image='someimage')
  368. service.image = lambda: {'Id': 'abc123'}
  369. new_container = service.recreate_container(mock_container)
  370. mock_container.stop.assert_called_once_with(timeout=10)
  371. mock_container.rename_to_tmp_name.assert_called_once_with()
  372. new_container.start.assert_called_once_with()
  373. mock_container.remove.assert_called_once_with()
  374. @mock.patch('compose.service.Container', autospec=True)
  375. def test_recreate_container_with_timeout(self, _):
  376. mock_container = mock.create_autospec(Container)
  377. self.mock_client.inspect_image.return_value = {'Id': 'abc123'}
  378. service = Service('foo', client=self.mock_client, image='someimage')
  379. service.recreate_container(mock_container, timeout=1)
  380. mock_container.stop.assert_called_once_with(timeout=1)
  381. def test_parse_repository_tag(self):
  382. assert parse_repository_tag("root") == ("root", "", ":")
  383. assert parse_repository_tag("root:tag") == ("root", "tag", ":")
  384. assert parse_repository_tag("user/repo") == ("user/repo", "", ":")
  385. assert parse_repository_tag("user/repo:tag") == ("user/repo", "tag", ":")
  386. assert parse_repository_tag("url:5000/repo") == ("url:5000/repo", "", ":")
  387. assert parse_repository_tag("url:5000/repo:tag") == ("url:5000/repo", "tag", ":")
  388. assert parse_repository_tag("root@sha256:digest") == ("root", "sha256:digest", "@")
  389. assert parse_repository_tag("user/repo@sha256:digest") == ("user/repo", "sha256:digest", "@")
  390. assert parse_repository_tag("url:5000/repo@sha256:digest") == (
  391. "url:5000/repo", "sha256:digest", "@"
  392. )
  393. def test_create_container(self):
  394. service = Service('foo', client=self.mock_client, build={'context': '.'})
  395. self.mock_client.inspect_image.side_effect = [
  396. NoSuchImageError,
  397. {'Id': 'abc123'},
  398. ]
  399. self.mock_client.build.return_value = [
  400. '{"stream": "Successfully built abcd"}',
  401. ]
  402. with mock.patch('compose.service.log', autospec=True) as mock_log:
  403. service.create_container()
  404. assert mock_log.warn.called
  405. _, args, _ = mock_log.warn.mock_calls[0]
  406. assert 'was built because it did not already exist' in args[0]
  407. self.mock_client.build.assert_called_once_with(
  408. tag='default_foo',
  409. dockerfile=None,
  410. path='.',
  411. pull=False,
  412. forcerm=False,
  413. nocache=False,
  414. rm=True,
  415. buildargs={},
  416. labels=None,
  417. cache_from=None,
  418. network_mode=None,
  419. target=None,
  420. shmsize=None,
  421. extra_hosts=None,
  422. container_limits={'memory': None},
  423. gzip=False,
  424. )
  425. def test_ensure_image_exists_no_build(self):
  426. service = Service('foo', client=self.mock_client, build={'context': '.'})
  427. self.mock_client.inspect_image.return_value = {'Id': 'abc123'}
  428. service.ensure_image_exists(do_build=BuildAction.skip)
  429. assert not self.mock_client.build.called
  430. def test_ensure_image_exists_no_build_but_needs_build(self):
  431. service = Service('foo', client=self.mock_client, build={'context': '.'})
  432. self.mock_client.inspect_image.side_effect = NoSuchImageError
  433. with pytest.raises(NeedsBuildError):
  434. service.ensure_image_exists(do_build=BuildAction.skip)
  435. def test_ensure_image_exists_force_build(self):
  436. service = Service('foo', client=self.mock_client, build={'context': '.'})
  437. self.mock_client.inspect_image.return_value = {'Id': 'abc123'}
  438. self.mock_client.build.return_value = [
  439. '{"stream": "Successfully built abcd"}',
  440. ]
  441. with mock.patch('compose.service.log', autospec=True) as mock_log:
  442. service.ensure_image_exists(do_build=BuildAction.force)
  443. assert not mock_log.warn.called
  444. self.mock_client.build.assert_called_once_with(
  445. tag='default_foo',
  446. dockerfile=None,
  447. path='.',
  448. pull=False,
  449. forcerm=False,
  450. nocache=False,
  451. rm=True,
  452. buildargs={},
  453. labels=None,
  454. cache_from=None,
  455. network_mode=None,
  456. target=None,
  457. shmsize=None,
  458. extra_hosts=None,
  459. container_limits={'memory': None},
  460. gzip=False
  461. )
  462. def test_build_does_not_pull(self):
  463. self.mock_client.build.return_value = [
  464. b'{"stream": "Successfully built 12345"}',
  465. ]
  466. service = Service('foo', client=self.mock_client, build={'context': '.'})
  467. service.build()
  468. assert self.mock_client.build.call_count == 1
  469. assert not self.mock_client.build.call_args[1]['pull']
  470. def test_build_with_override_build_args(self):
  471. self.mock_client.build.return_value = [
  472. b'{"stream": "Successfully built 12345"}',
  473. ]
  474. build_args = {
  475. 'arg1': 'arg1_new_value',
  476. }
  477. service = Service('foo', client=self.mock_client,
  478. build={'context': '.', 'args': {'arg1': 'arg1', 'arg2': 'arg2'}})
  479. service.build(build_args_override=build_args)
  480. called_build_args = self.mock_client.build.call_args[1]['buildargs']
  481. assert called_build_args['arg1'] == build_args['arg1']
  482. assert called_build_args['arg2'] == 'arg2'
  483. def test_config_dict(self):
  484. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  485. service = Service(
  486. 'foo',
  487. image='example.com/foo',
  488. client=self.mock_client,
  489. network_mode=ServiceNetworkMode(Service('other')),
  490. networks={'default': None},
  491. links=[(Service('one'), 'one')],
  492. volumes_from=[VolumeFromSpec(Service('two'), 'rw', 'service')])
  493. config_dict = service.config_dict()
  494. expected = {
  495. 'image_id': 'abcd',
  496. 'options': {'image': 'example.com/foo'},
  497. 'links': [('one', 'one')],
  498. 'net': 'other',
  499. 'networks': {'default': None},
  500. 'volumes_from': [('two', 'rw')],
  501. }
  502. assert config_dict == expected
  503. def test_config_dict_with_network_mode_from_container(self):
  504. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  505. container = Container(
  506. self.mock_client,
  507. {'Id': 'aaabbb', 'Name': '/foo_1'})
  508. service = Service(
  509. 'foo',
  510. image='example.com/foo',
  511. client=self.mock_client,
  512. network_mode=ContainerNetworkMode(container))
  513. config_dict = service.config_dict()
  514. expected = {
  515. 'image_id': 'abcd',
  516. 'options': {'image': 'example.com/foo'},
  517. 'links': [],
  518. 'networks': {},
  519. 'net': 'aaabbb',
  520. 'volumes_from': [],
  521. }
  522. assert config_dict == expected
  523. def test_config_hash_matches_label(self):
  524. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  525. service = Service(
  526. 'foo',
  527. image='example.com/foo',
  528. client=self.mock_client,
  529. network_mode=NetworkMode('bridge'),
  530. networks={'bridge': {}},
  531. links=[(Service('one', client=self.mock_client), 'one')],
  532. volumes_from=[VolumeFromSpec(Service('two', client=self.mock_client), 'rw', 'service')]
  533. )
  534. config_hash = service.config_hash
  535. for api_version in set(API_VERSIONS.values()):
  536. self.mock_client.api_version = api_version
  537. assert service._get_container_create_options({}, 1)['labels'][LABEL_CONFIG_HASH] == (
  538. config_hash
  539. )
  540. def test_remove_image_none(self):
  541. web = Service('web', image='example', client=self.mock_client)
  542. assert not web.remove_image(ImageType.none)
  543. assert not self.mock_client.remove_image.called
  544. def test_remove_image_local_with_image_name_doesnt_remove(self):
  545. web = Service('web', image='example', client=self.mock_client)
  546. assert not web.remove_image(ImageType.local)
  547. assert not self.mock_client.remove_image.called
  548. def test_remove_image_local_without_image_name_does_remove(self):
  549. web = Service('web', build='.', client=self.mock_client)
  550. assert web.remove_image(ImageType.local)
  551. self.mock_client.remove_image.assert_called_once_with(web.image_name)
  552. def test_remove_image_all_does_remove(self):
  553. web = Service('web', image='example', client=self.mock_client)
  554. assert web.remove_image(ImageType.all)
  555. self.mock_client.remove_image.assert_called_once_with(web.image_name)
  556. def test_remove_image_with_error(self):
  557. self.mock_client.remove_image.side_effect = error = APIError(
  558. message="testing",
  559. response={},
  560. explanation="Boom")
  561. web = Service('web', image='example', client=self.mock_client)
  562. with mock.patch('compose.service.log', autospec=True) as mock_log:
  563. assert not web.remove_image(ImageType.all)
  564. mock_log.error.assert_called_once_with(
  565. "Failed to remove image for service %s: %s", web.name, error)
  566. def test_specifies_host_port_with_no_ports(self):
  567. service = Service(
  568. 'foo',
  569. image='foo')
  570. assert not service.specifies_host_port()
  571. def test_specifies_host_port_with_container_port(self):
  572. service = Service(
  573. 'foo',
  574. image='foo',
  575. ports=["2000"])
  576. assert not service.specifies_host_port()
  577. def test_specifies_host_port_with_host_port(self):
  578. service = Service(
  579. 'foo',
  580. image='foo',
  581. ports=["1000:2000"])
  582. assert service.specifies_host_port()
  583. def test_specifies_host_port_with_host_ip_no_port(self):
  584. service = Service(
  585. 'foo',
  586. image='foo',
  587. ports=["127.0.0.1::2000"])
  588. assert not service.specifies_host_port()
  589. def test_specifies_host_port_with_host_ip_and_port(self):
  590. service = Service(
  591. 'foo',
  592. image='foo',
  593. ports=["127.0.0.1:1000:2000"])
  594. assert service.specifies_host_port()
  595. def test_specifies_host_port_with_container_port_range(self):
  596. service = Service(
  597. 'foo',
  598. image='foo',
  599. ports=["2000-3000"])
  600. assert not service.specifies_host_port()
  601. def test_specifies_host_port_with_host_port_range(self):
  602. service = Service(
  603. 'foo',
  604. image='foo',
  605. ports=["1000-2000:2000-3000"])
  606. assert service.specifies_host_port()
  607. def test_specifies_host_port_with_host_ip_no_port_range(self):
  608. service = Service(
  609. 'foo',
  610. image='foo',
  611. ports=["127.0.0.1::2000-3000"])
  612. assert not service.specifies_host_port()
  613. def test_specifies_host_port_with_host_ip_and_port_range(self):
  614. service = Service(
  615. 'foo',
  616. image='foo',
  617. ports=["127.0.0.1:1000-2000:2000-3000"])
  618. assert service.specifies_host_port()
  619. def test_image_name_from_config(self):
  620. image_name = 'example/web:latest'
  621. service = Service('foo', image=image_name)
  622. assert service.image_name == image_name
  623. def test_image_name_default(self):
  624. service = Service('foo', project='testing')
  625. assert service.image_name == 'testing_foo'
  626. @mock.patch('compose.service.log', autospec=True)
  627. def test_only_log_warning_when_host_ports_clash(self, mock_log):
  628. self.mock_client.inspect_image.return_value = {'Id': 'abcd'}
  629. ParallelStreamWriter.instance = None
  630. name = 'foo'
  631. service = Service(
  632. name,
  633. client=self.mock_client,
  634. ports=["8080:80"])
  635. service.scale(0)
  636. assert not mock_log.warn.called
  637. service.scale(1)
  638. assert not mock_log.warn.called
  639. service.scale(2)
  640. mock_log.warn.assert_called_once_with(
  641. 'The "{}" service specifies a port on the host. If multiple containers '
  642. 'for this service are created on a single host, the port will clash.'.format(name))
  643. def test_parse_proxy_config(self):
  644. default_proxy_config = {
  645. 'httpProxy': 'http://proxy.mycorp.com:3128',
  646. 'httpsProxy': 'https://user:[email protected]:3129',
  647. 'ftpProxy': 'http://ftpproxy.mycorp.com:21',
  648. 'noProxy': '*.intra.mycorp.com',
  649. }
  650. self.mock_client.base_url = 'http+docker://localunixsocket'
  651. self.mock_client._general_configs = {
  652. 'proxies': {
  653. 'default': default_proxy_config,
  654. }
  655. }
  656. service = Service('foo', client=self.mock_client)
  657. assert service._parse_proxy_config() == {
  658. 'HTTP_PROXY': default_proxy_config['httpProxy'],
  659. 'http_proxy': default_proxy_config['httpProxy'],
  660. 'HTTPS_PROXY': default_proxy_config['httpsProxy'],
  661. 'https_proxy': default_proxy_config['httpsProxy'],
  662. 'FTP_PROXY': default_proxy_config['ftpProxy'],
  663. 'ftp_proxy': default_proxy_config['ftpProxy'],
  664. 'NO_PROXY': default_proxy_config['noProxy'],
  665. 'no_proxy': default_proxy_config['noProxy'],
  666. }
  667. def test_parse_proxy_config_per_host(self):
  668. default_proxy_config = {
  669. 'httpProxy': 'http://proxy.mycorp.com:3128',
  670. 'httpsProxy': 'https://user:[email protected]:3129',
  671. 'ftpProxy': 'http://ftpproxy.mycorp.com:21',
  672. 'noProxy': '*.intra.mycorp.com',
  673. }
  674. host_specific_proxy_config = {
  675. 'httpProxy': 'http://proxy.example.com:3128',
  676. 'httpsProxy': 'https://user:[email protected]:3129',
  677. 'ftpProxy': 'http://ftpproxy.example.com:21',
  678. 'noProxy': '*.intra.example.com'
  679. }
  680. self.mock_client.base_url = 'http+docker://localunixsocket'
  681. self.mock_client._general_configs = {
  682. 'proxies': {
  683. 'default': default_proxy_config,
  684. 'tcp://example.docker.com:2376': host_specific_proxy_config,
  685. }
  686. }
  687. service = Service('foo', client=self.mock_client)
  688. assert service._parse_proxy_config() == {
  689. 'HTTP_PROXY': default_proxy_config['httpProxy'],
  690. 'http_proxy': default_proxy_config['httpProxy'],
  691. 'HTTPS_PROXY': default_proxy_config['httpsProxy'],
  692. 'https_proxy': default_proxy_config['httpsProxy'],
  693. 'FTP_PROXY': default_proxy_config['ftpProxy'],
  694. 'ftp_proxy': default_proxy_config['ftpProxy'],
  695. 'NO_PROXY': default_proxy_config['noProxy'],
  696. 'no_proxy': default_proxy_config['noProxy'],
  697. }
  698. self.mock_client._original_base_url = 'tcp://example.docker.com:2376'
  699. assert service._parse_proxy_config() == {
  700. 'HTTP_PROXY': host_specific_proxy_config['httpProxy'],
  701. 'http_proxy': host_specific_proxy_config['httpProxy'],
  702. 'HTTPS_PROXY': host_specific_proxy_config['httpsProxy'],
  703. 'https_proxy': host_specific_proxy_config['httpsProxy'],
  704. 'FTP_PROXY': host_specific_proxy_config['ftpProxy'],
  705. 'ftp_proxy': host_specific_proxy_config['ftpProxy'],
  706. 'NO_PROXY': host_specific_proxy_config['noProxy'],
  707. 'no_proxy': host_specific_proxy_config['noProxy'],
  708. }
  709. def test_build_service_with_proxy_config(self):
  710. default_proxy_config = {
  711. 'httpProxy': 'http://proxy.mycorp.com:3128',
  712. 'httpsProxy': 'https://user:[email protected]:3129',
  713. }
  714. buildargs = {
  715. 'HTTPS_PROXY': 'https://rdcf.th08.jp:8911',
  716. 'https_proxy': 'https://rdcf.th08.jp:8911',
  717. }
  718. self.mock_client._general_configs = {
  719. 'proxies': {
  720. 'default': default_proxy_config,
  721. }
  722. }
  723. self.mock_client.base_url = 'http+docker://localunixsocket'
  724. self.mock_client.build.return_value = [
  725. b'{"stream": "Successfully built 12345"}',
  726. ]
  727. service = Service('foo', client=self.mock_client, build={'context': '.', 'args': buildargs})
  728. service.build()
  729. assert self.mock_client.build.call_count == 1
  730. assert self.mock_client.build.call_args[1]['buildargs'] == {
  731. 'HTTP_PROXY': default_proxy_config['httpProxy'],
  732. 'http_proxy': default_proxy_config['httpProxy'],
  733. 'HTTPS_PROXY': buildargs['HTTPS_PROXY'],
  734. 'https_proxy': buildargs['HTTPS_PROXY'],
  735. }
  736. def test_get_create_options_with_proxy_config(self):
  737. default_proxy_config = {
  738. 'httpProxy': 'http://proxy.mycorp.com:3128',
  739. 'httpsProxy': 'https://user:[email protected]:3129',
  740. 'ftpProxy': 'http://ftpproxy.mycorp.com:21',
  741. }
  742. self.mock_client._general_configs = {
  743. 'proxies': {
  744. 'default': default_proxy_config,
  745. }
  746. }
  747. self.mock_client.base_url = 'http+docker://localunixsocket'
  748. override_options = {
  749. 'environment': {
  750. 'FTP_PROXY': 'ftp://xdge.exo.au:21',
  751. 'ftp_proxy': 'ftp://xdge.exo.au:21',
  752. }
  753. }
  754. environment = {
  755. 'HTTPS_PROXY': 'https://rdcf.th08.jp:8911',
  756. 'https_proxy': 'https://rdcf.th08.jp:8911',
  757. }
  758. service = Service('foo', client=self.mock_client, environment=environment)
  759. create_opts = service._get_container_create_options(override_options, 1)
  760. assert set(create_opts['environment']) == set(format_environment({
  761. 'HTTP_PROXY': default_proxy_config['httpProxy'],
  762. 'http_proxy': default_proxy_config['httpProxy'],
  763. 'HTTPS_PROXY': environment['HTTPS_PROXY'],
  764. 'https_proxy': environment['HTTPS_PROXY'],
  765. 'FTP_PROXY': override_options['environment']['FTP_PROXY'],
  766. 'ftp_proxy': override_options['environment']['FTP_PROXY'],
  767. }))
  768. class TestServiceNetwork(unittest.TestCase):
  769. def setUp(self):
  770. self.mock_client = mock.create_autospec(docker.APIClient)
  771. self.mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  772. self.mock_client._general_configs = {}
  773. def test_connect_container_to_networks_short_aliase_exists(self):
  774. service = Service(
  775. 'db',
  776. self.mock_client,
  777. 'myproject',
  778. image='foo',
  779. networks={'project_default': {}})
  780. container = Container(
  781. None,
  782. {
  783. 'Id': 'abcdef',
  784. 'NetworkSettings': {
  785. 'Networks': {
  786. 'project_default': {
  787. 'Aliases': ['analias', 'abcdef'],
  788. },
  789. },
  790. },
  791. },
  792. True)
  793. service.connect_container_to_networks(container)
  794. assert not self.mock_client.disconnect_container_from_network.call_count
  795. assert not self.mock_client.connect_container_to_network.call_count
  796. def sort_by_name(dictionary_list):
  797. return sorted(dictionary_list, key=lambda k: k['name'])
  798. class BuildUlimitsTestCase(unittest.TestCase):
  799. def test_build_ulimits_with_dict(self):
  800. ulimits = build_ulimits(
  801. {
  802. 'nofile': {'soft': 10000, 'hard': 20000},
  803. 'nproc': {'soft': 65535, 'hard': 65535}
  804. }
  805. )
  806. expected = [
  807. {'name': 'nofile', 'soft': 10000, 'hard': 20000},
  808. {'name': 'nproc', 'soft': 65535, 'hard': 65535}
  809. ]
  810. assert sort_by_name(ulimits) == sort_by_name(expected)
  811. def test_build_ulimits_with_ints(self):
  812. ulimits = build_ulimits({'nofile': 20000, 'nproc': 65535})
  813. expected = [
  814. {'name': 'nofile', 'soft': 20000, 'hard': 20000},
  815. {'name': 'nproc', 'soft': 65535, 'hard': 65535}
  816. ]
  817. assert sort_by_name(ulimits) == sort_by_name(expected)
  818. def test_build_ulimits_with_integers_and_dicts(self):
  819. ulimits = build_ulimits(
  820. {
  821. 'nproc': 65535,
  822. 'nofile': {'soft': 10000, 'hard': 20000}
  823. }
  824. )
  825. expected = [
  826. {'name': 'nofile', 'soft': 10000, 'hard': 20000},
  827. {'name': 'nproc', 'soft': 65535, 'hard': 65535}
  828. ]
  829. assert sort_by_name(ulimits) == sort_by_name(expected)
  830. class NetTestCase(unittest.TestCase):
  831. def setUp(self):
  832. self.mock_client = mock.create_autospec(docker.APIClient)
  833. self.mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  834. self.mock_client._general_configs = {}
  835. def test_network_mode(self):
  836. network_mode = NetworkMode('host')
  837. assert network_mode.id == 'host'
  838. assert network_mode.mode == 'host'
  839. assert network_mode.service_name is None
  840. def test_network_mode_container(self):
  841. container_id = 'abcd'
  842. network_mode = ContainerNetworkMode(Container(None, {'Id': container_id}))
  843. assert network_mode.id == container_id
  844. assert network_mode.mode == 'container:' + container_id
  845. assert network_mode.service_name is None
  846. def test_network_mode_service(self):
  847. container_id = 'bbbb'
  848. service_name = 'web'
  849. self.mock_client.containers.return_value = [
  850. {'Id': container_id, 'Name': container_id, 'Image': 'abcd'},
  851. ]
  852. service = Service(name=service_name, client=self.mock_client)
  853. network_mode = ServiceNetworkMode(service)
  854. assert network_mode.id == service_name
  855. assert network_mode.mode == 'container:' + container_id
  856. assert network_mode.service_name == service_name
  857. def test_network_mode_service_no_containers(self):
  858. service_name = 'web'
  859. self.mock_client.containers.return_value = []
  860. service = Service(name=service_name, client=self.mock_client)
  861. network_mode = ServiceNetworkMode(service)
  862. assert network_mode.id == service_name
  863. assert network_mode.mode is None
  864. assert network_mode.service_name == service_name
  865. class ServicePortsTest(unittest.TestCase):
  866. def test_formatted_ports(self):
  867. ports = [
  868. '3000',
  869. '0.0.0.0:4025-4030:23000-23005',
  870. ServicePort(6000, None, None, None, None),
  871. ServicePort(8080, 8080, None, None, None),
  872. ServicePort('20000', '20000', 'udp', 'ingress', None),
  873. ServicePort(30000, '30000', 'tcp', None, '127.0.0.1'),
  874. ]
  875. formatted = formatted_ports(ports)
  876. assert ports[0] in formatted
  877. assert ports[1] in formatted
  878. assert '6000/tcp' in formatted
  879. assert '8080:8080/tcp' in formatted
  880. assert '20000:20000/udp' in formatted
  881. assert '127.0.0.1:30000:30000/tcp' in formatted
  882. def build_mount(destination, source, mode='rw'):
  883. return {'Source': source, 'Destination': destination, 'Mode': mode}
  884. class ServiceVolumesTest(unittest.TestCase):
  885. def setUp(self):
  886. self.mock_client = mock.create_autospec(docker.APIClient)
  887. self.mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  888. self.mock_client._general_configs = {}
  889. def test_build_volume_binding(self):
  890. binding = build_volume_binding(VolumeSpec.parse('/outside:/inside', True))
  891. assert binding == ('/inside', '/outside:/inside:rw')
  892. def test_get_container_data_volumes(self):
  893. options = [VolumeSpec.parse(v) for v in [
  894. '/host/volume:/host/volume:ro',
  895. '/new/volume',
  896. '/existing/volume',
  897. 'named:/named/vol',
  898. '/dev/tmpfs'
  899. ]]
  900. self.mock_client.inspect_image.return_value = {
  901. 'ContainerConfig': {
  902. 'Volumes': {
  903. '/mnt/image/data': {},
  904. }
  905. }
  906. }
  907. container = Container(self.mock_client, {
  908. 'Image': 'ababab',
  909. 'Mounts': [
  910. {
  911. 'Source': '/host/volume',
  912. 'Destination': '/host/volume',
  913. 'Mode': '',
  914. 'RW': True,
  915. 'Name': 'hostvolume',
  916. }, {
  917. 'Source': '/var/lib/docker/aaaaaaaa',
  918. 'Destination': '/existing/volume',
  919. 'Mode': '',
  920. 'RW': True,
  921. 'Name': 'existingvolume',
  922. }, {
  923. 'Source': '/var/lib/docker/bbbbbbbb',
  924. 'Destination': '/removed/volume',
  925. 'Mode': '',
  926. 'RW': True,
  927. 'Name': 'removedvolume',
  928. }, {
  929. 'Source': '/var/lib/docker/cccccccc',
  930. 'Destination': '/mnt/image/data',
  931. 'Mode': '',
  932. 'RW': True,
  933. 'Name': 'imagedata',
  934. },
  935. ]
  936. }, has_been_inspected=True)
  937. expected = [
  938. VolumeSpec.parse('existingvolume:/existing/volume:rw'),
  939. VolumeSpec.parse('imagedata:/mnt/image/data:rw'),
  940. ]
  941. volumes, _ = get_container_data_volumes(container, options, ['/dev/tmpfs'], [])
  942. assert sorted(volumes) == sorted(expected)
  943. def test_merge_volume_bindings(self):
  944. options = [
  945. VolumeSpec.parse(v, True) for v in [
  946. '/host/volume:/host/volume:ro',
  947. '/host/rw/volume:/host/rw/volume',
  948. '/new/volume',
  949. '/existing/volume',
  950. '/dev/tmpfs'
  951. ]
  952. ]
  953. self.mock_client.inspect_image.return_value = {
  954. 'ContainerConfig': {'Volumes': {}}
  955. }
  956. previous_container = Container(self.mock_client, {
  957. 'Id': 'cdefab',
  958. 'Image': 'ababab',
  959. 'Mounts': [{
  960. 'Source': '/var/lib/docker/aaaaaaaa',
  961. 'Destination': '/existing/volume',
  962. 'Mode': '',
  963. 'RW': True,
  964. 'Name': 'existingvolume',
  965. }],
  966. }, has_been_inspected=True)
  967. expected = [
  968. '/host/volume:/host/volume:ro',
  969. '/host/rw/volume:/host/rw/volume:rw',
  970. 'existingvolume:/existing/volume:rw',
  971. ]
  972. binds, affinity = merge_volume_bindings(options, ['/dev/tmpfs'], previous_container, [])
  973. assert sorted(binds) == sorted(expected)
  974. assert affinity == {'affinity:container': '=cdefab'}
  975. def test_mount_same_host_path_to_two_volumes(self):
  976. service = Service(
  977. 'web',
  978. image='busybox',
  979. volumes=[
  980. VolumeSpec.parse('/host/path:/data1', True),
  981. VolumeSpec.parse('/host/path:/data2', True),
  982. ],
  983. client=self.mock_client,
  984. )
  985. self.mock_client.inspect_image.return_value = {
  986. 'Id': 'ababab',
  987. 'ContainerConfig': {
  988. 'Volumes': {}
  989. }
  990. }
  991. service._get_container_create_options(
  992. override_options={},
  993. number=1,
  994. )
  995. assert set(self.mock_client.create_host_config.call_args[1]['binds']) == set([
  996. '/host/path:/data1:rw',
  997. '/host/path:/data2:rw',
  998. ])
  999. def test_get_container_create_options_with_different_host_path_in_container_json(self):
  1000. service = Service(
  1001. 'web',
  1002. image='busybox',
  1003. volumes=[VolumeSpec.parse('/host/path:/data')],
  1004. client=self.mock_client,
  1005. )
  1006. volume_name = 'abcdefff1234'
  1007. self.mock_client.inspect_image.return_value = {
  1008. 'Id': 'ababab',
  1009. 'ContainerConfig': {
  1010. 'Volumes': {
  1011. '/data': {},
  1012. }
  1013. }
  1014. }
  1015. self.mock_client.inspect_container.return_value = {
  1016. 'Id': '123123123',
  1017. 'Image': 'ababab',
  1018. 'Mounts': [
  1019. {
  1020. 'Destination': '/data',
  1021. 'Source': '/mnt/sda1/host/path',
  1022. 'Mode': '',
  1023. 'RW': True,
  1024. 'Driver': 'local',
  1025. 'Name': volume_name,
  1026. },
  1027. ]
  1028. }
  1029. service._get_container_create_options(
  1030. override_options={},
  1031. number=1,
  1032. previous_container=Container(self.mock_client, {'Id': '123123123'}),
  1033. )
  1034. assert (
  1035. self.mock_client.create_host_config.call_args[1]['binds'] ==
  1036. ['{}:/data:rw'.format(volume_name)]
  1037. )
  1038. def test_warn_on_masked_volume_no_warning_when_no_container_volumes(self):
  1039. volumes_option = [VolumeSpec('/home/user', '/path', 'rw')]
  1040. container_volumes = []
  1041. service = 'service_name'
  1042. with mock.patch('compose.service.log', autospec=True) as mock_log:
  1043. warn_on_masked_volume(volumes_option, container_volumes, service)
  1044. assert not mock_log.warn.called
  1045. def test_warn_on_masked_volume_when_masked(self):
  1046. volumes_option = [VolumeSpec('/home/user', '/path', 'rw')]
  1047. container_volumes = [
  1048. VolumeSpec('/var/lib/docker/path', '/path', 'rw'),
  1049. VolumeSpec('/var/lib/docker/path', '/other', 'rw'),
  1050. ]
  1051. service = 'service_name'
  1052. with mock.patch('compose.service.log', autospec=True) as mock_log:
  1053. warn_on_masked_volume(volumes_option, container_volumes, service)
  1054. mock_log.warn.assert_called_once_with(mock.ANY)
  1055. def test_warn_on_masked_no_warning_with_same_path(self):
  1056. volumes_option = [VolumeSpec('/home/user', '/path', 'rw')]
  1057. container_volumes = [VolumeSpec('/home/user', '/path', 'rw')]
  1058. service = 'service_name'
  1059. with mock.patch('compose.service.log', autospec=True) as mock_log:
  1060. warn_on_masked_volume(volumes_option, container_volumes, service)
  1061. assert not mock_log.warn.called
  1062. def test_warn_on_masked_no_warning_with_container_only_option(self):
  1063. volumes_option = [VolumeSpec(None, '/path', 'rw')]
  1064. container_volumes = [
  1065. VolumeSpec('/var/lib/docker/volume/path', '/path', 'rw')
  1066. ]
  1067. service = 'service_name'
  1068. with mock.patch('compose.service.log', autospec=True) as mock_log:
  1069. warn_on_masked_volume(volumes_option, container_volumes, service)
  1070. assert not mock_log.warn.called
  1071. def test_create_with_special_volume_mode(self):
  1072. self.mock_client.inspect_image.return_value = {'Id': 'imageid'}
  1073. self.mock_client.create_container.return_value = {'Id': 'containerid'}
  1074. volume = '/tmp:/foo:z'
  1075. Service(
  1076. 'web',
  1077. client=self.mock_client,
  1078. image='busybox',
  1079. volumes=[VolumeSpec.parse(volume, True)],
  1080. ).create_container()
  1081. assert self.mock_client.create_container.call_count == 1
  1082. assert self.mock_client.create_host_config.call_args[1]['binds'] == [volume]
  1083. class ServiceSecretTest(unittest.TestCase):
  1084. def setUp(self):
  1085. self.mock_client = mock.create_autospec(docker.APIClient)
  1086. self.mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  1087. self.mock_client._general_configs = {}
  1088. def test_get_secret_volumes(self):
  1089. secret1 = {
  1090. 'secret': ServiceSecret.parse({'source': 'secret1', 'target': 'b.txt'}),
  1091. 'file': 'a.txt'
  1092. }
  1093. service = Service(
  1094. 'web',
  1095. client=self.mock_client,
  1096. image='busybox',
  1097. secrets=[secret1]
  1098. )
  1099. volumes = service.get_secret_volumes()
  1100. assert volumes[0].source == secret1['file']
  1101. assert volumes[0].target == '{}/{}'.format(SECRETS_PATH, secret1['secret'].target)
  1102. def test_get_secret_volumes_abspath(self):
  1103. secret1 = {
  1104. 'secret': ServiceSecret.parse({'source': 'secret1', 'target': '/d.txt'}),
  1105. 'file': 'c.txt'
  1106. }
  1107. service = Service(
  1108. 'web',
  1109. client=self.mock_client,
  1110. image='busybox',
  1111. secrets=[secret1]
  1112. )
  1113. volumes = service.get_secret_volumes()
  1114. assert volumes[0].source == secret1['file']
  1115. assert volumes[0].target == secret1['secret'].target
  1116. def test_get_secret_volumes_no_target(self):
  1117. secret1 = {
  1118. 'secret': ServiceSecret.parse({'source': 'secret1'}),
  1119. 'file': 'c.txt'
  1120. }
  1121. service = Service(
  1122. 'web',
  1123. client=self.mock_client,
  1124. image='busybox',
  1125. secrets=[secret1]
  1126. )
  1127. volumes = service.get_secret_volumes()
  1128. assert volumes[0].source == secret1['file']
  1129. assert volumes[0].target == '{}/{}'.format(SECRETS_PATH, secret1['secret'].source)