cli_test.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. # encoding: utf-8
  2. from __future__ import absolute_import
  3. from __future__ import unicode_literals
  4. import os
  5. import shutil
  6. import tempfile
  7. from io import StringIO
  8. import docker
  9. import py
  10. import pytest
  11. from docker.constants import DEFAULT_DOCKER_API_VERSION
  12. from .. import mock
  13. from .. import unittest
  14. from ..helpers import build_config
  15. from compose.cli.command import get_project
  16. from compose.cli.command import get_project_name
  17. from compose.cli.docopt_command import NoSuchCommand
  18. from compose.cli.errors import UserError
  19. from compose.cli.main import TopLevelCommand
  20. from compose.const import IS_WINDOWS_PLATFORM
  21. from compose.project import Project
  22. class CLITestCase(unittest.TestCase):
  23. def test_default_project_name(self):
  24. test_dir = py._path.local.LocalPath('tests/fixtures/simple-composefile')
  25. with test_dir.as_cwd():
  26. project_name = get_project_name('.')
  27. self.assertEqual('simplecomposefile', project_name)
  28. def test_project_name_with_explicit_base_dir(self):
  29. base_dir = 'tests/fixtures/simple-composefile'
  30. project_name = get_project_name(base_dir)
  31. self.assertEqual('simplecomposefile', project_name)
  32. def test_project_name_with_explicit_uppercase_base_dir(self):
  33. base_dir = 'tests/fixtures/UpperCaseDir'
  34. project_name = get_project_name(base_dir)
  35. self.assertEqual('uppercasedir', project_name)
  36. def test_project_name_with_explicit_project_name(self):
  37. name = 'explicit-project-name'
  38. project_name = get_project_name(None, project_name=name)
  39. self.assertEqual('explicitprojectname', project_name)
  40. @mock.patch.dict(os.environ)
  41. def test_project_name_from_environment_new_var(self):
  42. name = 'namefromenv'
  43. os.environ['COMPOSE_PROJECT_NAME'] = name
  44. project_name = get_project_name(None)
  45. self.assertEqual(project_name, name)
  46. def test_project_name_with_empty_environment_var(self):
  47. base_dir = 'tests/fixtures/simple-composefile'
  48. with mock.patch.dict(os.environ):
  49. os.environ['COMPOSE_PROJECT_NAME'] = ''
  50. project_name = get_project_name(base_dir)
  51. self.assertEqual('simplecomposefile', project_name)
  52. @mock.patch.dict(os.environ)
  53. def test_project_name_with_environment_file(self):
  54. base_dir = tempfile.mkdtemp()
  55. try:
  56. name = 'namefromenvfile'
  57. with open(os.path.join(base_dir, '.env'), 'w') as f:
  58. f.write('COMPOSE_PROJECT_NAME={}'.format(name))
  59. project_name = get_project_name(base_dir)
  60. assert project_name == name
  61. # Environment has priority over .env file
  62. os.environ['COMPOSE_PROJECT_NAME'] = 'namefromenv'
  63. assert get_project_name(base_dir) == os.environ['COMPOSE_PROJECT_NAME']
  64. finally:
  65. shutil.rmtree(base_dir)
  66. def test_get_project(self):
  67. base_dir = 'tests/fixtures/longer-filename-composefile'
  68. project = get_project(base_dir)
  69. self.assertEqual(project.name, 'longerfilenamecomposefile')
  70. self.assertTrue(project.client)
  71. self.assertTrue(project.services)
  72. def test_command_help(self):
  73. with mock.patch('sys.stdout', new=StringIO()) as fake_stdout:
  74. TopLevelCommand.help({'COMMAND': 'up'})
  75. assert "Usage: up" in fake_stdout.getvalue()
  76. def test_command_help_nonexistent(self):
  77. with pytest.raises(NoSuchCommand):
  78. TopLevelCommand.help({'COMMAND': 'nonexistent'})
  79. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason="requires dockerpty")
  80. @mock.patch('compose.cli.main.RunOperation', autospec=True)
  81. @mock.patch('compose.cli.main.PseudoTerminal', autospec=True)
  82. def test_run_interactive_passes_logs_false(self, mock_pseudo_terminal, mock_run_operation):
  83. mock_client = mock.create_autospec(docker.APIClient)
  84. mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  85. project = Project.from_config(
  86. name='composetest',
  87. client=mock_client,
  88. config_data=build_config({
  89. 'service': {'image': 'busybox'}
  90. }),
  91. )
  92. command = TopLevelCommand(project)
  93. with pytest.raises(SystemExit):
  94. command.run({
  95. 'SERVICE': 'service',
  96. 'COMMAND': None,
  97. '-e': [],
  98. '--user': None,
  99. '--no-deps': None,
  100. '-d': False,
  101. '-T': None,
  102. '--entrypoint': None,
  103. '--service-ports': None,
  104. '--publish': [],
  105. '--volume': [],
  106. '--rm': None,
  107. '--name': None,
  108. '--workdir': None,
  109. })
  110. _, _, call_kwargs = mock_run_operation.mock_calls[0]
  111. assert call_kwargs['logs'] is False
  112. def test_run_service_with_restart_always(self):
  113. mock_client = mock.create_autospec(docker.APIClient)
  114. mock_client.api_version = DEFAULT_DOCKER_API_VERSION
  115. project = Project.from_config(
  116. name='composetest',
  117. client=mock_client,
  118. config_data=build_config({
  119. 'service': {
  120. 'image': 'busybox',
  121. 'restart': 'always',
  122. }
  123. }),
  124. )
  125. command = TopLevelCommand(project)
  126. command.run({
  127. 'SERVICE': 'service',
  128. 'COMMAND': None,
  129. '-e': [],
  130. '--user': None,
  131. '--no-deps': None,
  132. '-d': True,
  133. '-T': None,
  134. '--entrypoint': None,
  135. '--service-ports': None,
  136. '--publish': [],
  137. '--volume': [],
  138. '--rm': None,
  139. '--name': None,
  140. '--workdir': None,
  141. })
  142. self.assertEqual(
  143. mock_client.create_host_config.call_args[1]['restart_policy']['Name'],
  144. 'always'
  145. )
  146. command = TopLevelCommand(project)
  147. command.run({
  148. 'SERVICE': 'service',
  149. 'COMMAND': None,
  150. '-e': [],
  151. '--user': None,
  152. '--no-deps': None,
  153. '-d': True,
  154. '-T': None,
  155. '--entrypoint': None,
  156. '--service-ports': None,
  157. '--publish': [],
  158. '--volume': [],
  159. '--rm': True,
  160. '--name': None,
  161. '--workdir': None,
  162. })
  163. self.assertFalse(
  164. mock_client.create_host_config.call_args[1].get('restart_policy')
  165. )
  166. def test_command_manual_and_service_ports_together(self):
  167. project = Project.from_config(
  168. name='composetest',
  169. client=None,
  170. config_data=build_config({
  171. 'service': {'image': 'busybox'},
  172. }),
  173. )
  174. command = TopLevelCommand(project)
  175. with self.assertRaises(UserError):
  176. command.run({
  177. 'SERVICE': 'service',
  178. 'COMMAND': None,
  179. '-e': [],
  180. '--user': None,
  181. '--no-deps': None,
  182. '-d': True,
  183. '-T': None,
  184. '--entrypoint': None,
  185. '--service-ports': True,
  186. '--publish': ['80:80'],
  187. '--rm': None,
  188. '--name': None,
  189. })