Move networking from share manager to driver interface

Several things are implemented:
- Allocation/deallocation now handled by drivers instead of share manager.
It provides flexibility for drivers.
- Network plugin interface was updated to support new approach for
configuration options setting.

Config opts for network plugin can be defined via three sources:
a) using separate config group
b) using config group of back end
c) using DEFAULT config group
Variants (a) and (b) are mutually exclusive, there are switched by
opt 'network_config_group' that belongs to share driver interface.

Implements bp network-helper

Change-Id: I3b05369f01777675c1b834af5ee076d8b7219a0f
This commit is contained in:
Valeriy Ponomaryov 2014-12-12 22:25:42 +02:00
parent 1a2ec3da9b
commit 36db3f3917
13 changed files with 613 additions and 541 deletions

View File

@ -1,4 +1,5 @@
# Copyright 2013 Openstack Foundation # Copyright 2013 Openstack Foundation
# Copyright 2014 Mirantis Inc.
# All Rights Reserved. # All Rights Reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # Licensed under the Apache License, Version 2.0 (the "License"); you may
@ -15,26 +16,42 @@
import abc import abc
from oslo.config import cfg from oslo.config import cfg
import oslo.utils.importutils from oslo.utils import importutils
from manila.db import base as db_base
network_opts = [ network_opts = [
cfg.StrOpt('network_api_class', cfg.StrOpt(
default='manila.network.neutron.' 'network_api_class',
'neutron_network_plugin.NeutronNetworkPlugin', default='manila.network.neutron.'
help='The full class name of the Networking API class to use.'), 'neutron_network_plugin.NeutronNetworkPlugin',
deprecated_group='DEFAULT',
help='The full class name of the Networking API class to use.'),
] ]
cfg.CONF.register_opts(network_opts) CONF = cfg.CONF
def API(): def API(config_group_name=None):
importutils = oslo.utils.importutils """Selects class and config group of network plugin.
network_api_class = cfg.CONF.network_api_class
:param config_group_name: name of config group to be used for
registration of networking opts.
:returns: instance of networking plugin class
"""
CONF.register_opts(network_opts, group=config_group_name)
if config_group_name:
network_api_class = getattr(CONF, config_group_name).network_api_class
else:
network_api_class = CONF.network_api_class
cls = importutils.import_class(network_api_class) cls = importutils.import_class(network_api_class)
return cls() return cls(config_group_name=config_group_name)
class NetworkBaseAPI(object): class NetworkBaseAPI(db_base.Base):
def __init__(self, db_driver=None):
super(NetworkBaseAPI, self).__init__(db_driver=db_driver)
@abc.abstractmethod @abc.abstractmethod
def allocate_network(self, context, network_id, subnet_id, **kwargs): def allocate_network(self, context, network_id, subnet_id, **kwargs):

View File

@ -1,51 +0,0 @@
# Copyright 2012 OpenStack Foundation
# All Rights Reserved
#
# 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 neutronclient.common import exceptions
from neutronclient.v2_0 import client as clientv20
from oslo.config import cfg
CONF = cfg.CONF
def _get_client(token=None):
params = {
'endpoint_url': CONF.neutron_url,
'timeout': CONF.neutron_url_timeout,
'insecure': CONF.neutron_api_insecure,
'ca_cert': CONF.neutron_ca_certificates_file,
}
if token:
params['token'] = token
params['auth_strategy'] = None
else:
params['username'] = CONF.neutron_admin_username
params['tenant_name'] = CONF.neutron_admin_tenant_name
params['password'] = CONF.neutron_admin_password
params['auth_url'] = CONF.neutron_admin_auth_url
params['auth_strategy'] = CONF.neutron_auth_strategy
return clientv20.Client(**params)
def get_client(context):
if context.is_admin:
token = None
elif not context.auth_token:
raise exceptions.Unauthorized()
else:
token = context.auth_token
return _get_client(token=token)

View File

@ -1,4 +1,5 @@
# Copyright 2013 OpenStack Foundation # Copyright 2013 OpenStack Foundation
# Copyright 2014 Mirantis Inc.
# All Rights Reserved # All Rights Reserved
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # Licensed under the Apache License, Version 2.0 (the "License"); you may
@ -14,77 +15,111 @@
# under the License. # under the License.
from neutronclient.common import exceptions as neutron_client_exc from neutronclient.common import exceptions as neutron_client_exc
from neutronclient.v2_0 import client as clientv20
from oslo.config import cfg from oslo.config import cfg
from manila import context from manila import context
from manila.db import base from manila.db import base
from manila import exception from manila import exception
from manila.i18n import _LE from manila.i18n import _LE
from manila.network import neutron
from manila.network.neutron import constants as neutron_constants from manila.network.neutron import constants as neutron_constants
from manila.openstack.common import log as logging from manila.openstack.common import log as logging
neutron_opts = [ neutron_opts = [
cfg.StrOpt('neutron_url', cfg.StrOpt(
default='http://127.0.0.1:9696', 'neutron_url',
deprecated_name='quantum_url', default='http://127.0.0.1:9696',
help='URL for connecting to neutron.'), deprecated_group='DEFAULT',
cfg.IntOpt('neutron_url_timeout', help='URL for connecting to neutron.'),
default=30, cfg.IntOpt(
deprecated_name='quantum_url_timeout', 'neutron_url_timeout',
help='Timeout value for connecting to neutron in seconds.'), default=30,
cfg.StrOpt('neutron_admin_username', deprecated_group='DEFAULT',
default='neutron', help='Timeout value for connecting to neutron in seconds.'),
deprecated_name='quantum_admin_username', cfg.StrOpt(
help='Username for connecting to neutron in admin context.'), 'neutron_admin_username',
cfg.StrOpt('neutron_admin_password', default='neutron',
deprecated_name='quantum_admin_password', deprecated_group='DEFAULT',
help='Password for connecting to neutron in admin context.', help='Username for connecting to neutron in admin context.'),
secret=True), cfg.StrOpt(
cfg.StrOpt('neutron_admin_tenant_name', 'neutron_admin_password',
default='service', help='Password for connecting to neutron in admin context.',
deprecated_name='quantum_admin_tenant_name', deprecated_group='DEFAULT',
help='Tenant name for connecting to neutron in admin context.'), secret=True),
cfg.StrOpt('neutron_region_name', cfg.StrOpt(
deprecated_name='quantum_region_name', 'neutron_admin_tenant_name',
help='Region name for connecting to neutron in admin context.'), default='service',
cfg.StrOpt('neutron_admin_auth_url', deprecated_group='DEFAULT',
deprecated_name='quantum_admin_auth_url', help='Tenant name for connecting to neutron in admin context.'),
default='http://localhost:5000/v2.0', cfg.StrOpt(
help='Auth URL for connecting to neutron in admin context.'), 'neutron_admin_auth_url',
cfg.BoolOpt('neutron_api_insecure', default='http://localhost:5000/v2.0',
default=False, deprecated_group='DEFAULT',
deprecated_name='quantum_api_insecure', help='Auth URL for connecting to neutron in admin context.'),
help='If set, ignore any SSL validation issues.'), cfg.BoolOpt(
cfg.StrOpt('neutron_auth_strategy', 'neutron_api_insecure',
default='keystone', default=False,
deprecated_name='quantum_auth_strategy', deprecated_group='DEFAULT',
help='Auth strategy for connecting to ' help='If set, ignore any SSL validation issues.'),
'neutron in admin context.'), cfg.StrOpt(
# TODO(berrange) temporary hack until Neutron can pass over the 'neutron_auth_strategy',
# name of the OVS bridge it is configured with default='keystone',
cfg.StrOpt('neutron_ovs_bridge', deprecated_group='DEFAULT',
default='br-int', help='Auth strategy for connecting to neutron in admin context.'),
deprecated_name='quantum_ovs_bridge', cfg.StrOpt(
help='Name of integration bridge used by Open vSwitch.'), 'neutron_ca_certificates_file',
cfg.StrOpt('neutron_ca_certificates_file', deprecated_group='DEFAULT',
help='Location of CA certificates file to use for ' help='Location of CA certificates file to use for '
'neutron client requests.'), 'neutron client requests.'),
] ]
CONF = cfg.CONF CONF = cfg.CONF
CONF.register_opts(neutron_opts)
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
class API(base.Base): class API(base.Base):
"""API for interacting with the neutron 2.x API.""" """API for interacting with the neutron 2.x API.
def __init__(self): :param configuration: instance of config or config group.
"""
def __init__(self, config_group_name=None):
super(API, self).__init__() super(API, self).__init__()
if config_group_name is None:
config_group_name = 'DEFAULT'
CONF.register_opts(neutron_opts, group=config_group_name)
self.configuration = getattr(CONF, config_group_name, CONF)
self.last_neutron_extension_sync = None self.last_neutron_extension_sync = None
self.extensions = {} self.extensions = {}
self.client = neutron.get_client(context.get_admin_context()) self.client = self.get_client(context.get_admin_context())
def _get_client(self, token=None):
params = {
'endpoint_url': self.configuration.neutron_url,
'timeout': self.configuration.neutron_url_timeout,
'insecure': self.configuration.neutron_api_insecure,
'ca_cert': self.configuration.neutron_ca_certificates_file,
}
if token:
params['token'] = token
params['auth_strategy'] = None
else:
params['username'] = self.configuration.neutron_admin_username
params['tenant_name'] = (
self.configuration.neutron_admin_tenant_name)
params['password'] = self.configuration.neutron_admin_password
params['auth_url'] = self.configuration.neutron_admin_auth_url
params['auth_strategy'] = self.configuration.neutron_auth_strategy
return clientv20.Client(**params)
def get_client(self, context):
if context.is_admin:
token = None
elif not context.auth_token:
raise neutron_client_exc.Unauthorized()
else:
token = context.auth_token
return self._get_client(token=token)
@property @property
def admin_tenant_id(self): def admin_tenant_id(self):

View File

@ -13,25 +13,22 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from oslo.config import cfg
from manila.common import constants from manila.common import constants
from manila.db import base as db_base
from manila import exception from manila import exception
from manila import network as manila_network from manila import network
from manila.network.neutron import api as neutron_api from manila.network.neutron import api as neutron_api
from manila.network.neutron import constants as neutron_constants from manila.network.neutron import constants as neutron_constants
from manila.openstack.common import log as logging from manila.openstack.common import log as logging
CONF = cfg.CONF
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
class NeutronNetworkPlugin(manila_network.NetworkBaseAPI, db_base.Base): class NeutronNetworkPlugin(network.NetworkBaseAPI):
def __init__(self): def __init__(self, *args, **kwargs):
super(NeutronNetworkPlugin, self).__init__() db_driver = kwargs.pop('db_driver', None)
self.neutron_api = neutron_api.API() super(NeutronNetworkPlugin, self).__init__(db_driver=db_driver)
self.neutron_api = neutron_api.API(*args, **kwargs)
def allocate_network(self, context, share_server, share_network, **kwargs): def allocate_network(self, context, share_server, share_network, **kwargs):
"""Allocate network resources using given network information. """Allocate network resources using given network information.

View File

@ -19,40 +19,53 @@ Drivers for shares.
import time import time
from manila import exception
from manila.i18n import _LE
from manila.openstack.common import log as logging
from manila import utils
from oslo.config import cfg from oslo.config import cfg
from manila import exception
from manila.i18n import _LE
from manila import network
from manila.openstack.common import log as logging
from manila import utils
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
share_opts = [ share_opts = [
# NOTE(rushiagr): Reasonable to define this option at only one place. # NOTE(rushiagr): Reasonable to define this option at only one place.
cfg.IntOpt('num_shell_tries', cfg.IntOpt(
default=3, 'num_shell_tries',
help='Number of times to attempt to run flakey shell ' default=3,
'commands.'), help='Number of times to attempt to run flakey shell commands.'),
cfg.IntOpt('reserved_share_percentage', cfg.IntOpt(
default=0, 'reserved_share_percentage',
help='The percentage of backend capacity reserved.'), default=0,
cfg.StrOpt('share_backend_name', help='The percentage of backend capacity reserved.'),
default=None, cfg.StrOpt(
help='The backend name for a given driver implementation.'), 'share_backend_name',
default=None,
help='The backend name for a given driver implementation.'),
cfg.StrOpt(
'network_config_group',
default=None,
help="Name of the configuration group in the Manila conf file "
"to look for network config options."
"If not set, the share backend's config group will be used."
"If an option is not found within provided group, then"
"'DEFAULT' group will be used for search of option."),
] ]
ssh_opts = [ ssh_opts = [
cfg.IntOpt('ssh_conn_timeout', cfg.IntOpt(
default=60, 'ssh_conn_timeout',
help='Backend server SSH connection timeout.'), default=60,
cfg.IntOpt('ssh_min_pool_conn', help='Backend server SSH connection timeout.'),
default=1, cfg.IntOpt(
help='Minimum number of connections in the SSH pool.'), 'ssh_min_pool_conn',
cfg.IntOpt('ssh_max_pool_conn', default=1,
default=10, help='Minimum number of connections in the SSH pool.'),
help='Maximum number of connections in the SSH pool.'), cfg.IntOpt(
'ssh_max_pool_conn',
default=10,
help='Maximum number of connections in the SSH pool.'),
] ]
CONF = cfg.CONF CONF = cfg.CONF
@ -100,6 +113,11 @@ class ShareDriver(object):
self.configuration = kwargs.get('configuration', None) self.configuration = kwargs.get('configuration', None)
if self.configuration: if self.configuration:
self.configuration.append_config_values(share_opts) self.configuration.append_config_values(share_opts)
network_config_group = (self.configuration.network_config_group or
self.configuration.config_group)
else:
network_config_group = None
self.network_api = network.API(config_group_name=network_config_group)
def create_share(self, context, share, share_server=None): def create_share(self, context, share, share_server=None):
"""Is called to create share.""" """Is called to create share."""
@ -153,8 +171,30 @@ class ShareDriver(object):
return self._stats return self._stats
def get_network_allocations_number(self): def get_network_allocations_number(self):
"""Returns number of network allocations for creating VIFs.""" """Returns number of network allocations for creating VIFs.
pass
Drivers that use Nova for share servers should return zero (0) here
same as Generic driver does.
Because Nova will handle network resources allocation.
Drivers that handle networking itself should calculate it according
to their own requirements. It can have 1+ network interfaces.
"""
raise NotImplementedError()
def allocate_network(self, context, share_server, share_network,
count=None, **kwargs):
"""Allocate network resources using given network information."""
if count is None:
count = self.get_network_allocations_number()
if count:
kwargs.update(count=count)
self.network_api.allocate_network(
context, share_server, share_network, **kwargs)
def deallocate_network(self, context, share_server_id):
"""Deallocate network resources for the given share server."""
if self.get_network_allocations_number():
self.network_api.deallocate_network(context, share_server_id)
def setup_server(self, network_info, metadata=None): def setup_server(self, network_info, metadata=None):
"""Set up and configures share server with given network parameters.""" """Set up and configures share server with given network parameters."""

View File

@ -63,7 +63,7 @@ CONF.register_opts(EMC_NAS_OPTS)
class EMCShareDriver(driver.ShareDriver): class EMCShareDriver(driver.ShareDriver):
"""EMC specific NAS driver. Allows for NFS and CIFS NAS storage usage.""" """EMC specific NAS driver. Allows for NFS and CIFS NAS storage usage."""
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super(EMCShareDriver, self).__init__() super(EMCShareDriver, self).__init__(*args, **kwargs)
self.configuration = kwargs.get('configuration', None) self.configuration = kwargs.get('configuration', None)
if self.configuration: if self.configuration:
self.configuration.append_config_values(EMC_NAS_OPTS) self.configuration.append_config_values(EMC_NAS_OPTS)

View File

@ -265,6 +265,9 @@ class GlusterfsShareDriver(driver.ExecuteMixin, driver.ShareDriver):
data['free_capacity_gb'] = (smpv.f_bavail * smpv.f_frsize) >> 30 data['free_capacity_gb'] = (smpv.f_bavail * smpv.f_frsize) >> 30
self._stats = data self._stats = data
def get_network_allocations_number(self):
return 0
def create_share(self, ctx, share, share_server=None): def create_share(self, ctx, share, share_server=None):
"""Create a sub-directory/share in the GlusterFS volume.""" """Create a sub-directory/share in the GlusterFS volume."""
local_share_path = self._get_local_share_path(share) local_share_path = self._get_local_share_path(share)

View File

@ -373,6 +373,9 @@ class GlusterfsNativeShareDriver(driver.ExecuteMixin, driver.ShareDriver):
self._restart_gluster_vol(gluster_addr) self._restart_gluster_vol(gluster_addr)
def get_network_allocations_number(self):
return 0
def create_share(self, context, share, share_server=None): def create_share(self, context, share, share_server=None):
"""Create a share using GlusterFS volume. """Create a share using GlusterFS volume.

View File

@ -418,6 +418,9 @@ class GPFSShareDriver(driver.ExecuteMixin, driver.ShareDriver):
LOG.error(msg) LOG.error(msg)
raise exception.GPFSException(msg) raise exception.GPFSException(msg)
def get_network_allocations_number(self):
return 0
def create_share(self, ctx, share, share_server=None): def create_share(self, ctx, share, share_server=None):
"""Create GPFS directory that will be represented as share.""" """Create GPFS directory that will be represented as share."""
self._create_share(share) self._create_share(share)

View File

@ -32,7 +32,6 @@ from manila.i18n import _LE
from manila.i18n import _LI from manila.i18n import _LI
from manila.i18n import _LW from manila.i18n import _LW
from manila import manager from manila import manager
from manila import network
from manila.openstack.common import log as logging from manila.openstack.common import log as logging
from manila import quota from manila import quota
import manila.share.configuration import manila.share.configuration
@ -71,7 +70,6 @@ class ShareManager(manager.SchedulerDependentManager):
share_driver = self.configuration.share_driver share_driver = self.configuration.share_driver
self.driver = importutils.import_object( self.driver = importutils.import_object(
share_driver, self.db, configuration=self.configuration) share_driver, self.db, configuration=self.configuration)
self.network_api = network.API()
def init_host(self): def init_host(self):
"""Initialization for a standalone service.""" """Initialization for a standalone service."""
@ -436,22 +434,14 @@ class ShareManager(manager.SchedulerDependentManager):
return network_info return network_info
def _setup_server(self, context, share_server, metadata=None): def _setup_server(self, context, share_server, metadata=None):
# NOTE(vponomaryov): set network_allocations to 0 before 'try' block
# for case we get exception calling appropriate method. This value will
# be used in exception handling and for case 'setup_server' method was
# not called we won't make redundant actions.
allocation_number = 0
try: try:
share_network = self.db.share_network_get( share_network = self.db.share_network_get(
context, share_server['share_network_id']) context, share_server['share_network_id'])
allocation_number = self.driver.get_network_allocations_number() self.driver.allocate_network(context, share_server, share_network)
if allocation_number:
self.network_api.allocate_network( # Get share_network again in case it was updated.
context, share_server, share_network, share_network = self.db.share_network_get(
count=allocation_number) context, share_server['share_network_id'])
# If we reach here, then share_network was updated
share_network = self.db.share_network_get(
context, share_server['share_network_id'])
network_info = self._form_server_setup_info(context, share_server, network_info = self._form_server_setup_info(context, share_server,
share_network) share_network)
@ -483,9 +473,7 @@ class ShareManager(manager.SchedulerDependentManager):
self.db.share_server_update(context, share_server['id'], self.db.share_server_update(context, share_server['id'],
{'status': constants.STATUS_ERROR}) {'status': constants.STATUS_ERROR})
if allocation_number: self.driver.deallocate_network(context, share_server['id'])
self.network_api.deallocate_network(
context, share_server['id'])
def delete_share_server(self, context, share_server): def delete_share_server(self, context, share_server):
@ -526,9 +514,4 @@ class ShareManager(manager.SchedulerDependentManager):
_teardown_server() _teardown_server()
LOG.info(_LI("Share server deleted successfully.")) LOG.info(_LI("Share server deleted successfully."))
# NOTE(vponomaryov): share servers created by Nova do not need self.driver.deallocate_network(context, share_server['id'])
# explicit network allocations release. It is done by Nova itself.
# So, all drivers that use 'service_instance' module do not need
# following operation.
if self.driver.get_network_allocations_number():
self.network_api.deallocate_network(context, share_server['id'])

View File

@ -1,4 +1,5 @@
# Copyright 2013 OpenStack Foundation # Copyright 2013 OpenStack Foundation
# Copyright 2014 Mirantis Inc.
# All Rights Reserved # All Rights Reserved
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # Licensed under the Apache License, Version 2.0 (the "License"); you may
@ -21,7 +22,6 @@ from oslo.config import cfg
from manila import context from manila import context
from manila.db import base from manila.db import base
from manila import exception from manila import exception
from manila.network import neutron
from manila.network.neutron import api as neutron_api from manila.network.neutron import api as neutron_api
from manila.network.neutron import constants as neutron_constants from manila.network.neutron import constants as neutron_constants
from manila import test from manila import test
@ -85,370 +85,485 @@ class NeutronApiTest(test.TestCase):
def setUp(self): def setUp(self):
super(NeutronApiTest, self).setUp() super(NeutronApiTest, self).setUp()
self._create_neutron_api() self.context = context.get_admin_context()
self.stubs.Set(base, 'Base', fakes.FakeModel)
@mock.patch.object(base, 'Base', fakes.FakeModel) self.stubs.Set(
@mock.patch.object(context, 'get_admin_context', clientv20, 'Client', mock.Mock(return_value=FakeNeutronClient()))
mock.Mock(return_value='context'))
@mock.patch.object(neutron, 'get_client',
mock.Mock(return_value=FakeNeutronClient()))
def _create_neutron_api(self):
self.neutron_api = neutron_api.API() self.neutron_api = neutron_api.API()
@mock.patch.object(base.Base, '__init__', mock.Mock())
@mock.patch.object(context, 'get_admin_context',
mock.Mock(return_value='context'))
@mock.patch.object(neutron, 'get_client', mock.Mock())
def test_create_api_object(self): def test_create_api_object(self):
neutron_api.API() # instantiate Neutron API object
neutron_api_instance = neutron_api.API()
context.get_admin_context.assert_called_once_with() # Verify results
neutron.get_client.assert_called_once_with('context') self.assertTrue(clientv20.Client.called)
base.Base.__init__.assert_called_once_with() self.assertTrue(hasattr(neutron_api_instance, 'client'))
self.assertTrue(hasattr(neutron_api_instance, 'configuration'))
def test_create_api_object_custom_config_group(self):
# Set up test data
fake_config_group_name = 'fake_config_group_name'
# instantiate Neutron API object
obj = neutron_api.API(fake_config_group_name)
# Verify results
self.assertTrue(clientv20.Client.called)
self.assertTrue(hasattr(obj, 'client'))
self.assertTrue(hasattr(obj, 'configuration'))
self.assertEqual(
fake_config_group_name, obj.configuration._group.name)
def test_create_port_with_all_args(self): def test_create_port_with_all_args(self):
port_args = {'tenant_id': 'test tenant', 'network_id': 'test net', # Set up test data
'host_id': 'test host', 'subnet_id': 'test subnet', self.stubs.Set(self.neutron_api, '_has_port_binding_extension',
'fixed_ip': 'test ip', 'device_owner': 'test owner', mock.Mock(return_value=True))
'device_id': 'test device', 'mac_address': 'test mac', port_args = {
'security_group_ids': 'test group', 'tenant_id': 'test tenant', 'network_id': 'test net',
'dhcp_opts': 'test dhcp'} 'host_id': 'test host', 'subnet_id': 'test subnet',
'fixed_ip': 'test ip', 'device_owner': 'test owner',
'device_id': 'test device', 'mac_address': 'test mac',
'security_group_ids': 'test group',
'dhcp_opts': 'test dhcp',
}
with mock.patch.object(self.neutron_api, '_has_port_binding_extension', # Execute method 'create_port'
mock.Mock(return_value=True)): port = self.neutron_api.create_port(**port_args)
port = self.neutron_api.create_port(**port_args)
self.assertEqual(port['tenant_id'], port_args['tenant_id']) # Verify results
self.assertEqual(port['network_id'], self.assertEqual(port['tenant_id'], port_args['tenant_id'])
port_args['network_id']) self.assertEqual(port['network_id'], port_args['network_id'])
self.assertEqual(port['binding:host_id'], self.assertEqual(port['binding:host_id'], port_args['host_id'])
port_args['host_id']) self.assertEqual(port['fixed_ips'][0]['subnet_id'],
self.assertEqual(port['fixed_ips'][0]['subnet_id'], port_args['subnet_id'])
port_args['subnet_id']) self.assertEqual(port['fixed_ips'][0]['ip_address'],
self.assertEqual(port['fixed_ips'][0]['ip_address'], port_args['fixed_ip'])
port_args['fixed_ip']) self.assertEqual(port['device_owner'], port_args['device_owner'])
self.assertEqual(port['device_owner'], self.assertEqual(port['device_id'], port_args['device_id'])
port_args['device_owner']) self.assertEqual(port['mac_address'], port_args['mac_address'])
self.assertEqual(port['device_id'], port_args['device_id']) self.assertEqual(port['security_groups'],
self.assertEqual(port['mac_address'], port_args['security_group_ids'])
port_args['mac_address']) self.assertEqual(port['extra_dhcp_opts'], port_args['dhcp_opts'])
self.assertEqual(port['security_groups'], self.neutron_api._has_port_binding_extension.assert_called_once_with()
port_args['security_group_ids']) self.assertTrue(clientv20.Client.called)
self.assertEqual(port['extra_dhcp_opts'],
port_args['dhcp_opts'])
def test_create_port_with_required_args(self): def test_create_port_with_required_args(self):
# Set up test data
self.stubs.Set(self.neutron_api, '_has_port_binding_extension',
mock.Mock(return_value=True))
port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'} port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'}
with mock.patch.object(self.neutron_api, '_has_port_binding_extension', # Execute method 'create_port'
mock.Mock(return_value=True)): port = self.neutron_api.create_port(**port_args)
port = self.neutron_api.create_port(**port_args) # Verify results
self.assertEqual(port['tenant_id'], port_args['tenant_id']) self.assertEqual(port['tenant_id'], port_args['tenant_id'])
self.assertEqual(port['network_id'], self.assertEqual(port['network_id'],
port_args['network_id']) port_args['network_id'])
self.neutron_api._has_port_binding_extension.assert_called_once_with()
self.assertTrue(clientv20.Client.called)
@mock.patch.object(neutron_api.LOG, 'exception', mock.Mock()) @mock.patch.object(neutron_api.LOG, 'exception', mock.Mock())
def test_create_port_exception(self): def test_create_port_exception(self):
# Set up test data
self.stubs.Set(
self.neutron_api, '_has_port_binding_extension',
mock.Mock(return_value=True))
self.stubs.Set(
self.neutron_api.client, 'create_port',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'} port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'}
client_create_port_mock = mock.Mock(
side_effect=neutron_client_exc.NeutronClientException)
with mock.patch.object(self.neutron_api, '_has_port_binding_extension', # Execute method 'create_port'
mock.Mock(return_value=True)): self.assertRaises(exception.NetworkException,
with mock.patch.object(self.neutron_api.client, 'create_port', self.neutron_api.create_port,
client_create_port_mock): **port_args)
self.assertRaises(exception.NetworkException, # Verify results
self.neutron_api.create_port, self.neutron_api._has_port_binding_extension.assert_called_once_with()
**port_args) self.assertTrue(neutron_api.LOG.exception.called)
self.assertTrue(neutron_api.LOG.exception.called) self.assertTrue(clientv20.Client.called)
self.assertTrue(self.neutron_api.client.create_port.called)
@mock.patch.object(neutron_api.LOG, 'exception', mock.Mock()) @mock.patch.object(neutron_api.LOG, 'exception', mock.Mock())
def test_create_port_exception_status_409(self): def test_create_port_exception_status_409(self):
# Set up test data
self.stubs.Set(
self.neutron_api, '_has_port_binding_extension',
mock.Mock(return_value=True))
self.stubs.Set(
self.neutron_api.client, 'create_port',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException(
status_code=409)))
port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'} port_args = {'tenant_id': 'test tenant', 'network_id': 'test net'}
client_create_port_mock = mock.Mock(
side_effect=neutron_client_exc.NeutronClientException(
status_code=409))
with mock.patch.object(self.neutron_api, '_has_port_binding_extension', # Execute method 'create_port'
mock.Mock(return_value=True)): self.assertRaises(exception.PortLimitExceeded,
with mock.patch.object(self.neutron_api.client, 'create_port', self.neutron_api.create_port,
client_create_port_mock): **port_args)
self.assertRaises(exception.PortLimitExceeded, # Verify results
self.neutron_api.create_port, self.neutron_api._has_port_binding_extension.assert_called_once_with()
**port_args) self.assertTrue(neutron_api.LOG.exception.called)
self.assertTrue(neutron_api.LOG.exception.called) self.assertTrue(clientv20.Client.called)
self.assertTrue(self.neutron_api.client.create_port.called)
def test_delete_port(self): def test_delete_port(self):
# Set up test data
self.stubs.Set(self.neutron_api.client, 'delete_port', mock.Mock())
port_id = 'test port id' port_id = 'test port id'
with mock.patch.object(self.neutron_api.client, 'delete_port',
mock.Mock()) as client_delete_port_mock:
self.neutron_api.delete_port(port_id) # Execute method 'delete_port'
client_delete_port_mock.assert_called_once_with(port_id) self.neutron_api.delete_port(port_id)
# Verify results
self.neutron_api.client.delete_port.assert_called_once_with(port_id)
self.assertTrue(clientv20.Client.called)
def test_list_ports(self): def test_list_ports(self):
# Set up test data
search_opts = {'test_option': 'test_value'} search_opts = {'test_option': 'test_value'}
fake_ports = [{'fake port': 'fake port info'}] fake_ports = [{'fake port': 'fake port info'}]
client_list_ports_mock = mock.Mock(return_value={'ports': fake_ports}) self.stubs.Set(
self.neutron_api.client, 'list_ports',
mock.Mock(return_value={'ports': fake_ports}))
with mock.patch.object(self.neutron_api.client, 'list_ports', # Execute method 'list_ports'
client_list_ports_mock): ports = self.neutron_api.list_ports(**search_opts)
ports = self.neutron_api.list_ports(**search_opts) # Verify results
client_list_ports_mock.assert_called_once_with(**search_opts) self.assertEqual(fake_ports, ports)
self.assertEqual(ports, fake_ports) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.list_ports.assert_called_once_with(
**search_opts)
def test_show_port(self): def test_show_port(self):
# Set up test data
port_id = 'test port id' port_id = 'test port id'
fake_port = {'fake port': 'fake port info'} fake_port = {'fake port': 'fake port info'}
client_show_port_mock = mock.Mock(return_value={'port': fake_port}) self.stubs.Set(
self.neutron_api.client, 'show_port',
mock.Mock(return_value={'port': fake_port}))
with mock.patch.object(self.neutron_api.client, 'show_port', # Execute method 'show_port'
client_show_port_mock): port = self.neutron_api.show_port(port_id)
port = self.neutron_api.show_port(port_id) # Verify results
client_show_port_mock.assert_called_once_with(port_id) self.assertEqual(fake_port, port)
self.assertEqual(port, fake_port) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.show_port.assert_called_once_with(port_id)
def test_get_network(self): def test_get_network(self):
# Set up test data
network_id = 'test network id' network_id = 'test network id'
fake_network = {'fake network': 'fake network info'} fake_network = {'fake network': 'fake network info'}
client_show_network_mock = mock.Mock( self.stubs.Set(
return_value={'network': fake_network}) self.neutron_api.client, 'show_network',
mock.Mock(return_value={'network': fake_network}))
with mock.patch.object(self.neutron_api.client, 'show_network', # Execute method 'get_network'
client_show_network_mock): network = self.neutron_api.get_network(network_id)
network = self.neutron_api.get_network(network_id) # Verify results
client_show_network_mock.assert_called_once_with(network_id) self.assertEqual(fake_network, network)
self.assertEqual(network, fake_network) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.show_network.assert_called_once_with(
network_id)
def test_get_subnet(self): def test_get_subnet(self):
# Set up test data
subnet_id = 'fake subnet id' subnet_id = 'fake subnet id'
self.stubs.Set(
self.neutron_api.client, 'show_subnet',
mock.Mock(return_value={'subnet': {}}))
with mock.patch.object(self.neutron_api.client, 'show_subnet', # Execute method 'get_subnet'
mock.Mock(return_value={'subnet': {}})): subnet = self.neutron_api.get_subnet(subnet_id)
subnet = self.neutron_api.get_subnet(subnet_id) # Verify results
self.neutron_api.client.show_subnet.assert_called_once_with( self.assertEqual({}, subnet)
subnet_id) self.assertTrue(clientv20.Client.called)
self.assertEqual(subnet, {}) self.neutron_api.client.show_subnet.assert_called_once_with(
subnet_id)
def test_get_all_network(self): def test_get_all_network(self):
# Set up test data
fake_networks = [{'fake network': 'fake network info'}] fake_networks = [{'fake network': 'fake network info'}]
client_list_networks_mock = mock.Mock( self.stubs.Set(
return_value={'networks': fake_networks}) self.neutron_api.client, 'list_networks',
mock.Mock(return_value={'networks': fake_networks}))
with mock.patch.object(self.neutron_api.client, 'list_networks', # Execute method 'get_all_networks'
client_list_networks_mock): networks = self.neutron_api.get_all_networks()
networks = self.neutron_api.get_all_networks() # Verify results
client_list_networks_mock.assert_any_call() self.assertEqual(fake_networks, networks)
self.assertEqual(networks, fake_networks) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.list_networks.assert_called_once_with()
def test_list_extensions(self): def test_list_extensions(self):
extensions = [{'name': neutron_constants.PORTBINDING_EXT}, # Set up test data
{'name': neutron_constants.PROVIDER_NW_EXT}] extensions = [
{'name': neutron_constants.PORTBINDING_EXT},
{'name': neutron_constants.PROVIDER_NW_EXT},
]
self.stubs.Set(
self.neutron_api.client, 'list_extensions',
mock.Mock(return_value={'extensions': extensions}))
with mock.patch.object( # Execute method 'list_extensions'
self.neutron_api.client, result = self.neutron_api.list_extensions()
'list_extensions',
mock.Mock(return_value={'extensions': extensions})):
result = self.neutron_api.list_extensions() # Verify results
self.neutron_api.client.list_extensions.assert_any_call() self.assertTrue(clientv20.Client.called)
self.assertTrue(neutron_constants.PORTBINDING_EXT in result) self.neutron_api.client.list_extensions.assert_called_once_with()
self.assertTrue(neutron_constants.PROVIDER_NW_EXT in result) self.assertIn(neutron_constants.PORTBINDING_EXT, result)
self.assertEqual(result[neutron_constants.PORTBINDING_EXT], self.assertIn(neutron_constants.PROVIDER_NW_EXT, result)
extensions[0]) self.assertEqual(
self.assertEqual(result[neutron_constants.PROVIDER_NW_EXT], extensions[0], result[neutron_constants.PORTBINDING_EXT])
extensions[1]) self.assertEqual(
extensions[1], result[neutron_constants.PROVIDER_NW_EXT])
def test_create_network(self): def test_create_network(self):
# Set up test data
net_args = {'tenant_id': 'test tenant', 'name': 'test name'} net_args = {'tenant_id': 'test tenant', 'name': 'test name'}
# Execute method 'network_create'
network = self.neutron_api.network_create(**net_args) network = self.neutron_api.network_create(**net_args)
self.assertEqual(network['tenant_id'], net_args['tenant_id'])
self.assertEqual(network['name'], net_args['name']) # Verify results
self.assertEqual(net_args['tenant_id'], network['tenant_id'])
self.assertEqual(net_args['name'], network['name'])
self.assertTrue(clientv20.Client.called)
def test_create_subnet(self): def test_create_subnet(self):
subnet_args = {'tenant_id': 'test tenant', 'name': 'test name', # Set up test data
'net_id': 'test net id', 'cidr': '10.0.0.0/24'} subnet_args = {
'tenant_id': 'test tenant',
'name': 'test name',
'net_id': 'test net id',
'cidr': '10.0.0.0/24',
}
# Execute method 'subnet_create'
subnet = self.neutron_api.subnet_create(**subnet_args) subnet = self.neutron_api.subnet_create(**subnet_args)
self.assertEqual(subnet['tenant_id'], subnet_args['tenant_id'])
self.assertEqual(subnet['name'], subnet_args['name']) # Verify results
self.assertEqual(subnet_args['tenant_id'], subnet['tenant_id'])
self.assertEqual(subnet_args['name'], subnet['name'])
self.assertTrue(clientv20.Client.called)
def test_create_router(self): def test_create_router(self):
# Set up test data
router_args = {'tenant_id': 'test tenant', 'name': 'test name'} router_args = {'tenant_id': 'test tenant', 'name': 'test name'}
# Execute method 'router_create'
router = self.neutron_api.router_create(**router_args) router = self.neutron_api.router_create(**router_args)
self.assertEqual(router['tenant_id'], router_args['tenant_id'])
self.assertEqual(router['name'], router_args['name']) # Verify results
self.assertEqual(router_args['tenant_id'], router['tenant_id'])
self.assertEqual(router_args['name'], router['name'])
self.assertTrue(clientv20.Client.called)
def test_list_routers(self): def test_list_routers(self):
# Set up test data
fake_routers = [{'fake router': 'fake router info'}] fake_routers = [{'fake router': 'fake router info'}]
client_list_routers_mock = mock.Mock( self.stubs.Set(
return_value={'routers': fake_routers}) self.neutron_api.client, 'list_routers',
mock.Mock(return_value={'routers': fake_routers}))
with mock.patch.object(self.neutron_api.client, 'list_routers', # Execute method 'router_list'
client_list_routers_mock): networks = self.neutron_api.router_list()
networks = self.neutron_api.router_list() # Verify results
client_list_routers_mock.assert_any_call() self.assertEqual(fake_routers, networks)
self.assertEqual(networks, fake_routers) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.list_routers.assert_called_once_with()
def test_create_network_exception(self): def test_create_network_exception(self):
# Set up test data
net_args = {'tenant_id': 'test tenant', 'name': 'test name'} net_args = {'tenant_id': 'test tenant', 'name': 'test name'}
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'create_network', self.neutron_api.client, 'create_network',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'network_create'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.network_create, self.neutron_api.network_create,
**net_args) **net_args)
# Verify results
self.neutron_api.client.create_network.assert_called_once_with(
{'network': net_args})
self.assertTrue(clientv20.Client.called)
def test_create_subnet_exception(self): def test_create_subnet_exception(self):
subnet_args = {'tenant_id': 'test tenant', 'name': 'test name', # Set up test data
'net_id': 'test net id', 'cidr': '10.0.0.0/24'} subnet_args = {
'tenant_id': 'test tenant',
'name': 'test name',
'net_id': 'test net id',
'cidr': '10.0.0.0/24',
}
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'create_subnet', self.neutron_api.client, 'create_subnet',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'subnet_create'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.subnet_create, self.neutron_api.subnet_create,
**subnet_args) **subnet_args)
# Verify results
expected_data = {
'network_id': subnet_args['net_id'],
'tenant_id': subnet_args['tenant_id'],
'cidr': subnet_args['cidr'],
'name': subnet_args['name'],
'ip_version': 4,
}
self.neutron_api.client.create_subnet.assert_called_once_with(
{'subnet': expected_data})
self.assertTrue(clientv20.Client.called)
def test_create_router_exception(self): def test_create_router_exception(self):
# Set up test data
router_args = {'tenant_id': 'test tenant', 'name': 'test name'} router_args = {'tenant_id': 'test tenant', 'name': 'test name'}
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'create_router', self.neutron_api.client, 'create_router',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'router_create'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.router_create, self.neutron_api.router_create,
**router_args) **router_args)
# Verify results
self.neutron_api.client.create_router.assert_called_once_with(
{'router': router_args})
self.assertTrue(clientv20.Client.called)
def test_update_port_fixed_ips(self): def test_update_port_fixed_ips(self):
# Set up test data
port_id = 'test_port' port_id = 'test_port'
fixed_ips = {'fixed_ips': [{'subnet_id': 'test subnet'}]} fixed_ips = {'fixed_ips': [{'subnet_id': 'test subnet'}]}
# Execute method 'update_port_fixed_ips'
port = self.neutron_api.update_port_fixed_ips(port_id, fixed_ips) port = self.neutron_api.update_port_fixed_ips(port_id, fixed_ips)
self.assertEqual(port, fixed_ips)
# Verify results
self.assertEqual(fixed_ips, port)
self.assertTrue(clientv20.Client.called)
def test_update_port_fixed_ips_exception(self): def test_update_port_fixed_ips_exception(self):
# Set up test data
port_id = 'test_port' port_id = 'test_port'
fixed_ips = {'fixed_ips': [{'subnet_id': 'test subnet'}]} fixed_ips = {'fixed_ips': [{'subnet_id': 'test subnet'}]}
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'update_port', self.neutron_api.client, 'update_port',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'update_port_fixed_ips'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.update_port_fixed_ips, self.neutron_api.update_port_fixed_ips,
port_id, fixed_ips) port_id, fixed_ips)
# Verify results
self.neutron_api.client.update_port.assert_called_once_with(
port_id, {'port': fixed_ips})
self.assertTrue(clientv20.Client.called)
def test_router_update_routes(self): def test_router_update_routes(self):
# Set up test data
router_id = 'test_router' router_id = 'test_router'
routes = {'routes': [{'destination': '0.0.0.0/0', routes = {
'nexthop': '8.8.8.8'}]} 'routes': [
{'destination': '0.0.0.0/0', 'nexthop': '8.8.8.8', },
],
}
# Execute method 'router_update_routes'
router = self.neutron_api.router_update_routes(router_id, routes) router = self.neutron_api.router_update_routes(router_id, routes)
self.assertEqual(router, routes)
# Verify results
self.assertEqual(routes, router)
self.assertTrue(clientv20.Client.called)
def test_router_update_routes_exception(self): def test_router_update_routes_exception(self):
# Set up test data
router_id = 'test_router' router_id = 'test_router'
routes = {'routes': [{'destination': '0.0.0.0/0', routes = {
'nexthop': '8.8.8.8'}]} 'routes': [
{'destination': '0.0.0.0/0', 'nexthop': '8.8.8.8', },
],
}
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'update_router', self.neutron_api.client, 'update_router',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'router_update_routes'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.router_update_routes, self.neutron_api.router_update_routes,
router_id, routes) router_id, routes)
# Verify results
self.neutron_api.client.update_router.assert_called_once_with(
router_id, {'router': routes})
self.assertTrue(clientv20.Client.called)
def test_show_router(self): def test_show_router(self):
# Set up test data
router_id = 'test router id' router_id = 'test router id'
fake_router = {'fake router': 'fake router info'} fake_router = {'fake router': 'fake router info'}
client_show_router_mock = mock.Mock(return_value={'router': self.stubs.Set(
fake_router}) self.neutron_api.client, 'show_router',
mock.Mock(return_value={'router': fake_router}))
with mock.patch.object(self.neutron_api.client, 'show_router', # Execute method 'show_router'
client_show_router_mock): port = self.neutron_api.show_router(router_id)
port = self.neutron_api.show_router(router_id) # Verify results
client_show_router_mock.assert_called_once_with(router_id) self.assertEqual(fake_router, port)
self.assertEqual(port, fake_router) self.assertTrue(clientv20.Client.called)
self.neutron_api.client.show_router.assert_called_once_with(router_id)
def test_router_add_interface(self): def test_router_add_interface(self):
# Set up test data
router_id = 'test port id' router_id = 'test port id'
subnet_id = 'test subnet id' subnet_id = 'test subnet id'
port_id = 'test port id' port_id = 'test port id'
with mock.patch.object( self.stubs.Set(
self.neutron_api.client, 'add_interface_router', self.neutron_api.client, 'add_interface_router', mock.Mock())
mock.Mock()) as client_add_interface_router_mock:
self.neutron_api.router_add_interface(router_id, # Execute method 'router_add_interface'
subnet_id, self.neutron_api.router_add_interface(router_id, subnet_id, port_id)
port_id)
client_add_interface_router_mock.assert_called_once_with( # Verify results
port_id, {'subnet_id': subnet_id, 'port_id': port_id}) self.neutron_api.client.add_interface_router.assert_called_once_with(
port_id, {'subnet_id': subnet_id, 'port_id': port_id})
self.assertTrue(clientv20.Client.called)
def test_router_add_interface_exception(self): def test_router_add_interface_exception(self):
# Set up test data
router_id = 'test port id' router_id = 'test port id'
subnet_id = 'test subnet id' subnet_id = 'test subnet id'
port_id = 'test port id' port_id = 'test port id'
self.stubs.Set( self.stubs.Set(
self.neutron_api.client, 'add_interface_router', self.neutron_api.client, 'add_interface_router',
mock.Mock(side_effect=neutron_client_exc.NeutronClientException)) mock.Mock(side_effect=neutron_client_exc.NeutronClientException))
# Execute method 'router_add_interface'
self.assertRaises( self.assertRaises(
exception.NetworkException, exception.NetworkException,
self.neutron_api.router_add_interface, self.neutron_api.router_add_interface,
router_id, subnet_id, port_id) router_id, subnet_id, port_id)
# Verify results
class TestNeutronClient(test.TestCase): self.neutron_api.client.add_interface_router.assert_called_once_with(
router_id, {'subnet_id': subnet_id, 'port_id': port_id})
@mock.patch.object(clientv20.Client, '__init__', self.assertTrue(clientv20.Client.called)
mock.Mock(return_value=None))
def test_get_client_with_token(self):
client_args = {'endpoint_url': CONF.neutron_url,
'timeout': CONF.neutron_url_timeout,
'insecure': CONF.neutron_api_insecure,
'ca_cert': CONF.neutron_ca_certificates_file,
'token': 'test_token',
'auth_strategy': None}
my_context = context.RequestContext('test_user', 'test_tenant',
auth_token='test_token',
is_admin=False)
neutron.get_client(my_context)
clientv20.Client.__init__.assert_called_once_with(**client_args)
@mock.patch.object(clientv20.Client, '__init__',
mock.Mock(return_value=None))
def test_get_client_no_token(self):
my_context = context.RequestContext('test_user', 'test_tenant',
is_admin=False)
self.assertRaises(neutron_client_exc.Unauthorized,
neutron.get_client,
my_context)
@mock.patch.object(clientv20.Client, '__init__',
mock.Mock(return_value=None))
def test_get_client_admin_context(self):
client_args = {'endpoint_url': CONF.neutron_url,
'timeout': CONF.neutron_url_timeout,
'insecure': CONF.neutron_api_insecure,
'ca_cert': CONF.neutron_ca_certificates_file,
'username': CONF.neutron_admin_username,
'tenant_name': CONF.neutron_admin_tenant_name,
'password': CONF.neutron_admin_password,
'auth_url': CONF.neutron_admin_auth_url,
'auth_strategy': CONF.neutron_auth_strategy}
my_context = context.RequestContext('test_user', 'test_tenant',
is_admin=True)
neutron.get_client(my_context)
clientv20.Client.__init__.assert_called_once_with(**client_args)

View File

@ -1,4 +1,5 @@
# Copyright 2012 NetApp # Copyright 2012 NetApp
# Copyright 2014 Mirantis Inc.
# All Rights Reserved. # All Rights Reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # Licensed under the Apache License, Version 2.0 (the "License"); you may
@ -16,8 +17,11 @@
import time import time
import mock
from manila import exception from manila import exception
import manila.share.configuration from manila import network
from manila.share import configuration
from manila.share import driver from manila.share import driver
from manila import test from manila import test
from manila import utils from manila import utils
@ -32,6 +36,7 @@ def fake_sleep(duration):
class ShareDriverTestCase(test.TestCase): class ShareDriverTestCase(test.TestCase):
def setUp(self): def setUp(self):
super(ShareDriverTestCase, self).setUp() super(ShareDriverTestCase, self).setUp()
self.utils = utils self.utils = utils
@ -39,11 +44,40 @@ class ShareDriverTestCase(test.TestCase):
self.time = time self.time = time
self.stubs.Set(self.time, 'sleep', fake_sleep) self.stubs.Set(self.time, 'sleep', fake_sleep)
def tearDown(self):
super(ShareDriverTestCase, self).tearDown()
def test__try_execute(self): def test__try_execute(self):
execute_mixin = driver.ExecuteMixin( execute_mixin = driver.ExecuteMixin(
configuration=manila.share.configuration.Configuration(None)) configuration=configuration.Configuration(None))
self.assertRaises(exception.ProcessExecutionError, self.assertRaises(exception.ProcessExecutionError,
execute_mixin._try_execute) execute_mixin._try_execute)
def _instantiate_share_driver(self, network_config_group):
self.stubs.Set(network, 'API', mock.Mock())
config = mock.Mock()
config.append_config_values = mock.Mock()
config.config_group = 'fake_config_group'
config.network_config_group = network_config_group
share_driver = driver.ShareDriver(configuration=config)
self.assertTrue(hasattr(share_driver, 'configuration'))
config.append_config_values.assert_called_once_with(driver.share_opts)
if network_config_group:
network.API.assert_called_once_with(
config_group_name=config.network_config_group)
else:
network.API.assert_called_once_with(
config_group_name=config.config_group)
def test_instantiate_share_driver(self):
self._instantiate_share_driver(None)
def test_instantiate_share_driver_another_config_group(self):
self._instantiate_share_driver("fake_network_config_group")
def test_instantiate_share_driver_no_configuration(self):
self.stubs.Set(network, 'API', mock.Mock())
share_driver = driver.ShareDriver(configuration=None)
self.assertEqual(None, share_driver.configuration)
network.API.assert_called_once_with(config_group_name=None)

View File

@ -789,10 +789,8 @@ class ShareManagerTestCase(test.TestCase):
acs = db.share_access_get(self.context, access_id) acs = db.share_access_get(self.context, access_id)
self.assertEqual(acs['state'], 'error') self.assertEqual(acs['state'], 'error')
def test_setup_server_2_net_allocations(self): def test_setup_server(self):
# Setup required test data # Setup required test data
allocation_number = 2
context = "fake_context"
share_server = { share_server = {
'id': 'fake_id', 'id': 'fake_id',
'share_network_id': 'fake_sn_id', 'share_network_id': 'fake_sn_id',
@ -805,10 +803,7 @@ class ShareManagerTestCase(test.TestCase):
# mock required stuff # mock required stuff
self.stubs.Set(self.share_manager.db, 'share_network_get', self.stubs.Set(self.share_manager.db, 'share_network_get',
mock.Mock(return_value=share_network)) mock.Mock(return_value=share_network))
self.stubs.Set(self.share_manager.driver, self.stubs.Set(self.share_manager.driver, 'allocate_network',
'get_network_allocations_number',
mock.Mock(return_value=allocation_number))
self.stubs.Set(self.share_manager.network_api, 'allocate_network',
mock.Mock()) mock.Mock())
self.stubs.Set(self.share_manager, '_form_server_setup_info', self.stubs.Set(self.share_manager, '_form_server_setup_info',
mock.Mock(return_value=network_info)) mock.Mock(return_value=network_info))
@ -822,80 +817,29 @@ class ShareManagerTestCase(test.TestCase):
# execute method _setup_server # execute method _setup_server
result = self.share_manager._setup_server( result = self.share_manager._setup_server(
context, share_server, metadata=metadata) self.context, share_server, metadata=metadata)
# verify results # verify results
self.assertEqual(share_server, result) self.assertEqual(share_server, result)
self.share_manager.db.share_network_get.assert_has_calls([ self.share_manager.db.share_network_get.assert_has_calls([
mock.call(context, share_server['share_network_id']), mock.call(self.context, share_server['share_network_id']),
mock.call(context, share_server['share_network_id']), mock.call(self.context, share_server['share_network_id']),
]) ])
self.share_manager.driver.get_network_allocations_number.\ self.share_manager.driver.allocate_network.assert_called_once_with(
assert_called_once_with() self.context, share_server, share_network)
self.share_manager.network_api.allocate_network.\
assert_called_once_with(context, share_server, share_network,
count=allocation_number)
self.share_manager._form_server_setup_info.assert_called_once_with( self.share_manager._form_server_setup_info.assert_called_once_with(
context, share_server, share_network) self.context, share_server, share_network)
self.share_manager.driver.setup_server.assert_called_once_with( self.share_manager.driver.setup_server.assert_called_once_with(
network_info, metadata=metadata) network_info, metadata=metadata)
self.share_manager.db.share_server_backend_details_set.\ self.share_manager.db.share_server_backend_details_set.\
assert_called_once_with(context, share_server['id'], server_info) assert_called_once_with(
self.context, share_server['id'], server_info)
self.share_manager.db.share_server_update.assert_called_once_with( self.share_manager.db.share_server_update.assert_called_once_with(
context, share_server['id'], {'status': constants.STATUS_ACTIVE}) self.context, share_server['id'],
{'status': constants.STATUS_ACTIVE})
def test_setup_server_no_net_allocations(self): def test_setup_server_server_info_not_present(self):
# Setup required test data # Setup required test data
allocation_number = 0
context = "fake_context"
share_server = {
'id': 'fake_id',
'share_network_id': 'fake_sn_id',
}
metadata = {'fake_metadata_key': 'fake_metadata_value'}
share_network = {'id': 'fake_sn_id'}
network_info = {'fake_network_info_key': 'fake_network_info_value'}
server_info = {'fake_server_info_key': 'fake_server_info_value'}
# mock required stuff
self.stubs.Set(self.share_manager.db, 'share_network_get',
mock.Mock(return_value=share_network))
self.stubs.Set(self.share_manager.driver,
'get_network_allocations_number',
mock.Mock(return_value=allocation_number))
self.stubs.Set(self.share_manager, '_form_server_setup_info',
mock.Mock(return_value=network_info))
self.stubs.Set(self.share_manager.driver, 'setup_server',
mock.Mock(return_value=server_info))
self.stubs.Set(self.share_manager.db,
'share_server_backend_details_set',
mock.Mock())
self.stubs.Set(self.share_manager.db, 'share_server_update',
mock.Mock(return_value=share_server))
# execute method _setup_server
result = self.share_manager._setup_server(
context, share_server, metadata=metadata)
# verify results
self.assertEqual(share_server, result)
self.share_manager.db.share_network_get.assert_called_once_with(
context, share_server['share_network_id'])
self.share_manager.driver.get_network_allocations_number.\
assert_called_once_with()
self.share_manager._form_server_setup_info.assert_called_once_with(
context, share_server, share_network)
self.share_manager.driver.setup_server.assert_called_once_with(
network_info, metadata=metadata)
self.share_manager.db.share_server_backend_details_set.\
assert_called_once_with(context, share_server['id'], server_info)
self.share_manager.db.share_server_update.assert_called_once_with(
context, share_server['id'], {'status': constants.STATUS_ACTIVE})
def test_setup_server_server_info_not_present_no_net_allocations(self):
# Setup required test data
allocation_number = 0
context = "fake_context"
share_server = { share_server = {
'id': 'fake_id', 'id': 'fake_id',
'share_network_id': 'fake_sn_id', 'share_network_id': 'fake_sn_id',
@ -908,69 +852,36 @@ class ShareManagerTestCase(test.TestCase):
# mock required stuff # mock required stuff
self.stubs.Set(self.share_manager.db, 'share_network_get', self.stubs.Set(self.share_manager.db, 'share_network_get',
mock.Mock(return_value=share_network)) mock.Mock(return_value=share_network))
self.stubs.Set(self.share_manager.driver,
'get_network_allocations_number',
mock.Mock(return_value=allocation_number))
self.stubs.Set(self.share_manager, '_form_server_setup_info', self.stubs.Set(self.share_manager, '_form_server_setup_info',
mock.Mock(return_value=network_info)) mock.Mock(return_value=network_info))
self.stubs.Set(self.share_manager.driver, 'setup_server', self.stubs.Set(self.share_manager.driver, 'setup_server',
mock.Mock(return_value=server_info)) mock.Mock(return_value=server_info))
self.stubs.Set(self.share_manager.db, 'share_server_update', self.stubs.Set(self.share_manager.db, 'share_server_update',
mock.Mock(return_value=share_server)) mock.Mock(return_value=share_server))
self.stubs.Set(self.share_manager.driver, 'allocate_network',
# execute method _setup_server
result = self.share_manager._setup_server(
context, share_server, metadata=metadata)
# verify results
self.assertEqual(share_server, result)
self.share_manager.db.share_network_get.assert_called_once_with(
context, share_server['share_network_id'])
self.share_manager.driver.get_network_allocations_number.\
assert_called_once_with()
self.share_manager._form_server_setup_info.assert_called_once_with(
context, share_server, share_network)
self.share_manager.driver.setup_server.assert_called_once_with(
network_info, metadata=metadata)
self.share_manager.db.share_server_update.assert_called_once_with(
context, share_server['id'], {'status': constants.STATUS_ACTIVE})
def test_setup_server_exception_raised(self):
# Setup required test data
context = "fake_context"
share_server = {
'id': 'fake_id',
'share_network_id': 'fake_sn_id',
}
share_network = {'id': 'fake_sn_id'}
# mock required stuff
self.stubs.Set(self.share_manager.db, 'share_network_get',
mock.Mock(return_value=share_network))
self.stubs.Set(self.share_manager.driver,
'get_network_allocations_number',
mock.Mock(side_effect=exception.ManilaException()))
self.stubs.Set(self.share_manager.db, 'share_server_update',
mock.Mock()) mock.Mock())
# execute method _setup_server # execute method _setup_server
self.assertRaises( result = self.share_manager._setup_server(
exception.ManilaException, self.context, share_server, metadata=metadata)
self.share_manager._setup_server,
context,
share_server,
)
self.share_manager.db.share_network_get.assert_called_once_with(
context, share_server['share_network_id'])
self.share_manager.driver.get_network_allocations_number.\
assert_called_once_with()
self.share_manager.db.share_server_update.assert_called_once_with(
context, share_server['id'], {'status': constants.STATUS_ERROR})
def setup_server_raise_exception(self, allocation_number, # verify results
detail_data_proper): self.assertEqual(share_server, result)
self.share_manager.db.share_network_get.assert_has_calls([
mock.call(self.context, share_server['share_network_id']),
mock.call(self.context, share_server['share_network_id'])])
self.share_manager._form_server_setup_info.assert_called_once_with(
self.context, share_server, share_network)
self.share_manager.driver.setup_server.assert_called_once_with(
network_info, metadata=metadata)
self.share_manager.db.share_server_update.assert_called_once_with(
self.context, share_server['id'],
{'status': constants.STATUS_ACTIVE})
self.share_manager.driver.allocate_network.assert_called_once_with(
self.context, share_server, share_network)
def setup_server_raise_exception(self, detail_data_proper):
# Setup required test data # Setup required test data
context = "fake_context"
share_server = { share_server = {
'id': 'fake_id', 'id': 'fake_id',
'share_network_id': 'fake_sn_id', 'share_network_id': 'fake_sn_id',
@ -989,14 +900,10 @@ class ShareManagerTestCase(test.TestCase):
# Mock required parameters # Mock required parameters
self.stubs.Set(self.share_manager.db, 'share_network_get', self.stubs.Set(self.share_manager.db, 'share_network_get',
mock.Mock(return_value=share_network)) mock.Mock(return_value=share_network))
self.stubs.Set(self.share_manager.driver,
'get_network_allocations_number',
mock.Mock(return_value=allocation_number))
self.stubs.Set(self.share_manager.db, 'share_server_update', self.stubs.Set(self.share_manager.db, 'share_server_update',
mock.Mock()) mock.Mock())
if allocation_number: for m in ['deallocate_network', 'allocate_network']:
for m in ['deallocate_network', 'allocate_network']: self.stubs.Set(self.share_manager.driver, m, mock.Mock())
self.stubs.Set(self.share_manager.network_api, m, mock.Mock())
self.stubs.Set(self.share_manager, '_form_server_setup_info', self.stubs.Set(self.share_manager, '_form_server_setup_info',
mock.Mock(return_value=network_info)) mock.Mock(return_value=network_info))
self.stubs.Set(self.share_manager.db, self.stubs.Set(self.share_manager.db,
@ -1010,44 +917,30 @@ class ShareManagerTestCase(test.TestCase):
self.assertRaises( self.assertRaises(
exception.ManilaException, exception.ManilaException,
self.share_manager._setup_server, self.share_manager._setup_server,
context, self.context,
share_server, share_server,
) )
# verify results
if detail_data_proper: if detail_data_proper:
self.share_manager.db.share_server_backend_details_set.\ self.share_manager.db.share_server_backend_details_set.\
assert_called_once_with( assert_called_once_with(
context, share_server['id'], server_info) self.context, share_server['id'], server_info)
self.share_manager.driver.get_network_allocations_number.\
assert_called_once_with()
self.share_manager._form_server_setup_info.assert_called_once_with( self.share_manager._form_server_setup_info.assert_called_once_with(
context, share_server, share_network) self.context, share_server, share_network)
self.share_manager.db.share_server_update.assert_called_once_with( self.share_manager.db.share_server_update.assert_called_once_with(
context, share_server['id'], {'status': constants.STATUS_ERROR}) self.context, share_server['id'],
if allocation_number: {'status': constants.STATUS_ERROR})
self.share_manager.db.share_network_get.assert_has_calls([ self.share_manager.db.share_network_get.assert_has_calls([
mock.call(context, share_server['share_network_id']), mock.call(self.context, share_server['share_network_id']),
mock.call(context, share_server['share_network_id'])]) mock.call(self.context, share_server['share_network_id'])])
self.share_manager.network_api.allocate_network.assert_has_calls([ self.share_manager.driver.allocate_network.assert_has_calls([
mock.call(context, share_server, share_network, mock.call(self.context, share_server, share_network)])
count=allocation_number)]) self.share_manager.driver.deallocate_network.assert_has_calls(
self.share_manager.network_api.deallocate_network.assert_has_calls( mock.call(self.context, share_server['id']))
mock.call(context, share_server['id']))
else:
self.share_manager.db.share_network_get.assert_called_once_with(
context, share_server['share_network_id'])
def test_setup_server_incorrect_detail_data_an2(self): def test_setup_server_incorrect_detail_data(self):
# an2 - allocation number has value -> 2 self.setup_server_raise_exception(detail_data_proper=False)
self.setup_server_raise_exception(2, detail_data_proper=False)
def test_setup_server_incorrect_detail_data_an0(self): def test_setup_server_exception_in_driver(self):
# an0 - allocation number has value -> 0 self.setup_server_raise_exception(detail_data_proper=True)
self.setup_server_raise_exception(0, detail_data_proper=False)
def test_setup_server_exception_in_driver_an2(self):
# an2 - allocation number has value -> 2
self.setup_server_raise_exception(2, detail_data_proper=True)
def test_setup_server_exception_in_driver_an0(self):
# an0 - allocation number has value -> 0
self.setup_server_raise_exception(2, detail_data_proper=True)