cli_test.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  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_get_project(self):
  42. base_dir = 'tests/fixtures/longer-filename-composefile'
  43. project = get_project(base_dir)
  44. self.assertEqual(project.name, 'longerfilenamecomposefile')
  45. self.assertTrue(project.client)
  46. self.assertTrue(project.services)
  47. def test_help(self):
  48. command = TopLevelCommand()
  49. with self.assertRaises(SystemExit):
  50. command.dispatch(['-h'], None)
  51. def test_command_help(self):
  52. with self.assertRaises(SystemExit) as ctx:
  53. TopLevelCommand().dispatch(['help', 'up'], None)
  54. self.assertIn('Usage: up', str(ctx.exception))
  55. def test_command_help_nonexistent(self):
  56. with self.assertRaises(NoSuchCommand):
  57. TopLevelCommand().dispatch(['help', 'nonexistent'], None)
  58. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason="requires dockerpty")
  59. @mock.patch('compose.cli.main.PseudoTerminal', autospec=True)
  60. def test_run_with_environment_merged_with_options_list(self, mock_pseudo_terminal):
  61. command = TopLevelCommand()
  62. mock_client = mock.create_autospec(docker.Client)
  63. mock_project = mock.Mock(client=mock_client)
  64. mock_project.get_service.return_value = Service(
  65. 'service',
  66. client=mock_client,
  67. environment=['FOO=ONE', 'BAR=TWO'],
  68. image='someimage')
  69. command.run(mock_project, {
  70. 'SERVICE': 'service',
  71. 'COMMAND': None,
  72. '-e': ['BAR=NEW', 'OTHER=bär'.encode('utf-8')],
  73. '--user': None,
  74. '--no-deps': None,
  75. '-d': True,
  76. '-T': None,
  77. '--entrypoint': None,
  78. '--service-ports': None,
  79. '--publish': [],
  80. '--rm': None,
  81. '--name': None,
  82. })
  83. _, _, call_kwargs = mock_client.create_container.mock_calls[0]
  84. self.assertEqual(
  85. call_kwargs['environment'],
  86. {'FOO': 'ONE', 'BAR': 'NEW', 'OTHER': u'bär'})
  87. def test_run_service_with_restart_always(self):
  88. command = TopLevelCommand()
  89. mock_client = mock.create_autospec(docker.Client)
  90. mock_project = mock.Mock(client=mock_client)
  91. mock_project.get_service.return_value = Service(
  92. 'service',
  93. client=mock_client,
  94. restart={'Name': 'always', 'MaximumRetryCount': 0},
  95. image='someimage')
  96. command.run(mock_project, {
  97. 'SERVICE': 'service',
  98. 'COMMAND': None,
  99. '-e': [],
  100. '--user': None,
  101. '--no-deps': None,
  102. '-d': True,
  103. '-T': None,
  104. '--entrypoint': None,
  105. '--service-ports': None,
  106. '--publish': [],
  107. '--rm': None,
  108. '--name': None,
  109. })
  110. self.assertEquals(
  111. mock_client.create_host_config.call_args[1]['restart_policy']['Name'],
  112. 'always'
  113. )
  114. command = TopLevelCommand()
  115. mock_client = mock.create_autospec(docker.Client)
  116. mock_project = mock.Mock(client=mock_client)
  117. mock_project.get_service.return_value = Service(
  118. 'service',
  119. client=mock_client,
  120. restart='always',
  121. image='someimage')
  122. command.run(mock_project, {
  123. 'SERVICE': 'service',
  124. 'COMMAND': None,
  125. '-e': [],
  126. '--user': None,
  127. '--no-deps': None,
  128. '-d': True,
  129. '-T': None,
  130. '--entrypoint': None,
  131. '--service-ports': None,
  132. '--publish': [],
  133. '--rm': True,
  134. '--name': None,
  135. })
  136. self.assertFalse(
  137. mock_client.create_host_config.call_args[1].get('restart_policy')
  138. )
  139. def test_command_manula_and_service_ports_together(self):
  140. command = TopLevelCommand()
  141. mock_client = mock.create_autospec(docker.Client)
  142. mock_project = mock.Mock(client=mock_client)
  143. mock_project.get_service.return_value = Service(
  144. 'service',
  145. client=mock_client,
  146. restart='always',
  147. image='someimage',
  148. )
  149. with self.assertRaises(UserError):
  150. command.run(mock_project, {
  151. 'SERVICE': 'service',
  152. 'COMMAND': None,
  153. '-e': [],
  154. '--user': None,
  155. '--no-deps': None,
  156. '-d': True,
  157. '-T': None,
  158. '--entrypoint': None,
  159. '--service-ports': True,
  160. '--publish': ['80:80'],
  161. '--rm': None,
  162. '--name': None,
  163. })