project_test.py 51 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import os.path
  4. import random
  5. import py
  6. import pytest
  7. from docker.errors import NotFound
  8. from .. import mock
  9. from ..helpers import build_config as load_config
  10. from .testcases import DockerClientTestCase
  11. from compose.config import config
  12. from compose.config import ConfigurationError
  13. from compose.config import types
  14. from compose.config.config import V2_0
  15. from compose.config.config import V2_1
  16. from compose.config.config import V3_1
  17. from compose.config.types import VolumeFromSpec
  18. from compose.config.types import VolumeSpec
  19. from compose.const import LABEL_PROJECT
  20. from compose.const import LABEL_SERVICE
  21. from compose.container import Container
  22. from compose.errors import HealthCheckFailed
  23. from compose.errors import NoHealthCheckConfigured
  24. from compose.project import Project
  25. from compose.project import ProjectError
  26. from compose.service import ConvergenceStrategy
  27. from tests.integration.testcases import v2_1_only
  28. from tests.integration.testcases import v2_only
  29. from tests.integration.testcases import v3_only
  30. def build_config(**kwargs):
  31. return config.Config(
  32. version=kwargs.get('version'),
  33. services=kwargs.get('services'),
  34. volumes=kwargs.get('volumes'),
  35. networks=kwargs.get('networks'),
  36. secrets=kwargs.get('secrets'))
  37. class ProjectTest(DockerClientTestCase):
  38. def test_containers(self):
  39. web = self.create_service('web')
  40. db = self.create_service('db')
  41. project = Project('composetest', [web, db], self.client)
  42. project.up()
  43. containers = project.containers()
  44. self.assertEqual(len(containers), 2)
  45. def test_containers_with_service_names(self):
  46. web = self.create_service('web')
  47. db = self.create_service('db')
  48. project = Project('composetest', [web, db], self.client)
  49. project.up()
  50. containers = project.containers(['web'])
  51. self.assertEqual(
  52. [c.name for c in containers],
  53. ['composetest_web_1'])
  54. def test_containers_with_extra_service(self):
  55. web = self.create_service('web')
  56. web_1 = web.create_container()
  57. db = self.create_service('db')
  58. db_1 = db.create_container()
  59. self.create_service('extra').create_container()
  60. project = Project('composetest', [web, db], self.client)
  61. self.assertEqual(
  62. set(project.containers(stopped=True)),
  63. set([web_1, db_1]),
  64. )
  65. def test_volumes_from_service(self):
  66. project = Project.from_config(
  67. name='composetest',
  68. config_data=load_config({
  69. 'data': {
  70. 'image': 'busybox:latest',
  71. 'volumes': ['/var/data'],
  72. },
  73. 'db': {
  74. 'image': 'busybox:latest',
  75. 'volumes_from': ['data'],
  76. },
  77. }),
  78. client=self.client,
  79. )
  80. db = project.get_service('db')
  81. data = project.get_service('data')
  82. self.assertEqual(db.volumes_from, [VolumeFromSpec(data, 'rw', 'service')])
  83. def test_volumes_from_container(self):
  84. data_container = Container.create(
  85. self.client,
  86. image='busybox:latest',
  87. volumes=['/var/data'],
  88. name='composetest_data_container',
  89. labels={LABEL_PROJECT: 'composetest'},
  90. )
  91. project = Project.from_config(
  92. name='composetest',
  93. config_data=load_config({
  94. 'db': {
  95. 'image': 'busybox:latest',
  96. 'volumes_from': ['composetest_data_container'],
  97. },
  98. }),
  99. client=self.client,
  100. )
  101. db = project.get_service('db')
  102. self.assertEqual(db._get_volumes_from(), [data_container.id + ':rw'])
  103. @v2_only()
  104. def test_network_mode_from_service(self):
  105. project = Project.from_config(
  106. name='composetest',
  107. client=self.client,
  108. config_data=load_config({
  109. 'version': V2_0,
  110. 'services': {
  111. 'net': {
  112. 'image': 'busybox:latest',
  113. 'command': ["top"]
  114. },
  115. 'web': {
  116. 'image': 'busybox:latest',
  117. 'network_mode': 'service:net',
  118. 'command': ["top"]
  119. },
  120. },
  121. }),
  122. )
  123. project.up()
  124. web = project.get_service('web')
  125. net = project.get_service('net')
  126. self.assertEqual(web.network_mode.mode, 'container:' + net.containers()[0].id)
  127. @v2_only()
  128. def test_network_mode_from_container(self):
  129. def get_project():
  130. return Project.from_config(
  131. name='composetest',
  132. config_data=load_config({
  133. 'version': V2_0,
  134. 'services': {
  135. 'web': {
  136. 'image': 'busybox:latest',
  137. 'network_mode': 'container:composetest_net_container'
  138. },
  139. },
  140. }),
  141. client=self.client,
  142. )
  143. with pytest.raises(ConfigurationError) as excinfo:
  144. get_project()
  145. assert "container 'composetest_net_container' which does not exist" in excinfo.exconly()
  146. net_container = Container.create(
  147. self.client,
  148. image='busybox:latest',
  149. name='composetest_net_container',
  150. command='top',
  151. labels={LABEL_PROJECT: 'composetest'},
  152. )
  153. net_container.start()
  154. project = get_project()
  155. project.up()
  156. web = project.get_service('web')
  157. self.assertEqual(web.network_mode.mode, 'container:' + net_container.id)
  158. def test_net_from_service_v1(self):
  159. project = Project.from_config(
  160. name='composetest',
  161. config_data=load_config({
  162. 'net': {
  163. 'image': 'busybox:latest',
  164. 'command': ["top"]
  165. },
  166. 'web': {
  167. 'image': 'busybox:latest',
  168. 'net': 'container:net',
  169. 'command': ["top"]
  170. },
  171. }),
  172. client=self.client,
  173. )
  174. project.up()
  175. web = project.get_service('web')
  176. net = project.get_service('net')
  177. self.assertEqual(web.network_mode.mode, 'container:' + net.containers()[0].id)
  178. def test_net_from_container_v1(self):
  179. def get_project():
  180. return Project.from_config(
  181. name='composetest',
  182. config_data=load_config({
  183. 'web': {
  184. 'image': 'busybox:latest',
  185. 'net': 'container:composetest_net_container'
  186. },
  187. }),
  188. client=self.client,
  189. )
  190. with pytest.raises(ConfigurationError) as excinfo:
  191. get_project()
  192. assert "container 'composetest_net_container' which does not exist" in excinfo.exconly()
  193. net_container = Container.create(
  194. self.client,
  195. image='busybox:latest',
  196. name='composetest_net_container',
  197. command='top',
  198. labels={LABEL_PROJECT: 'composetest'},
  199. )
  200. net_container.start()
  201. project = get_project()
  202. project.up()
  203. web = project.get_service('web')
  204. self.assertEqual(web.network_mode.mode, 'container:' + net_container.id)
  205. def test_start_pause_unpause_stop_kill_remove(self):
  206. web = self.create_service('web')
  207. db = self.create_service('db')
  208. project = Project('composetest', [web, db], self.client)
  209. project.start()
  210. self.assertEqual(len(web.containers()), 0)
  211. self.assertEqual(len(db.containers()), 0)
  212. web_container_1 = web.create_container()
  213. web_container_2 = web.create_container()
  214. db_container = db.create_container()
  215. project.start(service_names=['web'])
  216. self.assertEqual(
  217. set(c.name for c in project.containers()),
  218. set([web_container_1.name, web_container_2.name]))
  219. project.start()
  220. self.assertEqual(
  221. set(c.name for c in project.containers()),
  222. set([web_container_1.name, web_container_2.name, db_container.name]))
  223. project.pause(service_names=['web'])
  224. self.assertEqual(
  225. set([c.name for c in project.containers() if c.is_paused]),
  226. set([web_container_1.name, web_container_2.name]))
  227. project.pause()
  228. self.assertEqual(
  229. set([c.name for c in project.containers() if c.is_paused]),
  230. set([web_container_1.name, web_container_2.name, db_container.name]))
  231. project.unpause(service_names=['db'])
  232. self.assertEqual(len([c.name for c in project.containers() if c.is_paused]), 2)
  233. project.unpause()
  234. self.assertEqual(len([c.name for c in project.containers() if c.is_paused]), 0)
  235. project.stop(service_names=['web'], timeout=1)
  236. self.assertEqual(set(c.name for c in project.containers()), set([db_container.name]))
  237. project.kill(service_names=['db'])
  238. self.assertEqual(len(project.containers()), 0)
  239. self.assertEqual(len(project.containers(stopped=True)), 3)
  240. project.remove_stopped(service_names=['web'])
  241. self.assertEqual(len(project.containers(stopped=True)), 1)
  242. project.remove_stopped()
  243. self.assertEqual(len(project.containers(stopped=True)), 0)
  244. def test_create(self):
  245. web = self.create_service('web')
  246. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  247. project = Project('composetest', [web, db], self.client)
  248. project.create(['db'])
  249. self.assertEqual(len(project.containers()), 0)
  250. self.assertEqual(len(project.containers(stopped=True)), 1)
  251. self.assertEqual(len(db.containers()), 0)
  252. self.assertEqual(len(db.containers(stopped=True)), 1)
  253. self.assertEqual(len(web.containers(stopped=True)), 0)
  254. def test_create_twice(self):
  255. web = self.create_service('web')
  256. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  257. project = Project('composetest', [web, db], self.client)
  258. project.create(['db', 'web'])
  259. project.create(['db', 'web'])
  260. self.assertEqual(len(project.containers()), 0)
  261. self.assertEqual(len(project.containers(stopped=True)), 2)
  262. self.assertEqual(len(db.containers()), 0)
  263. self.assertEqual(len(db.containers(stopped=True)), 1)
  264. self.assertEqual(len(web.containers()), 0)
  265. self.assertEqual(len(web.containers(stopped=True)), 1)
  266. def test_create_with_links(self):
  267. db = self.create_service('db')
  268. web = self.create_service('web', links=[(db, 'db')])
  269. project = Project('composetest', [db, web], self.client)
  270. project.create(['web'])
  271. self.assertEqual(len(project.containers()), 0)
  272. self.assertEqual(len(project.containers(stopped=True)), 2)
  273. self.assertEqual(len(db.containers()), 0)
  274. self.assertEqual(len(db.containers(stopped=True)), 1)
  275. self.assertEqual(len(web.containers()), 0)
  276. self.assertEqual(len(web.containers(stopped=True)), 1)
  277. def test_create_strategy_always(self):
  278. db = self.create_service('db')
  279. project = Project('composetest', [db], self.client)
  280. project.create(['db'])
  281. old_id = project.containers(stopped=True)[0].id
  282. project.create(['db'], strategy=ConvergenceStrategy.always)
  283. self.assertEqual(len(project.containers()), 0)
  284. self.assertEqual(len(project.containers(stopped=True)), 1)
  285. db_container = project.containers(stopped=True)[0]
  286. self.assertNotEqual(db_container.id, old_id)
  287. def test_create_strategy_never(self):
  288. db = self.create_service('db')
  289. project = Project('composetest', [db], self.client)
  290. project.create(['db'])
  291. old_id = project.containers(stopped=True)[0].id
  292. project.create(['db'], strategy=ConvergenceStrategy.never)
  293. self.assertEqual(len(project.containers()), 0)
  294. self.assertEqual(len(project.containers(stopped=True)), 1)
  295. db_container = project.containers(stopped=True)[0]
  296. self.assertEqual(db_container.id, old_id)
  297. def test_project_up(self):
  298. web = self.create_service('web')
  299. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  300. project = Project('composetest', [web, db], self.client)
  301. project.start()
  302. self.assertEqual(len(project.containers()), 0)
  303. project.up(['db'])
  304. self.assertEqual(len(project.containers()), 1)
  305. self.assertEqual(len(db.containers()), 1)
  306. self.assertEqual(len(web.containers()), 0)
  307. def test_project_up_starts_uncreated_services(self):
  308. db = self.create_service('db')
  309. web = self.create_service('web', links=[(db, 'db')])
  310. project = Project('composetest', [db, web], self.client)
  311. project.up(['db'])
  312. self.assertEqual(len(project.containers()), 1)
  313. project.up()
  314. self.assertEqual(len(project.containers()), 2)
  315. self.assertEqual(len(db.containers()), 1)
  316. self.assertEqual(len(web.containers()), 1)
  317. def test_recreate_preserves_volumes(self):
  318. web = self.create_service('web')
  319. db = self.create_service('db', volumes=[VolumeSpec.parse('/etc')])
  320. project = Project('composetest', [web, db], self.client)
  321. project.start()
  322. self.assertEqual(len(project.containers()), 0)
  323. project.up(['db'])
  324. self.assertEqual(len(project.containers()), 1)
  325. old_db_id = project.containers()[0].id
  326. db_volume_path = project.containers()[0].get('Volumes./etc')
  327. project.up(strategy=ConvergenceStrategy.always)
  328. self.assertEqual(len(project.containers()), 2)
  329. db_container = [c for c in project.containers() if 'db' in c.name][0]
  330. self.assertNotEqual(db_container.id, old_db_id)
  331. self.assertEqual(db_container.get('Volumes./etc'), db_volume_path)
  332. def test_project_up_with_no_recreate_running(self):
  333. web = self.create_service('web')
  334. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  335. project = Project('composetest', [web, db], self.client)
  336. project.start()
  337. self.assertEqual(len(project.containers()), 0)
  338. project.up(['db'])
  339. self.assertEqual(len(project.containers()), 1)
  340. old_db_id = project.containers()[0].id
  341. container, = project.containers()
  342. db_volume_path = container.get_mount('/var/db')['Source']
  343. project.up(strategy=ConvergenceStrategy.never)
  344. self.assertEqual(len(project.containers()), 2)
  345. db_container = [c for c in project.containers() if 'db' in c.name][0]
  346. self.assertEqual(db_container.id, old_db_id)
  347. self.assertEqual(
  348. db_container.get_mount('/var/db')['Source'],
  349. db_volume_path)
  350. def test_project_up_with_no_recreate_stopped(self):
  351. web = self.create_service('web')
  352. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  353. project = Project('composetest', [web, db], self.client)
  354. project.start()
  355. self.assertEqual(len(project.containers()), 0)
  356. project.up(['db'])
  357. project.kill()
  358. old_containers = project.containers(stopped=True)
  359. self.assertEqual(len(old_containers), 1)
  360. old_container, = old_containers
  361. old_db_id = old_container.id
  362. db_volume_path = old_container.get_mount('/var/db')['Source']
  363. project.up(strategy=ConvergenceStrategy.never)
  364. new_containers = project.containers(stopped=True)
  365. self.assertEqual(len(new_containers), 2)
  366. self.assertEqual([c.is_running for c in new_containers], [True, True])
  367. db_container = [c for c in new_containers if 'db' in c.name][0]
  368. self.assertEqual(db_container.id, old_db_id)
  369. self.assertEqual(
  370. db_container.get_mount('/var/db')['Source'],
  371. db_volume_path)
  372. def test_project_up_without_all_services(self):
  373. console = self.create_service('console')
  374. db = self.create_service('db')
  375. project = Project('composetest', [console, db], self.client)
  376. project.start()
  377. self.assertEqual(len(project.containers()), 0)
  378. project.up()
  379. self.assertEqual(len(project.containers()), 2)
  380. self.assertEqual(len(db.containers()), 1)
  381. self.assertEqual(len(console.containers()), 1)
  382. def test_project_up_starts_links(self):
  383. console = self.create_service('console')
  384. db = self.create_service('db', volumes=[VolumeSpec.parse('/var/db')])
  385. web = self.create_service('web', links=[(db, 'db')])
  386. project = Project('composetest', [web, db, console], self.client)
  387. project.start()
  388. self.assertEqual(len(project.containers()), 0)
  389. project.up(['web'])
  390. self.assertEqual(len(project.containers()), 2)
  391. self.assertEqual(len(web.containers()), 1)
  392. self.assertEqual(len(db.containers()), 1)
  393. self.assertEqual(len(console.containers()), 0)
  394. def test_project_up_starts_depends(self):
  395. project = Project.from_config(
  396. name='composetest',
  397. config_data=load_config({
  398. 'console': {
  399. 'image': 'busybox:latest',
  400. 'command': ["top"],
  401. },
  402. 'data': {
  403. 'image': 'busybox:latest',
  404. 'command': ["top"]
  405. },
  406. 'db': {
  407. 'image': 'busybox:latest',
  408. 'command': ["top"],
  409. 'volumes_from': ['data'],
  410. },
  411. 'web': {
  412. 'image': 'busybox:latest',
  413. 'command': ["top"],
  414. 'links': ['db'],
  415. },
  416. }),
  417. client=self.client,
  418. )
  419. project.start()
  420. self.assertEqual(len(project.containers()), 0)
  421. project.up(['web'])
  422. self.assertEqual(len(project.containers()), 3)
  423. self.assertEqual(len(project.get_service('web').containers()), 1)
  424. self.assertEqual(len(project.get_service('db').containers()), 1)
  425. self.assertEqual(len(project.get_service('data').containers()), 1)
  426. self.assertEqual(len(project.get_service('console').containers()), 0)
  427. def test_project_up_with_no_deps(self):
  428. project = Project.from_config(
  429. name='composetest',
  430. config_data=load_config({
  431. 'console': {
  432. 'image': 'busybox:latest',
  433. 'command': ["top"],
  434. },
  435. 'data': {
  436. 'image': 'busybox:latest',
  437. 'command': ["top"]
  438. },
  439. 'db': {
  440. 'image': 'busybox:latest',
  441. 'command': ["top"],
  442. 'volumes_from': ['data'],
  443. },
  444. 'web': {
  445. 'image': 'busybox:latest',
  446. 'command': ["top"],
  447. 'links': ['db'],
  448. },
  449. }),
  450. client=self.client,
  451. )
  452. project.start()
  453. self.assertEqual(len(project.containers()), 0)
  454. project.up(['db'], start_deps=False)
  455. self.assertEqual(len(project.containers(stopped=True)), 2)
  456. self.assertEqual(len(project.get_service('web').containers()), 0)
  457. self.assertEqual(len(project.get_service('db').containers()), 1)
  458. self.assertEqual(len(project.get_service('data').containers()), 0)
  459. self.assertEqual(len(project.get_service('data').containers(stopped=True)), 1)
  460. self.assertEqual(len(project.get_service('console').containers()), 0)
  461. def test_unscale_after_restart(self):
  462. web = self.create_service('web')
  463. project = Project('composetest', [web], self.client)
  464. project.start()
  465. service = project.get_service('web')
  466. service.scale(1)
  467. self.assertEqual(len(service.containers()), 1)
  468. service.scale(3)
  469. self.assertEqual(len(service.containers()), 3)
  470. project.up()
  471. service = project.get_service('web')
  472. self.assertEqual(len(service.containers()), 3)
  473. service.scale(1)
  474. self.assertEqual(len(service.containers()), 1)
  475. project.up()
  476. service = project.get_service('web')
  477. self.assertEqual(len(service.containers()), 1)
  478. # does scale=0 ,makes any sense? after recreating at least 1 container is running
  479. service.scale(0)
  480. project.up()
  481. service = project.get_service('web')
  482. self.assertEqual(len(service.containers()), 1)
  483. @v2_only()
  484. def test_project_up_networks(self):
  485. config_data = build_config(
  486. version=V2_0,
  487. services=[{
  488. 'name': 'web',
  489. 'image': 'busybox:latest',
  490. 'command': 'top',
  491. 'networks': {
  492. 'foo': None,
  493. 'bar': None,
  494. 'baz': {'aliases': ['extra']},
  495. },
  496. }],
  497. networks={
  498. 'foo': {'driver': 'bridge'},
  499. 'bar': {'driver': None},
  500. 'baz': {},
  501. },
  502. )
  503. project = Project.from_config(
  504. client=self.client,
  505. name='composetest',
  506. config_data=config_data,
  507. )
  508. project.up()
  509. containers = project.containers()
  510. assert len(containers) == 1
  511. container, = containers
  512. for net_name in ['foo', 'bar', 'baz']:
  513. full_net_name = 'composetest_{}'.format(net_name)
  514. network_data = self.client.inspect_network(full_net_name)
  515. assert network_data['Name'] == full_net_name
  516. aliases_key = 'NetworkSettings.Networks.{net}.Aliases'
  517. assert 'web' in container.get(aliases_key.format(net='composetest_foo'))
  518. assert 'web' in container.get(aliases_key.format(net='composetest_baz'))
  519. assert 'extra' in container.get(aliases_key.format(net='composetest_baz'))
  520. foo_data = self.client.inspect_network('composetest_foo')
  521. assert foo_data['Driver'] == 'bridge'
  522. @v2_only()
  523. def test_up_with_ipam_config(self):
  524. config_data = build_config(
  525. version=V2_0,
  526. services=[{
  527. 'name': 'web',
  528. 'image': 'busybox:latest',
  529. 'networks': {'front': None},
  530. }],
  531. networks={
  532. 'front': {
  533. 'driver': 'bridge',
  534. 'driver_opts': {
  535. "com.docker.network.bridge.enable_icc": "false",
  536. },
  537. 'ipam': {
  538. 'driver': 'default',
  539. 'config': [{
  540. "subnet": "172.28.0.0/16",
  541. "ip_range": "172.28.5.0/24",
  542. "gateway": "172.28.5.254",
  543. "aux_addresses": {
  544. "a": "172.28.1.5",
  545. "b": "172.28.1.6",
  546. "c": "172.28.1.7",
  547. },
  548. }],
  549. },
  550. },
  551. },
  552. )
  553. project = Project.from_config(
  554. client=self.client,
  555. name='composetest',
  556. config_data=config_data,
  557. )
  558. project.up()
  559. network = self.client.networks(names=['composetest_front'])[0]
  560. assert network['Options'] == {
  561. "com.docker.network.bridge.enable_icc": "false"
  562. }
  563. assert network['IPAM'] == {
  564. 'Driver': 'default',
  565. 'Options': None,
  566. 'Config': [{
  567. 'Subnet': "172.28.0.0/16",
  568. 'IPRange': "172.28.5.0/24",
  569. 'Gateway': "172.28.5.254",
  570. 'AuxiliaryAddresses': {
  571. 'a': '172.28.1.5',
  572. 'b': '172.28.1.6',
  573. 'c': '172.28.1.7',
  574. },
  575. }],
  576. }
  577. @v2_only()
  578. def test_up_with_network_static_addresses(self):
  579. config_data = build_config(
  580. version=V2_0,
  581. services=[{
  582. 'name': 'web',
  583. 'image': 'busybox:latest',
  584. 'command': 'top',
  585. 'networks': {
  586. 'static_test': {
  587. 'ipv4_address': '172.16.100.100',
  588. 'ipv6_address': 'fe80::1001:102'
  589. }
  590. },
  591. }],
  592. networks={
  593. 'static_test': {
  594. 'driver': 'bridge',
  595. 'driver_opts': {
  596. "com.docker.network.enable_ipv6": "true",
  597. },
  598. 'ipam': {
  599. 'driver': 'default',
  600. 'config': [
  601. {"subnet": "172.16.100.0/24",
  602. "gateway": "172.16.100.1"},
  603. {"subnet": "fe80::/64",
  604. "gateway": "fe80::1001:1"}
  605. ]
  606. }
  607. }
  608. }
  609. )
  610. project = Project.from_config(
  611. client=self.client,
  612. name='composetest',
  613. config_data=config_data,
  614. )
  615. project.up(detached=True)
  616. network = self.client.networks(names=['static_test'])[0]
  617. service_container = project.get_service('web').containers()[0]
  618. assert network['Options'] == {
  619. "com.docker.network.enable_ipv6": "true"
  620. }
  621. IPAMConfig = (service_container.inspect().get('NetworkSettings', {}).
  622. get('Networks', {}).get('composetest_static_test', {}).
  623. get('IPAMConfig', {}))
  624. assert IPAMConfig.get('IPv4Address') == '172.16.100.100'
  625. assert IPAMConfig.get('IPv6Address') == 'fe80::1001:102'
  626. @v2_1_only()
  627. def test_up_with_enable_ipv6(self):
  628. self.require_api_version('1.23')
  629. config_data = build_config(
  630. version=V2_0,
  631. services=[{
  632. 'name': 'web',
  633. 'image': 'busybox:latest',
  634. 'command': 'top',
  635. 'networks': {
  636. 'static_test': {
  637. 'ipv6_address': 'fe80::1001:102'
  638. }
  639. },
  640. }],
  641. networks={
  642. 'static_test': {
  643. 'driver': 'bridge',
  644. 'enable_ipv6': True,
  645. 'ipam': {
  646. 'driver': 'default',
  647. 'config': [
  648. {"subnet": "fe80::/64",
  649. "gateway": "fe80::1001:1"}
  650. ]
  651. }
  652. }
  653. }
  654. )
  655. project = Project.from_config(
  656. client=self.client,
  657. name='composetest',
  658. config_data=config_data,
  659. )
  660. project.up(detached=True)
  661. network = self.client.networks(names=['static_test'])[0]
  662. service_container = project.get_service('web').containers()[0]
  663. assert network['EnableIPv6'] is True
  664. ipam_config = (service_container.inspect().get('NetworkSettings', {}).
  665. get('Networks', {}).get('composetest_static_test', {}).
  666. get('IPAMConfig', {}))
  667. assert ipam_config.get('IPv6Address') == 'fe80::1001:102'
  668. @v2_only()
  669. def test_up_with_network_static_addresses_missing_subnet(self):
  670. config_data = build_config(
  671. version=V2_0,
  672. services=[{
  673. 'name': 'web',
  674. 'image': 'busybox:latest',
  675. 'networks': {
  676. 'static_test': {
  677. 'ipv4_address': '172.16.100.100',
  678. 'ipv6_address': 'fe80::1001:101'
  679. }
  680. },
  681. }],
  682. networks={
  683. 'static_test': {
  684. 'driver': 'bridge',
  685. 'driver_opts': {
  686. "com.docker.network.enable_ipv6": "true",
  687. },
  688. 'ipam': {
  689. 'driver': 'default',
  690. },
  691. },
  692. },
  693. )
  694. project = Project.from_config(
  695. client=self.client,
  696. name='composetest',
  697. config_data=config_data,
  698. )
  699. with self.assertRaises(ProjectError):
  700. project.up()
  701. @v2_1_only()
  702. def test_up_with_network_link_local_ips(self):
  703. config_data = build_config(
  704. version=V2_1,
  705. services=[{
  706. 'name': 'web',
  707. 'image': 'busybox:latest',
  708. 'networks': {
  709. 'linklocaltest': {
  710. 'link_local_ips': ['169.254.8.8']
  711. }
  712. }
  713. }],
  714. networks={
  715. 'linklocaltest': {'driver': 'bridge'}
  716. }
  717. )
  718. project = Project.from_config(
  719. client=self.client,
  720. name='composetest',
  721. config_data=config_data
  722. )
  723. project.up(detached=True)
  724. service_container = project.get_service('web').containers(stopped=True)[0]
  725. ipam_config = service_container.inspect().get(
  726. 'NetworkSettings', {}
  727. ).get(
  728. 'Networks', {}
  729. ).get(
  730. 'composetest_linklocaltest', {}
  731. ).get('IPAMConfig', {})
  732. assert 'LinkLocalIPs' in ipam_config
  733. assert ipam_config['LinkLocalIPs'] == ['169.254.8.8']
  734. @v2_1_only()
  735. def test_up_with_isolation(self):
  736. self.require_api_version('1.24')
  737. config_data = build_config(
  738. version=V2_1,
  739. services=[{
  740. 'name': 'web',
  741. 'image': 'busybox:latest',
  742. 'isolation': 'default'
  743. }],
  744. )
  745. project = Project.from_config(
  746. client=self.client,
  747. name='composetest',
  748. config_data=config_data
  749. )
  750. project.up(detached=True)
  751. service_container = project.get_service('web').containers(stopped=True)[0]
  752. assert service_container.inspect()['HostConfig']['Isolation'] == 'default'
  753. @v2_1_only()
  754. def test_up_with_invalid_isolation(self):
  755. self.require_api_version('1.24')
  756. config_data = build_config(
  757. version=V2_1,
  758. services=[{
  759. 'name': 'web',
  760. 'image': 'busybox:latest',
  761. 'isolation': 'foobar'
  762. }],
  763. )
  764. project = Project.from_config(
  765. client=self.client,
  766. name='composetest',
  767. config_data=config_data
  768. )
  769. with self.assertRaises(ProjectError):
  770. project.up()
  771. @v2_only()
  772. def test_project_up_with_network_internal(self):
  773. self.require_api_version('1.23')
  774. config_data = build_config(
  775. version=V2_0,
  776. services=[{
  777. 'name': 'web',
  778. 'image': 'busybox:latest',
  779. 'networks': {'internal': None},
  780. }],
  781. networks={
  782. 'internal': {'driver': 'bridge', 'internal': True},
  783. },
  784. )
  785. project = Project.from_config(
  786. client=self.client,
  787. name='composetest',
  788. config_data=config_data,
  789. )
  790. project.up()
  791. network = self.client.networks(names=['composetest_internal'])[0]
  792. assert network['Internal'] is True
  793. @v2_1_only()
  794. def test_project_up_with_network_label(self):
  795. self.require_api_version('1.23')
  796. network_name = 'network_with_label'
  797. config_data = build_config(
  798. version=V2_0,
  799. services=[{
  800. 'name': 'web',
  801. 'image': 'busybox:latest',
  802. 'networks': {network_name: None}
  803. }],
  804. networks={
  805. network_name: {'labels': {'label_key': 'label_val'}}
  806. }
  807. )
  808. project = Project.from_config(
  809. client=self.client,
  810. name='composetest',
  811. config_data=config_data
  812. )
  813. project.up()
  814. networks = [
  815. n for n in self.client.networks()
  816. if n['Name'].startswith('composetest_')
  817. ]
  818. assert [n['Name'] for n in networks] == ['composetest_{}'.format(network_name)]
  819. assert 'label_key' in networks[0]['Labels']
  820. assert networks[0]['Labels']['label_key'] == 'label_val'
  821. @v2_only()
  822. def test_project_up_volumes(self):
  823. vol_name = '{0:x}'.format(random.getrandbits(32))
  824. full_vol_name = 'composetest_{0}'.format(vol_name)
  825. config_data = build_config(
  826. version=V2_0,
  827. services=[{
  828. 'name': 'web',
  829. 'image': 'busybox:latest',
  830. 'command': 'top'
  831. }],
  832. volumes={vol_name: {'driver': 'local'}},
  833. )
  834. project = Project.from_config(
  835. name='composetest',
  836. config_data=config_data, client=self.client
  837. )
  838. project.up()
  839. self.assertEqual(len(project.containers()), 1)
  840. volume_data = self.client.inspect_volume(full_vol_name)
  841. self.assertEqual(volume_data['Name'], full_vol_name)
  842. self.assertEqual(volume_data['Driver'], 'local')
  843. @v2_1_only()
  844. def test_project_up_with_volume_labels(self):
  845. self.require_api_version('1.23')
  846. volume_name = 'volume_with_label'
  847. config_data = build_config(
  848. version=V2_0,
  849. services=[{
  850. 'name': 'web',
  851. 'image': 'busybox:latest',
  852. 'volumes': [VolumeSpec.parse('{}:/data'.format(volume_name))]
  853. }],
  854. volumes={
  855. volume_name: {
  856. 'labels': {
  857. 'label_key': 'label_val'
  858. }
  859. }
  860. },
  861. )
  862. project = Project.from_config(
  863. client=self.client,
  864. name='composetest',
  865. config_data=config_data,
  866. )
  867. project.up()
  868. volumes = [
  869. v for v in self.client.volumes().get('Volumes', [])
  870. if v['Name'].startswith('composetest_')
  871. ]
  872. assert [v['Name'] for v in volumes] == ['composetest_{}'.format(volume_name)]
  873. assert 'label_key' in volumes[0]['Labels']
  874. assert volumes[0]['Labels']['label_key'] == 'label_val'
  875. @v2_only()
  876. def test_project_up_logging_with_multiple_files(self):
  877. base_file = config.ConfigFile(
  878. 'base.yml',
  879. {
  880. 'version': V2_0,
  881. 'services': {
  882. 'simple': {'image': 'busybox:latest', 'command': 'top'},
  883. 'another': {
  884. 'image': 'busybox:latest',
  885. 'command': 'top',
  886. 'logging': {
  887. 'driver': "json-file",
  888. 'options': {
  889. 'max-size': "10m"
  890. }
  891. }
  892. }
  893. }
  894. })
  895. override_file = config.ConfigFile(
  896. 'override.yml',
  897. {
  898. 'version': V2_0,
  899. 'services': {
  900. 'another': {
  901. 'logging': {
  902. 'driver': "none"
  903. }
  904. }
  905. }
  906. })
  907. details = config.ConfigDetails('.', [base_file, override_file])
  908. tmpdir = py.test.ensuretemp('logging_test')
  909. self.addCleanup(tmpdir.remove)
  910. with tmpdir.as_cwd():
  911. config_data = config.load(details)
  912. project = Project.from_config(
  913. name='composetest', config_data=config_data, client=self.client
  914. )
  915. project.up()
  916. containers = project.containers()
  917. self.assertEqual(len(containers), 2)
  918. another = project.get_service('another').containers()[0]
  919. log_config = another.get('HostConfig.LogConfig')
  920. self.assertTrue(log_config)
  921. self.assertEqual(log_config.get('Type'), 'none')
  922. @v2_only()
  923. def test_project_up_port_mappings_with_multiple_files(self):
  924. base_file = config.ConfigFile(
  925. 'base.yml',
  926. {
  927. 'version': V2_0,
  928. 'services': {
  929. 'simple': {
  930. 'image': 'busybox:latest',
  931. 'command': 'top',
  932. 'ports': ['1234:1234']
  933. },
  934. },
  935. })
  936. override_file = config.ConfigFile(
  937. 'override.yml',
  938. {
  939. 'version': V2_0,
  940. 'services': {
  941. 'simple': {
  942. 'ports': ['1234:1234']
  943. }
  944. }
  945. })
  946. details = config.ConfigDetails('.', [base_file, override_file])
  947. config_data = config.load(details)
  948. project = Project.from_config(
  949. name='composetest', config_data=config_data, client=self.client
  950. )
  951. project.up()
  952. containers = project.containers()
  953. self.assertEqual(len(containers), 1)
  954. @v2_only()
  955. def test_initialize_volumes(self):
  956. vol_name = '{0:x}'.format(random.getrandbits(32))
  957. full_vol_name = 'composetest_{0}'.format(vol_name)
  958. config_data = build_config(
  959. version=V2_0,
  960. services=[{
  961. 'name': 'web',
  962. 'image': 'busybox:latest',
  963. 'command': 'top'
  964. }],
  965. volumes={vol_name: {}},
  966. )
  967. project = Project.from_config(
  968. name='composetest',
  969. config_data=config_data, client=self.client
  970. )
  971. project.volumes.initialize()
  972. volume_data = self.client.inspect_volume(full_vol_name)
  973. assert volume_data['Name'] == full_vol_name
  974. assert volume_data['Driver'] == 'local'
  975. @v2_only()
  976. def test_project_up_implicit_volume_driver(self):
  977. vol_name = '{0:x}'.format(random.getrandbits(32))
  978. full_vol_name = 'composetest_{0}'.format(vol_name)
  979. config_data = build_config(
  980. version=V2_0,
  981. services=[{
  982. 'name': 'web',
  983. 'image': 'busybox:latest',
  984. 'command': 'top'
  985. }],
  986. volumes={vol_name: {}},
  987. )
  988. project = Project.from_config(
  989. name='composetest',
  990. config_data=config_data, client=self.client
  991. )
  992. project.up()
  993. volume_data = self.client.inspect_volume(full_vol_name)
  994. self.assertEqual(volume_data['Name'], full_vol_name)
  995. self.assertEqual(volume_data['Driver'], 'local')
  996. @v3_only()
  997. def test_project_up_with_secrets(self):
  998. create_host_file(self.client, os.path.abspath('tests/fixtures/secrets/default'))
  999. config_data = build_config(
  1000. version=V3_1,
  1001. services=[{
  1002. 'name': 'web',
  1003. 'image': 'busybox:latest',
  1004. 'command': 'cat /run/secrets/special',
  1005. 'secrets': [
  1006. types.ServiceSecret.parse({'source': 'super', 'target': 'special'}),
  1007. ],
  1008. }],
  1009. secrets={
  1010. 'super': {
  1011. 'file': os.path.abspath('tests/fixtures/secrets/default'),
  1012. },
  1013. },
  1014. )
  1015. project = Project.from_config(
  1016. client=self.client,
  1017. name='composetest',
  1018. config_data=config_data,
  1019. )
  1020. project.up()
  1021. project.stop()
  1022. containers = project.containers(stopped=True)
  1023. assert len(containers) == 1
  1024. container, = containers
  1025. output = container.logs()
  1026. assert output == b"This is the secret\n"
  1027. @v2_only()
  1028. def test_initialize_volumes_invalid_volume_driver(self):
  1029. vol_name = '{0:x}'.format(random.getrandbits(32))
  1030. config_data = build_config(
  1031. version=V2_0,
  1032. services=[{
  1033. 'name': 'web',
  1034. 'image': 'busybox:latest',
  1035. 'command': 'top'
  1036. }],
  1037. volumes={vol_name: {'driver': 'foobar'}},
  1038. )
  1039. project = Project.from_config(
  1040. name='composetest',
  1041. config_data=config_data, client=self.client
  1042. )
  1043. with self.assertRaises(config.ConfigurationError):
  1044. project.volumes.initialize()
  1045. @v2_only()
  1046. def test_initialize_volumes_updated_driver(self):
  1047. vol_name = '{0:x}'.format(random.getrandbits(32))
  1048. full_vol_name = 'composetest_{0}'.format(vol_name)
  1049. config_data = build_config(
  1050. version=V2_0,
  1051. services=[{
  1052. 'name': 'web',
  1053. 'image': 'busybox:latest',
  1054. 'command': 'top'
  1055. }],
  1056. volumes={vol_name: {'driver': 'local'}},
  1057. )
  1058. project = Project.from_config(
  1059. name='composetest',
  1060. config_data=config_data, client=self.client
  1061. )
  1062. project.volumes.initialize()
  1063. volume_data = self.client.inspect_volume(full_vol_name)
  1064. self.assertEqual(volume_data['Name'], full_vol_name)
  1065. self.assertEqual(volume_data['Driver'], 'local')
  1066. config_data = config_data._replace(
  1067. volumes={vol_name: {'driver': 'smb'}}
  1068. )
  1069. project = Project.from_config(
  1070. name='composetest',
  1071. config_data=config_data,
  1072. client=self.client
  1073. )
  1074. with self.assertRaises(config.ConfigurationError) as e:
  1075. project.volumes.initialize()
  1076. assert 'Configuration for volume {0} specifies driver smb'.format(
  1077. vol_name
  1078. ) in str(e.exception)
  1079. @v2_only()
  1080. def test_initialize_volumes_updated_blank_driver(self):
  1081. vol_name = '{0:x}'.format(random.getrandbits(32))
  1082. full_vol_name = 'composetest_{0}'.format(vol_name)
  1083. config_data = build_config(
  1084. version=V2_0,
  1085. services=[{
  1086. 'name': 'web',
  1087. 'image': 'busybox:latest',
  1088. 'command': 'top'
  1089. }],
  1090. volumes={vol_name: {'driver': 'local'}},
  1091. )
  1092. project = Project.from_config(
  1093. name='composetest',
  1094. config_data=config_data, client=self.client
  1095. )
  1096. project.volumes.initialize()
  1097. volume_data = self.client.inspect_volume(full_vol_name)
  1098. self.assertEqual(volume_data['Name'], full_vol_name)
  1099. self.assertEqual(volume_data['Driver'], 'local')
  1100. config_data = config_data._replace(
  1101. volumes={vol_name: {}}
  1102. )
  1103. project = Project.from_config(
  1104. name='composetest',
  1105. config_data=config_data,
  1106. client=self.client
  1107. )
  1108. project.volumes.initialize()
  1109. volume_data = self.client.inspect_volume(full_vol_name)
  1110. self.assertEqual(volume_data['Name'], full_vol_name)
  1111. self.assertEqual(volume_data['Driver'], 'local')
  1112. @v2_only()
  1113. def test_initialize_volumes_external_volumes(self):
  1114. # Use composetest_ prefix so it gets garbage-collected in tearDown()
  1115. vol_name = 'composetest_{0:x}'.format(random.getrandbits(32))
  1116. full_vol_name = 'composetest_{0}'.format(vol_name)
  1117. self.client.create_volume(vol_name)
  1118. config_data = build_config(
  1119. version=V2_0,
  1120. services=[{
  1121. 'name': 'web',
  1122. 'image': 'busybox:latest',
  1123. 'command': 'top'
  1124. }],
  1125. volumes={
  1126. vol_name: {'external': True, 'external_name': vol_name}
  1127. },
  1128. )
  1129. project = Project.from_config(
  1130. name='composetest',
  1131. config_data=config_data, client=self.client
  1132. )
  1133. project.volumes.initialize()
  1134. with self.assertRaises(NotFound):
  1135. self.client.inspect_volume(full_vol_name)
  1136. @v2_only()
  1137. def test_initialize_volumes_inexistent_external_volume(self):
  1138. vol_name = '{0:x}'.format(random.getrandbits(32))
  1139. config_data = build_config(
  1140. version=V2_0,
  1141. services=[{
  1142. 'name': 'web',
  1143. 'image': 'busybox:latest',
  1144. 'command': 'top'
  1145. }],
  1146. volumes={
  1147. vol_name: {'external': True, 'external_name': vol_name}
  1148. },
  1149. )
  1150. project = Project.from_config(
  1151. name='composetest',
  1152. config_data=config_data, client=self.client
  1153. )
  1154. with self.assertRaises(config.ConfigurationError) as e:
  1155. project.volumes.initialize()
  1156. assert 'Volume {0} declared as external'.format(
  1157. vol_name
  1158. ) in str(e.exception)
  1159. @v2_only()
  1160. def test_project_up_named_volumes_in_binds(self):
  1161. vol_name = '{0:x}'.format(random.getrandbits(32))
  1162. full_vol_name = 'composetest_{0}'.format(vol_name)
  1163. base_file = config.ConfigFile(
  1164. 'base.yml',
  1165. {
  1166. 'version': V2_0,
  1167. 'services': {
  1168. 'simple': {
  1169. 'image': 'busybox:latest',
  1170. 'command': 'top',
  1171. 'volumes': ['{0}:/data'.format(vol_name)]
  1172. },
  1173. },
  1174. 'volumes': {
  1175. vol_name: {'driver': 'local'}
  1176. }
  1177. })
  1178. config_details = config.ConfigDetails('.', [base_file])
  1179. config_data = config.load(config_details)
  1180. project = Project.from_config(
  1181. name='composetest', config_data=config_data, client=self.client
  1182. )
  1183. service = project.services[0]
  1184. self.assertEqual(service.name, 'simple')
  1185. volumes = service.options.get('volumes')
  1186. self.assertEqual(len(volumes), 1)
  1187. self.assertEqual(volumes[0].external, full_vol_name)
  1188. project.up()
  1189. engine_volumes = self.client.volumes()['Volumes']
  1190. container = service.get_container()
  1191. assert [mount['Name'] for mount in container.get('Mounts')] == [full_vol_name]
  1192. assert next((v for v in engine_volumes if v['Name'] == vol_name), None) is None
  1193. def test_project_up_orphans(self):
  1194. config_dict = {
  1195. 'service1': {
  1196. 'image': 'busybox:latest',
  1197. 'command': 'top',
  1198. }
  1199. }
  1200. config_data = load_config(config_dict)
  1201. project = Project.from_config(
  1202. name='composetest', config_data=config_data, client=self.client
  1203. )
  1204. project.up()
  1205. config_dict['service2'] = config_dict['service1']
  1206. del config_dict['service1']
  1207. config_data = load_config(config_dict)
  1208. project = Project.from_config(
  1209. name='composetest', config_data=config_data, client=self.client
  1210. )
  1211. with mock.patch('compose.project.log') as mock_log:
  1212. project.up()
  1213. mock_log.warning.assert_called_once_with(mock.ANY)
  1214. assert len([
  1215. ctnr for ctnr in project._labeled_containers()
  1216. if ctnr.labels.get(LABEL_SERVICE) == 'service1'
  1217. ]) == 1
  1218. project.up(remove_orphans=True)
  1219. assert len([
  1220. ctnr for ctnr in project._labeled_containers()
  1221. if ctnr.labels.get(LABEL_SERVICE) == 'service1'
  1222. ]) == 0
  1223. @v2_1_only()
  1224. def test_project_up_healthy_dependency(self):
  1225. config_dict = {
  1226. 'version': '2.1',
  1227. 'services': {
  1228. 'svc1': {
  1229. 'image': 'busybox:latest',
  1230. 'command': 'top',
  1231. 'healthcheck': {
  1232. 'test': 'exit 0',
  1233. 'retries': 1,
  1234. 'timeout': '10s',
  1235. 'interval': '0.1s'
  1236. },
  1237. },
  1238. 'svc2': {
  1239. 'image': 'busybox:latest',
  1240. 'command': 'top',
  1241. 'depends_on': {
  1242. 'svc1': {'condition': 'service_healthy'},
  1243. }
  1244. }
  1245. }
  1246. }
  1247. config_data = load_config(config_dict)
  1248. project = Project.from_config(
  1249. name='composetest', config_data=config_data, client=self.client
  1250. )
  1251. project.up()
  1252. containers = project.containers()
  1253. assert len(containers) == 2
  1254. svc1 = project.get_service('svc1')
  1255. svc2 = project.get_service('svc2')
  1256. assert 'svc1' in svc2.get_dependency_names()
  1257. assert svc1.is_healthy()
  1258. @v2_1_only()
  1259. def test_project_up_unhealthy_dependency(self):
  1260. config_dict = {
  1261. 'version': '2.1',
  1262. 'services': {
  1263. 'svc1': {
  1264. 'image': 'busybox:latest',
  1265. 'command': 'top',
  1266. 'healthcheck': {
  1267. 'test': 'exit 1',
  1268. 'retries': 1,
  1269. 'timeout': '10s',
  1270. 'interval': '0.1s'
  1271. },
  1272. },
  1273. 'svc2': {
  1274. 'image': 'busybox:latest',
  1275. 'command': 'top',
  1276. 'depends_on': {
  1277. 'svc1': {'condition': 'service_healthy'},
  1278. }
  1279. }
  1280. }
  1281. }
  1282. config_data = load_config(config_dict)
  1283. project = Project.from_config(
  1284. name='composetest', config_data=config_data, client=self.client
  1285. )
  1286. with pytest.raises(ProjectError):
  1287. project.up()
  1288. containers = project.containers()
  1289. assert len(containers) == 1
  1290. svc1 = project.get_service('svc1')
  1291. svc2 = project.get_service('svc2')
  1292. assert 'svc1' in svc2.get_dependency_names()
  1293. with pytest.raises(HealthCheckFailed):
  1294. svc1.is_healthy()
  1295. @v2_1_only()
  1296. def test_project_up_no_healthcheck_dependency(self):
  1297. config_dict = {
  1298. 'version': '2.1',
  1299. 'services': {
  1300. 'svc1': {
  1301. 'image': 'busybox:latest',
  1302. 'command': 'top',
  1303. 'healthcheck': {
  1304. 'disable': True
  1305. },
  1306. },
  1307. 'svc2': {
  1308. 'image': 'busybox:latest',
  1309. 'command': 'top',
  1310. 'depends_on': {
  1311. 'svc1': {'condition': 'service_healthy'},
  1312. }
  1313. }
  1314. }
  1315. }
  1316. config_data = load_config(config_dict)
  1317. project = Project.from_config(
  1318. name='composetest', config_data=config_data, client=self.client
  1319. )
  1320. with pytest.raises(ProjectError):
  1321. project.up()
  1322. containers = project.containers()
  1323. assert len(containers) == 1
  1324. svc1 = project.get_service('svc1')
  1325. svc2 = project.get_service('svc2')
  1326. assert 'svc1' in svc2.get_dependency_names()
  1327. with pytest.raises(NoHealthCheckConfigured):
  1328. svc1.is_healthy()
  1329. def create_host_file(client, filename):
  1330. dirname = os.path.dirname(filename)
  1331. with open(filename, 'r') as fh:
  1332. content = fh.read()
  1333. container = client.create_container(
  1334. 'busybox:latest',
  1335. ['sh', '-c', 'echo -n "{}" > {}'.format(content, filename)],
  1336. volumes={dirname: {}},
  1337. host_config=client.create_host_config(
  1338. binds={dirname: {'bind': dirname, 'ro': False}},
  1339. network_mode='none',
  1340. ),
  1341. )
  1342. try:
  1343. client.start(container)
  1344. exitcode = client.wait(container)
  1345. if exitcode != 0:
  1346. output = client.logs(container)
  1347. raise Exception(
  1348. "Container exited with code {}:\n{}".format(exitcode, output))
  1349. finally:
  1350. client.remove_container(container, force=True)