cli_test.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. # encoding: utf-8
  2. from __future__ import absolute_import
  3. from __future__ import unicode_literals
  4. import os
  5. import docker
  6. import py
  7. import pytest
  8. from .. import mock
  9. from .. import unittest
  10. from compose.cli.command import get_project
  11. from compose.cli.command import get_project_name
  12. from compose.cli.docopt_command import NoSuchCommand
  13. from compose.cli.errors import UserError
  14. from compose.cli.main import TopLevelCommand
  15. from compose.const import IS_WINDOWS_PLATFORM
  16. from compose.service import Service
  17. class CLITestCase(unittest.TestCase):
  18. def test_default_project_name(self):
  19. test_dir = py._path.local.LocalPath('tests/fixtures/simple-composefile')
  20. with test_dir.as_cwd():
  21. project_name = get_project_name('.')
  22. self.assertEquals('simplecomposefile', project_name)
  23. def test_project_name_with_explicit_base_dir(self):
  24. base_dir = 'tests/fixtures/simple-composefile'
  25. project_name = get_project_name(base_dir)
  26. self.assertEquals('simplecomposefile', project_name)
  27. def test_project_name_with_explicit_uppercase_base_dir(self):
  28. base_dir = 'tests/fixtures/UpperCaseDir'
  29. project_name = get_project_name(base_dir)
  30. self.assertEquals('uppercasedir', project_name)
  31. def test_project_name_with_explicit_project_name(self):
  32. name = 'explicit-project-name'
  33. project_name = get_project_name(None, project_name=name)
  34. self.assertEquals('explicitprojectname', project_name)
  35. def test_project_name_from_environment_new_var(self):
  36. name = 'namefromenv'
  37. with mock.patch.dict(os.environ):
  38. os.environ['COMPOSE_PROJECT_NAME'] = name
  39. project_name = get_project_name(None)
  40. self.assertEquals(project_name, name)
  41. def test_project_name_with_empty_environment_var(self):
  42. base_dir = 'tests/fixtures/simple-composefile'
  43. with mock.patch.dict(os.environ):
  44. os.environ['COMPOSE_PROJECT_NAME'] = ''
  45. project_name = get_project_name(base_dir)
  46. self.assertEquals('simplecomposefile', project_name)
  47. def test_get_project(self):
  48. base_dir = 'tests/fixtures/longer-filename-composefile'
  49. project = get_project(base_dir)
  50. self.assertEqual(project.name, 'longerfilenamecomposefile')
  51. self.assertTrue(project.client)
  52. self.assertTrue(project.services)
  53. def test_help(self):
  54. command = TopLevelCommand()
  55. with self.assertRaises(SystemExit):
  56. command.dispatch(['-h'])
  57. def test_command_help(self):
  58. with self.assertRaises(SystemExit) as ctx:
  59. TopLevelCommand().dispatch(['help', 'up'])
  60. self.assertIn('Usage: up', str(ctx.exception))
  61. def test_command_help_nonexistent(self):
  62. with self.assertRaises(NoSuchCommand):
  63. TopLevelCommand().dispatch(['help', 'nonexistent'])
  64. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason="requires dockerpty")
  65. @mock.patch('compose.cli.main.RunOperation', autospec=True)
  66. @mock.patch('compose.cli.main.PseudoTerminal', autospec=True)
  67. def test_run_interactive_passes_logs_false(self, mock_pseudo_terminal, mock_run_operation):
  68. command = TopLevelCommand()
  69. mock_client = mock.create_autospec(docker.Client)
  70. mock_project = mock.Mock(client=mock_client)
  71. mock_project.get_service.return_value = Service(
  72. 'service',
  73. client=mock_client,
  74. environment=['FOO=ONE', 'BAR=TWO'],
  75. image='someimage')
  76. with pytest.raises(SystemExit):
  77. command.run(mock_project, {
  78. 'SERVICE': 'service',
  79. 'COMMAND': None,
  80. '-e': ['BAR=NEW', 'OTHER=bär'.encode('utf-8')],
  81. '--user': None,
  82. '--no-deps': None,
  83. '-d': False,
  84. '-T': None,
  85. '--entrypoint': None,
  86. '--service-ports': None,
  87. '--publish': [],
  88. '--rm': None,
  89. '--name': None,
  90. })
  91. _, _, call_kwargs = mock_run_operation.mock_calls[0]
  92. assert call_kwargs['logs'] is False
  93. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason="requires dockerpty")
  94. @mock.patch('compose.cli.main.PseudoTerminal', autospec=True)
  95. def test_run_with_environment_merged_with_options_list(self, mock_pseudo_terminal):
  96. command = TopLevelCommand()
  97. mock_client = mock.create_autospec(docker.Client)
  98. mock_project = mock.Mock(client=mock_client)
  99. mock_project.get_service.return_value = Service(
  100. 'service',
  101. client=mock_client,
  102. environment=['FOO=ONE', 'BAR=TWO'],
  103. image='someimage')
  104. command.run(mock_project, {
  105. 'SERVICE': 'service',
  106. 'COMMAND': None,
  107. '-e': ['BAR=NEW', 'OTHER=bär'.encode('utf-8')],
  108. '--user': None,
  109. '--no-deps': None,
  110. '-d': True,
  111. '-T': None,
  112. '--entrypoint': None,
  113. '--service-ports': None,
  114. '--publish': [],
  115. '--rm': None,
  116. '--name': None,
  117. })
  118. _, _, call_kwargs = mock_client.create_container.mock_calls[0]
  119. assert (
  120. sorted(call_kwargs['environment']) ==
  121. sorted(['FOO=ONE', 'BAR=NEW', 'OTHER=bär'])
  122. )
  123. def test_run_service_with_restart_always(self):
  124. command = TopLevelCommand()
  125. mock_client = mock.create_autospec(docker.Client)
  126. mock_project = mock.Mock(client=mock_client)
  127. mock_project.get_service.return_value = Service(
  128. 'service',
  129. client=mock_client,
  130. restart={'Name': 'always', 'MaximumRetryCount': 0},
  131. image='someimage')
  132. command.run(mock_project, {
  133. 'SERVICE': 'service',
  134. 'COMMAND': None,
  135. '-e': [],
  136. '--user': None,
  137. '--no-deps': None,
  138. '-d': True,
  139. '-T': None,
  140. '--entrypoint': None,
  141. '--service-ports': None,
  142. '--publish': [],
  143. '--rm': None,
  144. '--name': None,
  145. })
  146. self.assertEquals(
  147. mock_client.create_host_config.call_args[1]['restart_policy']['Name'],
  148. 'always'
  149. )
  150. command = TopLevelCommand()
  151. mock_client = mock.create_autospec(docker.Client)
  152. mock_project = mock.Mock(client=mock_client)
  153. mock_project.get_service.return_value = Service(
  154. 'service',
  155. client=mock_client,
  156. restart='always',
  157. image='someimage')
  158. command.run(mock_project, {
  159. 'SERVICE': 'service',
  160. 'COMMAND': None,
  161. '-e': [],
  162. '--user': None,
  163. '--no-deps': None,
  164. '-d': True,
  165. '-T': None,
  166. '--entrypoint': None,
  167. '--service-ports': None,
  168. '--publish': [],
  169. '--rm': True,
  170. '--name': None,
  171. })
  172. self.assertFalse(
  173. mock_client.create_host_config.call_args[1].get('restart_policy')
  174. )
  175. def test_command_manula_and_service_ports_together(self):
  176. command = TopLevelCommand()
  177. mock_client = mock.create_autospec(docker.Client)
  178. mock_project = mock.Mock(client=mock_client)
  179. mock_project.get_service.return_value = Service(
  180. 'service',
  181. client=mock_client,
  182. restart='always',
  183. image='someimage',
  184. )
  185. with self.assertRaises(UserError):
  186. command.run(mock_project, {
  187. 'SERVICE': 'service',
  188. 'COMMAND': None,
  189. '-e': [],
  190. '--user': None,
  191. '--no-deps': None,
  192. '-d': True,
  193. '-T': None,
  194. '--entrypoint': None,
  195. '--service-ports': True,
  196. '--publish': ['80:80'],
  197. '--rm': None,
  198. '--name': None,
  199. })