1229 lines
54 KiB
Python
1229 lines
54 KiB
Python
# Copyright 2013 Mirantis, Inc.
|
|
# Copyright 2013 OpenStack Foundation
|
|
#
|
|
# 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 cinderclient import api_versions as cinder_api_versions
|
|
from cinderclient import apiclient as cinder_apiclient
|
|
from cinderclient import exceptions as cinder_exception
|
|
from cinderclient.v2 import limits as cinder_limits
|
|
from keystoneauth1 import loading as ks_loading
|
|
from keystoneauth1 import session
|
|
from keystoneclient import exceptions as keystone_exception
|
|
import mock
|
|
from oslo_utils.fixture import uuidsentinel as uuids
|
|
from oslo_utils import timeutils
|
|
|
|
import nova.conf
|
|
from nova import context
|
|
from nova import exception
|
|
from nova import test
|
|
from nova.tests.unit.fake_instance import fake_instance_obj
|
|
from nova.volume import cinder
|
|
|
|
|
|
CONF = nova.conf.CONF
|
|
|
|
|
|
class FakeVolume(object):
|
|
|
|
def __init__(self, volume_id, size=1, attachments=None, multiattach=False):
|
|
self.id = volume_id
|
|
self.name = 'volume_name'
|
|
self.description = 'volume_description'
|
|
self.status = 'available'
|
|
self.created_at = timeutils.utcnow()
|
|
self.size = size
|
|
self.availability_zone = 'nova'
|
|
self.attachments = attachments or []
|
|
self.volume_type = 99
|
|
self.bootable = False
|
|
self.snapshot_id = 'snap_id_1'
|
|
self.metadata = {}
|
|
self.multiattach = multiattach
|
|
|
|
def get(self, volume_id):
|
|
return self.volume_id
|
|
|
|
|
|
class FakeSnapshot(object):
|
|
|
|
def __init__(self, snapshot_id, volume_id, size=1):
|
|
self.id = snapshot_id
|
|
self.name = 'snapshot_name'
|
|
self.description = 'snapshot_description'
|
|
self.status = 'available'
|
|
self.size = size
|
|
self.created_at = timeutils.utcnow()
|
|
self.progress = '99%'
|
|
self.volume_id = volume_id
|
|
self.project_id = 'fake_project'
|
|
|
|
|
|
class FakeVolumeType(object):
|
|
def __init__(self, volume_type_name, volume_type_id):
|
|
self.id = volume_type_id
|
|
self.name = volume_type_name
|
|
|
|
|
|
class FakeAttachment(object):
|
|
|
|
def __init__(self):
|
|
self.id = uuids.attachment_id
|
|
self.status = 'attaching'
|
|
self.instance = uuids.instance_uuid
|
|
self.volume_id = uuids.volume_id
|
|
self.attached_at = timeutils.utcnow()
|
|
self.detached_at = None
|
|
self.attach_mode = 'rw'
|
|
self.connection_info = {'driver_volume_type': 'fake_type',
|
|
'target_lun': '1',
|
|
'foo': 'bar',
|
|
'attachment_id': uuids.attachment_id}
|
|
self.att = {'id': self.id,
|
|
'status': self.status,
|
|
'instance': self.instance,
|
|
'volume_id': self.volume_id,
|
|
'attached_at': self.attached_at,
|
|
'detached_at': self.detached_at,
|
|
'attach_mode': self.attach_mode,
|
|
'connection_info': self.connection_info}
|
|
|
|
def get(self, key, default=None):
|
|
return self.att.get(key, default)
|
|
|
|
def __setitem__(self, key, value):
|
|
self.att[key] = value
|
|
|
|
def __getitem__(self, key):
|
|
return self.att[key]
|
|
|
|
def to_dict(self):
|
|
return self.att
|
|
|
|
|
|
class CinderApiTestCase(test.NoDBTestCase):
|
|
def setUp(self):
|
|
super(CinderApiTestCase, self).setUp()
|
|
|
|
self.api = cinder.API()
|
|
self.ctx = context.get_admin_context()
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get(self, mock_cinderclient):
|
|
volume_id = 'volume_id1'
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.get(self.ctx, volume_id)
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx, microversion=None)
|
|
mock_volumes.get.assert_called_once_with(volume_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_failed_notfound(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.get.side_effect = (
|
|
cinder_exception.NotFound(404, '404'))
|
|
|
|
self.assertRaises(exception.VolumeNotFound,
|
|
self.api.get, self.ctx, 'id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_failed_badrequest(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.get.side_effect = (
|
|
cinder_exception.BadRequest(400, '400'))
|
|
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.get, self.ctx, 'id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_failed_connection_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.get.side_effect = (
|
|
cinder_exception.ConnectionError(''))
|
|
|
|
self.assertRaises(exception.CinderConnectionFailed,
|
|
self.api.get, self.ctx, 'id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_with_shared_targets(self, mock_cinderclient):
|
|
"""Tests getting a volume at microversion 3.48 which includes the
|
|
shared_targets and service_uuid parameters in the volume response body.
|
|
"""
|
|
mock_volume = mock.MagicMock(
|
|
shared_targets=False, service_uuid=uuids.service_uuid)
|
|
mock_volumes = mock.MagicMock()
|
|
mock_volumes.get.return_value = mock_volume
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
vol = self.api.get(self.ctx, uuids.volume_id, microversion='3.48')
|
|
mock_cinderclient.assert_called_once_with(
|
|
self.ctx, microversion='3.48')
|
|
mock_volumes.get.assert_called_once_with(uuids.volume_id)
|
|
self.assertIn('shared_targets', vol)
|
|
self.assertFalse(vol['shared_targets'])
|
|
self.assertEqual(uuids.service_uuid, vol['service_uuid'])
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.48'))
|
|
def test_get_microversion_not_supported(self, mock_cinderclient):
|
|
"""Tests getting a volume at microversion 3.48 but that version
|
|
is not available.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.get, self.ctx, uuids.volume_id,
|
|
microversion='3.48')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create(self, mock_cinderclient):
|
|
volume = FakeVolume('id1')
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
mock_volumes.create.return_value = volume
|
|
|
|
created_volume = self.api.create(self.ctx, 1, '', '')
|
|
self.assertEqual('id1', created_volume['id'])
|
|
self.assertEqual(1, created_volume['size'])
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.create.assert_called_once_with(1, availability_zone=None,
|
|
description='',
|
|
imageRef=None,
|
|
metadata=None, name='',
|
|
snapshot_id=None,
|
|
volume_type=None)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.create.side_effect = (
|
|
cinder_exception.BadRequest(400, '400'))
|
|
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.create, self.ctx, 1, '', '')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create_failed_not_found(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.create.side_effect = (
|
|
cinder_exception.NotFound(404, 'Volume type can not be found.'))
|
|
|
|
ex = self.assertRaises(exception.NotFound,
|
|
self.api.create, self.ctx, 1, '', '')
|
|
self.assertEqual('Volume type can not be found.', str(ex))
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create_over_quota_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.create.side_effect = (
|
|
cinder_exception.OverLimit(413))
|
|
self.assertRaises(exception.OverQuota, self.api.create, self.ctx,
|
|
1, '', '')
|
|
mock_cinderclient.return_value.volumes.create.assert_called_once_with(
|
|
1, imageRef=None, availability_zone=None,
|
|
volume_type=None, description='', snapshot_id=None, name='',
|
|
metadata=None)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_all(self, mock_cinderclient):
|
|
volume1 = FakeVolume('id1')
|
|
volume2 = FakeVolume('id2')
|
|
|
|
volume_list = [volume1, volume2]
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
mock_volumes.list.return_value = volume_list
|
|
|
|
volumes = self.api.get_all(self.ctx)
|
|
self.assertEqual(2, len(volumes))
|
|
self.assertEqual(['id1', 'id2'], [vol['id'] for vol in volumes])
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.list.assert_called_once_with(detailed=True,
|
|
search_opts={})
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_all_with_search(self, mock_cinderclient):
|
|
volume1 = FakeVolume('id1')
|
|
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
mock_volumes.list.return_value = [volume1]
|
|
|
|
volumes = self.api.get_all(self.ctx, search_opts={'id': 'id1'})
|
|
self.assertEqual(1, len(volumes))
|
|
self.assertEqual('id1', volumes[0]['id'])
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.list.assert_called_once_with(detailed=True,
|
|
search_opts={'id': 'id1'})
|
|
|
|
@mock.patch.object(cinder.az, 'get_instance_availability_zone',
|
|
return_value='zone1')
|
|
def test_check_availability_zone_differs(self, mock_get_instance_az):
|
|
self.flags(cross_az_attach=False, group='cinder')
|
|
volume = {'id': uuids.volume_id,
|
|
'status': 'available',
|
|
'attach_status': 'detached',
|
|
'availability_zone': 'zone2'}
|
|
instance = fake_instance_obj(self.ctx)
|
|
# Simulate _provision_instances in the compute API; the instance is not
|
|
# created in the API so the instance will not have an id attribute set.
|
|
delattr(instance, 'id')
|
|
|
|
self.assertRaises(exception.InvalidVolume,
|
|
self.api.check_availability_zone,
|
|
self.ctx, volume, instance)
|
|
mock_get_instance_az.assert_called_once_with(self.ctx, instance)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_reserve_volume(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.reserve_volume(self.ctx, 'id1')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.reserve.assert_called_once_with('id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_unreserve_volume(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.unreserve_volume(self.ctx, 'id1')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.unreserve.assert_called_once_with('id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_begin_detaching(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.begin_detaching(self.ctx, 'id1')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.begin_detaching.assert_called_once_with('id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_roll_detaching(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.roll_detaching(self.ctx, 'id1')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.roll_detaching.assert_called_once_with('id1')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attach(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.attach(self.ctx, 'id1', 'uuid', 'point')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.attach.assert_called_once_with('id1', 'uuid', 'point',
|
|
mode='rw')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attach_with_mode(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.attach(self.ctx, 'id1', 'uuid', 'point', mode='ro')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.attach.assert_called_once_with('id1', 'uuid', 'point',
|
|
mode='ro')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_create(self, mock_cinderclient):
|
|
"""Tests the happy path for creating a volume attachment without a
|
|
mountpoint.
|
|
"""
|
|
attachment_ref = {'id': uuids.attachment_id,
|
|
'connection_info': {}}
|
|
expected_attachment_ref = {'id': uuids.attachment_id,
|
|
'connection_info': {}}
|
|
mock_cinderclient.return_value.attachments.create.return_value = (
|
|
attachment_ref)
|
|
result = self.api.attachment_create(
|
|
self.ctx, uuids.volume_id, uuids.instance_id)
|
|
self.assertEqual(expected_attachment_ref, result)
|
|
mock_cinderclient.return_value.attachments.create.\
|
|
assert_called_once_with(uuids.volume_id, None, uuids.instance_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_create_with_mountpoint(self, mock_cinderclient):
|
|
"""Tests the happy path for creating a volume attachment with a
|
|
mountpoint.
|
|
"""
|
|
attachment_ref = {'id': uuids.attachment_id,
|
|
'connection_info': {}}
|
|
expected_attachment_ref = {'id': uuids.attachment_id,
|
|
'connection_info': {}}
|
|
mock_cinderclient.return_value.attachments.create.return_value = (
|
|
attachment_ref)
|
|
original_connector = {'host': 'fake-host'}
|
|
updated_connector = dict(original_connector, mountpoint='/dev/vdb')
|
|
result = self.api.attachment_create(
|
|
self.ctx, uuids.volume_id, uuids.instance_id,
|
|
connector=original_connector, mountpoint='/dev/vdb')
|
|
self.assertEqual(expected_attachment_ref, result)
|
|
# Make sure the original connector wasn't modified.
|
|
self.assertNotIn('mountpoint', original_connector)
|
|
# Make sure the mountpoint was passed through via the connector.
|
|
mock_cinderclient.return_value.attachments.create.\
|
|
assert_called_once_with(uuids.volume_id, updated_connector,
|
|
uuids.instance_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_create_volume_not_found(self, mock_cinderclient):
|
|
"""Tests that the translate_volume_exception decorator is used."""
|
|
# fake out the volume not found error
|
|
mock_cinderclient.return_value.attachments.create.side_effect = (
|
|
cinder_exception.NotFound(404))
|
|
self.assertRaises(exception.VolumeNotFound, self.api.attachment_create,
|
|
self.ctx, uuids.volume_id, uuids.instance_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.44'))
|
|
def test_attachment_create_unsupported_api_version(self,
|
|
mock_cinderclient):
|
|
"""Tests that CinderAPIVersionNotAvailable is passed back through
|
|
if 3.44 isn't available.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.attachment_create,
|
|
self.ctx, uuids.volume_id, uuids.instance_id)
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_update(self, mock_cinderclient):
|
|
"""Tests the happy path for updating a volume attachment without
|
|
a mountpoint.
|
|
"""
|
|
fake_attachment = FakeAttachment()
|
|
connector = {'host': 'fake-host'}
|
|
expected_attachment_ref = {
|
|
'id': uuids.attachment_id,
|
|
'volume_id': fake_attachment.volume_id,
|
|
'attach_mode': 'rw',
|
|
'connection_info': {
|
|
'attached_at': fake_attachment.attached_at,
|
|
'data': {'foo': 'bar', 'target_lun': '1'},
|
|
'detached_at': None,
|
|
'driver_volume_type': 'fake_type',
|
|
'instance': fake_attachment.instance,
|
|
'status': 'attaching',
|
|
'volume_id': fake_attachment.volume_id}}
|
|
mock_cinderclient.return_value.attachments.update.return_value = (
|
|
fake_attachment)
|
|
result = self.api.attachment_update(
|
|
self.ctx, uuids.attachment_id, connector=connector)
|
|
self.assertEqual(expected_attachment_ref, result)
|
|
# Make sure the connector wasn't modified.
|
|
self.assertNotIn('mountpoint', connector)
|
|
mock_cinderclient.return_value.attachments.update.\
|
|
assert_called_once_with(uuids.attachment_id, connector)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_update_with_mountpoint(self, mock_cinderclient):
|
|
"""Tests the happy path for updating a volume attachment with
|
|
a mountpoint.
|
|
"""
|
|
fake_attachment = FakeAttachment()
|
|
original_connector = {'host': 'fake-host'}
|
|
updated_connector = dict(original_connector, mountpoint='/dev/vdb')
|
|
expected_attachment_ref = {
|
|
'id': uuids.attachment_id,
|
|
'volume_id': fake_attachment.volume_id,
|
|
'attach_mode': 'rw',
|
|
'connection_info': {
|
|
'attached_at': fake_attachment.attached_at,
|
|
'data': {'foo': 'bar', 'target_lun': '1'},
|
|
'detached_at': None,
|
|
'driver_volume_type': 'fake_type',
|
|
'instance': fake_attachment.instance,
|
|
'status': 'attaching',
|
|
'volume_id': fake_attachment.volume_id}}
|
|
mock_cinderclient.return_value.attachments.update.return_value = (
|
|
fake_attachment)
|
|
result = self.api.attachment_update(
|
|
self.ctx, uuids.attachment_id, connector=original_connector,
|
|
mountpoint='/dev/vdb')
|
|
self.assertEqual(expected_attachment_ref, result)
|
|
# Make sure the original connector wasn't modified.
|
|
self.assertNotIn('mountpoint', original_connector)
|
|
# Make sure the mountpoint was passed through via the connector.
|
|
mock_cinderclient.return_value.attachments.update.\
|
|
assert_called_once_with(uuids.attachment_id, updated_connector)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_update_attachment_not_found(self, mock_cinderclient):
|
|
"""Tests that the translate_attachment_exception decorator is used."""
|
|
# fake out the volume not found error
|
|
mock_cinderclient.return_value.attachments.update.side_effect = (
|
|
cinder_exception.NotFound(404))
|
|
self.assertRaises(exception.VolumeAttachmentNotFound,
|
|
self.api.attachment_update,
|
|
self.ctx, uuids.attachment_id,
|
|
connector={'host': 'fake-host'})
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_update_attachment_no_connector(self,
|
|
mock_cinderclient):
|
|
"""Tests that the translate_cinder_exception decorator is used."""
|
|
# fake out the volume bad request error
|
|
mock_cinderclient.return_value.attachments.update.side_effect = (
|
|
cinder_exception.BadRequest(400))
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.attachment_update,
|
|
self.ctx, uuids.attachment_id, connector=None)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.44'))
|
|
def test_attachment_update_unsupported_api_version(self,
|
|
mock_cinderclient):
|
|
"""Tests that CinderAPIVersionNotAvailable is passed back through
|
|
if 3.44 isn't available.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.attachment_update,
|
|
self.ctx, uuids.attachment_id, connector={})
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_delete(self, mock_cinderclient):
|
|
mock_attachments = mock.MagicMock()
|
|
mock_cinderclient.return_value = \
|
|
mock.MagicMock(attachments=mock_attachments)
|
|
|
|
attachment_id = uuids.attachment
|
|
self.api.attachment_delete(self.ctx, attachment_id)
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
mock_attachments.delete.assert_called_once_with(attachment_id)
|
|
|
|
@mock.patch('nova.volume.cinder.LOG')
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_delete_failed(self, mock_cinderclient, mock_log):
|
|
mock_cinderclient.return_value.attachments.delete.side_effect = (
|
|
cinder_exception.NotFound(404, '404'))
|
|
|
|
attachment_id = uuids.attachment
|
|
ex = self.assertRaises(exception.VolumeAttachmentNotFound,
|
|
self.api.attachment_delete,
|
|
self.ctx,
|
|
attachment_id)
|
|
|
|
self.assertEqual(404, ex.code)
|
|
self.assertIn(attachment_id, str(ex))
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.44'))
|
|
def test_attachment_delete_unsupported_api_version(self,
|
|
mock_cinderclient):
|
|
"""Tests that CinderAPIVersionNotAvailable is passed back through
|
|
if 3.44 isn't available.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.attachment_delete,
|
|
self.ctx, uuids.attachment_id)
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_apiclient.exceptions.InternalServerError)
|
|
def test_attachment_delete_internal_server_error(self, mock_cinderclient):
|
|
|
|
self.assertRaises(cinder_apiclient.exceptions.InternalServerError,
|
|
self.api.attachment_delete,
|
|
self.ctx, uuids.attachment_id)
|
|
|
|
self.assertEqual(5, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_delete_internal_server_error_do_not_raise(
|
|
self, mock_cinderclient):
|
|
# generate exception, and then have a normal return on the next retry
|
|
mock_cinderclient.return_value.attachments.delete.side_effect = [
|
|
cinder_apiclient.exceptions.InternalServerError, None]
|
|
|
|
attachment_id = uuids.attachment
|
|
self.api.attachment_delete(self.ctx, attachment_id)
|
|
|
|
self.assertEqual(2, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_exception.BadRequest(code=400))
|
|
def test_attachment_delete_bad_request_exception(self, mock_cinderclient):
|
|
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.attachment_delete,
|
|
self.ctx, uuids.attachment_id)
|
|
|
|
self.assertEqual(1, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_complete(self, mock_cinderclient):
|
|
mock_attachments = mock.MagicMock()
|
|
mock_cinderclient.return_value = \
|
|
mock.MagicMock(attachments=mock_attachments)
|
|
|
|
attachment_id = uuids.attachment
|
|
self.api.attachment_complete(self.ctx, attachment_id)
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
mock_attachments.complete.assert_called_once_with(attachment_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_complete_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.attachments.complete.side_effect = (
|
|
cinder_exception.NotFound(404, '404'))
|
|
|
|
attachment_id = uuids.attachment
|
|
ex = self.assertRaises(exception.VolumeAttachmentNotFound,
|
|
self.api.attachment_complete,
|
|
self.ctx,
|
|
attachment_id)
|
|
|
|
self.assertEqual(404, ex.code)
|
|
self.assertIn(attachment_id, str(ex))
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.44'))
|
|
def test_attachment_complete_unsupported_api_version(self,
|
|
mock_cinderclient):
|
|
"""Tests that CinderAPIVersionNotAvailable is passed back.
|
|
|
|
If microversion 3.44 isn't available that should result in a
|
|
CinderAPIVersionNotAvailable exception.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.attachment_complete,
|
|
self.ctx, uuids.attachment_id)
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_detach(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(version='2',
|
|
volumes=mock_volumes)
|
|
|
|
self.api.detach(self.ctx, 'id1', instance_uuid='fake_uuid',
|
|
attachment_id='fakeid')
|
|
|
|
mock_cinderclient.assert_called_with(self.ctx)
|
|
mock_volumes.detach.assert_called_once_with('id1', 'fakeid')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_detach_no_attachment_id(self, mock_cinderclient):
|
|
attachment = {'server_id': 'fake_uuid',
|
|
'attachment_id': 'fakeid'
|
|
}
|
|
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(version='2',
|
|
volumes=mock_volumes)
|
|
mock_cinderclient.return_value.volumes.get.return_value = \
|
|
FakeVolume('id1', attachments=[attachment])
|
|
|
|
self.api.detach(self.ctx, 'id1', instance_uuid='fake_uuid')
|
|
|
|
mock_cinderclient.assert_called_with(self.ctx, microversion=None)
|
|
mock_volumes.detach.assert_called_once_with('id1', None)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_detach_no_attachment_id_multiattach(self, mock_cinderclient):
|
|
attachment = {'server_id': 'fake_uuid',
|
|
'attachment_id': 'fakeid'
|
|
}
|
|
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(version='2',
|
|
volumes=mock_volumes)
|
|
mock_cinderclient.return_value.volumes.get.return_value = \
|
|
FakeVolume('id1', attachments=[attachment], multiattach=True)
|
|
|
|
self.api.detach(self.ctx, 'id1', instance_uuid='fake_uuid')
|
|
|
|
mock_cinderclient.assert_called_with(self.ctx, microversion=None)
|
|
mock_volumes.detach.assert_called_once_with('id1', 'fakeid')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_apiclient.exceptions.InternalServerError)
|
|
def test_detach_internal_server_error(self, mock_cinderclient):
|
|
|
|
self.assertRaises(cinder_apiclient.exceptions.InternalServerError,
|
|
self.api.detach,
|
|
self.ctx, 'id1', instance_uuid='fake_uuid')
|
|
|
|
self.assertEqual(5, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_detach_internal_server_error_do_not_raise(
|
|
self, mock_cinderclient):
|
|
# generate exception, and then have a normal return on the next retry
|
|
mock_cinderclient.return_value.volumes.detach.side_effect = [
|
|
cinder_apiclient.exceptions.InternalServerError, None]
|
|
|
|
self.api.detach(self.ctx, 'id1', instance_uuid='fake_uuid',
|
|
attachment_id='fakeid')
|
|
|
|
self.assertEqual(2, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_exception.BadRequest(code=400))
|
|
def test_detach_bad_request_exception(self, mock_cinderclient):
|
|
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.detach,
|
|
self.ctx, 'id1', instance_uuid='fake_uuid')
|
|
|
|
self.assertEqual(1, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_get(self, mock_cinderclient):
|
|
mock_attachment = mock.MagicMock()
|
|
mock_cinderclient.return_value = \
|
|
mock.MagicMock(attachments=mock_attachment)
|
|
|
|
attachment_id = uuids.attachment
|
|
self.api.attachment_get(self.ctx, attachment_id)
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
mock_attachment.show.assert_called_once_with(attachment_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_attachment_get_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.attachments.show.side_effect = (
|
|
cinder_exception.NotFound(404, '404'))
|
|
|
|
attachment_id = uuids.attachment
|
|
ex = self.assertRaises(exception.VolumeAttachmentNotFound,
|
|
self.api.attachment_get,
|
|
self.ctx,
|
|
attachment_id)
|
|
|
|
self.assertEqual(404, ex.code)
|
|
self.assertIn(attachment_id, str(ex))
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=exception.CinderAPIVersionNotAvailable(
|
|
version='3.44'))
|
|
def test_attachment_get_unsupported_api_version(self, mock_cinderclient):
|
|
"""Tests that CinderAPIVersionNotAvailable is passed back.
|
|
|
|
If microversion 3.44 isn't available that should result in a
|
|
CinderAPIVersionNotAvailable exception.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
self.api.attachment_get,
|
|
self.ctx, uuids.attachment_id)
|
|
mock_cinderclient.assert_called_once_with(self.ctx, '3.44',
|
|
skip_version_check=True)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_initialize_connection(self, mock_cinderclient):
|
|
connection_info = {'foo': 'bar'}
|
|
mock_cinderclient.return_value.volumes. \
|
|
initialize_connection.return_value = connection_info
|
|
|
|
volume_id = 'fake_vid'
|
|
connector = {'host': 'fakehost1'}
|
|
actual = self.api.initialize_connection(self.ctx, volume_id, connector)
|
|
|
|
expected = connection_info
|
|
expected['connector'] = connector
|
|
self.assertEqual(expected, actual)
|
|
|
|
mock_cinderclient.return_value.volumes. \
|
|
initialize_connection.assert_called_once_with(volume_id, connector)
|
|
|
|
@mock.patch('nova.volume.cinder.LOG')
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_initialize_connection_exception_no_code(
|
|
self, mock_cinderclient, mock_log):
|
|
mock_cinderclient.return_value.volumes. \
|
|
initialize_connection.side_effect = (
|
|
cinder_exception.ClientException(500, "500"))
|
|
mock_cinderclient.return_value.volumes. \
|
|
terminate_connection.side_effect = (
|
|
test.TestingException)
|
|
|
|
connector = {'host': 'fakehost1'}
|
|
self.assertRaises(cinder_exception.ClientException,
|
|
self.api.initialize_connection,
|
|
self.ctx,
|
|
'id1',
|
|
connector)
|
|
self.assertIsNone(mock_log.error.call_args_list[1][0][1]['code'])
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_initialize_connection_rollback(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.\
|
|
initialize_connection.side_effect = (
|
|
cinder_exception.ClientException(500, "500"))
|
|
|
|
connector = {'host': 'host1'}
|
|
ex = self.assertRaises(cinder_exception.ClientException,
|
|
self.api.initialize_connection,
|
|
self.ctx,
|
|
'id1',
|
|
connector)
|
|
self.assertEqual(500, ex.code)
|
|
mock_cinderclient.return_value.volumes.\
|
|
terminate_connection.assert_called_once_with('id1', connector)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_initialize_connection_no_rollback(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volumes.\
|
|
initialize_connection.side_effect = test.TestingException
|
|
|
|
connector = {'host': 'host1'}
|
|
self.assertRaises(test.TestingException,
|
|
self.api.initialize_connection,
|
|
self.ctx,
|
|
'id1',
|
|
connector)
|
|
self.assertFalse(mock_cinderclient.return_value.volumes.
|
|
terminate_connection.called)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_terminate_connection(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.terminate_connection(self.ctx, 'id1', 'connector')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.terminate_connection.assert_called_once_with('id1',
|
|
'connector')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_apiclient.exceptions.InternalServerError)
|
|
def test_terminate_connection_internal_server_error(
|
|
self, mock_cinderclient):
|
|
self.assertRaises(cinder_apiclient.exceptions.InternalServerError,
|
|
self.api.terminate_connection,
|
|
self.ctx, 'id1', 'connector')
|
|
|
|
self.assertEqual(5, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_terminate_connection_internal_server_error_do_not_raise(
|
|
self, mock_cinderclient):
|
|
# generate exception, and then have a normal return on the next retry
|
|
mock_cinderclient.return_value.volumes.terminate_connection.\
|
|
side_effect = [cinder_apiclient.exceptions.InternalServerError,
|
|
None]
|
|
|
|
self.api.terminate_connection(self.ctx, 'id1', 'connector')
|
|
|
|
self.assertEqual(2, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient',
|
|
side_effect=cinder_exception.BadRequest(code=400))
|
|
def test_terminate_connection_bad_request_exception(
|
|
self, mock_cinderclient):
|
|
self.assertRaises(exception.InvalidInput,
|
|
self.api.terminate_connection,
|
|
self.ctx, 'id1', 'connector')
|
|
|
|
self.assertEqual(1, mock_cinderclient.call_count)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_delete(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.delete(self.ctx, 'id1')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.delete.assert_called_once_with('id1')
|
|
|
|
def test_update(self):
|
|
self.assertRaises(NotImplementedError,
|
|
self.api.update, self.ctx, '', '')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_absolute_limits_forbidden(self, cinderclient):
|
|
"""Tests to make sure we gracefully handle a Forbidden error raised
|
|
from python-cinderclient when getting limits.
|
|
"""
|
|
cinderclient.return_value.limits.get.side_effect = (
|
|
cinder_exception.Forbidden(403))
|
|
self.assertRaises(
|
|
exception.Forbidden, self.api.get_absolute_limits, self.ctx)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_absolute_limits(self, cinderclient):
|
|
"""Tests the happy path of getting the absolute limits."""
|
|
expected_limits = {
|
|
"totalSnapshotsUsed": 0,
|
|
"maxTotalBackups": 10,
|
|
"maxTotalVolumeGigabytes": 1000,
|
|
"maxTotalSnapshots": 10,
|
|
"maxTotalBackupGigabytes": 1000,
|
|
"totalBackupGigabytesUsed": 0,
|
|
"maxTotalVolumes": 10,
|
|
"totalVolumesUsed": 0,
|
|
"totalBackupsUsed": 0,
|
|
"totalGigabytesUsed": 0
|
|
}
|
|
limits_obj = cinder_limits.Limits(None, {'absolute': expected_limits})
|
|
cinderclient.return_value.limits.get.return_value = limits_obj
|
|
actual_limits = self.api.get_absolute_limits(self.ctx)
|
|
self.assertDictEqual(expected_limits, actual_limits)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_snapshot(self, mock_cinderclient):
|
|
snapshot_id = 'snapshot_id'
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
|
|
self.api.get_snapshot(self.ctx, snapshot_id)
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.get.assert_called_once_with(snapshot_id)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_snapshot_failed_notfound(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volume_snapshots.get.side_effect = (
|
|
cinder_exception.NotFound(404, '404'))
|
|
|
|
self.assertRaises(exception.SnapshotNotFound,
|
|
self.api.get_snapshot, self.ctx, 'snapshot_id')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_snapshot_connection_failed(self, mock_cinderclient):
|
|
mock_cinderclient.return_value.volume_snapshots.get.side_effect = (
|
|
cinder_exception.ConnectionError(''))
|
|
|
|
self.assertRaises(exception.CinderConnectionFailed,
|
|
self.api.get_snapshot, self.ctx, 'snapshot_id')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_all_snapshots(self, mock_cinderclient):
|
|
snapshot1 = FakeSnapshot('snapshot_id1', 'id1')
|
|
snapshot2 = FakeSnapshot('snapshot_id2', 'id2')
|
|
|
|
snapshot_list = [snapshot1, snapshot2]
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
mock_volume_snapshots.list.return_value = snapshot_list
|
|
|
|
snapshots = self.api.get_all_snapshots(self.ctx)
|
|
self.assertEqual(2, len(snapshots))
|
|
self.assertEqual(['snapshot_id1', 'snapshot_id2'],
|
|
[snap['id'] for snap in snapshots])
|
|
self.assertEqual(['id1', 'id2'],
|
|
[snap['volume_id'] for snap in snapshots])
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.list.assert_called_once_with(detailed=True)
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create_snapshot(self, mock_cinderclient):
|
|
snapshot = FakeSnapshot('snapshot_id1', 'id1')
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
mock_volume_snapshots.create.return_value = snapshot
|
|
|
|
created_snapshot = self.api.create_snapshot(self.ctx,
|
|
'id1',
|
|
'name',
|
|
'description')
|
|
|
|
self.assertEqual('snapshot_id1', created_snapshot['id'])
|
|
self.assertEqual('id1', created_snapshot['volume_id'])
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.create.assert_called_once_with('id1', False,
|
|
'name',
|
|
'description')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_create_force(self, mock_cinderclient):
|
|
snapshot = FakeSnapshot('snapshot_id1', 'id1')
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
mock_volume_snapshots.create.return_value = snapshot
|
|
|
|
created_snapshot = self.api.create_snapshot_force(self.ctx,
|
|
'id1',
|
|
'name',
|
|
'description')
|
|
|
|
self.assertEqual('snapshot_id1', created_snapshot['id'])
|
|
self.assertEqual('id1', created_snapshot['volume_id'])
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.create.assert_called_once_with('id1', True,
|
|
'name',
|
|
'description')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_delete_snapshot(self, mock_cinderclient):
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
|
|
self.api.delete_snapshot(self.ctx, 'snapshot_id')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.delete.assert_called_once_with('snapshot_id')
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_update_snapshot_status(self, mock_cinderclient):
|
|
mock_volume_snapshots = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_snapshots=mock_volume_snapshots)
|
|
|
|
self.api.update_snapshot_status(self.ctx, 'snapshot_id', 'error')
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_snapshots.update_snapshot_status.assert_called_once_with(
|
|
'snapshot_id', {'status': 'error', 'progress': '90%'})
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_all_volume_types(self, mock_cinderclient):
|
|
volume_type1 = FakeVolumeType('lvm_1', 'volume_type_id1')
|
|
volume_type2 = FakeVolumeType('lvm_2', 'volume_type_id2')
|
|
volume_type_list = [volume_type1, volume_type2]
|
|
|
|
mock_volume_types = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(
|
|
volume_types=mock_volume_types)
|
|
mock_volume_types.list.return_value = volume_type_list
|
|
|
|
volume_types = self.api.get_all_volume_types(self.ctx)
|
|
self.assertEqual(2, len(volume_types))
|
|
self.assertEqual(['volume_type_id1', 'volume_type_id2'],
|
|
[vol_type['id'] for vol_type in volume_types])
|
|
self.assertEqual(['lvm_1', 'lvm_2'],
|
|
[vol_type['name'] for vol_type in volume_types])
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volume_types.list.assert_called_once_with()
|
|
|
|
@mock.patch('nova.volume.cinder.cinderclient')
|
|
def test_get_volume_encryption_metadata(self, mock_cinderclient):
|
|
mock_volumes = mock.MagicMock()
|
|
mock_cinderclient.return_value = mock.MagicMock(volumes=mock_volumes)
|
|
|
|
self.api.get_volume_encryption_metadata(self.ctx,
|
|
{'encryption_key_id':
|
|
'fake_key'})
|
|
|
|
mock_cinderclient.assert_called_once_with(self.ctx)
|
|
mock_volumes.get_encryption_metadata.assert_called_once_with(
|
|
{'encryption_key_id': 'fake_key'})
|
|
|
|
def test_translate_cinder_exception_no_error(self):
|
|
my_func = mock.Mock()
|
|
my_func.__name__ = 'my_func'
|
|
my_func.return_value = 'foo'
|
|
|
|
res = cinder.translate_cinder_exception(my_func)('fizzbuzz',
|
|
'bar', 'baz')
|
|
|
|
self.assertEqual('foo', res)
|
|
my_func.assert_called_once_with('fizzbuzz', 'bar', 'baz')
|
|
|
|
def test_translate_cinder_exception_cinder_connection_error(self):
|
|
self._do_translate_cinder_exception_test(
|
|
cinder_exception.ConnectionError,
|
|
exception.CinderConnectionFailed)
|
|
|
|
def test_translate_cinder_exception_keystone_connection_error(self):
|
|
self._do_translate_cinder_exception_test(
|
|
keystone_exception.ConnectionError,
|
|
exception.CinderConnectionFailed)
|
|
|
|
def test_translate_cinder_exception_cinder_bad_request(self):
|
|
self._do_translate_cinder_exception_test(
|
|
cinder_exception.BadRequest(400, '400'),
|
|
exception.InvalidInput)
|
|
|
|
def test_translate_cinder_exception_keystone_bad_request(self):
|
|
self._do_translate_cinder_exception_test(
|
|
keystone_exception.BadRequest,
|
|
exception.InvalidInput)
|
|
|
|
def test_translate_cinder_exception_cinder_forbidden(self):
|
|
self._do_translate_cinder_exception_test(
|
|
cinder_exception.Forbidden(403, '403'),
|
|
exception.Forbidden)
|
|
|
|
def test_translate_cinder_exception_keystone_forbidden(self):
|
|
self._do_translate_cinder_exception_test(
|
|
keystone_exception.Forbidden,
|
|
exception.Forbidden)
|
|
|
|
def test_translate_mixed_exception_over_limit(self):
|
|
self._do_translate_mixed_exception_test(
|
|
cinder_exception.OverLimit(''),
|
|
exception.OverQuota)
|
|
|
|
def test_translate_mixed_exception_volume_not_found(self):
|
|
self._do_translate_mixed_exception_test(
|
|
cinder_exception.NotFound(''),
|
|
exception.VolumeNotFound)
|
|
|
|
def test_translate_mixed_exception_keystone_not_found(self):
|
|
self._do_translate_mixed_exception_test(
|
|
keystone_exception.NotFound,
|
|
exception.VolumeNotFound)
|
|
|
|
def test_translate_create_exception_keystone_not_found(self):
|
|
self._do_translate_create_exception_test(
|
|
keystone_exception.NotFound,
|
|
exception.NotFound)
|
|
|
|
def test_translate_create_exception_volume_not_found(self):
|
|
self._do_translate_create_exception_test(
|
|
cinder_exception.NotFound('Volume type could not be found'),
|
|
exception.NotFound)
|
|
|
|
def _do_translate_cinder_exception_test(self, raised_exc, expected_exc):
|
|
self._do_translate_exception_test(raised_exc, expected_exc,
|
|
cinder.translate_cinder_exception)
|
|
|
|
def _do_translate_mixed_exception_test(self, raised_exc, expected_exc):
|
|
self._do_translate_exception_test(raised_exc, expected_exc,
|
|
cinder.translate_mixed_exceptions)
|
|
|
|
def _do_translate_create_exception_test(self, raised_exc, expected_exc):
|
|
self._do_translate_exception_test(raised_exc, expected_exc,
|
|
cinder.translate_create_exception)
|
|
|
|
def _do_translate_exception_test(self, raised_exc, expected_exc, wrapper):
|
|
my_func = mock.Mock()
|
|
my_func.__name__ = 'my_func'
|
|
my_func.side_effect = raised_exc
|
|
|
|
self.assertRaises(expected_exc, wrapper(my_func), 'foo', 'bar', 'baz')
|
|
|
|
|
|
class CinderClientTestCase(test.NoDBTestCase):
|
|
"""Used to test constructing a cinder client object at various versions."""
|
|
|
|
def setUp(self):
|
|
super(CinderClientTestCase, self).setUp()
|
|
cinder.reset_globals()
|
|
self.ctxt = context.RequestContext('fake-user', 'fake-project')
|
|
# Mock out the keystoneauth stuff.
|
|
self.mock_session = mock.Mock(autospec=session.Session)
|
|
patcher = mock.patch('keystoneauth1.loading.'
|
|
'load_session_from_conf_options',
|
|
return_value=self.mock_session)
|
|
patcher.start()
|
|
self.addCleanup(patcher.stop)
|
|
|
|
@mock.patch('cinderclient.client.get_volume_api_from_url',
|
|
return_value='3')
|
|
def test_create_v3_client_no_microversion(self, get_volume_api):
|
|
"""Tests that creating a v3 client, which is the default, and without
|
|
specifying a microversion will default to 3.0 as the version to use.
|
|
"""
|
|
client = cinder.cinderclient(self.ctxt)
|
|
self.assertEqual(cinder_api_versions.APIVersion('3.0'),
|
|
client.api_version)
|
|
get_volume_api.assert_called_once_with(
|
|
self.mock_session.get_endpoint.return_value)
|
|
|
|
@mock.patch('nova.volume.cinder._get_highest_client_server_version',
|
|
# Fake the case that cinder is really old.
|
|
return_value=cinder_api_versions.APIVersion('2.0'))
|
|
@mock.patch('cinderclient.client.get_volume_api_from_url',
|
|
return_value='3')
|
|
def test_create_v3_client_with_microversion_too_new(self,
|
|
get_volume_api,
|
|
get_highest_version):
|
|
"""Tests that creating a v3 client and requesting a microversion that
|
|
is either too new for the server (or client) to support raises an
|
|
exception.
|
|
"""
|
|
self.assertRaises(exception.CinderAPIVersionNotAvailable,
|
|
cinder.cinderclient, self.ctxt, microversion='3.44')
|
|
get_volume_api.assert_called_once_with(
|
|
self.mock_session.get_endpoint.return_value)
|
|
get_highest_version.assert_called_once_with(
|
|
self.ctxt, self.mock_session.get_endpoint.return_value)
|
|
|
|
@mock.patch('nova.volume.cinder._get_highest_client_server_version',
|
|
return_value=cinder_api_versions.APIVersion(
|
|
cinder_api_versions.MAX_VERSION))
|
|
@mock.patch('cinderclient.client.get_volume_api_from_url',
|
|
return_value='3')
|
|
def test_create_v3_client_with_microversion_available(self,
|
|
get_volume_api,
|
|
get_highest_version):
|
|
"""Tests that creating a v3 client and requesting a microversion that
|
|
is available in the server and supported by the client will result in
|
|
creating a Client object with the requested microversion.
|
|
"""
|
|
client = cinder.cinderclient(self.ctxt, microversion='3.44')
|
|
self.assertEqual(cinder_api_versions.APIVersion('3.44'),
|
|
client.api_version)
|
|
get_volume_api.assert_called_once_with(
|
|
self.mock_session.get_endpoint.return_value)
|
|
get_highest_version.assert_called_once_with(
|
|
self.ctxt, self.mock_session.get_endpoint.return_value)
|
|
|
|
@mock.patch('nova.volume.cinder._get_highest_client_server_version',
|
|
new_callable=mock.NonCallableMock) # asserts not called
|
|
@mock.patch('cinderclient.client.get_volume_api_from_url',
|
|
return_value='3')
|
|
def test_create_v3_client_with_microversion_skip_version_check(
|
|
self, get_volume_api, get_highest_version):
|
|
"""Tests that creating a v3 client and requesting a microversion
|
|
but asking to skip the version discovery check is honored.
|
|
"""
|
|
client = cinder.cinderclient(self.ctxt, microversion='3.44',
|
|
skip_version_check=True)
|
|
self.assertEqual(cinder_api_versions.APIVersion('3.44'),
|
|
client.api_version)
|
|
get_volume_api.assert_called_once_with(
|
|
self.mock_session.get_endpoint.return_value)
|
|
|
|
@mock.patch('nova.volume.cinder.LOG.error')
|
|
@mock.patch.object(ks_loading, 'load_auth_from_conf_options')
|
|
def test_load_auth_plugin_failed(self, mock_load_from_conf, mock_log_err):
|
|
mock_load_from_conf.return_value = None
|
|
self.assertRaises(cinder_exception.Unauthorized,
|
|
cinder._load_auth_plugin, CONF)
|
|
mock_log_err.assert_called()
|
|
self.assertIn('The [cinder] section of your nova configuration file',
|
|
mock_log_err.call_args[0][0])
|
|
|
|
@mock.patch('nova.volume.cinder._ADMIN_AUTH')
|
|
def test_admin_context_without_token(self,
|
|
mock_admin_auth):
|
|
|
|
mock_admin_auth.return_value = '_FAKE_ADMIN_AUTH'
|
|
admin_ctx = context.get_admin_context()
|
|
params = cinder._get_cinderclient_parameters(admin_ctx)
|
|
self.assertEqual(params[0], mock_admin_auth)
|