Remove passing hostname and ports to Docker
These two options are conflicting with network mode 'none'. Let's skip it on creating containers. Instead, we treat these properties as outputted attributes. In particular, after a docker container is created, we inspect the container and populate these fields. Depends-On: I063e91e22c891d86ac1733222bb6a8c48bb68a70 Change-Id: I931d5e5d4425b455fe3a825aceb5199fcac15fcb Related-Bug: #1650964
This commit is contained in:
parent
b60c861094
commit
179af2c3e1
@ -490,7 +490,7 @@ class DockerDriver(driver.ComputeDriver):
|
|||||||
flavor=None):
|
flavor=None):
|
||||||
image_name = self._get_image_name(context, instance, image_meta)
|
image_name = self._get_image_name(context, instance, image_meta)
|
||||||
args = {
|
args = {
|
||||||
'hostname': instance['display_name'],
|
'hostname': instance['display_name'][:63],
|
||||||
'mem_limit': self._get_memory_limit_bytes(instance),
|
'mem_limit': self._get_memory_limit_bytes(instance),
|
||||||
'cpu_shares': self._get_cpu_shares(instance),
|
'cpu_shares': self._get_cpu_shares(instance),
|
||||||
'network_disabled': True,
|
'network_disabled': True,
|
||||||
|
@ -19,11 +19,9 @@ _container_properties = {
|
|||||||
'cpu': parameter_types.cpu,
|
'cpu': parameter_types.cpu,
|
||||||
'memory': parameter_types.memory,
|
'memory': parameter_types.memory,
|
||||||
'workdir': parameter_types.workdir,
|
'workdir': parameter_types.workdir,
|
||||||
'hostname': parameter_types.hostname,
|
|
||||||
'image_pull_policy': parameter_types.image_pull_policy,
|
'image_pull_policy': parameter_types.image_pull_policy,
|
||||||
'labels': parameter_types.labels,
|
'labels': parameter_types.labels,
|
||||||
'environment': parameter_types.environment,
|
'environment': parameter_types.environment,
|
||||||
'ports': parameter_types.ports,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
container_create = {
|
container_create = {
|
||||||
|
@ -61,12 +61,6 @@ workdir = {
|
|||||||
'type': ['string', 'null']
|
'type': ['string', 'null']
|
||||||
}
|
}
|
||||||
|
|
||||||
hostname = {
|
|
||||||
'type': ['string', 'null'],
|
|
||||||
'minLength': 0,
|
|
||||||
'maxLength': 255
|
|
||||||
}
|
|
||||||
|
|
||||||
image_pull_policy = {
|
image_pull_policy = {
|
||||||
'type': ['string', 'null'],
|
'type': ['string', 'null'],
|
||||||
'enum': ['never', 'always', 'ifnotpresent', None]
|
'enum': ['never', 'always', 'ifnotpresent', None]
|
||||||
@ -80,16 +74,6 @@ environment = {
|
|||||||
'type': ['object', 'null']
|
'type': ['object', 'null']
|
||||||
}
|
}
|
||||||
|
|
||||||
ports = {
|
|
||||||
'type': ['array', 'null'],
|
|
||||||
'items': {
|
|
||||||
'type': ['integer', 'string'],
|
|
||||||
'pattern': '^[0-9]*$',
|
|
||||||
'minimum': 1, 'maximum': 65535,
|
|
||||||
'minLength': 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
image_id = {
|
image_id = {
|
||||||
'type': ['string', 'null'],
|
'type': ['string', 'null'],
|
||||||
'minLength': 2,
|
'minLength': 2,
|
||||||
|
@ -68,11 +68,9 @@ class DockerDriver(driver.ContainerDriver):
|
|||||||
|
|
||||||
kwargs = {
|
kwargs = {
|
||||||
'name': self.get_container_name(container),
|
'name': self.get_container_name(container),
|
||||||
'hostname': container.hostname,
|
|
||||||
'command': container.command,
|
'command': container.command,
|
||||||
'environment': container.environment,
|
'environment': container.environment,
|
||||||
'working_dir': container.workdir,
|
'working_dir': container.workdir,
|
||||||
'ports': container.ports,
|
|
||||||
'labels': container.labels,
|
'labels': container.labels,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,6 +138,19 @@ class DockerDriver(driver.ContainerDriver):
|
|||||||
else:
|
else:
|
||||||
container.status = fields.ContainerStatus.STOPPED
|
container.status = fields.ContainerStatus.STOPPED
|
||||||
|
|
||||||
|
config = response.get('Config')
|
||||||
|
if config:
|
||||||
|
# populate hostname
|
||||||
|
container.hostname = config.get('Hostname')
|
||||||
|
# populate ports
|
||||||
|
ports = []
|
||||||
|
exposed_ports = config.get('ExposedPorts')
|
||||||
|
if exposed_ports:
|
||||||
|
for key in exposed_ports:
|
||||||
|
port = key.split('/')[0]
|
||||||
|
ports.append(int(port))
|
||||||
|
container.ports = ports
|
||||||
|
|
||||||
@check_container_id
|
@check_container_id
|
||||||
def reboot(self, container, timeout):
|
def reboot(self, container, timeout):
|
||||||
with docker_utils.docker_client() as docker:
|
with docker_utils.docker_client() as docker:
|
||||||
@ -222,7 +233,8 @@ class DockerDriver(driver.ContainerDriver):
|
|||||||
def create_sandbox(self, context, container, image='kubernetes/pause'):
|
def create_sandbox(self, context, container, image='kubernetes/pause'):
|
||||||
with docker_utils.docker_client() as docker:
|
with docker_utils.docker_client() as docker:
|
||||||
name = self.get_sandbox_name(container)
|
name = self.get_sandbox_name(container)
|
||||||
response = docker.create_container(image, name=name)
|
response = docker.create_container(image, name=name,
|
||||||
|
hostname=name[:63])
|
||||||
sandbox_id = response['Id']
|
sandbox_id = response['Id']
|
||||||
docker.start(sandbox_id)
|
docker.start(sandbox_id)
|
||||||
return sandbox_id
|
return sandbox_id
|
||||||
|
@ -140,7 +140,7 @@ class Container(Base):
|
|||||||
environment = Column(JSONEncodedDict)
|
environment = Column(JSONEncodedDict)
|
||||||
workdir = Column(String(255))
|
workdir = Column(String(255))
|
||||||
ports = Column(JSONEncodedList)
|
ports = Column(JSONEncodedList)
|
||||||
hostname = Column(String(255))
|
hostname = Column(String(63))
|
||||||
labels = Column(JSONEncodedDict)
|
labels = Column(JSONEncodedDict)
|
||||||
meta = Column(JSONEncodedDict)
|
meta = Column(JSONEncodedDict)
|
||||||
addresses = Column(JSONEncodedDict)
|
addresses = Column(JSONEncodedDict)
|
||||||
|
@ -25,9 +25,7 @@ CONTAINER_CREATE = {
|
|||||||
'cpu': parameter_types.cpu,
|
'cpu': parameter_types.cpu,
|
||||||
'memory': parameter_types.memory,
|
'memory': parameter_types.memory,
|
||||||
'workdir': parameter_types.workdir,
|
'workdir': parameter_types.workdir,
|
||||||
'hostname': parameter_types.hostname,
|
|
||||||
'image_pull_policy': parameter_types.image_pull_policy,
|
'image_pull_policy': parameter_types.image_pull_policy,
|
||||||
'ports': parameter_types.ports,
|
|
||||||
'labels': parameter_types.labels,
|
'labels': parameter_types.labels,
|
||||||
'environment': parameter_types.environment
|
'environment': parameter_types.environment
|
||||||
},
|
},
|
||||||
@ -45,9 +43,7 @@ class TestSchemaValidations(base.BaseTestCase):
|
|||||||
request_to_validate = {'name': 'test1', 'image': 'nginx',
|
request_to_validate = {'name': 'test1', 'image': 'nginx',
|
||||||
'command': '/bin/sh', 'cpu': 1.0,
|
'command': '/bin/sh', 'cpu': 1.0,
|
||||||
'memory': '5', 'workdir': '/workdir',
|
'memory': '5', 'workdir': '/workdir',
|
||||||
'hostname': 'container1',
|
|
||||||
'image_pull_policy': 'never',
|
'image_pull_policy': 'never',
|
||||||
'ports': ['123', '1', 1, '65535', 65535, 123],
|
|
||||||
'labels': {'abc': 12, 'bcd': 'xyz'},
|
'labels': {'abc': 12, 'bcd': 'xyz'},
|
||||||
'environment': {'xyz': 'pqr', 'pqr': 2}}
|
'environment': {'xyz': 'pqr', 'pqr': 2}}
|
||||||
self.schema_validator.validate(request_to_validate)
|
self.schema_validator.validate(request_to_validate)
|
||||||
@ -56,9 +52,7 @@ class TestSchemaValidations(base.BaseTestCase):
|
|||||||
request_to_validate = {'name': None, 'image': 'nginx',
|
request_to_validate = {'name': None, 'image': 'nginx',
|
||||||
'command': None, 'cpu': None,
|
'command': None, 'cpu': None,
|
||||||
'memory': None, 'workdir': None,
|
'memory': None, 'workdir': None,
|
||||||
'hostname': None,
|
|
||||||
'image_pull_policy': None,
|
'image_pull_policy': None,
|
||||||
'ports': None,
|
|
||||||
'labels': None,
|
'labels': None,
|
||||||
'environment': None}
|
'environment': None}
|
||||||
self.schema_validator.validate(request_to_validate)
|
self.schema_validator.validate(request_to_validate)
|
||||||
@ -99,20 +93,6 @@ class TestSchemaValidations(base.BaseTestCase):
|
|||||||
request_to_validate = {'memory': value, 'image': 'nginx'}
|
request_to_validate = {'memory': value, 'image': 'nginx'}
|
||||||
self.schema_validator.validate(request_to_validate)
|
self.schema_validator.validate(request_to_validate)
|
||||||
|
|
||||||
def test_create_schema_invalid_ports(self):
|
|
||||||
invalid_ports = [56, 0, 1, 65535, "", [0, '0', 65536, '65536',
|
|
||||||
"", 'x', " "]]
|
|
||||||
for value in invalid_ports:
|
|
||||||
request_to_validate = {'image': 'nginx', 'ports': value}
|
|
||||||
# TODO(pksingh): if value inside port array is not valid,
|
|
||||||
# message like below is raised:
|
|
||||||
# 'Invalid input for field '2'. Value: '65536'.
|
|
||||||
# 65536 is greater than the maximum of 65535'
|
|
||||||
# I think field '2' in message is not informative.
|
|
||||||
with self.assertRaisesRegexp(exception.SchemaValidationError,
|
|
||||||
"Invalid input for field"):
|
|
||||||
self.schema_validator.validate(request_to_validate)
|
|
||||||
|
|
||||||
def test_create_schema_cpu(self):
|
def test_create_schema_cpu(self):
|
||||||
valid_cpu = [4, 5, '4', '5', '0.5', '123.50', 0.5, 123.50, None]
|
valid_cpu = [4, 5, '4', '5', '0.5', '123.50', 0.5, 123.50, None]
|
||||||
invalid_cpu = ['12a', 'abc', '0.a', 'a.90', "", " ", " 0.9 "]
|
invalid_cpu = ['12a', 'abc', '0.a', 'a.90', "", " ", " 0.9 "]
|
||||||
|
@ -75,11 +75,9 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
|
|
||||||
kwargs = {
|
kwargs = {
|
||||||
'name': 'zun-ea8e2a25-2901-438d-8157-de7ffd68d051',
|
'name': 'zun-ea8e2a25-2901-438d-8157-de7ffd68d051',
|
||||||
'hostname': 'testhost',
|
|
||||||
'command': 'fake_command',
|
'command': 'fake_command',
|
||||||
'environment': {'key1': 'val1', 'key2': 'val2'},
|
'environment': {'key1': 'val1', 'key2': 'val2'},
|
||||||
'working_dir': '/home/ubuntu',
|
'working_dir': '/home/ubuntu',
|
||||||
'ports': [80, 443],
|
|
||||||
'labels': {'key1': 'val1', 'key2': 'val2'},
|
'labels': {'key1': 'val1', 'key2': 'val2'},
|
||||||
'host_config': {'Id1': 'val1', 'key2': 'val2'},
|
'host_config': {'Id1': 'val1', 'key2': 'val2'},
|
||||||
}
|
}
|
||||||
@ -113,11 +111,9 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
**host_config)
|
**host_config)
|
||||||
|
|
||||||
kwargs = {
|
kwargs = {
|
||||||
'hostname': 'testhost',
|
|
||||||
'command': 'fake_command',
|
'command': 'fake_command',
|
||||||
'environment': {'key1': 'val1', 'key2': 'val2'},
|
'environment': {'key1': 'val1', 'key2': 'val2'},
|
||||||
'working_dir': '/home/ubuntu',
|
'working_dir': '/home/ubuntu',
|
||||||
'ports': [80, 443],
|
|
||||||
'labels': {'key1': 'val1', 'key2': 'val2'},
|
'labels': {'key1': 'val1', 'key2': 'val2'},
|
||||||
'host_config': {'Id1': 'val1', 'key2': 'val2'},
|
'host_config': {'Id1': 'val1', 'key2': 'val2'},
|
||||||
'name': 'zun-ea8e2a25-2901-438d-8157-de7ffd68d051',
|
'name': 'zun-ea8e2a25-2901-438d-8157-de7ffd68d051',
|
||||||
@ -165,7 +161,7 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
self.mock_docker.list_instances.assert_called_once_with()
|
self.mock_docker.list_instances.assert_called_once_with()
|
||||||
|
|
||||||
def test_show_success(self):
|
def test_show_success(self):
|
||||||
self.mock_docker.inspect_container = mock.Mock()
|
self.mock_docker.inspect_container = mock.Mock(return_value={})
|
||||||
db_container = db_utils.create_test_container(context=self.context)
|
db_container = db_utils.create_test_container(context=self.context)
|
||||||
self.driver.show(db_container)
|
self.driver.show(db_container)
|
||||||
self.mock_docker.inspect_container.assert_called_once_with(
|
self.mock_docker.inspect_container.assert_called_once_with(
|
||||||
@ -266,7 +262,7 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
|
|
||||||
def test_kill_successful_signal_is_none(self):
|
def test_kill_successful_signal_is_none(self):
|
||||||
self.mock_docker.kill = mock.Mock()
|
self.mock_docker.kill = mock.Mock()
|
||||||
self.mock_docker.inspect_container = mock.Mock()
|
self.mock_docker.inspect_container = mock.Mock(return_value={})
|
||||||
db_container = db_utils.create_test_container(context=self.context)
|
db_container = db_utils.create_test_container(context=self.context)
|
||||||
self.driver.kill(db_container, signal=None)
|
self.driver.kill(db_container, signal=None)
|
||||||
self.mock_docker.kill.assert_called_once_with(
|
self.mock_docker.kill.assert_called_once_with(
|
||||||
@ -276,7 +272,7 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
|
|
||||||
def test_kill_successful_signal_is_not_none(self):
|
def test_kill_successful_signal_is_not_none(self):
|
||||||
self.mock_docker.kill = mock.Mock()
|
self.mock_docker.kill = mock.Mock()
|
||||||
self.mock_docker.inspect_container = mock.Mock()
|
self.mock_docker.inspect_container = mock.Mock(return_value={})
|
||||||
db_container = db_utils.create_test_container(context=self.context)
|
db_container = db_utils.create_test_container(context=self.context)
|
||||||
self.driver.kill(db_container, signal='test')
|
self.driver.kill(db_container, signal='test')
|
||||||
self.mock_docker.kill.assert_called_once_with(
|
self.mock_docker.kill.assert_called_once_with(
|
||||||
@ -318,7 +314,8 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
|
|
||||||
@mock.patch('zun.container.docker.driver.DockerDriver.get_sandbox_name')
|
@mock.patch('zun.container.docker.driver.DockerDriver.get_sandbox_name')
|
||||||
def test_create_sandbox(self, mock_get_sandbox_name):
|
def test_create_sandbox(self, mock_get_sandbox_name):
|
||||||
mock_get_sandbox_name.return_value = 'my_test_sandbox'
|
sandbox_name = 'my_test_sandbox'
|
||||||
|
mock_get_sandbox_name.return_value = sandbox_name
|
||||||
self.mock_docker.create_container = mock.Mock(
|
self.mock_docker.create_container = mock.Mock(
|
||||||
return_value={'Id': 'val1', 'key1': 'val2'})
|
return_value={'Id': 'val1', 'key1': 'val2'})
|
||||||
self.mock_docker.start()
|
self.mock_docker.start()
|
||||||
@ -327,7 +324,22 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
db_container,
|
db_container,
|
||||||
'kubernetes/pause')
|
'kubernetes/pause')
|
||||||
self.mock_docker.create_container.assert_called_once_with(
|
self.mock_docker.create_container.assert_called_once_with(
|
||||||
'kubernetes/pause', name='my_test_sandbox')
|
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name)
|
||||||
|
self.assertEqual(result_sandbox_id, 'val1')
|
||||||
|
|
||||||
|
@mock.patch('zun.container.docker.driver.DockerDriver.get_sandbox_name')
|
||||||
|
def test_create_sandbox_with_long_name(self, mock_get_sandbox_name):
|
||||||
|
sandbox_name = 'x' * 100
|
||||||
|
mock_get_sandbox_name.return_value = sandbox_name
|
||||||
|
self.mock_docker.create_container = mock.Mock(
|
||||||
|
return_value={'Id': 'val1', 'key1': 'val2'})
|
||||||
|
self.mock_docker.start()
|
||||||
|
db_container = db_utils.create_test_container(context=self.context)
|
||||||
|
result_sandbox_id = self.driver.create_sandbox(self.context,
|
||||||
|
db_container,
|
||||||
|
'kubernetes/pause')
|
||||||
|
self.mock_docker.create_container.assert_called_once_with(
|
||||||
|
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name[:63])
|
||||||
self.assertEqual(result_sandbox_id, 'val1')
|
self.assertEqual(result_sandbox_id, 'val1')
|
||||||
|
|
||||||
def test_delete_sandbox(self):
|
def test_delete_sandbox(self):
|
||||||
|
Loading…
Reference in New Issue
Block a user