2015-08-26 22:21:31 +02:00
|
|
|
from __future__ import absolute_import
|
2015-10-30 21:22:51 +01:00
|
|
|
from __future__ import unicode_literals
|
2015-08-26 22:21:31 +02:00
|
|
|
|
2015-10-29 19:06:50 +01:00
|
|
|
import logging
|
|
|
|
|
2018-01-18 02:08:13 +01:00
|
|
|
import docker
|
2016-03-08 20:42:51 +01:00
|
|
|
import pytest
|
|
|
|
|
2015-08-26 22:21:31 +02:00
|
|
|
from compose import container
|
2015-09-02 17:07:59 +02:00
|
|
|
from compose.cli.errors import UserError
|
2015-10-29 19:06:50 +01:00
|
|
|
from compose.cli.formatter import ConsoleWarningFormatter
|
2019-07-22 17:27:10 +02:00
|
|
|
from compose.cli.main import build_one_off_container_options
|
2018-02-27 20:14:28 +01:00
|
|
|
from compose.cli.main import call_docker
|
2015-09-02 17:07:59 +02:00
|
|
|
from compose.cli.main import convergence_strategy_from_opts
|
2016-03-02 23:04:52 +01:00
|
|
|
from compose.cli.main import filter_containers_to_service_names
|
2019-07-22 17:27:10 +02:00
|
|
|
from compose.cli.main import get_docker_start_call
|
2015-10-29 19:06:50 +01:00
|
|
|
from compose.cli.main import setup_console_handler
|
2018-01-18 02:08:13 +01:00
|
|
|
from compose.cli.main import warn_for_swarm_mode
|
2015-09-02 17:07:59 +02:00
|
|
|
from compose.service import ConvergenceStrategy
|
2015-08-26 22:21:31 +02:00
|
|
|
from tests import mock
|
|
|
|
|
|
|
|
|
|
|
|
def mock_container(service, number):
|
|
|
|
return mock.create_autospec(
|
|
|
|
container.Container,
|
|
|
|
service=service,
|
|
|
|
number=number,
|
|
|
|
name_without_project='{0}_{1}'.format(service, number))
|
|
|
|
|
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
@pytest.fixture
|
|
|
|
def logging_handler():
|
|
|
|
stream = mock.Mock()
|
|
|
|
stream.isatty.return_value = True
|
|
|
|
return logging.StreamHandler(stream=stream)
|
|
|
|
|
|
|
|
|
|
|
|
class TestCLIMainTestCase(object):
|
2015-08-26 22:21:31 +02:00
|
|
|
|
2016-03-02 23:04:52 +01:00
|
|
|
def test_filter_containers_to_service_names(self):
|
2015-08-26 22:21:31 +02:00
|
|
|
containers = [
|
|
|
|
mock_container('web', 1),
|
|
|
|
mock_container('web', 2),
|
|
|
|
mock_container('db', 1),
|
|
|
|
mock_container('other', 1),
|
|
|
|
mock_container('another', 1),
|
|
|
|
]
|
|
|
|
service_names = ['web', 'db']
|
2016-03-02 23:04:52 +01:00
|
|
|
actual = filter_containers_to_service_names(containers, service_names)
|
|
|
|
assert actual == containers[:3]
|
2015-08-26 22:21:31 +02:00
|
|
|
|
2016-03-02 23:04:52 +01:00
|
|
|
def test_filter_containers_to_service_names_all(self):
|
2015-09-02 22:08:18 +02:00
|
|
|
containers = [
|
|
|
|
mock_container('web', 1),
|
|
|
|
mock_container('db', 1),
|
|
|
|
mock_container('other', 1),
|
|
|
|
]
|
|
|
|
service_names = []
|
2016-03-02 23:04:52 +01:00
|
|
|
actual = filter_containers_to_service_names(containers, service_names)
|
|
|
|
assert actual == containers
|
2015-09-02 17:07:59 +02:00
|
|
|
|
2018-01-18 02:08:13 +01:00
|
|
|
def test_warning_in_swarm_mode(self):
|
|
|
|
mock_client = mock.create_autospec(docker.APIClient)
|
|
|
|
mock_client.info.return_value = {'Swarm': {'LocalNodeState': 'active'}}
|
|
|
|
|
|
|
|
with mock.patch('compose.cli.main.log') as fake_log:
|
|
|
|
warn_for_swarm_mode(mock_client)
|
2019-05-15 19:45:40 +02:00
|
|
|
assert fake_log.warning.call_count == 1
|
2018-01-18 02:08:13 +01:00
|
|
|
|
2019-07-22 17:27:10 +02:00
|
|
|
def test_build_one_off_container_options(self):
|
|
|
|
command = 'build myservice'
|
|
|
|
detach = False
|
|
|
|
options = {
|
|
|
|
'-e': ['MYVAR=MYVALUE'],
|
|
|
|
'-T': True,
|
|
|
|
'--label': ['MYLABEL'],
|
|
|
|
'--entrypoint': 'bash',
|
|
|
|
'--user': 'MYUSER',
|
|
|
|
'--service-ports': [],
|
|
|
|
'--publish': '',
|
|
|
|
'--name': 'MYNAME',
|
|
|
|
'--workdir': '.',
|
|
|
|
'--volume': [],
|
|
|
|
'stdin_open': False,
|
|
|
|
}
|
|
|
|
|
|
|
|
expected_container_options = {
|
|
|
|
'command': command,
|
|
|
|
'tty': False,
|
|
|
|
'stdin_open': False,
|
|
|
|
'detach': detach,
|
|
|
|
'entrypoint': 'bash',
|
|
|
|
'environment': {'MYVAR': 'MYVALUE'},
|
|
|
|
'labels': {'MYLABEL': ''},
|
|
|
|
'name': 'MYNAME',
|
|
|
|
'ports': [],
|
|
|
|
'restart': None,
|
|
|
|
'user': 'MYUSER',
|
|
|
|
'working_dir': '.',
|
|
|
|
}
|
|
|
|
|
|
|
|
container_options = build_one_off_container_options(options, detach, command)
|
|
|
|
assert container_options == expected_container_options
|
|
|
|
|
|
|
|
def test_get_docker_start_call(self):
|
|
|
|
container_id = 'my_container_id'
|
|
|
|
|
|
|
|
mock_container_options = {'detach': False, 'stdin_open': True}
|
|
|
|
expected_docker_start_call = ['start', '--attach', '--interactive', container_id]
|
|
|
|
docker_start_call = get_docker_start_call(mock_container_options, container_id)
|
|
|
|
assert expected_docker_start_call == docker_start_call
|
|
|
|
|
|
|
|
mock_container_options = {'detach': False, 'stdin_open': False}
|
|
|
|
expected_docker_start_call = ['start', '--attach', container_id]
|
|
|
|
docker_start_call = get_docker_start_call(mock_container_options, container_id)
|
|
|
|
assert expected_docker_start_call == docker_start_call
|
|
|
|
|
|
|
|
mock_container_options = {'detach': True, 'stdin_open': True}
|
|
|
|
expected_docker_start_call = ['start', '--interactive', container_id]
|
|
|
|
docker_start_call = get_docker_start_call(mock_container_options, container_id)
|
|
|
|
assert expected_docker_start_call == docker_start_call
|
|
|
|
|
|
|
|
mock_container_options = {'detach': True, 'stdin_open': False}
|
|
|
|
expected_docker_start_call = ['start', container_id]
|
|
|
|
docker_start_call = get_docker_start_call(mock_container_options, container_id)
|
|
|
|
assert expected_docker_start_call == docker_start_call
|
|
|
|
|
2015-10-29 19:06:50 +01:00
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
class TestSetupConsoleHandlerTestCase(object):
|
2015-10-29 19:06:50 +01:00
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
def test_with_tty_verbose(self, logging_handler):
|
|
|
|
setup_console_handler(logging_handler, True)
|
|
|
|
assert type(logging_handler.formatter) == ConsoleWarningFormatter
|
|
|
|
assert '%(name)s' in logging_handler.formatter._fmt
|
|
|
|
assert '%(funcName)s' in logging_handler.formatter._fmt
|
2015-10-29 19:06:50 +01:00
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
def test_with_tty_not_verbose(self, logging_handler):
|
|
|
|
setup_console_handler(logging_handler, False)
|
|
|
|
assert type(logging_handler.formatter) == ConsoleWarningFormatter
|
|
|
|
assert '%(name)s' not in logging_handler.formatter._fmt
|
|
|
|
assert '%(funcName)s' not in logging_handler.formatter._fmt
|
2015-10-29 19:06:50 +01:00
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
def test_with_not_a_tty(self, logging_handler):
|
|
|
|
logging_handler.stream.isatty.return_value = False
|
|
|
|
setup_console_handler(logging_handler, False)
|
|
|
|
assert type(logging_handler.formatter) == logging.Formatter
|
2015-10-29 19:06:50 +01:00
|
|
|
|
|
|
|
|
2016-03-08 20:42:51 +01:00
|
|
|
class TestConvergeStrategyFromOptsTestCase(object):
|
2015-09-02 17:07:59 +02:00
|
|
|
|
|
|
|
def test_invalid_opts(self):
|
|
|
|
options = {'--force-recreate': True, '--no-recreate': True}
|
2016-03-08 20:42:51 +01:00
|
|
|
with pytest.raises(UserError):
|
2015-09-02 17:07:59 +02:00
|
|
|
convergence_strategy_from_opts(options)
|
|
|
|
|
|
|
|
def test_always(self):
|
|
|
|
options = {'--force-recreate': True, '--no-recreate': False}
|
2016-03-08 20:42:51 +01:00
|
|
|
assert (
|
|
|
|
convergence_strategy_from_opts(options) ==
|
2015-09-02 17:07:59 +02:00
|
|
|
ConvergenceStrategy.always
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_never(self):
|
|
|
|
options = {'--force-recreate': False, '--no-recreate': True}
|
2016-03-08 20:42:51 +01:00
|
|
|
assert (
|
|
|
|
convergence_strategy_from_opts(options) ==
|
2015-09-02 17:07:59 +02:00
|
|
|
ConvergenceStrategy.never
|
|
|
|
)
|
|
|
|
|
|
|
|
def test_changed(self):
|
|
|
|
options = {'--force-recreate': False, '--no-recreate': False}
|
2016-03-08 20:42:51 +01:00
|
|
|
assert (
|
|
|
|
convergence_strategy_from_opts(options) ==
|
2015-09-02 17:07:59 +02:00
|
|
|
ConvergenceStrategy.changed
|
|
|
|
)
|
2018-02-27 20:14:28 +01:00
|
|
|
|
|
|
|
|
|
|
|
def mock_find_executable(exe):
|
|
|
|
return exe
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('compose.cli.main.find_executable', mock_find_executable)
|
|
|
|
class TestCallDocker(object):
|
|
|
|
def test_simple_no_options(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
2019-05-23 16:59:21 +02:00
|
|
|
call_docker(['ps'], {}, {})
|
2018-02-27 20:14:28 +01:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == ['docker', 'ps']
|
|
|
|
|
|
|
|
def test_simple_tls_option(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
2019-05-23 16:59:21 +02:00
|
|
|
call_docker(['ps'], {'--tls': True}, {})
|
2018-02-27 20:14:28 +01:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == ['docker', '--tls', 'ps']
|
|
|
|
|
|
|
|
def test_advanced_tls_options(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
|
|
|
call_docker(['ps'], {
|
|
|
|
'--tls': True,
|
|
|
|
'--tlscacert': './ca.pem',
|
|
|
|
'--tlscert': './cert.pem',
|
|
|
|
'--tlskey': './key.pem',
|
2019-05-23 16:59:21 +02:00
|
|
|
}, {})
|
2018-02-27 20:14:28 +01:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == [
|
|
|
|
'docker', '--tls', '--tlscacert', './ca.pem', '--tlscert',
|
|
|
|
'./cert.pem', '--tlskey', './key.pem', 'ps'
|
|
|
|
]
|
|
|
|
|
|
|
|
def test_with_host_option(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
2019-05-23 16:59:21 +02:00
|
|
|
call_docker(['ps'], {'--host': 'tcp://mydocker.net:2333'}, {})
|
2018-02-27 20:14:28 +01:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == [
|
|
|
|
'docker', '--host', 'tcp://mydocker.net:2333', 'ps'
|
|
|
|
]
|
2018-03-26 23:25:31 +02:00
|
|
|
|
2018-10-25 01:08:56 +02:00
|
|
|
def test_with_http_host(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
2019-05-23 16:59:21 +02:00
|
|
|
call_docker(['ps'], {'--host': 'http://mydocker.net:2333'}, {})
|
2018-10-25 01:08:56 +02:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == [
|
|
|
|
'docker', '--host', 'tcp://mydocker.net:2333', 'ps',
|
|
|
|
]
|
|
|
|
|
2018-03-26 23:25:31 +02:00
|
|
|
def test_with_host_option_shorthand_equal(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
2019-05-23 16:59:21 +02:00
|
|
|
call_docker(['ps'], {'--host': '=tcp://mydocker.net:2333'}, {})
|
2018-03-26 23:25:31 +02:00
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == [
|
|
|
|
'docker', '--host', 'tcp://mydocker.net:2333', 'ps'
|
|
|
|
]
|
2019-05-23 16:59:21 +02:00
|
|
|
|
|
|
|
def test_with_env(self):
|
|
|
|
with mock.patch('subprocess.call') as fake_call:
|
|
|
|
call_docker(['ps'], {}, {'DOCKER_HOST': 'tcp://mydocker.net:2333'})
|
|
|
|
|
|
|
|
assert fake_call.call_args[0][0] == [
|
|
|
|
'docker', 'ps'
|
|
|
|
]
|
|
|
|
assert fake_call.call_args[1]['env'] == {'DOCKER_HOST': 'tcp://mydocker.net:2333'}
|