cli_test.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. from __future__ import absolute_import
  2. import sys
  3. import os
  4. from six import StringIO
  5. from mock import patch
  6. from .testcases import DockerClientTestCase
  7. from compose.cli.main import TopLevelCommand
  8. class CLITestCase(DockerClientTestCase):
  9. def setUp(self):
  10. super(CLITestCase, self).setUp()
  11. self.old_sys_exit = sys.exit
  12. sys.exit = lambda code=0: None
  13. self.command = TopLevelCommand()
  14. self.command.base_dir = 'tests/fixtures/simple-composefile'
  15. def tearDown(self):
  16. sys.exit = self.old_sys_exit
  17. self.project.kill()
  18. self.project.remove_stopped()
  19. @property
  20. def project(self):
  21. # Hack: allow project to be overridden. This needs refactoring so that
  22. # the project object is built exactly once, by the command object, and
  23. # accessed by the test case object.
  24. if hasattr(self, '_project'):
  25. return self._project
  26. return self.command.get_project(self.command.get_config_path())
  27. def test_help(self):
  28. old_base_dir = self.command.base_dir
  29. self.command.base_dir = 'tests/fixtures/no-composefile'
  30. with self.assertRaises(SystemExit) as exc_context:
  31. self.command.dispatch(['help', 'up'], None)
  32. self.assertIn('Usage: up [options] [SERVICE...]', str(exc_context.exception))
  33. # self.project.kill() fails during teardown
  34. # unless there is a composefile.
  35. self.command.base_dir = old_base_dir
  36. # TODO: address the "Inappropriate ioctl for device" warnings in test output
  37. @patch('sys.stdout', new_callable=StringIO)
  38. def test_ps(self, mock_stdout):
  39. self.project.get_service('simple').create_container()
  40. self.command.dispatch(['ps'], None)
  41. self.assertIn('simplecomposefile_simple_1', mock_stdout.getvalue())
  42. @patch('sys.stdout', new_callable=StringIO)
  43. def test_ps_default_composefile(self, mock_stdout):
  44. self.command.base_dir = 'tests/fixtures/multiple-composefiles'
  45. self.command.dispatch(['up', '-d'], None)
  46. self.command.dispatch(['ps'], None)
  47. output = mock_stdout.getvalue()
  48. self.assertIn('multiplecomposefiles_simple_1', output)
  49. self.assertIn('multiplecomposefiles_another_1', output)
  50. self.assertNotIn('multiplecomposefiles_yetanother_1', output)
  51. @patch('sys.stdout', new_callable=StringIO)
  52. def test_ps_alternate_composefile(self, mock_stdout):
  53. self.command.base_dir = 'tests/fixtures/multiple-composefiles'
  54. self.command.dispatch(['-f', 'compose2.yml', 'up', '-d'], None)
  55. self.command.dispatch(['-f', 'compose2.yml', 'ps'], None)
  56. output = mock_stdout.getvalue()
  57. self.assertNotIn('multiplecomposefiles_simple_1', output)
  58. self.assertNotIn('multiplecomposefiles_another_1', output)
  59. self.assertIn('multiplecomposefiles_yetanother_1', output)
  60. @patch('compose.service.log')
  61. def test_pull(self, mock_logging):
  62. self.command.dispatch(['pull'], None)
  63. mock_logging.info.assert_any_call('Pulling simple (busybox:latest)...')
  64. mock_logging.info.assert_any_call('Pulling another (busybox:latest)...')
  65. @patch('sys.stdout', new_callable=StringIO)
  66. def test_build_no_cache(self, mock_stdout):
  67. self.command.base_dir = 'tests/fixtures/simple-dockerfile'
  68. self.command.dispatch(['build', 'simple'], None)
  69. mock_stdout.truncate(0)
  70. cache_indicator = 'Using cache'
  71. self.command.dispatch(['build', 'simple'], None)
  72. output = mock_stdout.getvalue()
  73. self.assertIn(cache_indicator, output)
  74. mock_stdout.truncate(0)
  75. self.command.dispatch(['build', '--no-cache', 'simple'], None)
  76. output = mock_stdout.getvalue()
  77. self.assertNotIn(cache_indicator, output)
  78. def test_up(self):
  79. self.command.dispatch(['up', '-d'], None)
  80. service = self.project.get_service('simple')
  81. another = self.project.get_service('another')
  82. self.assertEqual(len(service.containers()), 1)
  83. self.assertEqual(len(another.containers()), 1)
  84. # Ensure containers don't have stdin and stdout connected in -d mode
  85. config = service.containers()[0].inspect()['Config']
  86. self.assertFalse(config['AttachStderr'])
  87. self.assertFalse(config['AttachStdout'])
  88. self.assertFalse(config['AttachStdin'])
  89. def test_up_with_links(self):
  90. self.command.base_dir = 'tests/fixtures/links-composefile'
  91. self.command.dispatch(['up', '-d', 'web'], None)
  92. web = self.project.get_service('web')
  93. db = self.project.get_service('db')
  94. console = self.project.get_service('console')
  95. self.assertEqual(len(web.containers()), 1)
  96. self.assertEqual(len(db.containers()), 1)
  97. self.assertEqual(len(console.containers()), 0)
  98. def test_up_with_no_deps(self):
  99. self.command.base_dir = 'tests/fixtures/links-composefile'
  100. self.command.dispatch(['up', '-d', '--no-deps', 'web'], None)
  101. web = self.project.get_service('web')
  102. db = self.project.get_service('db')
  103. console = self.project.get_service('console')
  104. self.assertEqual(len(web.containers()), 1)
  105. self.assertEqual(len(db.containers()), 0)
  106. self.assertEqual(len(console.containers()), 0)
  107. def test_up_with_recreate(self):
  108. self.command.dispatch(['up', '-d'], None)
  109. service = self.project.get_service('simple')
  110. self.assertEqual(len(service.containers()), 1)
  111. old_ids = [c.id for c in service.containers()]
  112. self.command.dispatch(['up', '-d'], None)
  113. self.assertEqual(len(service.containers()), 1)
  114. new_ids = [c.id for c in service.containers()]
  115. self.assertNotEqual(old_ids, new_ids)
  116. def test_up_with_keep_old(self):
  117. self.command.dispatch(['up', '-d'], None)
  118. service = self.project.get_service('simple')
  119. self.assertEqual(len(service.containers()), 1)
  120. old_ids = [c.id for c in service.containers()]
  121. self.command.dispatch(['up', '-d', '--no-recreate'], None)
  122. self.assertEqual(len(service.containers()), 1)
  123. new_ids = [c.id for c in service.containers()]
  124. self.assertEqual(old_ids, new_ids)
  125. @patch('dockerpty.start')
  126. def test_run_service_without_links(self, mock_stdout):
  127. self.command.base_dir = 'tests/fixtures/links-composefile'
  128. self.command.dispatch(['run', 'console', '/bin/true'], None)
  129. self.assertEqual(len(self.project.containers()), 0)
  130. # Ensure stdin/out was open
  131. container = self.project.containers(stopped=True, one_off=True)[0]
  132. config = container.inspect()['Config']
  133. self.assertTrue(config['AttachStderr'])
  134. self.assertTrue(config['AttachStdout'])
  135. self.assertTrue(config['AttachStdin'])
  136. @patch('dockerpty.start')
  137. def test_run_service_with_links(self, __):
  138. self.command.base_dir = 'tests/fixtures/links-composefile'
  139. self.command.dispatch(['run', 'web', '/bin/true'], None)
  140. db = self.project.get_service('db')
  141. console = self.project.get_service('console')
  142. self.assertEqual(len(db.containers()), 1)
  143. self.assertEqual(len(console.containers()), 0)
  144. @patch('dockerpty.start')
  145. def test_run_with_no_deps(self, __):
  146. self.command.base_dir = 'tests/fixtures/links-composefile'
  147. self.command.dispatch(['run', '--no-deps', 'web', '/bin/true'], None)
  148. db = self.project.get_service('db')
  149. self.assertEqual(len(db.containers()), 0)
  150. @patch('dockerpty.start')
  151. def test_run_does_not_recreate_linked_containers(self, __):
  152. self.command.base_dir = 'tests/fixtures/links-composefile'
  153. self.command.dispatch(['up', '-d', 'db'], None)
  154. db = self.project.get_service('db')
  155. self.assertEqual(len(db.containers()), 1)
  156. old_ids = [c.id for c in db.containers()]
  157. self.command.dispatch(['run', 'web', '/bin/true'], None)
  158. self.assertEqual(len(db.containers()), 1)
  159. new_ids = [c.id for c in db.containers()]
  160. self.assertEqual(old_ids, new_ids)
  161. @patch('dockerpty.start')
  162. def test_run_without_command(self, __):
  163. self.command.base_dir = 'tests/fixtures/commands-composefile'
  164. self.check_build('tests/fixtures/simple-dockerfile', tag='composetest_test')
  165. for c in self.project.containers(stopped=True, one_off=True):
  166. c.remove()
  167. self.command.dispatch(['run', 'implicit'], None)
  168. service = self.project.get_service('implicit')
  169. containers = service.containers(stopped=True, one_off=True)
  170. self.assertEqual(
  171. [c.human_readable_command for c in containers],
  172. [u'/bin/sh -c echo "success"'],
  173. )
  174. self.command.dispatch(['run', 'explicit'], None)
  175. service = self.project.get_service('explicit')
  176. containers = service.containers(stopped=True, one_off=True)
  177. self.assertEqual(
  178. [c.human_readable_command for c in containers],
  179. [u'/bin/true'],
  180. )
  181. @patch('dockerpty.start')
  182. def test_run_service_with_entrypoint_overridden(self, _):
  183. self.command.base_dir = 'tests/fixtures/dockerfile_with_entrypoint'
  184. name = 'service'
  185. self.command.dispatch(
  186. ['run', '--entrypoint', '/bin/echo', name, 'helloworld'],
  187. None
  188. )
  189. service = self.project.get_service(name)
  190. container = service.containers(stopped=True, one_off=True)[0]
  191. self.assertEqual(
  192. container.human_readable_command,
  193. u'/bin/echo helloworld'
  194. )
  195. @patch('dockerpty.start')
  196. def test_run_service_with_user_overridden(self, _):
  197. self.command.base_dir = 'tests/fixtures/user-composefile'
  198. name = 'service'
  199. user = 'sshd'
  200. args = ['run', '--user={}'.format(user), name]
  201. self.command.dispatch(args, None)
  202. service = self.project.get_service(name)
  203. container = service.containers(stopped=True, one_off=True)[0]
  204. self.assertEqual(user, container.get('Config.User'))
  205. @patch('dockerpty.start')
  206. def test_run_service_with_user_overridden_short_form(self, _):
  207. self.command.base_dir = 'tests/fixtures/user-composefile'
  208. name = 'service'
  209. user = 'sshd'
  210. args = ['run', '-u', user, name]
  211. self.command.dispatch(args, None)
  212. service = self.project.get_service(name)
  213. container = service.containers(stopped=True, one_off=True)[0]
  214. self.assertEqual(user, container.get('Config.User'))
  215. @patch('dockerpty.start')
  216. def test_run_service_with_environement_overridden(self, _):
  217. name = 'service'
  218. self.command.base_dir = 'tests/fixtures/environment-composefile'
  219. self.command.dispatch(
  220. ['run', '-e', 'foo=notbar', '-e', 'allo=moto=bobo',
  221. '-e', 'alpha=beta', name],
  222. None
  223. )
  224. service = self.project.get_service(name)
  225. container = service.containers(stopped=True, one_off=True)[0]
  226. # env overriden
  227. self.assertEqual('notbar', container.environment['foo'])
  228. # keep environement from yaml
  229. self.assertEqual('world', container.environment['hello'])
  230. # added option from command line
  231. self.assertEqual('beta', container.environment['alpha'])
  232. # make sure a value with a = don't crash out
  233. self.assertEqual('moto=bobo', container.environment['allo'])
  234. @patch('dockerpty.start')
  235. def test_run_service_without_map_ports(self, __):
  236. # create one off container
  237. self.command.base_dir = 'tests/fixtures/ports-composefile'
  238. self.command.dispatch(['run', '-d', 'simple'], None)
  239. container = self.project.get_service('simple').containers(one_off=True)[0]
  240. # get port information
  241. port_random = container.get_local_port(3000)
  242. port_assigned = container.get_local_port(3001)
  243. # close all one off containers we just created
  244. container.stop()
  245. # check the ports
  246. self.assertEqual(port_random, None)
  247. self.assertEqual(port_assigned, None)
  248. @patch('dockerpty.start')
  249. def test_run_service_with_map_ports(self, __):
  250. # create one off container
  251. self.command.base_dir = 'tests/fixtures/ports-composefile'
  252. self.command.dispatch(['run', '-d', '--service-ports', 'simple'], None)
  253. container = self.project.get_service('simple').containers(one_off=True)[0]
  254. # get port information
  255. port_random = container.get_local_port(3000)
  256. port_assigned = container.get_local_port(3001)
  257. # close all one off containers we just created
  258. container.stop()
  259. # check the ports
  260. self.assertNotEqual(port_random, None)
  261. self.assertIn("0.0.0.0", port_random)
  262. self.assertEqual(port_assigned, "0.0.0.0:9999")
  263. def test_rm(self):
  264. service = self.project.get_service('simple')
  265. service.create_container()
  266. service.kill()
  267. self.assertEqual(len(service.containers(stopped=True)), 1)
  268. self.command.dispatch(['rm', '--force'], None)
  269. self.assertEqual(len(service.containers(stopped=True)), 0)
  270. service = self.project.get_service('simple')
  271. service.create_container()
  272. service.kill()
  273. self.assertEqual(len(service.containers(stopped=True)), 1)
  274. self.command.dispatch(['rm', '-f'], None)
  275. self.assertEqual(len(service.containers(stopped=True)), 0)
  276. def test_kill(self):
  277. self.command.dispatch(['up', '-d'], None)
  278. service = self.project.get_service('simple')
  279. self.assertEqual(len(service.containers()), 1)
  280. self.assertTrue(service.containers()[0].is_running)
  281. self.command.dispatch(['kill'], None)
  282. self.assertEqual(len(service.containers(stopped=True)), 1)
  283. self.assertFalse(service.containers(stopped=True)[0].is_running)
  284. def test_kill_signal_sigint(self):
  285. self.command.dispatch(['up', '-d'], None)
  286. service = self.project.get_service('simple')
  287. self.assertEqual(len(service.containers()), 1)
  288. self.assertTrue(service.containers()[0].is_running)
  289. self.command.dispatch(['kill', '-s', 'SIGINT'], None)
  290. self.assertEqual(len(service.containers()), 1)
  291. # The container is still running. It has been only interrupted
  292. self.assertTrue(service.containers()[0].is_running)
  293. def test_kill_interrupted_service(self):
  294. self.command.dispatch(['up', '-d'], None)
  295. service = self.project.get_service('simple')
  296. self.command.dispatch(['kill', '-s', 'SIGINT'], None)
  297. self.assertTrue(service.containers()[0].is_running)
  298. self.command.dispatch(['kill', '-s', 'SIGKILL'], None)
  299. self.assertEqual(len(service.containers(stopped=True)), 1)
  300. self.assertFalse(service.containers(stopped=True)[0].is_running)
  301. def test_restart(self):
  302. service = self.project.get_service('simple')
  303. container = service.create_container()
  304. service.start_container(container)
  305. started_at = container.dictionary['State']['StartedAt']
  306. self.command.dispatch(['restart'], None)
  307. container.inspect()
  308. self.assertNotEqual(
  309. container.dictionary['State']['FinishedAt'],
  310. '0001-01-01T00:00:00Z',
  311. )
  312. self.assertNotEqual(
  313. container.dictionary['State']['StartedAt'],
  314. started_at,
  315. )
  316. def test_scale(self):
  317. project = self.project
  318. self.command.scale(project, {'SERVICE=NUM': ['simple=1']})
  319. self.assertEqual(len(project.get_service('simple').containers()), 1)
  320. self.command.scale(project, {'SERVICE=NUM': ['simple=3', 'another=2']})
  321. self.assertEqual(len(project.get_service('simple').containers()), 3)
  322. self.assertEqual(len(project.get_service('another').containers()), 2)
  323. self.command.scale(project, {'SERVICE=NUM': ['simple=1', 'another=1']})
  324. self.assertEqual(len(project.get_service('simple').containers()), 1)
  325. self.assertEqual(len(project.get_service('another').containers()), 1)
  326. self.command.scale(project, {'SERVICE=NUM': ['simple=1', 'another=1']})
  327. self.assertEqual(len(project.get_service('simple').containers()), 1)
  328. self.assertEqual(len(project.get_service('another').containers()), 1)
  329. self.command.scale(project, {'SERVICE=NUM': ['simple=0', 'another=0']})
  330. self.assertEqual(len(project.get_service('simple').containers()), 0)
  331. self.assertEqual(len(project.get_service('another').containers()), 0)
  332. def test_port(self):
  333. self.command.base_dir = 'tests/fixtures/ports-composefile'
  334. self.command.dispatch(['up', '-d'], None)
  335. container = self.project.get_service('simple').get_container()
  336. @patch('sys.stdout', new_callable=StringIO)
  337. def get_port(number, mock_stdout):
  338. self.command.dispatch(['port', 'simple', str(number)], None)
  339. return mock_stdout.getvalue().rstrip()
  340. self.assertEqual(get_port(3000), container.get_local_port(3000))
  341. self.assertEqual(get_port(3001), "0.0.0.0:9999")
  342. self.assertEqual(get_port(3002), "")
  343. def test_env_file_relative_to_compose_file(self):
  344. config_path = os.path.abspath('tests/fixtures/env-file/docker-compose.yml')
  345. self.command.dispatch(['-f', config_path, 'up', '-d'], None)
  346. self._project = self.command.get_project(config_path)
  347. containers = self.project.containers(stopped=True)
  348. self.assertEqual(len(containers), 1)
  349. self.assertIn("FOO=1", containers[0].get('Config.Env'))