cli_test.py 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712
  1. from __future__ import absolute_import
  2. import os
  3. import shlex
  4. import subprocess
  5. from collections import namedtuple
  6. from operator import attrgetter
  7. from .. import mock
  8. from compose.cli.command import get_project
  9. from compose.cli.docker_client import docker_client
  10. from tests.integration.testcases import DockerClientTestCase
  11. ProcessResult = namedtuple('ProcessResult', 'stdout stderr')
  12. BUILD_CACHE_TEXT = 'Using cache'
  13. BUILD_PULL_TEXT = 'Status: Image is up to date for busybox:latest'
  14. class CLITestCase(DockerClientTestCase):
  15. def setUp(self):
  16. super(CLITestCase, self).setUp()
  17. self.base_dir = 'tests/fixtures/simple-composefile'
  18. def tearDown(self):
  19. self.project.kill()
  20. self.project.remove_stopped()
  21. for container in self.project.containers(stopped=True, one_off=True):
  22. container.remove(force=True)
  23. super(CLITestCase, self).tearDown()
  24. @property
  25. def project(self):
  26. # Hack: allow project to be overridden
  27. if not hasattr(self, '_project'):
  28. self._project = get_project(self.base_dir)
  29. return self._project
  30. def dispatch(self, options, project_options=None, returncode=0):
  31. project_options = project_options or []
  32. proc = subprocess.Popen(
  33. ['docker-compose'] + project_options + options,
  34. stdout=subprocess.PIPE,
  35. stderr=subprocess.PIPE,
  36. cwd=self.base_dir)
  37. print("Running process: %s" % proc.pid)
  38. stdout, stderr = proc.communicate()
  39. if proc.returncode != returncode:
  40. print(stderr)
  41. assert proc.returncode == returncode
  42. return ProcessResult(stdout.decode('utf-8'), stderr.decode('utf-8'))
  43. def test_help(self):
  44. old_base_dir = self.base_dir
  45. self.base_dir = 'tests/fixtures/no-composefile'
  46. result = self.dispatch(['help', 'up'], returncode=1)
  47. assert 'Usage: up [options] [SERVICE...]' in result.stderr
  48. # self.project.kill() fails during teardown
  49. # unless there is a composefile.
  50. self.base_dir = old_base_dir
  51. def test_ps(self):
  52. self.project.get_service('simple').create_container()
  53. result = self.dispatch(['ps'])
  54. assert 'simplecomposefile_simple_1' in result.stdout
  55. def test_ps_default_composefile(self):
  56. self.base_dir = 'tests/fixtures/multiple-composefiles'
  57. self.dispatch(['up', '-d'])
  58. result = self.dispatch(['ps'])
  59. self.assertIn('multiplecomposefiles_simple_1', result.stdout)
  60. self.assertIn('multiplecomposefiles_another_1', result.stdout)
  61. self.assertNotIn('multiplecomposefiles_yetanother_1', result.stdout)
  62. def test_ps_alternate_composefile(self):
  63. config_path = os.path.abspath(
  64. 'tests/fixtures/multiple-composefiles/compose2.yml')
  65. self._project = get_project(self.base_dir, [config_path])
  66. self.base_dir = 'tests/fixtures/multiple-composefiles'
  67. self.dispatch(['-f', 'compose2.yml', 'up', '-d'])
  68. result = self.dispatch(['-f', 'compose2.yml', 'ps'])
  69. self.assertNotIn('multiplecomposefiles_simple_1', result.stdout)
  70. self.assertNotIn('multiplecomposefiles_another_1', result.stdout)
  71. self.assertIn('multiplecomposefiles_yetanother_1', result.stdout)
  72. def test_pull(self):
  73. result = self.dispatch(['pull'])
  74. assert sorted(result.stderr.split('\n'))[1:] == [
  75. 'Pulling another (busybox:latest)...',
  76. 'Pulling simple (busybox:latest)...',
  77. ]
  78. def test_pull_with_digest(self):
  79. result = self.dispatch(['-f', 'digest.yml', 'pull'])
  80. assert 'Pulling simple (busybox:latest)...' in result.stderr
  81. assert ('Pulling digest (busybox@'
  82. 'sha256:38a203e1986cf79639cfb9b2e1d6e773de84002feea2d4eb006b520'
  83. '04ee8502d)...') in result.stderr
  84. def test_pull_with_ignore_pull_failures(self):
  85. result = self.dispatch([
  86. '-f', 'ignore-pull-failures.yml',
  87. 'pull', '--ignore-pull-failures'])
  88. assert 'Pulling simple (busybox:latest)...' in result.stderr
  89. assert 'Pulling another (nonexisting-image:latest)...' in result.stderr
  90. assert 'Error: image library/nonexisting-image:latest not found' in result.stderr
  91. def test_build_plain(self):
  92. self.base_dir = 'tests/fixtures/simple-dockerfile'
  93. self.dispatch(['build', 'simple'])
  94. result = self.dispatch(['build', 'simple'])
  95. assert BUILD_CACHE_TEXT in result.stdout
  96. assert BUILD_PULL_TEXT not in result.stdout
  97. def test_build_no_cache(self):
  98. self.base_dir = 'tests/fixtures/simple-dockerfile'
  99. self.dispatch(['build', 'simple'])
  100. result = self.dispatch(['build', '--no-cache', 'simple'])
  101. assert BUILD_CACHE_TEXT not in result.stdout
  102. assert BUILD_PULL_TEXT not in result.stdout
  103. def test_build_pull(self):
  104. self.base_dir = 'tests/fixtures/simple-dockerfile'
  105. self.dispatch(['build', 'simple'], None)
  106. result = self.dispatch(['build', '--pull', 'simple'])
  107. assert BUILD_CACHE_TEXT in result.stdout
  108. assert BUILD_PULL_TEXT in result.stdout
  109. def test_build_no_cache_pull(self):
  110. self.base_dir = 'tests/fixtures/simple-dockerfile'
  111. self.dispatch(['build', 'simple'])
  112. result = self.dispatch(['build', '--no-cache', '--pull', 'simple'])
  113. assert BUILD_CACHE_TEXT not in result.stdout
  114. assert BUILD_PULL_TEXT in result.stdout
  115. def test_up_detached(self):
  116. self.dispatch(['up', '-d'])
  117. service = self.project.get_service('simple')
  118. another = self.project.get_service('another')
  119. self.assertEqual(len(service.containers()), 1)
  120. self.assertEqual(len(another.containers()), 1)
  121. # Ensure containers don't have stdin and stdout connected in -d mode
  122. container, = service.containers()
  123. self.assertFalse(container.get('Config.AttachStderr'))
  124. self.assertFalse(container.get('Config.AttachStdout'))
  125. self.assertFalse(container.get('Config.AttachStdin'))
  126. def test_up_attached(self):
  127. self.base_dir = 'tests/fixtures/echo-services'
  128. result = self.dispatch(['up', '--no-color'])
  129. assert 'simple_1 | simple' in result.stdout
  130. assert 'another_1 | another' in result.stdout
  131. def test_up_without_networking(self):
  132. self.require_api_version('1.21')
  133. self.base_dir = 'tests/fixtures/links-composefile'
  134. self.dispatch(['up', '-d'], None)
  135. client = docker_client(version='1.21')
  136. networks = client.networks(names=[self.project.name])
  137. self.assertEqual(len(networks), 0)
  138. for service in self.project.get_services():
  139. containers = service.containers()
  140. self.assertEqual(len(containers), 1)
  141. self.assertNotEqual(containers[0].get('Config.Hostname'), service.name)
  142. web_container = self.project.get_service('web').containers()[0]
  143. self.assertTrue(web_container.get('HostConfig.Links'))
  144. def test_up_with_networking(self):
  145. self.require_api_version('1.21')
  146. self.base_dir = 'tests/fixtures/links-composefile'
  147. self.dispatch(['--x-networking', 'up', '-d'], None)
  148. client = docker_client(version='1.21')
  149. services = self.project.get_services()
  150. networks = client.networks(names=[self.project.name])
  151. for n in networks:
  152. self.addCleanup(client.remove_network, n['Id'])
  153. self.assertEqual(len(networks), 1)
  154. self.assertEqual(networks[0]['Driver'], 'bridge')
  155. network = client.inspect_network(networks[0]['Id'])
  156. self.assertEqual(len(network['Containers']), len(services))
  157. for service in services:
  158. containers = service.containers()
  159. self.assertEqual(len(containers), 1)
  160. self.assertIn(containers[0].id, network['Containers'])
  161. self.assertEqual(containers[0].get('Config.Hostname'), service.name)
  162. web_container = self.project.get_service('web').containers()[0]
  163. self.assertFalse(web_container.get('HostConfig.Links'))
  164. def test_up_with_links(self):
  165. self.base_dir = 'tests/fixtures/links-composefile'
  166. self.dispatch(['up', '-d', 'web'], None)
  167. web = self.project.get_service('web')
  168. db = self.project.get_service('db')
  169. console = self.project.get_service('console')
  170. self.assertEqual(len(web.containers()), 1)
  171. self.assertEqual(len(db.containers()), 1)
  172. self.assertEqual(len(console.containers()), 0)
  173. def test_up_with_no_deps(self):
  174. self.base_dir = 'tests/fixtures/links-composefile'
  175. self.dispatch(['up', '-d', '--no-deps', 'web'], None)
  176. web = self.project.get_service('web')
  177. db = self.project.get_service('db')
  178. console = self.project.get_service('console')
  179. self.assertEqual(len(web.containers()), 1)
  180. self.assertEqual(len(db.containers()), 0)
  181. self.assertEqual(len(console.containers()), 0)
  182. def test_up_with_force_recreate(self):
  183. self.dispatch(['up', '-d'], None)
  184. service = self.project.get_service('simple')
  185. self.assertEqual(len(service.containers()), 1)
  186. old_ids = [c.id for c in service.containers()]
  187. self.dispatch(['up', '-d', '--force-recreate'], None)
  188. self.assertEqual(len(service.containers()), 1)
  189. new_ids = [c.id for c in service.containers()]
  190. self.assertNotEqual(old_ids, new_ids)
  191. def test_up_with_no_recreate(self):
  192. self.dispatch(['up', '-d'], None)
  193. service = self.project.get_service('simple')
  194. self.assertEqual(len(service.containers()), 1)
  195. old_ids = [c.id for c in service.containers()]
  196. self.dispatch(['up', '-d', '--no-recreate'], None)
  197. self.assertEqual(len(service.containers()), 1)
  198. new_ids = [c.id for c in service.containers()]
  199. self.assertEqual(old_ids, new_ids)
  200. def test_up_with_force_recreate_and_no_recreate(self):
  201. self.dispatch(
  202. ['up', '-d', '--force-recreate', '--no-recreate'],
  203. returncode=1)
  204. def test_up_with_timeout(self):
  205. self.dispatch(['up', '-d', '-t', '1'], None)
  206. service = self.project.get_service('simple')
  207. another = self.project.get_service('another')
  208. self.assertEqual(len(service.containers()), 1)
  209. self.assertEqual(len(another.containers()), 1)
  210. # Ensure containers don't have stdin and stdout connected in -d mode
  211. config = service.containers()[0].inspect()['Config']
  212. self.assertFalse(config['AttachStderr'])
  213. self.assertFalse(config['AttachStdout'])
  214. self.assertFalse(config['AttachStdin'])
  215. def test_run_service_without_links(self):
  216. self.base_dir = 'tests/fixtures/links-composefile'
  217. self.dispatch(['run', 'console', '/bin/true'])
  218. self.assertEqual(len(self.project.containers()), 0)
  219. # Ensure stdin/out was open
  220. container = self.project.containers(stopped=True, one_off=True)[0]
  221. config = container.inspect()['Config']
  222. self.assertTrue(config['AttachStderr'])
  223. self.assertTrue(config['AttachStdout'])
  224. self.assertTrue(config['AttachStdin'])
  225. def test_run_service_with_links(self):
  226. self.base_dir = 'tests/fixtures/links-composefile'
  227. self.dispatch(['run', 'web', '/bin/true'], None)
  228. db = self.project.get_service('db')
  229. console = self.project.get_service('console')
  230. self.assertEqual(len(db.containers()), 1)
  231. self.assertEqual(len(console.containers()), 0)
  232. def test_run_with_no_deps(self):
  233. self.base_dir = 'tests/fixtures/links-composefile'
  234. self.dispatch(['run', '--no-deps', 'web', '/bin/true'])
  235. db = self.project.get_service('db')
  236. self.assertEqual(len(db.containers()), 0)
  237. def test_run_does_not_recreate_linked_containers(self):
  238. self.base_dir = 'tests/fixtures/links-composefile'
  239. self.dispatch(['up', '-d', 'db'])
  240. db = self.project.get_service('db')
  241. self.assertEqual(len(db.containers()), 1)
  242. old_ids = [c.id for c in db.containers()]
  243. self.dispatch(['run', 'web', '/bin/true'], None)
  244. self.assertEqual(len(db.containers()), 1)
  245. new_ids = [c.id for c in db.containers()]
  246. self.assertEqual(old_ids, new_ids)
  247. def test_run_without_command(self):
  248. self.base_dir = 'tests/fixtures/commands-composefile'
  249. self.check_build('tests/fixtures/simple-dockerfile', tag='composetest_test')
  250. self.dispatch(['run', 'implicit'])
  251. service = self.project.get_service('implicit')
  252. containers = service.containers(stopped=True, one_off=True)
  253. self.assertEqual(
  254. [c.human_readable_command for c in containers],
  255. [u'/bin/sh -c echo "success"'],
  256. )
  257. self.dispatch(['run', 'explicit'])
  258. service = self.project.get_service('explicit')
  259. containers = service.containers(stopped=True, one_off=True)
  260. self.assertEqual(
  261. [c.human_readable_command for c in containers],
  262. [u'/bin/true'],
  263. )
  264. def test_run_service_with_entrypoint_overridden(self):
  265. self.base_dir = 'tests/fixtures/dockerfile_with_entrypoint'
  266. name = 'service'
  267. self.dispatch(['run', '--entrypoint', '/bin/echo', name, 'helloworld'])
  268. service = self.project.get_service(name)
  269. container = service.containers(stopped=True, one_off=True)[0]
  270. self.assertEqual(
  271. shlex.split(container.human_readable_command),
  272. [u'/bin/echo', u'helloworld'],
  273. )
  274. def test_run_service_with_user_overridden(self):
  275. self.base_dir = 'tests/fixtures/user-composefile'
  276. name = 'service'
  277. user = 'sshd'
  278. self.dispatch(['run', '--user={user}'.format(user=user), name], returncode=1)
  279. service = self.project.get_service(name)
  280. container = service.containers(stopped=True, one_off=True)[0]
  281. self.assertEqual(user, container.get('Config.User'))
  282. def test_run_service_with_user_overridden_short_form(self):
  283. self.base_dir = 'tests/fixtures/user-composefile'
  284. name = 'service'
  285. user = 'sshd'
  286. self.dispatch(['run', '-u', user, name], returncode=1)
  287. service = self.project.get_service(name)
  288. container = service.containers(stopped=True, one_off=True)[0]
  289. self.assertEqual(user, container.get('Config.User'))
  290. def test_run_service_with_environement_overridden(self):
  291. name = 'service'
  292. self.base_dir = 'tests/fixtures/environment-composefile'
  293. self.dispatch([
  294. 'run', '-e', 'foo=notbar',
  295. '-e', 'allo=moto=bobo',
  296. '-e', 'alpha=beta',
  297. name,
  298. '/bin/true',
  299. ])
  300. service = self.project.get_service(name)
  301. container = service.containers(stopped=True, one_off=True)[0]
  302. # env overriden
  303. self.assertEqual('notbar', container.environment['foo'])
  304. # keep environement from yaml
  305. self.assertEqual('world', container.environment['hello'])
  306. # added option from command line
  307. self.assertEqual('beta', container.environment['alpha'])
  308. # make sure a value with a = don't crash out
  309. self.assertEqual('moto=bobo', container.environment['allo'])
  310. def test_run_service_without_map_ports(self):
  311. # create one off container
  312. self.base_dir = 'tests/fixtures/ports-composefile'
  313. self.dispatch(['run', '-d', 'simple'])
  314. container = self.project.get_service('simple').containers(one_off=True)[0]
  315. # get port information
  316. port_random = container.get_local_port(3000)
  317. port_assigned = container.get_local_port(3001)
  318. # close all one off containers we just created
  319. container.stop()
  320. # check the ports
  321. self.assertEqual(port_random, None)
  322. self.assertEqual(port_assigned, None)
  323. def test_run_service_with_map_ports(self):
  324. # create one off container
  325. self.base_dir = 'tests/fixtures/ports-composefile'
  326. self.dispatch(['run', '-d', '--service-ports', 'simple'])
  327. container = self.project.get_service('simple').containers(one_off=True)[0]
  328. # get port information
  329. port_random = container.get_local_port(3000)
  330. port_assigned = container.get_local_port(3001)
  331. port_range = container.get_local_port(3002), container.get_local_port(3003)
  332. # close all one off containers we just created
  333. container.stop()
  334. # check the ports
  335. self.assertNotEqual(port_random, None)
  336. self.assertIn("0.0.0.0", port_random)
  337. self.assertEqual(port_assigned, "0.0.0.0:49152")
  338. self.assertEqual(port_range[0], "0.0.0.0:49153")
  339. self.assertEqual(port_range[1], "0.0.0.0:49154")
  340. def test_run_service_with_explicitly_maped_ports(self):
  341. # create one off container
  342. self.base_dir = 'tests/fixtures/ports-composefile'
  343. self.dispatch(['run', '-d', '-p', '30000:3000', '--publish', '30001:3001', 'simple'])
  344. container = self.project.get_service('simple').containers(one_off=True)[0]
  345. # get port information
  346. port_short = container.get_local_port(3000)
  347. port_full = container.get_local_port(3001)
  348. # close all one off containers we just created
  349. container.stop()
  350. # check the ports
  351. self.assertEqual(port_short, "0.0.0.0:30000")
  352. self.assertEqual(port_full, "0.0.0.0:30001")
  353. def test_run_service_with_explicitly_maped_ip_ports(self):
  354. # create one off container
  355. self.base_dir = 'tests/fixtures/ports-composefile'
  356. self.dispatch(['run', '-d', '-p', '127.0.0.1:30000:3000', '--publish', '127.0.0.1:30001:3001', 'simple'], None)
  357. container = self.project.get_service('simple').containers(one_off=True)[0]
  358. # get port information
  359. port_short = container.get_local_port(3000)
  360. port_full = container.get_local_port(3001)
  361. # close all one off containers we just created
  362. container.stop()
  363. # check the ports
  364. self.assertEqual(port_short, "127.0.0.1:30000")
  365. self.assertEqual(port_full, "127.0.0.1:30001")
  366. def test_run_with_custom_name(self):
  367. self.base_dir = 'tests/fixtures/environment-composefile'
  368. name = 'the-container-name'
  369. self.dispatch(['run', '--name', name, 'service', '/bin/true'])
  370. service = self.project.get_service('service')
  371. container, = service.containers(stopped=True, one_off=True)
  372. self.assertEqual(container.name, name)
  373. def test_run_with_networking(self):
  374. self.require_api_version('1.21')
  375. client = docker_client(version='1.21')
  376. self.base_dir = 'tests/fixtures/simple-dockerfile'
  377. self.dispatch(['--x-networking', 'run', 'simple', 'true'], None)
  378. service = self.project.get_service('simple')
  379. container, = service.containers(stopped=True, one_off=True)
  380. networks = client.networks(names=[self.project.name])
  381. for n in networks:
  382. self.addCleanup(client.remove_network, n['Id'])
  383. self.assertEqual(len(networks), 1)
  384. self.assertEqual(container.human_readable_command, u'true')
  385. def test_rm(self):
  386. service = self.project.get_service('simple')
  387. service.create_container()
  388. service.kill()
  389. self.assertEqual(len(service.containers(stopped=True)), 1)
  390. self.dispatch(['rm', '--force'], None)
  391. self.assertEqual(len(service.containers(stopped=True)), 0)
  392. service = self.project.get_service('simple')
  393. service.create_container()
  394. service.kill()
  395. self.assertEqual(len(service.containers(stopped=True)), 1)
  396. self.dispatch(['rm', '-f'], None)
  397. self.assertEqual(len(service.containers(stopped=True)), 0)
  398. def test_stop(self):
  399. self.dispatch(['up', '-d'], None)
  400. service = self.project.get_service('simple')
  401. self.assertEqual(len(service.containers()), 1)
  402. self.assertTrue(service.containers()[0].is_running)
  403. self.dispatch(['stop', '-t', '1'], None)
  404. self.assertEqual(len(service.containers(stopped=True)), 1)
  405. self.assertFalse(service.containers(stopped=True)[0].is_running)
  406. def test_pause_unpause(self):
  407. self.dispatch(['up', '-d'], None)
  408. service = self.project.get_service('simple')
  409. self.assertFalse(service.containers()[0].is_paused)
  410. self.dispatch(['pause'], None)
  411. self.assertTrue(service.containers()[0].is_paused)
  412. self.dispatch(['unpause'], None)
  413. self.assertFalse(service.containers()[0].is_paused)
  414. def test_logs_invalid_service_name(self):
  415. self.dispatch(['logs', 'madeupname'], returncode=1)
  416. def test_kill(self):
  417. self.dispatch(['up', '-d'], None)
  418. service = self.project.get_service('simple')
  419. self.assertEqual(len(service.containers()), 1)
  420. self.assertTrue(service.containers()[0].is_running)
  421. self.dispatch(['kill'], None)
  422. self.assertEqual(len(service.containers(stopped=True)), 1)
  423. self.assertFalse(service.containers(stopped=True)[0].is_running)
  424. def test_kill_signal_sigstop(self):
  425. self.dispatch(['up', '-d'], None)
  426. service = self.project.get_service('simple')
  427. self.assertEqual(len(service.containers()), 1)
  428. self.assertTrue(service.containers()[0].is_running)
  429. self.dispatch(['kill', '-s', 'SIGSTOP'], None)
  430. self.assertEqual(len(service.containers()), 1)
  431. # The container is still running. It has only been paused
  432. self.assertTrue(service.containers()[0].is_running)
  433. def test_kill_stopped_service(self):
  434. self.dispatch(['up', '-d'], None)
  435. service = self.project.get_service('simple')
  436. self.dispatch(['kill', '-s', 'SIGSTOP'], None)
  437. self.assertTrue(service.containers()[0].is_running)
  438. self.dispatch(['kill', '-s', 'SIGKILL'], None)
  439. self.assertEqual(len(service.containers(stopped=True)), 1)
  440. self.assertFalse(service.containers(stopped=True)[0].is_running)
  441. def test_restart(self):
  442. service = self.project.get_service('simple')
  443. container = service.create_container()
  444. container.start()
  445. started_at = container.dictionary['State']['StartedAt']
  446. self.dispatch(['restart', '-t', '1'], None)
  447. container.inspect()
  448. self.assertNotEqual(
  449. container.dictionary['State']['FinishedAt'],
  450. '0001-01-01T00:00:00Z',
  451. )
  452. self.assertNotEqual(
  453. container.dictionary['State']['StartedAt'],
  454. started_at,
  455. )
  456. def test_scale(self):
  457. project = self.project
  458. self.dispatch(['scale', 'simple=1'])
  459. self.assertEqual(len(project.get_service('simple').containers()), 1)
  460. self.dispatch(['scale', 'simple=3', 'another=2'])
  461. self.assertEqual(len(project.get_service('simple').containers()), 3)
  462. self.assertEqual(len(project.get_service('another').containers()), 2)
  463. self.dispatch(['scale', 'simple=1', 'another=1'])
  464. self.assertEqual(len(project.get_service('simple').containers()), 1)
  465. self.assertEqual(len(project.get_service('another').containers()), 1)
  466. self.dispatch(['scale', 'simple=1', 'another=1'])
  467. self.assertEqual(len(project.get_service('simple').containers()), 1)
  468. self.assertEqual(len(project.get_service('another').containers()), 1)
  469. self.dispatch(['scale', 'simple=0', 'another=0'])
  470. self.assertEqual(len(project.get_service('simple').containers()), 0)
  471. self.assertEqual(len(project.get_service('another').containers()), 0)
  472. def test_port(self):
  473. self.base_dir = 'tests/fixtures/ports-composefile'
  474. self.dispatch(['up', '-d'], None)
  475. container = self.project.get_service('simple').get_container()
  476. def get_port(number):
  477. result = self.dispatch(['port', 'simple', str(number)])
  478. return result.stdout.rstrip()
  479. self.assertEqual(get_port(3000), container.get_local_port(3000))
  480. self.assertEqual(get_port(3001), "0.0.0.0:49152")
  481. self.assertEqual(get_port(3002), "0.0.0.0:49153")
  482. def test_port_with_scale(self):
  483. self.base_dir = 'tests/fixtures/ports-composefile-scale'
  484. self.dispatch(['scale', 'simple=2'], None)
  485. containers = sorted(
  486. self.project.containers(service_names=['simple']),
  487. key=attrgetter('name'))
  488. def get_port(number, index=None):
  489. if index is None:
  490. result = self.dispatch(['port', 'simple', str(number)])
  491. else:
  492. result = self.dispatch(['port', '--index=' + str(index), 'simple', str(number)])
  493. return result.stdout.rstrip()
  494. self.assertEqual(get_port(3000), containers[0].get_local_port(3000))
  495. self.assertEqual(get_port(3000, index=1), containers[0].get_local_port(3000))
  496. self.assertEqual(get_port(3000, index=2), containers[1].get_local_port(3000))
  497. self.assertEqual(get_port(3002), "")
  498. def test_env_file_relative_to_compose_file(self):
  499. config_path = os.path.abspath('tests/fixtures/env-file/docker-compose.yml')
  500. self.dispatch(['-f', config_path, 'up', '-d'], None)
  501. self._project = get_project(self.base_dir, [config_path])
  502. containers = self.project.containers(stopped=True)
  503. self.assertEqual(len(containers), 1)
  504. self.assertIn("FOO=1", containers[0].get('Config.Env'))
  505. @mock.patch.dict(os.environ)
  506. def test_home_and_env_var_in_volume_path(self):
  507. os.environ['VOLUME_NAME'] = 'my-volume'
  508. os.environ['HOME'] = '/tmp/home-dir'
  509. self.base_dir = 'tests/fixtures/volume-path-interpolation'
  510. self.dispatch(['up', '-d'], None)
  511. container = self.project.containers(stopped=True)[0]
  512. actual_host_path = container.get('Volumes')['/container-path']
  513. components = actual_host_path.split('/')
  514. assert components[-2:] == ['home-dir', 'my-volume']
  515. def test_up_with_default_override_file(self):
  516. self.base_dir = 'tests/fixtures/override-files'
  517. self.dispatch(['up', '-d'], None)
  518. containers = self.project.containers()
  519. self.assertEqual(len(containers), 2)
  520. web, db = containers
  521. self.assertEqual(web.human_readable_command, 'top')
  522. self.assertEqual(db.human_readable_command, 'top')
  523. def test_up_with_multiple_files(self):
  524. self.base_dir = 'tests/fixtures/override-files'
  525. config_paths = [
  526. 'docker-compose.yml',
  527. 'docker-compose.override.yml',
  528. 'extra.yml',
  529. ]
  530. self._project = get_project(self.base_dir, config_paths)
  531. self.dispatch(
  532. [
  533. '-f', config_paths[0],
  534. '-f', config_paths[1],
  535. '-f', config_paths[2],
  536. 'up', '-d',
  537. ],
  538. None)
  539. containers = self.project.containers()
  540. self.assertEqual(len(containers), 3)
  541. web, other, db = containers
  542. self.assertEqual(web.human_readable_command, 'top')
  543. self.assertTrue({'db', 'other'} <= set(web.links()))
  544. self.assertEqual(db.human_readable_command, 'top')
  545. self.assertEqual(other.human_readable_command, 'top')
  546. def test_up_with_extends(self):
  547. self.base_dir = 'tests/fixtures/extends'
  548. self.dispatch(['up', '-d'], None)
  549. self.assertEqual(
  550. set([s.name for s in self.project.services]),
  551. set(['mydb', 'myweb']),
  552. )
  553. # Sort by name so we get [db, web]
  554. containers = sorted(
  555. self.project.containers(stopped=True),
  556. key=lambda c: c.name,
  557. )
  558. self.assertEqual(len(containers), 2)
  559. web = containers[1]
  560. self.assertEqual(set(web.links()), set(['db', 'mydb_1', 'extends_mydb_1']))
  561. expected_env = set([
  562. "FOO=1",
  563. "BAR=2",
  564. "BAZ=2",
  565. ])
  566. self.assertTrue(expected_env <= set(web.get('Config.Env')))