mirror of https://github.com/docker/compose.git
Refactor network_mode logic out of Service.
Signed-off-by: Daniel Nephin <dnephin@docker.com>
This commit is contained in:
parent
d92f323e6d
commit
805f6a7683
|
@ -9,7 +9,10 @@ from .config import get_service_name_from_net, ConfigurationError
|
||||||
from .const import DEFAULT_TIMEOUT, LABEL_PROJECT, LABEL_SERVICE, LABEL_ONE_OFF
|
from .const import DEFAULT_TIMEOUT, LABEL_PROJECT, LABEL_SERVICE, LABEL_ONE_OFF
|
||||||
from .container import Container
|
from .container import Container
|
||||||
from .legacy import check_for_legacy_containers
|
from .legacy import check_for_legacy_containers
|
||||||
|
from .service import ContainerNet
|
||||||
|
from .service import Net
|
||||||
from .service import Service
|
from .service import Service
|
||||||
|
from .service import ServiceNet
|
||||||
from .utils import parallel_execute
|
from .utils import parallel_execute
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
@ -180,18 +183,18 @@ class Project(object):
|
||||||
def get_net(self, service_dict):
|
def get_net(self, service_dict):
|
||||||
net = service_dict.pop('net', None)
|
net = service_dict.pop('net', None)
|
||||||
if not net:
|
if not net:
|
||||||
return
|
return Net(None)
|
||||||
|
|
||||||
net_name = get_service_name_from_net(net)
|
net_name = get_service_name_from_net(net)
|
||||||
if not net_name:
|
if not net_name:
|
||||||
return net
|
return Net(net)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
return self.get_service(net_name)
|
return ServiceNet(self.get_service(net_name))
|
||||||
except NoSuchService:
|
except NoSuchService:
|
||||||
pass
|
pass
|
||||||
try:
|
try:
|
||||||
return Container.from_id(self.client, net_name)
|
return ContainerNet(Container.from_id(self.client, net_name))
|
||||||
except APIError:
|
except APIError:
|
||||||
raise ConfigurationError(
|
raise ConfigurationError(
|
||||||
'Service "%s" is trying to use the network of "%s", '
|
'Service "%s" is trying to use the network of "%s", '
|
||||||
|
|
|
@ -107,7 +107,7 @@ class Service(object):
|
||||||
self.project = project
|
self.project = project
|
||||||
self.links = links or []
|
self.links = links or []
|
||||||
self.volumes_from = volumes_from or []
|
self.volumes_from = volumes_from or []
|
||||||
self.net = net or None
|
self.net = net or Net(None)
|
||||||
self.options = options
|
self.options = options
|
||||||
|
|
||||||
def containers(self, stopped=False, one_off=False):
|
def containers(self, stopped=False, one_off=False):
|
||||||
|
@ -478,12 +478,12 @@ class Service(object):
|
||||||
'options': self.options,
|
'options': self.options,
|
||||||
'image_id': self.image()['Id'],
|
'image_id': self.image()['Id'],
|
||||||
'links': [(service.name, alias) for service, alias in self.links],
|
'links': [(service.name, alias) for service, alias in self.links],
|
||||||
'net': self.get_net_name() or getattr(self.net, 'id', self.net),
|
'net': self.net.id,
|
||||||
'volumes_from': self.get_volumes_from_names(),
|
'volumes_from': self.get_volumes_from_names(),
|
||||||
}
|
}
|
||||||
|
|
||||||
def get_dependency_names(self):
|
def get_dependency_names(self):
|
||||||
net_name = self.get_net_name()
|
net_name = self.net.service_name
|
||||||
return (self.get_linked_names() +
|
return (self.get_linked_names() +
|
||||||
self.get_volumes_from_names() +
|
self.get_volumes_from_names() +
|
||||||
([net_name] if net_name else []))
|
([net_name] if net_name else []))
|
||||||
|
@ -494,12 +494,6 @@ class Service(object):
|
||||||
def get_volumes_from_names(self):
|
def get_volumes_from_names(self):
|
||||||
return [s.name for s in self.volumes_from if isinstance(s, Service)]
|
return [s.name for s in self.volumes_from if isinstance(s, Service)]
|
||||||
|
|
||||||
def get_net_name(self):
|
|
||||||
if isinstance(self.net, Service):
|
|
||||||
return self.net.name
|
|
||||||
else:
|
|
||||||
return
|
|
||||||
|
|
||||||
def get_container_name(self, number, one_off=False):
|
def get_container_name(self, number, one_off=False):
|
||||||
# TODO: Implement issue #652 here
|
# TODO: Implement issue #652 here
|
||||||
return build_container_name(self.project, self.name, number, one_off)
|
return build_container_name(self.project, self.name, number, one_off)
|
||||||
|
@ -551,25 +545,6 @@ class Service(object):
|
||||||
|
|
||||||
return volumes_from
|
return volumes_from
|
||||||
|
|
||||||
def _get_net(self):
|
|
||||||
if not self.net:
|
|
||||||
return None
|
|
||||||
|
|
||||||
if isinstance(self.net, Service):
|
|
||||||
containers = self.net.containers()
|
|
||||||
if len(containers) > 0:
|
|
||||||
net = 'container:' + containers[0].id
|
|
||||||
else:
|
|
||||||
log.warning("Warning: Service %s is trying to use reuse the network stack "
|
|
||||||
"of another service that is not running." % (self.net.name))
|
|
||||||
net = None
|
|
||||||
elif isinstance(self.net, Container):
|
|
||||||
net = 'container:' + self.net.id
|
|
||||||
else:
|
|
||||||
net = self.net
|
|
||||||
|
|
||||||
return net
|
|
||||||
|
|
||||||
def _get_container_create_options(
|
def _get_container_create_options(
|
||||||
self,
|
self,
|
||||||
override_options,
|
override_options,
|
||||||
|
@ -690,7 +665,7 @@ class Service(object):
|
||||||
binds=options.get('binds'),
|
binds=options.get('binds'),
|
||||||
volumes_from=self._get_volumes_from(),
|
volumes_from=self._get_volumes_from(),
|
||||||
privileged=privileged,
|
privileged=privileged,
|
||||||
network_mode=self._get_net(),
|
network_mode=self.net.mode,
|
||||||
devices=devices,
|
devices=devices,
|
||||||
dns=dns,
|
dns=dns,
|
||||||
dns_search=dns_search,
|
dns_search=dns_search,
|
||||||
|
@ -785,6 +760,61 @@ class Service(object):
|
||||||
stream_output(output, sys.stdout)
|
stream_output(output, sys.stdout)
|
||||||
|
|
||||||
|
|
||||||
|
class Net(object):
|
||||||
|
"""A `standard` network mode (ex: host, bridge)"""
|
||||||
|
|
||||||
|
service_name = None
|
||||||
|
|
||||||
|
def __init__(self, net):
|
||||||
|
self.net = net
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self):
|
||||||
|
return self.net
|
||||||
|
|
||||||
|
mode = id
|
||||||
|
|
||||||
|
|
||||||
|
class ContainerNet(object):
|
||||||
|
"""A network mode that uses a containers network stack."""
|
||||||
|
|
||||||
|
service_name = None
|
||||||
|
|
||||||
|
def __init__(self, container):
|
||||||
|
self.container = container
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self):
|
||||||
|
return self.container.id
|
||||||
|
|
||||||
|
@property
|
||||||
|
def mode(self):
|
||||||
|
return 'container:' + self.container.id
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceNet(object):
|
||||||
|
"""A network mode that uses a service's network stack."""
|
||||||
|
|
||||||
|
def __init__(self, service):
|
||||||
|
self.service = service
|
||||||
|
|
||||||
|
@property
|
||||||
|
def id(self):
|
||||||
|
return self.service.name
|
||||||
|
|
||||||
|
service_name = id
|
||||||
|
|
||||||
|
@property
|
||||||
|
def mode(self):
|
||||||
|
containers = self.service.containers()
|
||||||
|
if containers:
|
||||||
|
return 'container:' + containers[0].id
|
||||||
|
|
||||||
|
log.warn("Warning: Service %s is trying to use reuse the network stack "
|
||||||
|
"of another service that is not running." % (self.id))
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
# Names
|
# Names
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -220,7 +220,7 @@ class ProjectTest(unittest.TestCase):
|
||||||
}
|
}
|
||||||
], self.mock_client)
|
], self.mock_client)
|
||||||
service = project.get_service('test')
|
service = project.get_service('test')
|
||||||
self.assertEqual(service._get_net(), None)
|
self.assertEqual(service.net.id, None)
|
||||||
self.assertNotIn('NetworkMode', service._get_container_host_config({}))
|
self.assertNotIn('NetworkMode', service._get_container_host_config({}))
|
||||||
|
|
||||||
def test_use_net_from_container(self):
|
def test_use_net_from_container(self):
|
||||||
|
@ -235,7 +235,7 @@ class ProjectTest(unittest.TestCase):
|
||||||
}
|
}
|
||||||
], self.mock_client)
|
], self.mock_client)
|
||||||
service = project.get_service('test')
|
service = project.get_service('test')
|
||||||
self.assertEqual(service._get_net(), 'container:' + container_id)
|
self.assertEqual(service.net.mode, 'container:' + container_id)
|
||||||
|
|
||||||
def test_use_net_from_service(self):
|
def test_use_net_from_service(self):
|
||||||
container_name = 'test_aaa_1'
|
container_name = 'test_aaa_1'
|
||||||
|
@ -260,7 +260,7 @@ class ProjectTest(unittest.TestCase):
|
||||||
], self.mock_client)
|
], self.mock_client)
|
||||||
|
|
||||||
service = project.get_service('test')
|
service = project.get_service('test')
|
||||||
self.assertEqual(service._get_net(), 'container:' + container_name)
|
self.assertEqual(service.net.mode, 'container:' + container_name)
|
||||||
|
|
||||||
def test_container_without_name(self):
|
def test_container_without_name(self):
|
||||||
self.mock_client.containers.return_value = [
|
self.mock_client.containers.return_value = [
|
||||||
|
|
|
@ -7,21 +7,27 @@ import mock
|
||||||
import docker
|
import docker
|
||||||
from docker.utils import LogConfig
|
from docker.utils import LogConfig
|
||||||
|
|
||||||
from compose.service import Service
|
from .. import mock
|
||||||
|
from .. import unittest
|
||||||
|
from compose.const import LABEL_CONFIG_HASH
|
||||||
|
from compose.const import LABEL_ONE_OFF
|
||||||
|
from compose.const import LABEL_PROJECT
|
||||||
|
from compose.const import LABEL_SERVICE
|
||||||
from compose.container import Container
|
from compose.container import Container
|
||||||
from compose.const import LABEL_SERVICE, LABEL_PROJECT, LABEL_ONE_OFF
|
from compose.service import ConfigError
|
||||||
from compose.service import (
|
from compose.service import ContainerNet
|
||||||
ConfigError,
|
from compose.service import NeedsBuildError
|
||||||
NeedsBuildError,
|
from compose.service import Net
|
||||||
NoSuchImageError,
|
from compose.service import NoSuchImageError
|
||||||
build_port_bindings,
|
from compose.service import Service
|
||||||
build_volume_binding,
|
from compose.service import ServiceNet
|
||||||
get_container_data_volumes,
|
from compose.service import build_port_bindings
|
||||||
merge_volume_bindings,
|
from compose.service import build_volume_binding
|
||||||
parse_repository_tag,
|
from compose.service import get_container_data_volumes
|
||||||
parse_volume_spec,
|
from compose.service import merge_volume_bindings
|
||||||
split_port,
|
from compose.service import parse_repository_tag
|
||||||
)
|
from compose.service import parse_volume_spec
|
||||||
|
from compose.service import split_port
|
||||||
|
|
||||||
|
|
||||||
class ServiceTest(unittest.TestCase):
|
class ServiceTest(unittest.TestCase):
|
||||||
|
@ -393,7 +399,7 @@ class ServiceTest(unittest.TestCase):
|
||||||
'foo',
|
'foo',
|
||||||
image='example.com/foo',
|
image='example.com/foo',
|
||||||
client=self.mock_client,
|
client=self.mock_client,
|
||||||
net=Service('other'),
|
net=ServiceNet(Service('other')),
|
||||||
links=[(Service('one'), 'one')],
|
links=[(Service('one'), 'one')],
|
||||||
volumes_from=[Service('two')])
|
volumes_from=[Service('two')])
|
||||||
|
|
||||||
|
@ -429,6 +435,49 @@ class ServiceTest(unittest.TestCase):
|
||||||
self.assertEqual(config_dict, expected)
|
self.assertEqual(config_dict, expected)
|
||||||
|
|
||||||
|
|
||||||
|
class NetTestCase(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_net(self):
|
||||||
|
net = Net('host')
|
||||||
|
self.assertEqual(net.id, 'host')
|
||||||
|
self.assertEqual(net.mode, 'host')
|
||||||
|
self.assertEqual(net.service_name, None)
|
||||||
|
|
||||||
|
def test_net_container(self):
|
||||||
|
container_id = 'abcd'
|
||||||
|
net = ContainerNet(Container(None, {'Id': container_id}))
|
||||||
|
self.assertEqual(net.id, container_id)
|
||||||
|
self.assertEqual(net.mode, 'container:' + container_id)
|
||||||
|
self.assertEqual(net.service_name, None)
|
||||||
|
|
||||||
|
def test_net_service(self):
|
||||||
|
container_id = 'bbbb'
|
||||||
|
service_name = 'web'
|
||||||
|
mock_client = mock.create_autospec(docker.Client)
|
||||||
|
mock_client.containers.return_value = [
|
||||||
|
{'Id': container_id, 'Name': container_id, 'Image': 'abcd'},
|
||||||
|
]
|
||||||
|
|
||||||
|
service = Service(name=service_name, client=mock_client)
|
||||||
|
net = ServiceNet(service)
|
||||||
|
|
||||||
|
self.assertEqual(net.id, service_name)
|
||||||
|
self.assertEqual(net.mode, 'container:' + container_id)
|
||||||
|
self.assertEqual(net.service_name, service_name)
|
||||||
|
|
||||||
|
def test_net_service_no_containers(self):
|
||||||
|
service_name = 'web'
|
||||||
|
mock_client = mock.create_autospec(docker.Client)
|
||||||
|
mock_client.containers.return_value = []
|
||||||
|
|
||||||
|
service = Service(name=service_name, client=mock_client)
|
||||||
|
net = ServiceNet(service)
|
||||||
|
|
||||||
|
self.assertEqual(net.id, service_name)
|
||||||
|
self.assertEqual(net.mode, None)
|
||||||
|
self.assertEqual(net.service_name, service_name)
|
||||||
|
|
||||||
|
|
||||||
def mock_get_image(images):
|
def mock_get_image(images):
|
||||||
if images:
|
if images:
|
||||||
return images[0]
|
return images[0]
|
||||||
|
|
Loading…
Reference in New Issue