Merge "Implement kuryr integration"
This commit is contained in:
commit
937d6df5df
@ -11,6 +11,7 @@ pbr!=2.1.0,>=2.0.0 # Apache-2.0
|
|||||||
pecan!=1.0.2,!=1.0.3,!=1.0.4,!=1.2,>=1.0.0 # BSD
|
pecan!=1.0.2,!=1.0.3,!=1.0.4,!=1.2,>=1.0.0 # BSD
|
||||||
python-etcd>=0.4.3 # MIT License
|
python-etcd>=0.4.3 # MIT License
|
||||||
python-glanceclient>=2.5.0 # Apache-2.0
|
python-glanceclient>=2.5.0 # Apache-2.0
|
||||||
|
python-neutronclient>=5.1.0 # Apache-2.0
|
||||||
python-novaclient>=7.1.0 # Apache-2.0
|
python-novaclient>=7.1.0 # Apache-2.0
|
||||||
oslo.i18n>=2.1.0 # Apache-2.0
|
oslo.i18n>=2.1.0 # Apache-2.0
|
||||||
oslo.log>=3.22.0 # Apache-2.0
|
oslo.log>=3.22.0 # Apache-2.0
|
||||||
|
@ -69,6 +69,9 @@ zun.image.driver =
|
|||||||
glance = zun.image.glance.driver:GlanceDriver
|
glance = zun.image.glance.driver:GlanceDriver
|
||||||
docker = zun.image.docker.driver:DockerDriver
|
docker = zun.image.docker.driver:DockerDriver
|
||||||
|
|
||||||
|
zun.network.driver =
|
||||||
|
kuryr = zun.network.kuryr_network:KuryrNetwork
|
||||||
|
|
||||||
tempest.test_plugins =
|
tempest.test_plugins =
|
||||||
zun_tests = zun.tests.tempest.plugin:ZunTempestPlugin
|
zun_tests = zun.tests.tempest.plugin:ZunTempestPlugin
|
||||||
|
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
from glanceclient import client as glanceclient
|
from glanceclient import client as glanceclient
|
||||||
|
from neutronclient.v2_0 import client as neutronclient
|
||||||
from novaclient import client as novaclient
|
from novaclient import client as novaclient
|
||||||
|
|
||||||
from zun.common import exception
|
from zun.common import exception
|
||||||
@ -28,6 +29,7 @@ class OpenStackClients(object):
|
|||||||
self._keystone = None
|
self._keystone = None
|
||||||
self._glance = None
|
self._glance = None
|
||||||
self._nova = None
|
self._nova = None
|
||||||
|
self._neutron = None
|
||||||
|
|
||||||
def url_for(self, **kwargs):
|
def url_for(self, **kwargs):
|
||||||
return self.keystone().session.get_endpoint(**kwargs)
|
return self.keystone().session.get_endpoint(**kwargs)
|
||||||
@ -93,3 +95,15 @@ class OpenStackClients(object):
|
|||||||
self._nova = novaclient.Client(nova_api_version, session=session)
|
self._nova = novaclient.Client(nova_api_version, session=session)
|
||||||
|
|
||||||
return self._nova
|
return self._nova
|
||||||
|
|
||||||
|
@exception.wrap_keystone_exception
|
||||||
|
def neutron(self):
|
||||||
|
if self._neutron:
|
||||||
|
return self._neutron
|
||||||
|
|
||||||
|
session = self.keystone().session
|
||||||
|
endpoint_type = self._get_client_option('neutron', 'endpoint_type')
|
||||||
|
self._neutron = neutronclient.Client(session=session,
|
||||||
|
endpoint_type=endpoint_type)
|
||||||
|
|
||||||
|
return self._neutron
|
||||||
|
@ -21,6 +21,8 @@ from zun.conf import database
|
|||||||
from zun.conf import docker
|
from zun.conf import docker
|
||||||
from zun.conf import glance_client
|
from zun.conf import glance_client
|
||||||
from zun.conf import image_driver
|
from zun.conf import image_driver
|
||||||
|
from zun.conf import network
|
||||||
|
from zun.conf import neutron_client
|
||||||
from zun.conf import nova_client
|
from zun.conf import nova_client
|
||||||
from zun.conf import path
|
from zun.conf import path
|
||||||
from zun.conf import profiler
|
from zun.conf import profiler
|
||||||
@ -45,3 +47,5 @@ services.register_opts(CONF)
|
|||||||
zun_client.register_opts(CONF)
|
zun_client.register_opts(CONF)
|
||||||
ssl.register_opts(CONF)
|
ssl.register_opts(CONF)
|
||||||
profiler.register_opts(CONF)
|
profiler.register_opts(CONF)
|
||||||
|
neutron_client.register_opts(CONF)
|
||||||
|
network.register_opts(CONF)
|
||||||
|
@ -20,7 +20,7 @@ docker_group = cfg.OptGroup(name='docker',
|
|||||||
|
|
||||||
docker_opts = [
|
docker_opts = [
|
||||||
cfg.StrOpt('docker_remote_api_version',
|
cfg.StrOpt('docker_remote_api_version',
|
||||||
default='1.22',
|
default='1.23',
|
||||||
help='Docker remote api version. Override it according to '
|
help='Docker remote api version. Override it according to '
|
||||||
'specific docker api version in your environment.'),
|
'specific docker api version in your environment.'),
|
||||||
cfg.IntOpt('default_timeout',
|
cfg.IntOpt('default_timeout',
|
||||||
|
34
zun/conf/network.py
Normal file
34
zun/conf/network.py
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
|
||||||
|
network_group = cfg.OptGroup(name='network',
|
||||||
|
title='Options for the container network')
|
||||||
|
|
||||||
|
network_opts = [
|
||||||
|
cfg.StrOpt('driver',
|
||||||
|
default='kuryr',
|
||||||
|
help='Defines which driver to use for container network.'),
|
||||||
|
]
|
||||||
|
|
||||||
|
ALL_OPTS = (network_opts)
|
||||||
|
|
||||||
|
|
||||||
|
def register_opts(conf):
|
||||||
|
conf.register_group(network_group)
|
||||||
|
conf.register_opts(ALL_OPTS, group=network_group)
|
||||||
|
|
||||||
|
|
||||||
|
def list_opts():
|
||||||
|
return {network_group: ALL_OPTS}
|
51
zun/conf/neutron_client.py
Normal file
51
zun/conf/neutron_client.py
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
from zun.common.i18n import _
|
||||||
|
|
||||||
|
|
||||||
|
neutron_group = cfg.OptGroup(name='neutron_client',
|
||||||
|
title='Options for the Neutron client')
|
||||||
|
|
||||||
|
common_security_opts = [
|
||||||
|
cfg.StrOpt('ca_file',
|
||||||
|
help=_('Optional CA cert file to use in SSL connections.')),
|
||||||
|
cfg.StrOpt('cert_file',
|
||||||
|
help=_('Optional PEM-formatted certificate chain file.')),
|
||||||
|
cfg.StrOpt('key_file',
|
||||||
|
help=_('Optional PEM-formatted file that contains the '
|
||||||
|
'private key.')),
|
||||||
|
cfg.BoolOpt('insecure',
|
||||||
|
default=False,
|
||||||
|
help=_("If set, then the server's certificate will not "
|
||||||
|
"be verified."))]
|
||||||
|
|
||||||
|
neutron_client_opts = [
|
||||||
|
cfg.StrOpt('endpoint_type',
|
||||||
|
default='publicURL',
|
||||||
|
help=_(
|
||||||
|
'Type of endpoint in Identity service catalog to use '
|
||||||
|
'for communication with the OpenStack service.'))]
|
||||||
|
|
||||||
|
|
||||||
|
ALL_OPTS = (neutron_client_opts + common_security_opts)
|
||||||
|
|
||||||
|
|
||||||
|
def register_opts(conf):
|
||||||
|
conf.register_group(neutron_group)
|
||||||
|
conf.register_opts(ALL_OPTS, group=neutron_group)
|
||||||
|
|
||||||
|
|
||||||
|
def list_opts():
|
||||||
|
return {neutron_group: ALL_OPTS}
|
@ -18,6 +18,7 @@ from docker import errors
|
|||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
from oslo_utils import timeutils
|
from oslo_utils import timeutils
|
||||||
|
|
||||||
|
from zun.common import clients
|
||||||
from zun.common import consts
|
from zun.common import consts
|
||||||
from zun.common import exception
|
from zun.common import exception
|
||||||
from zun.common.i18n import _
|
from zun.common.i18n import _
|
||||||
@ -27,8 +28,10 @@ from zun.common.utils import check_container_id
|
|||||||
import zun.conf
|
import zun.conf
|
||||||
from zun.container.docker import utils as docker_utils
|
from zun.container.docker import utils as docker_utils
|
||||||
from zun.container import driver
|
from zun.container import driver
|
||||||
|
from zun.network import network as zun_network
|
||||||
from zun import objects
|
from zun import objects
|
||||||
|
|
||||||
|
|
||||||
CONF = zun.conf.CONF
|
CONF = zun.conf.CONF
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
ATTACH_FLAG = "/attach/ws?logs=0&stream=1&stdin=1&stdout=1&stderr=1"
|
ATTACH_FLAG = "/attach/ws?logs=0&stream=1&stdin=1&stdout=1&stderr=1"
|
||||||
@ -457,17 +460,60 @@ class DockerDriver(driver.ContainerDriver):
|
|||||||
value = unicode(value)
|
value = unicode(value)
|
||||||
return value.encode('utf-8')
|
return value.encode('utf-8')
|
||||||
|
|
||||||
def create_sandbox(self, context, container, image='kubernetes/pause'):
|
def create_sandbox(self, context, container, image='kubernetes/pause',
|
||||||
|
networks=None):
|
||||||
with docker_utils.docker_client() as docker:
|
with docker_utils.docker_client() as docker:
|
||||||
|
network_api = zun_network.api(context=context, docker_api=docker)
|
||||||
|
if networks is None:
|
||||||
|
# Find an available neutron net and create docker network by
|
||||||
|
# wrapping the neutron net.
|
||||||
|
neutron_net = self._get_available_network(context)
|
||||||
|
network = self._get_or_create_docker_network(
|
||||||
|
context, network_api, neutron_net['id'])
|
||||||
|
networks = [network['Name']]
|
||||||
|
|
||||||
name = self.get_sandbox_name(container)
|
name = self.get_sandbox_name(container)
|
||||||
response = docker.create_container(image, name=name,
|
sandbox = docker.create_container(image, name=name,
|
||||||
hostname=name[:63])
|
hostname=name[:63])
|
||||||
sandbox_id = response['Id']
|
# Container connects to the bridge network by default so disconnect
|
||||||
docker.start(sandbox_id)
|
# the container from it before connecting it to neutron network.
|
||||||
return sandbox_id
|
# This avoids potential conflict between these two networks.
|
||||||
|
network_api.disconnect_container_from_network(sandbox, 'bridge')
|
||||||
|
for network in networks:
|
||||||
|
network_api.connect_container_to_network(sandbox, network)
|
||||||
|
docker.start(sandbox['Id'])
|
||||||
|
return sandbox['Id']
|
||||||
|
|
||||||
|
def _get_available_network(self, context):
|
||||||
|
neutron = clients.OpenStackClients(context).neutron()
|
||||||
|
search_opts = {'tenant_id': context.project_id, 'shared': False}
|
||||||
|
nets = neutron.list_networks(**search_opts).get('networks', [])
|
||||||
|
if not nets:
|
||||||
|
raise exception.ZunException(_(
|
||||||
|
"There is no neutron network available"))
|
||||||
|
nets.sort(key=lambda x: x['created_at'])
|
||||||
|
return nets[0]
|
||||||
|
|
||||||
|
def _get_or_create_docker_network(self, context, network_api,
|
||||||
|
neutron_net_id):
|
||||||
|
# Append project_id to the network name to avoid name collision
|
||||||
|
# across projects.
|
||||||
|
docker_net_name = neutron_net_id + '-' + context.project_id
|
||||||
|
docker_networks = network_api.list_networks(names=[docker_net_name])
|
||||||
|
if not docker_networks:
|
||||||
|
network_api.create_network(neutron_net_id=neutron_net_id,
|
||||||
|
name=docker_net_name)
|
||||||
|
docker_networks = network_api.list_networks(
|
||||||
|
names=[docker_net_name])
|
||||||
|
|
||||||
|
return docker_networks[0]
|
||||||
|
|
||||||
def delete_sandbox(self, context, sandbox_id):
|
def delete_sandbox(self, context, sandbox_id):
|
||||||
with docker_utils.docker_client() as docker:
|
with docker_utils.docker_client() as docker:
|
||||||
|
network_api = zun_network.api(context=context, docker_api=docker)
|
||||||
|
sandbox = docker.inspect_container(sandbox_id)
|
||||||
|
for network in sandbox["NetworkSettings"]["Networks"]:
|
||||||
|
network_api.disconnect_container_from_network(sandbox, network)
|
||||||
try:
|
try:
|
||||||
docker.remove_container(sandbox_id, force=True)
|
docker.remove_container(sandbox_id, force=True)
|
||||||
except errors.APIError as api_error:
|
except errors.APIError as api_error:
|
||||||
@ -501,15 +547,15 @@ class DockerDriver(driver.ContainerDriver):
|
|||||||
def get_addresses(self, context, container):
|
def get_addresses(self, context, container):
|
||||||
sandbox_id = self.get_sandbox_id(container)
|
sandbox_id = self.get_sandbox_id(container)
|
||||||
with docker_utils.docker_client() as docker:
|
with docker_utils.docker_client() as docker:
|
||||||
|
addresses = {}
|
||||||
response = docker.inspect_container(sandbox_id)
|
response = docker.inspect_container(sandbox_id)
|
||||||
addr = response["NetworkSettings"]["IPAddress"]
|
networks = response["NetworkSettings"]["Networks"]
|
||||||
addresses = {
|
for name, network in networks.items():
|
||||||
'default': [
|
addresses[name] = [
|
||||||
{
|
{'addr': network["IPAddress"]},
|
||||||
'addr': addr,
|
{'addr': network["GlobalIPv6Address"]},
|
||||||
},
|
]
|
||||||
],
|
|
||||||
}
|
|
||||||
return addresses
|
return addresses
|
||||||
|
|
||||||
def get_container_numbers(self):
|
def get_container_numbers(self):
|
||||||
|
0
zun/network/__init__.py
Normal file
0
zun/network/__init__.py
Normal file
162
zun/network/kuryr_network.py
Normal file
162
zun/network/kuryr_network.py
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import ipaddress
|
||||||
|
import six
|
||||||
|
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from zun.common import clients
|
||||||
|
from zun.common import exception
|
||||||
|
from zun.common.i18n import _
|
||||||
|
import zun.conf
|
||||||
|
from zun.network import network
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
CONF = zun.conf.CONF
|
||||||
|
|
||||||
|
|
||||||
|
class KuryrNetwork(network.Network):
|
||||||
|
|
||||||
|
def init(self, context, docker_api):
|
||||||
|
self.docker = docker_api
|
||||||
|
self.neutron = clients.OpenStackClients(context).neutron()
|
||||||
|
|
||||||
|
def create_network(self, name, neutron_net_id):
|
||||||
|
"""Create a docker network with Kuryr driver.
|
||||||
|
|
||||||
|
The docker network to be created will be based on the specified
|
||||||
|
neutron net. It is assumed that the neutron net will have one
|
||||||
|
or two subnets. If there are two subnets, it must be a ipv4
|
||||||
|
subnet and a ipv6 subnet and containers created from this network
|
||||||
|
will have both ipv4 and ipv6 addresses.
|
||||||
|
|
||||||
|
What this method does is finding the subnets under the specified
|
||||||
|
neutron net, retrieving the cidr, gateway, subnetpool of each
|
||||||
|
subnet, and compile the list of parameters for docker.create_network.
|
||||||
|
"""
|
||||||
|
# find a v4 and/or v6 subnet of the network
|
||||||
|
subnets = self.neutron.list_subnets(network_id=neutron_net_id)
|
||||||
|
subnets = subnets.get('subnets', [])
|
||||||
|
v4_subnet = self._get_subnet(subnets, ip_version=4)
|
||||||
|
v6_subnet = self._get_subnet(subnets, ip_version=6)
|
||||||
|
if not v4_subnet and not v6_subnet:
|
||||||
|
raise exception.ZunException(_(
|
||||||
|
"The Neutron network %s has no subnet") % neutron_net_id)
|
||||||
|
|
||||||
|
ipam_options = {
|
||||||
|
"Driver": "kuryr",
|
||||||
|
"Options": {},
|
||||||
|
"Config": []
|
||||||
|
}
|
||||||
|
if v4_subnet:
|
||||||
|
ipam_options["Options"]['neutron.pool.uuid'] = (
|
||||||
|
v4_subnet.get('subnetpool_id'))
|
||||||
|
ipam_options["Config"].append({
|
||||||
|
"Subnet": v4_subnet['cidr'],
|
||||||
|
"Gateway": v4_subnet['gateway_ip']
|
||||||
|
})
|
||||||
|
if v6_subnet:
|
||||||
|
ipam_options["Options"]['neutron.pool.v6.uuid'] = (
|
||||||
|
v6_subnet.get('subnetpool_id'))
|
||||||
|
ipam_options["Config"].append({
|
||||||
|
"Subnet": v6_subnet['cidr'],
|
||||||
|
"Gateway": v6_subnet['gateway_ip']
|
||||||
|
})
|
||||||
|
|
||||||
|
options = {
|
||||||
|
'neutron.net.uuid': neutron_net_id
|
||||||
|
}
|
||||||
|
if v4_subnet:
|
||||||
|
options['neutron.pool.uuid'] = v4_subnet.get('subnetpool_id')
|
||||||
|
if v6_subnet:
|
||||||
|
options['neutron.pool.v6.uuid'] = v6_subnet.get('subnetpool_id')
|
||||||
|
|
||||||
|
docker_network = self.docker.create_network(
|
||||||
|
name=name,
|
||||||
|
driver='kuryr',
|
||||||
|
enable_ipv6=True if v6_subnet else False,
|
||||||
|
options=options,
|
||||||
|
ipam=ipam_options)
|
||||||
|
|
||||||
|
return docker_network
|
||||||
|
|
||||||
|
def _get_subnet(self, subnets, ip_version):
|
||||||
|
subnets = [s for s in subnets if s['ip_version'] == ip_version]
|
||||||
|
if len(subnets) == 0:
|
||||||
|
return None
|
||||||
|
elif len(subnets) == 1:
|
||||||
|
return subnets[0]
|
||||||
|
else:
|
||||||
|
raise exception.ZunException(_(
|
||||||
|
"Multiple Neutron subnets exist with ip version %s") %
|
||||||
|
ip_version)
|
||||||
|
|
||||||
|
def delete_network(self, network_name):
|
||||||
|
self.docker.delete_network(network_name)
|
||||||
|
|
||||||
|
def inspect_network(self, network_name):
|
||||||
|
return self.docker.inspect_network(network_name)
|
||||||
|
|
||||||
|
def list_networks(self, **kwargs):
|
||||||
|
return self.docker.networks(**kwargs)
|
||||||
|
|
||||||
|
def connect_container_to_network(self, container, network_name):
|
||||||
|
"""Connect container to the network
|
||||||
|
|
||||||
|
This method will create a neutron port, retrieve the ip address(es)
|
||||||
|
of the port, and pass them to docker.connect_container_to_network.
|
||||||
|
"""
|
||||||
|
network = self.inspect_network(network_name)
|
||||||
|
neutron_net_id = network['Options']['neutron.net.uuid']
|
||||||
|
neutron_port = self.neutron.create_port({'port': {
|
||||||
|
'network_id': neutron_net_id,
|
||||||
|
}})
|
||||||
|
|
||||||
|
ipv4_address = None
|
||||||
|
ipv6_address = None
|
||||||
|
for fixed_ip in neutron_port['port']['fixed_ips']:
|
||||||
|
ip_address = fixed_ip['ip_address']
|
||||||
|
ip = ipaddress.ip_address(six.text_type(ip_address))
|
||||||
|
if ip.version == 4:
|
||||||
|
ipv4_address = ip_address
|
||||||
|
else:
|
||||||
|
ipv6_address = ip_address
|
||||||
|
|
||||||
|
kwargs = {}
|
||||||
|
if ipv4_address:
|
||||||
|
kwargs['ipv4_address'] = ipv4_address
|
||||||
|
if ipv6_address:
|
||||||
|
kwargs['ipv6_address'] = ipv6_address
|
||||||
|
self.docker.connect_container_to_network(
|
||||||
|
container['Id'], network_name, **kwargs)
|
||||||
|
|
||||||
|
def disconnect_container_from_network(self, container, network_name):
|
||||||
|
container_id = container['Id']
|
||||||
|
neutron_ports = None
|
||||||
|
# TODO(hongbin): Use objects instead of an ad hoc dict.
|
||||||
|
if "NetworkSettings" in container:
|
||||||
|
network = container["NetworkSettings"]["Networks"][network_name]
|
||||||
|
endpoint_id = network["EndpointID"]
|
||||||
|
# Kuryr set the port's device_id as endpoint_id so we leverge it
|
||||||
|
neutron_ports = self.neutron.list_ports(device_id=endpoint_id)
|
||||||
|
neutron_ports = neutron_ports.get('ports', [])
|
||||||
|
if not neutron_ports:
|
||||||
|
LOG.warning("Cannot find the neutron port that bind container "
|
||||||
|
"%s to network %s", container_id, network_name)
|
||||||
|
|
||||||
|
self.docker.disconnect_container_from_network(container_id,
|
||||||
|
network_name)
|
||||||
|
if neutron_ports:
|
||||||
|
port_id = neutron_ports[0]['id']
|
||||||
|
self.neutron.delete_port(port_id)
|
59
zun/network/network.py
Normal file
59
zun/network/network.py
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import abc
|
||||||
|
import six
|
||||||
|
|
||||||
|
from stevedore import driver
|
||||||
|
|
||||||
|
import zun.conf
|
||||||
|
|
||||||
|
|
||||||
|
CONF = zun.conf.CONF
|
||||||
|
|
||||||
|
|
||||||
|
def api(*args, **kwargs):
|
||||||
|
network_driver = CONF.network.driver
|
||||||
|
network_api = driver.DriverManager(
|
||||||
|
"zun.network.driver",
|
||||||
|
network_driver,
|
||||||
|
invoke_on_load=True).driver
|
||||||
|
|
||||||
|
network_api.init(*args, **kwargs)
|
||||||
|
return network_api
|
||||||
|
|
||||||
|
|
||||||
|
@six.add_metaclass(abc.ABCMeta)
|
||||||
|
class Network(object):
|
||||||
|
"""The base class that all Network classes should inherit from."""
|
||||||
|
|
||||||
|
def init(self, context, *args, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def create_network(self, *args, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def delete_network(self, network_name, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def inspect_network(self, network_name, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def list_networks(self, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def connect_container_to_network(self, container, network_name, **kwargs):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def disconnect_container_from_network(self, container, network_name,
|
||||||
|
**kwargs):
|
||||||
|
raise NotImplementedError()
|
@ -337,32 +337,35 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
self.mock_docker.resize.assert_called_once_with(
|
self.mock_docker.resize.assert_called_once_with(
|
||||||
mock_container.container_id, 100, 100)
|
mock_container.container_id, 100, 100)
|
||||||
|
|
||||||
|
@mock.patch('zun.network.kuryr_network.KuryrNetwork'
|
||||||
|
'.connect_container_to_network')
|
||||||
@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_connect):
|
||||||
sandbox_name = 'my_test_sandbox'
|
sandbox_name = 'my_test_sandbox'
|
||||||
mock_get_sandbox_name.return_value = sandbox_name
|
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()
|
|
||||||
mock_container = mock.MagicMock()
|
mock_container = mock.MagicMock()
|
||||||
result_sandbox_id = self.driver.create_sandbox(self.context,
|
with mock.patch.object(self.driver, '_get_available_network'):
|
||||||
mock_container,
|
result_sandbox_id = self.driver.create_sandbox(
|
||||||
'kubernetes/pause')
|
self.context, mock_container, 'kubernetes/pause')
|
||||||
self.mock_docker.create_container.assert_called_once_with(
|
self.mock_docker.create_container.assert_called_once_with(
|
||||||
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name)
|
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name)
|
||||||
self.assertEqual(result_sandbox_id, 'val1')
|
self.assertEqual(result_sandbox_id, 'val1')
|
||||||
|
|
||||||
|
@mock.patch('zun.network.kuryr_network.KuryrNetwork'
|
||||||
|
'.connect_container_to_network')
|
||||||
@mock.patch('zun.container.docker.driver.DockerDriver.get_sandbox_name')
|
@mock.patch('zun.container.docker.driver.DockerDriver.get_sandbox_name')
|
||||||
def test_create_sandbox_with_long_name(self, mock_get_sandbox_name):
|
def test_create_sandbox_with_long_name(self, mock_get_sandbox_name,
|
||||||
|
mock_connect):
|
||||||
sandbox_name = 'x' * 100
|
sandbox_name = 'x' * 100
|
||||||
mock_get_sandbox_name.return_value = sandbox_name
|
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()
|
|
||||||
mock_container = mock.MagicMock()
|
mock_container = mock.MagicMock()
|
||||||
result_sandbox_id = self.driver.create_sandbox(self.context,
|
with mock.patch.object(self.driver, '_get_available_network'):
|
||||||
mock_container,
|
result_sandbox_id = self.driver.create_sandbox(
|
||||||
'kubernetes/pause')
|
self.context, mock_container, 'kubernetes/pause')
|
||||||
self.mock_docker.create_container.assert_called_once_with(
|
self.mock_docker.create_container.assert_called_once_with(
|
||||||
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name[:63])
|
'kubernetes/pause', name=sandbox_name, hostname=sandbox_name[:63])
|
||||||
self.assertEqual(result_sandbox_id, 'val1')
|
self.assertEqual(result_sandbox_id, 'val1')
|
||||||
@ -417,14 +420,18 @@ class TestDockerDriver(base.DriverTestCase):
|
|||||||
def test_get_addresses(self, mock_get_sandbox_id):
|
def test_get_addresses(self, mock_get_sandbox_id):
|
||||||
mock_get_sandbox_id.return_value = 'test_sandbox_id'
|
mock_get_sandbox_id.return_value = 'test_sandbox_id'
|
||||||
self.mock_docker.inspect_container = mock.Mock(
|
self.mock_docker.inspect_container = mock.Mock(
|
||||||
return_value={'NetworkSettings': {'IPAddress': '127.0.0.1'}})
|
return_value={'NetworkSettings': {'Networks': {'default': {
|
||||||
|
'IPAddress': '127.0.0.1',
|
||||||
|
'GlobalIPv6Address': 'fe80::4',
|
||||||
|
}}}})
|
||||||
mock_container = mock.MagicMock()
|
mock_container = mock.MagicMock()
|
||||||
result_addresses = self.driver.get_addresses(self.context,
|
result_addresses = self.driver.get_addresses(self.context,
|
||||||
mock_container)
|
mock_container)
|
||||||
self.mock_docker.inspect_container.assert_called_once_with(
|
self.mock_docker.inspect_container.assert_called_once_with(
|
||||||
'test_sandbox_id')
|
'test_sandbox_id')
|
||||||
self.assertEqual(result_addresses,
|
expected_addresses = {'default': [
|
||||||
{'default': [{'addr': '127.0.0.1', }, ], })
|
{'addr': '127.0.0.1'}, {'addr': 'fe80::4'}]}
|
||||||
|
self.assertEqual(expected_addresses, result_addresses)
|
||||||
|
|
||||||
def test_execute_resize(self):
|
def test_execute_resize(self):
|
||||||
self.mock_docker.exec_resize = mock.Mock()
|
self.mock_docker.exec_resize = mock.Mock()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user