51fe944449
As mentioned in [1], we should avoid using six.iteritems(iter) to achieve iterators. We can use iter.items() instead, as it will return iterators in PY3 as well. [1] http://lists.openstack.org/pipermail/openstack-dev/2015-June/066391.html Change-Id: Ieadd5fa69b3f6058c0641a5b96ef3e8e1e3f6cc2
471 lines
21 KiB
Python
471 lines
21 KiB
Python
# Copyright 2012 Red Hat, Inc.
|
|
#
|
|
# 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 django.conf import settings
|
|
from django.test.utils import override_settings
|
|
|
|
import cinderclient as cinder_client
|
|
|
|
from openstack_dashboard import api
|
|
from openstack_dashboard.test import helpers as test
|
|
|
|
|
|
class CinderApiTests(test.APITestCase):
|
|
|
|
def test_volume_list(self):
|
|
search_opts = {'all_tenants': 1}
|
|
detailed = True
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts,).AndReturn(volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=detailed,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes = api.cinder.volume_list(self.request,
|
|
search_opts=search_opts)
|
|
self.assertEqual(len(volumes), len(api_volumes))
|
|
|
|
def test_volume_list_paged(self):
|
|
search_opts = {'all_tenants': 1}
|
|
detailed = True
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts,).AndReturn(volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=detailed,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, has_more, has_prev = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts)
|
|
self.assertEqual(len(volumes), len(api_volumes))
|
|
self.assertFalse(has_more)
|
|
self.assertFalse(has_prev)
|
|
|
|
@override_settings(API_RESULT_PAGE_SIZE=2)
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_volume_list_paginate_first_page(self):
|
|
api.cinder.VERSIONS._active = None
|
|
page_size = settings.API_RESULT_PAGE_SIZE
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
|
|
search_opts = {'all_tenants': 1}
|
|
mox_volumes = volumes[:page_size + 1]
|
|
expected_volumes = mox_volumes[:-1]
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts, limit=page_size + 1,
|
|
sort='created_at:desc', marker=None).\
|
|
AndReturn(mox_volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=True,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, more_data, prev_data = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts, paginate=True)
|
|
self.assertEqual(len(expected_volumes), len(api_volumes))
|
|
self.assertTrue(more_data)
|
|
self.assertFalse(prev_data)
|
|
|
|
@override_settings(API_RESULT_PAGE_SIZE=2)
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_volume_list_paginate_second_page(self):
|
|
api.cinder.VERSIONS._active = None
|
|
page_size = settings.API_RESULT_PAGE_SIZE
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
|
|
search_opts = {'all_tenants': 1}
|
|
mox_volumes = volumes[page_size:page_size * 2 + 1]
|
|
expected_volumes = mox_volumes[:-1]
|
|
marker = expected_volumes[0].id
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts, limit=page_size + 1,
|
|
sort='created_at:desc', marker=marker).\
|
|
AndReturn(mox_volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=True,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, more_data, prev_data = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts, marker=marker,
|
|
paginate=True)
|
|
self.assertEqual(len(expected_volumes), len(api_volumes))
|
|
self.assertTrue(more_data)
|
|
self.assertTrue(prev_data)
|
|
|
|
@override_settings(API_RESULT_PAGE_SIZE=2)
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_volume_list_paginate_last_page(self):
|
|
api.cinder.VERSIONS._active = None
|
|
page_size = settings.API_RESULT_PAGE_SIZE
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
|
|
search_opts = {'all_tenants': 1}
|
|
mox_volumes = volumes[-1 * page_size:]
|
|
expected_volumes = mox_volumes
|
|
marker = expected_volumes[0].id
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts, limit=page_size + 1,
|
|
sort='created_at:desc', marker=marker).\
|
|
AndReturn(mox_volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=True,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, more_data, prev_data = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts, marker=marker,
|
|
paginate=True)
|
|
self.assertEqual(len(expected_volumes), len(api_volumes))
|
|
self.assertFalse(more_data)
|
|
self.assertTrue(prev_data)
|
|
|
|
@override_settings(API_RESULT_PAGE_SIZE=2)
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_volume_list_paginate_back_from_some_page(self):
|
|
api.cinder.VERSIONS._active = None
|
|
page_size = settings.API_RESULT_PAGE_SIZE
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
|
|
search_opts = {'all_tenants': 1}
|
|
mox_volumes = volumes[page_size:page_size * 2 + 1]
|
|
expected_volumes = mox_volumes[:-1]
|
|
marker = expected_volumes[0].id
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts, limit=page_size + 1,
|
|
sort='created_at:asc', marker=marker).\
|
|
AndReturn(mox_volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=True,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, more_data, prev_data = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts, sort_dir="asc",
|
|
marker=marker, paginate=True)
|
|
self.assertEqual(len(expected_volumes), len(api_volumes))
|
|
self.assertTrue(more_data)
|
|
self.assertTrue(prev_data)
|
|
|
|
@override_settings(API_RESULT_PAGE_SIZE=2)
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_volume_list_paginate_back_to_first_page(self):
|
|
api.cinder.VERSIONS._active = None
|
|
page_size = settings.API_RESULT_PAGE_SIZE
|
|
volumes = self.cinder_volumes.list()
|
|
volume_transfers = self.cinder_volume_transfers.list()
|
|
|
|
search_opts = {'all_tenants': 1}
|
|
mox_volumes = volumes[:page_size]
|
|
expected_volumes = mox_volumes
|
|
marker = expected_volumes[0].id
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volumes = self.mox.CreateMockAnything()
|
|
cinderclient.volumes.list(search_opts=search_opts, limit=page_size + 1,
|
|
sort='created_at:asc', marker=marker).\
|
|
AndReturn(mox_volumes)
|
|
cinderclient.transfers = self.mox.CreateMockAnything()
|
|
cinderclient.transfers.list(
|
|
detailed=True,
|
|
search_opts=search_opts,).AndReturn(volume_transfers)
|
|
self.mox.ReplayAll()
|
|
|
|
api_volumes, more_data, prev_data = api.cinder.volume_list_paged(
|
|
self.request, search_opts=search_opts, sort_dir="asc",
|
|
marker=marker, paginate=True)
|
|
self.assertEqual(len(expected_volumes), len(api_volumes))
|
|
self.assertTrue(more_data)
|
|
self.assertFalse(prev_data)
|
|
|
|
def test_volume_snapshot_list(self):
|
|
search_opts = {'all_tenants': 1}
|
|
volume_snapshots = self.cinder_volume_snapshots.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volume_snapshots = self.mox.CreateMockAnything()
|
|
cinderclient.volume_snapshots.list(search_opts=search_opts).\
|
|
AndReturn(volume_snapshots)
|
|
self.mox.ReplayAll()
|
|
|
|
api.cinder.volume_snapshot_list(self.request, search_opts=search_opts)
|
|
|
|
def test_volume_snapshot_list_no_volume_configured(self):
|
|
# remove volume from service catalog
|
|
catalog = self.service_catalog
|
|
for service in catalog:
|
|
if service["type"] == "volume":
|
|
self.service_catalog.remove(service)
|
|
search_opts = {'all_tenants': 1}
|
|
volume_snapshots = self.cinder_volume_snapshots.list()
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volume_snapshots = self.mox.CreateMockAnything()
|
|
cinderclient.volume_snapshots.list(search_opts=search_opts).\
|
|
AndReturn(volume_snapshots)
|
|
self.mox.ReplayAll()
|
|
|
|
api.cinder.volume_snapshot_list(self.request, search_opts=search_opts)
|
|
|
|
def test_volume_type_list_with_qos_associations(self):
|
|
volume_types = self.cinder_volume_types.list()
|
|
# Due to test data limitations, we can only run this test using
|
|
# one qos spec, which is associated with one volume type.
|
|
# If we use multiple qos specs, the test data will always
|
|
# return the same associated volume type, which is invalid
|
|
# and prevented by the UI.
|
|
qos_specs_full = self.cinder_qos_specs.list()
|
|
qos_specs_only_one = [qos_specs_full[0]]
|
|
associations = self.cinder_qos_spec_associations.list()
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volume_types = self.mox.CreateMockAnything()
|
|
cinderclient.volume_types.list().AndReturn(volume_types)
|
|
cinderclient.qos_specs = self.mox.CreateMockAnything()
|
|
cinderclient.qos_specs.list().AndReturn(qos_specs_only_one)
|
|
cinderclient.qos_specs.get_associations = self.mox.CreateMockAnything()
|
|
cinderclient.qos_specs.get_associations(qos_specs_only_one[0].id).\
|
|
AndReturn(associations)
|
|
self.mox.ReplayAll()
|
|
|
|
assoc_vol_types = \
|
|
api.cinder.volume_type_list_with_qos_associations(self.request)
|
|
associate_spec = assoc_vol_types[0].associated_qos_spec
|
|
self.assertEqual(associate_spec, qos_specs_only_one[0].name)
|
|
|
|
def test_volume_type_get_with_qos_association(self):
|
|
volume_type = self.cinder_volume_types.first()
|
|
qos_specs_full = self.cinder_qos_specs.list()
|
|
qos_specs_only_one = [qos_specs_full[0]]
|
|
associations = self.cinder_qos_spec_associations.list()
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volume_types = self.mox.CreateMockAnything()
|
|
cinderclient.volume_types.get(volume_type.id).AndReturn(volume_type)
|
|
cinderclient.qos_specs = self.mox.CreateMockAnything()
|
|
cinderclient.qos_specs.list().AndReturn(qos_specs_only_one)
|
|
cinderclient.qos_specs.get_associations = self.mox.CreateMockAnything()
|
|
cinderclient.qos_specs.get_associations(qos_specs_only_one[0].id).\
|
|
AndReturn(associations)
|
|
self.mox.ReplayAll()
|
|
|
|
assoc_vol_type = \
|
|
api.cinder.volume_type_get_with_qos_association(self.request,
|
|
volume_type.id)
|
|
associate_spec = assoc_vol_type.associated_qos_spec
|
|
self.assertEqual(associate_spec, qos_specs_only_one[0].name)
|
|
|
|
def test_absolute_limits_with_negative_values(self):
|
|
values = {"maxTotalVolumes": -1, "totalVolumesUsed": -1}
|
|
expected_results = {"maxTotalVolumes": float("inf"),
|
|
"totalVolumesUsed": 0}
|
|
|
|
limits = self.mox.CreateMockAnything()
|
|
limits.absolute = []
|
|
for key, val in values.items():
|
|
limit = self.mox.CreateMockAnything()
|
|
limit.name = key
|
|
limit.value = val
|
|
limits.absolute.append(limit)
|
|
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.limits = self.mox.CreateMockAnything()
|
|
cinderclient.limits.get().AndReturn(limits)
|
|
self.mox.ReplayAll()
|
|
|
|
ret_val = api.cinder.tenant_absolute_limits(self.request)
|
|
for key in expected_results.keys():
|
|
self.assertEqual(expected_results[key], ret_val[key])
|
|
|
|
def test_pool_list(self):
|
|
pools = self.cinder_pools.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.pools = self.mox.CreateMockAnything()
|
|
cinderclient.pools.list(detailed=True).AndReturn(pools)
|
|
self.mox.ReplayAll()
|
|
|
|
# No assertions are necessary. Verification is handled by mox.
|
|
api.cinder.pool_list(self.request, detailed=True)
|
|
|
|
def test_volume_type_default(self):
|
|
volume_type = self.cinder_volume_types.first()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.volume_types = self.mox.CreateMockAnything()
|
|
cinderclient.volume_types.default().AndReturn(volume_type)
|
|
self.mox.ReplayAll()
|
|
|
|
default_volume_type = api.cinder.volume_type_default(self.request)
|
|
self.assertEqual(default_volume_type, volume_type)
|
|
|
|
def test_cgroup_list(self):
|
|
cgroups = self.cinder_consistencygroups.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.consistencygroups = self.mox.CreateMockAnything()
|
|
cinderclient.consistencygroups.list(search_opts=None).\
|
|
AndReturn(cgroups)
|
|
self.mox.ReplayAll()
|
|
api_cgroups = api.cinder.volume_cgroup_list(self.request)
|
|
self.assertEqual(len(cgroups), len(api_cgroups))
|
|
|
|
def test_cgroup_get(self):
|
|
cgroup = self.cinder_consistencygroups.first()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.consistencygroups = self.mox.CreateMockAnything()
|
|
cinderclient.consistencygroups.get(cgroup.id).AndReturn(cgroup)
|
|
self.mox.ReplayAll()
|
|
api_cgroup = api.cinder.volume_cgroup_get(self.request, cgroup.id)
|
|
self.assertEqual(api_cgroup.name, cgroup.name)
|
|
self.assertEqual(api_cgroup.description, cgroup.description)
|
|
self.assertEqual(api_cgroup.volume_types, cgroup.volume_types)
|
|
|
|
def test_cgroup_list_with_vol_type_names(self):
|
|
cgroups = self.cinder_consistencygroups.list()
|
|
volume_types_list = self.cinder_volume_types.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.consistencygroups = self.mox.CreateMockAnything()
|
|
cinderclient.consistencygroups.list(search_opts=None).\
|
|
AndReturn(cgroups)
|
|
cinderclient.volume_types = self.mox.CreateMockAnything()
|
|
cinderclient.volume_types.list().AndReturn(volume_types_list)
|
|
self.mox.ReplayAll()
|
|
api_cgroups = api.cinder.volume_cgroup_list_with_vol_type_names(
|
|
self.request)
|
|
self.assertEqual(len(cgroups), len(api_cgroups))
|
|
for i in range(len(api_cgroups[0].volume_type_names)):
|
|
self.assertEqual(volume_types_list[i].name,
|
|
api_cgroups[0].volume_type_names[i])
|
|
|
|
def test_cgsnapshot_list(self):
|
|
cgsnapshots = self.cinder_cg_snapshots.list()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.cgsnapshots = self.mox.CreateMockAnything()
|
|
cinderclient.cgsnapshots.list(search_opts=None).\
|
|
AndReturn(cgsnapshots)
|
|
self.mox.ReplayAll()
|
|
api_cgsnapshots = api.cinder.volume_cg_snapshot_list(self.request)
|
|
self.assertEqual(len(cgsnapshots), len(api_cgsnapshots))
|
|
|
|
def test_cgsnapshot_get(self):
|
|
cgsnapshot = self.cinder_cg_snapshots.first()
|
|
cinderclient = self.stub_cinderclient()
|
|
cinderclient.cgsnapshots = self.mox.CreateMockAnything()
|
|
cinderclient.cgsnapshots.get(cgsnapshot.id).AndReturn(cgsnapshot)
|
|
self.mox.ReplayAll()
|
|
api_cgsnapshot = api.cinder.volume_cg_snapshot_get(self.request,
|
|
cgsnapshot.id)
|
|
self.assertEqual(api_cgsnapshot.name, cgsnapshot.name)
|
|
self.assertEqual(api_cgsnapshot.description, cgsnapshot.description)
|
|
self.assertEqual(api_cgsnapshot.consistencygroup_id,
|
|
cgsnapshot.consistencygroup_id)
|
|
|
|
|
|
class CinderApiVersionTests(test.TestCase):
|
|
|
|
def setUp(self):
|
|
super(CinderApiVersionTests, self).setUp()
|
|
# The version is set when the module is loaded. Reset the
|
|
# active version each time so that we can test with different
|
|
# versions.
|
|
api.cinder.VERSIONS._active = None
|
|
|
|
def test_default_client_is_v2(self):
|
|
client = api.cinder.cinderclient(self.request)
|
|
self.assertIsInstance(client, cinder_client.v2.client.Client)
|
|
|
|
@override_settings(OPENSTACK_API_VERSIONS={'volume': 2})
|
|
def test_v2_setting_returns_v2_client(self):
|
|
client = api.cinder.cinderclient(self.request)
|
|
self.assertIsInstance(client, cinder_client.v2.client.Client)
|
|
|
|
def test_get_v2_volume_attributes(self):
|
|
# Get a v2 volume
|
|
volume = self.cinder_volumes.get(name="v2_volume")
|
|
self.assertTrue(hasattr(volume._apiresource, 'name'))
|
|
self.assertFalse(hasattr(volume._apiresource, 'display_name'))
|
|
|
|
name = "A v2 test volume name"
|
|
description = "A v2 volume description"
|
|
setattr(volume._apiresource, 'name', name)
|
|
setattr(volume._apiresource, 'description', description)
|
|
self.assertEqual(name, volume.name)
|
|
self.assertEqual(description, volume.description)
|
|
|
|
def test_get_v2_snapshot_attributes(self):
|
|
# Get a v2 snapshot
|
|
snapshot = self.cinder_volume_snapshots.get(
|
|
description="v2 volume snapshot description")
|
|
self.assertFalse(hasattr(snapshot._apiresource, 'display_name'))
|
|
|
|
name = "A v2 test snapshot name"
|
|
description = "A v2 snapshot description"
|
|
setattr(snapshot._apiresource, 'name', name)
|
|
setattr(snapshot._apiresource, 'description', description)
|
|
self.assertEqual(name, snapshot.name)
|
|
self.assertEqual(description, snapshot.description)
|
|
|
|
def test_get_v2_snapshot_metadata(self):
|
|
# Get a v2 snapshot with metadata
|
|
snapshot = self.cinder_volume_snapshots.get(
|
|
description="v2 volume snapshot with metadata description")
|
|
self.assertTrue(hasattr(snapshot._apiresource, 'metadata'))
|
|
self.assertFalse(hasattr(snapshot._apiresource, 'display_name'))
|
|
|
|
key_name = "snapshot_meta_key"
|
|
key_value = "snapshot_meta_value"
|
|
metadata_value = {key_name: key_value}
|
|
setattr(snapshot._apiresource, 'metadata', metadata_value)
|
|
self.assertIn(key_name, snapshot.metadata.keys())
|
|
self.assertEqual(key_value, snapshot.metadata['snapshot_meta_key'])
|
|
|
|
def test_get_id_for_nameless_volume(self):
|
|
volume = self.cinder_volumes.first()
|
|
setattr(volume._apiresource, 'display_name', "")
|
|
self.assertEqual(volume.id, volume.name)
|
|
|
|
def test_adapt_dictionary_to_v2(self):
|
|
volume = self.cinder_volumes.first()
|
|
data = {'name': volume.name,
|
|
'description': volume.description,
|
|
'size': volume.size}
|
|
|
|
ret_data = api.cinder._replace_v2_parameters(data)
|
|
self.assertIn('name', ret_data.keys())
|
|
self.assertIn('description', ret_data.keys())
|
|
self.assertNotIn('display_name', ret_data.keys())
|
|
self.assertNotIn('display_description', ret_data.keys())
|