4dd1406289
Due to foreign key constraints for the resource_provider.root_provider_id in mysql and the way of column filling it is impossible to delete newly created RP now. Patch sets root_provider_id to NULL before deletion if root_provider_id == id. Closes-Bug: #1739571 Co-Authored-by: Andrey Volkov <avolkov@mirantis.com> Change-Id: I256c901fdc02b1f764ea9f8d1a1a708e82cc369a
486 lines
20 KiB
Python
486 lines
20 KiB
Python
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
import mock
|
|
import six
|
|
|
|
from oslo_utils import timeutils
|
|
|
|
import nova
|
|
from nova import context
|
|
from nova.db.sqlalchemy import api_models as models
|
|
from nova import exception
|
|
from nova.objects import fields
|
|
from nova.objects import resource_provider
|
|
from nova import test
|
|
from nova.tests.unit.objects import test_objects
|
|
from nova.tests import uuidsentinel as uuids
|
|
|
|
|
|
_RESOURCE_CLASS_NAME = 'DISK_GB'
|
|
_RESOURCE_CLASS_ID = 2
|
|
IPV4_ADDRESS_ID = fields.ResourceClass.STANDARD.index(
|
|
fields.ResourceClass.IPV4_ADDRESS)
|
|
VCPU_ID = fields.ResourceClass.STANDARD.index(
|
|
fields.ResourceClass.VCPU)
|
|
|
|
_RESOURCE_PROVIDER_ID = 1
|
|
_RESOURCE_PROVIDER_UUID = uuids.resource_provider
|
|
_RESOURCE_PROVIDER_NAME = six.text_type(uuids.resource_name)
|
|
_RESOURCE_PROVIDER_DB = {
|
|
'id': _RESOURCE_PROVIDER_ID,
|
|
'uuid': _RESOURCE_PROVIDER_UUID,
|
|
'name': _RESOURCE_PROVIDER_NAME,
|
|
'generation': 0,
|
|
'root_provider_uuid': _RESOURCE_PROVIDER_UUID,
|
|
'parent_provider_uuid': None,
|
|
'updated_at': None,
|
|
'created_at': timeutils.utcnow(with_timezone=True),
|
|
}
|
|
|
|
_RESOURCE_PROVIDER_ID2 = 2
|
|
_RESOURCE_PROVIDER_UUID2 = uuids.resource_provider2
|
|
_RESOURCE_PROVIDER_NAME2 = uuids.resource_name2
|
|
_RESOURCE_PROVIDER_DB2 = {
|
|
'id': _RESOURCE_PROVIDER_ID2,
|
|
'uuid': _RESOURCE_PROVIDER_UUID2,
|
|
'name': _RESOURCE_PROVIDER_NAME2,
|
|
'generation': 0,
|
|
'root_provider_uuid': _RESOURCE_PROVIDER_UUID,
|
|
'parent_provider_uuid': _RESOURCE_PROVIDER_UUID,
|
|
}
|
|
|
|
|
|
_INVENTORY_ID = 2
|
|
_INVENTORY_DB = {
|
|
'id': _INVENTORY_ID,
|
|
'resource_provider_id': _RESOURCE_PROVIDER_ID,
|
|
'resource_class_id': _RESOURCE_CLASS_ID,
|
|
'total': 16,
|
|
'reserved': 2,
|
|
'min_unit': 1,
|
|
'max_unit': 8,
|
|
'step_size': 1,
|
|
'allocation_ratio': 1.0,
|
|
'updated_at': None,
|
|
'created_at': timeutils.utcnow(with_timezone=True),
|
|
}
|
|
_ALLOCATION_ID = 2
|
|
_ALLOCATION_DB = {
|
|
'id': _ALLOCATION_ID,
|
|
'resource_provider_id': _RESOURCE_PROVIDER_ID,
|
|
'resource_class_id': _RESOURCE_CLASS_ID,
|
|
'consumer_id': uuids.fake_instance,
|
|
'used': 8,
|
|
'user_id': None,
|
|
'project_id': None,
|
|
}
|
|
|
|
|
|
def _fake_ensure_cache(ctxt):
|
|
cache = resource_provider._RC_CACHE = mock.MagicMock()
|
|
cache.string_from_id.return_value = _RESOURCE_CLASS_NAME
|
|
cache.id_from_string.return_value = _RESOURCE_CLASS_ID
|
|
|
|
|
|
class TestResourceProviderNoDB(test_objects._LocalTest):
|
|
USES_DB = False
|
|
|
|
def test_create_id_fail(self):
|
|
obj = resource_provider.ResourceProvider(context=self.context,
|
|
uuid=_RESOURCE_PROVIDER_UUID,
|
|
id=_RESOURCE_PROVIDER_ID)
|
|
self.assertRaises(exception.ObjectActionError,
|
|
obj.create)
|
|
|
|
def test_create_no_uuid_fail(self):
|
|
obj = resource_provider.ResourceProvider(context=self.context)
|
|
self.assertRaises(exception.ObjectActionError,
|
|
obj.create)
|
|
|
|
def test_create_with_root_provider_uuid_fail(self):
|
|
obj = resource_provider.ResourceProvider(
|
|
context=self.context,
|
|
uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME,
|
|
root_provider_uuid=_RESOURCE_PROVIDER_UUID,
|
|
)
|
|
|
|
exc = self.assertRaises(exception.ObjectActionError, obj.create)
|
|
self.assertIn('root provider UUID cannot be manually set', str(exc))
|
|
|
|
|
|
class TestProviderSummaryNoDB(test_objects._LocalTest):
|
|
USES_DB = False
|
|
|
|
def test_resource_class_names(self):
|
|
psum = resource_provider.ProviderSummary(mock.sentinel.ctx)
|
|
disk_psr = resource_provider.ProviderSummaryResource(
|
|
mock.sentinel.ctx, resource_class=fields.ResourceClass.DISK_GB,
|
|
capacity=100, used=0)
|
|
ram_psr = resource_provider.ProviderSummaryResource(
|
|
mock.sentinel.ctx, resource_class=fields.ResourceClass.MEMORY_MB,
|
|
capacity=1024, used=0)
|
|
psum.resources = [disk_psr, ram_psr]
|
|
expected = set(['DISK_GB', 'MEMORY_MB'])
|
|
self.assertEqual(expected, psum.resource_class_names)
|
|
|
|
|
|
class TestResourceProvider(test_objects._LocalTest):
|
|
|
|
def test_create_no_parent(self):
|
|
rp = resource_provider.ResourceProvider(
|
|
self.context, uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME)
|
|
rp.create()
|
|
self.assertIsInstance(rp.id, int)
|
|
self.assertEqual(_RESOURCE_PROVIDER_UUID, rp.uuid)
|
|
self.assertEqual(_RESOURCE_PROVIDER_NAME, rp.name)
|
|
self.assertEqual(_RESOURCE_PROVIDER_UUID, rp.root_provider_uuid)
|
|
self.assertIsNone(rp.parent_provider_uuid)
|
|
|
|
def test_create_in_db_with_parent_provider_uuid(self):
|
|
parent = resource_provider.ResourceProvider(
|
|
self.context, uuid=uuids.parent, name="parent")
|
|
parent.create()
|
|
child = resource_provider.ResourceProvider(
|
|
self.context, uuid=uuids.child, name="child",
|
|
parent_provider_uuid=uuids.parent)
|
|
child.create()
|
|
self.assertEqual(uuids.child, child.uuid)
|
|
self.assertEqual(uuids.parent, child.parent_provider_uuid)
|
|
self.assertEqual(uuids.parent, child.root_provider_uuid)
|
|
|
|
def test_save_immutable(self):
|
|
fields = {
|
|
'id': 1,
|
|
'uuid': _RESOURCE_PROVIDER_UUID,
|
|
'generation': 1,
|
|
'root_provider_uuid': _RESOURCE_PROVIDER_UUID,
|
|
}
|
|
for field in fields:
|
|
rp = resource_provider.ResourceProvider(context=self.context)
|
|
setattr(rp, field, fields[field])
|
|
self.assertRaises(exception.ObjectActionError, rp.save)
|
|
|
|
def test_get_by_uuid_from_db(self):
|
|
rp = resource_provider.ResourceProvider(context=self.context,
|
|
uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME)
|
|
rp.create()
|
|
retrieved_rp = resource_provider.ResourceProvider.get_by_uuid(
|
|
self.context, _RESOURCE_PROVIDER_UUID)
|
|
self.assertEqual(rp.uuid, retrieved_rp.uuid)
|
|
self.assertEqual(rp.name, retrieved_rp.name)
|
|
self.assertEqual(rp.root_provider_uuid,
|
|
retrieved_rp.root_provider_uuid)
|
|
|
|
def test_get_by_uuid_from_db_missing(self):
|
|
self.assertRaises(exception.NotFound,
|
|
resource_provider.ResourceProvider.get_by_uuid,
|
|
self.context, uuids.missing)
|
|
|
|
def test_destroy_with_traits(self):
|
|
"""Test deleting a resource provider that has a trait successfully.
|
|
"""
|
|
rp = resource_provider.ResourceProvider(self.context,
|
|
uuid=uuids.rp,
|
|
name='fake_rp1')
|
|
rp.create()
|
|
custom_trait = resource_provider.Trait(self.context,
|
|
uuid=uuids.trait,
|
|
name='CUSTOM_TRAIT_1')
|
|
custom_trait.create()
|
|
rp.set_traits([custom_trait])
|
|
|
|
trl = resource_provider.TraitList.get_all_by_resource_provider(
|
|
self.context, rp)
|
|
self.assertEqual(1, len(trl))
|
|
|
|
# Delete a resource provider that has a trait assosiation.
|
|
rp.destroy()
|
|
|
|
# Assert the record has been deleted
|
|
# in 'resource_provider_traits' table
|
|
# after Resource Provider object has been destroyed.
|
|
trl = resource_provider.TraitList.get_all_by_resource_provider(
|
|
self.context, rp)
|
|
self.assertEqual(0, len(trl))
|
|
# Assert that NotFound exception is raised.
|
|
self.assertRaises(exception.NotFound,
|
|
resource_provider.ResourceProvider.get_by_uuid,
|
|
self.context, uuids.rp)
|
|
|
|
def test_destroy(self):
|
|
|
|
def emulate_rp_mysql_delete(func):
|
|
def wrapped(context, _id):
|
|
rp = context.session.query(
|
|
models.ResourceProvider).\
|
|
filter(
|
|
models.ResourceProvider.id == _id).first()
|
|
self.assertIsNone(rp.root_provider_id)
|
|
return func(context, _id)
|
|
return wrapped
|
|
|
|
emulated = emulate_rp_mysql_delete(resource_provider._delete_rp_record)
|
|
|
|
rp = resource_provider.ResourceProvider(
|
|
self.context, uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME)
|
|
rp.create()
|
|
|
|
with mock.patch.object(
|
|
resource_provider, '_delete_rp_record', emulated):
|
|
rp.destroy()
|
|
|
|
|
|
class TestInventoryNoDB(test_objects._LocalTest):
|
|
USES_DB = False
|
|
|
|
@mock.patch('nova.objects.resource_provider._ensure_rc_cache',
|
|
side_effect=_fake_ensure_cache)
|
|
@mock.patch('nova.objects.resource_provider._get_inventory_by_provider_id')
|
|
def test_get_all_by_resource_provider(self, mock_get, mock_ensure_cache):
|
|
expected = [dict(_INVENTORY_DB,
|
|
resource_provider_id=_RESOURCE_PROVIDER_ID),
|
|
dict(_INVENTORY_DB,
|
|
id=_INVENTORY_DB['id'] + 1,
|
|
resource_provider_id=_RESOURCE_PROVIDER_ID)]
|
|
mock_get.return_value = expected
|
|
rp = resource_provider.ResourceProvider(id=_RESOURCE_PROVIDER_ID,
|
|
uuid=_RESOURCE_PROVIDER_UUID)
|
|
objs = resource_provider.InventoryList.get_all_by_resource_provider(
|
|
self.context, rp)
|
|
self.assertEqual(2, len(objs))
|
|
self.assertEqual(_INVENTORY_DB['id'], objs[0].id)
|
|
self.assertEqual(_INVENTORY_DB['id'] + 1, objs[1].id)
|
|
self.assertEqual(_RESOURCE_PROVIDER_ID, objs[0].resource_provider.id)
|
|
|
|
def test_non_negative_handling(self):
|
|
# NOTE(cdent): Just checking, useless to be actually
|
|
# comprehensive here.
|
|
rp = resource_provider.ResourceProvider(id=_RESOURCE_PROVIDER_ID,
|
|
uuid=_RESOURCE_PROVIDER_UUID)
|
|
kwargs = dict(resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
total=16,
|
|
reserved=2,
|
|
min_unit=1,
|
|
max_unit=-8,
|
|
step_size=1,
|
|
allocation_ratio=1.0)
|
|
self.assertRaises(ValueError, resource_provider.Inventory,
|
|
**kwargs)
|
|
|
|
def test_set_defaults(self):
|
|
rp = resource_provider.ResourceProvider(id=_RESOURCE_PROVIDER_ID,
|
|
uuid=_RESOURCE_PROVIDER_UUID)
|
|
kwargs = dict(resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
total=16)
|
|
inv = resource_provider.Inventory(self.context, **kwargs)
|
|
|
|
inv.obj_set_defaults()
|
|
self.assertEqual(0, inv.reserved)
|
|
self.assertEqual(1, inv.min_unit)
|
|
self.assertEqual(1, inv.max_unit)
|
|
self.assertEqual(1, inv.step_size)
|
|
self.assertEqual(1.0, inv.allocation_ratio)
|
|
|
|
def test_capacity(self):
|
|
rp = resource_provider.ResourceProvider(id=_RESOURCE_PROVIDER_ID,
|
|
uuid=_RESOURCE_PROVIDER_UUID)
|
|
kwargs = dict(resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
total=16,
|
|
reserved=16)
|
|
inv = resource_provider.Inventory(self.context, **kwargs)
|
|
inv.obj_set_defaults()
|
|
|
|
self.assertEqual(0, inv.capacity)
|
|
inv.reserved = 15
|
|
self.assertEqual(1, inv.capacity)
|
|
inv.allocation_ratio = 2.0
|
|
self.assertEqual(2, inv.capacity)
|
|
|
|
|
|
class TestInventoryList(test_objects._LocalTest):
|
|
|
|
def test_find(self):
|
|
rp = resource_provider.ResourceProvider(uuid=uuids.rp_uuid)
|
|
inv_list = resource_provider.InventoryList(objects=[
|
|
resource_provider.Inventory(
|
|
resource_provider=rp,
|
|
resource_class=fields.ResourceClass.VCPU,
|
|
total=24),
|
|
resource_provider.Inventory(
|
|
resource_provider=rp,
|
|
resource_class=fields.ResourceClass.MEMORY_MB,
|
|
total=10240),
|
|
])
|
|
|
|
found = inv_list.find(fields.ResourceClass.MEMORY_MB)
|
|
self.assertIsNotNone(found)
|
|
self.assertEqual(10240, found.total)
|
|
|
|
found = inv_list.find(fields.ResourceClass.VCPU)
|
|
self.assertIsNotNone(found)
|
|
self.assertEqual(24, found.total)
|
|
|
|
found = inv_list.find(fields.ResourceClass.DISK_GB)
|
|
self.assertIsNone(found)
|
|
|
|
# Try an integer resource class identifier...
|
|
self.assertRaises(ValueError, inv_list.find, VCPU_ID)
|
|
|
|
# Use an invalid string...
|
|
self.assertIsNone(inv_list.find('HOUSE'))
|
|
|
|
|
|
class TestAllocation(test_objects._LocalTest):
|
|
USES_DB = True
|
|
|
|
@mock.patch('nova.objects.resource_provider._ensure_rc_cache',
|
|
side_effect=_fake_ensure_cache)
|
|
def test_create(self, mock_ensure_cache):
|
|
rp = resource_provider.ResourceProvider(context=self.context,
|
|
uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME)
|
|
rp.create()
|
|
inv = resource_provider.Inventory(context=self.context,
|
|
resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
total=16,
|
|
reserved=2,
|
|
min_unit=1,
|
|
max_unit=8,
|
|
step_size=1,
|
|
allocation_ratio=1.0)
|
|
inv_list = resource_provider.InventoryList(context=self.context,
|
|
objects=[inv])
|
|
rp.set_inventory(inv_list)
|
|
obj = resource_provider.Allocation(context=self.context,
|
|
resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
consumer_id=uuids.fake_instance,
|
|
used=8)
|
|
alloc_list = resource_provider.AllocationList(self.context,
|
|
objects=[obj])
|
|
alloc_list.create_all()
|
|
|
|
rp_al = resource_provider.AllocationList
|
|
saved_allocations = rp_al.get_all_by_resource_provider(
|
|
self.context, rp)
|
|
self.assertEqual(1, len(saved_allocations))
|
|
self.assertEqual(obj.used, saved_allocations[0].used)
|
|
|
|
def test_create_with_id_fails(self):
|
|
rp = resource_provider.ResourceProvider(context=self.context,
|
|
uuid=_RESOURCE_PROVIDER_UUID,
|
|
name=_RESOURCE_PROVIDER_NAME)
|
|
rp.create()
|
|
inv = resource_provider.Inventory(context=self.context,
|
|
resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
total=16,
|
|
reserved=2,
|
|
min_unit=1,
|
|
max_unit=8,
|
|
step_size=1,
|
|
allocation_ratio=1.0)
|
|
inv_list = resource_provider.InventoryList(context=self.context,
|
|
objects=[inv])
|
|
rp.set_inventory(inv_list)
|
|
obj = resource_provider.Allocation(context=self.context,
|
|
id=99,
|
|
resource_provider=rp,
|
|
resource_class=_RESOURCE_CLASS_NAME,
|
|
consumer_id=uuids.fake_instance,
|
|
used=8)
|
|
alloc_list = resource_provider.AllocationList(self.context,
|
|
objects=[obj])
|
|
self.assertRaises(exception.ObjectActionError, alloc_list.create_all)
|
|
|
|
|
|
class TestAllocationListNoDB(test_objects._LocalTest):
|
|
USES_DB = False
|
|
|
|
@mock.patch('nova.objects.resource_provider._ensure_rc_cache',
|
|
side_effect=_fake_ensure_cache)
|
|
@mock.patch('nova.objects.resource_provider.'
|
|
'_get_allocations_by_provider_id',
|
|
return_value=[_ALLOCATION_DB])
|
|
def test_get_allocations(self, mock_get_allocations_from_db,
|
|
mock_ensure_cache):
|
|
rp = resource_provider.ResourceProvider(id=_RESOURCE_PROVIDER_ID,
|
|
uuid=uuids.resource_provider)
|
|
rp_alloc_list = resource_provider.AllocationList
|
|
allocations = rp_alloc_list.get_all_by_resource_provider(
|
|
self.context, rp)
|
|
|
|
self.assertEqual(1, len(allocations))
|
|
mock_get_allocations_from_db.assert_called_once_with(self.context,
|
|
rp.id)
|
|
self.assertEqual(_ALLOCATION_DB['used'], allocations[0].used)
|
|
|
|
|
|
class TestResourceClass(test.NoDBTestCase):
|
|
|
|
def setUp(self):
|
|
super(TestResourceClass, self).setUp()
|
|
self.user_id = 'fake-user'
|
|
self.project_id = 'fake-project'
|
|
self.context = context.RequestContext(self.user_id, self.project_id)
|
|
|
|
def test_cannot_create_with_id(self):
|
|
rc = resource_provider.ResourceClass(self.context, id=1,
|
|
name='CUSTOM_IRON_NFV')
|
|
exc = self.assertRaises(exception.ObjectActionError, rc.create)
|
|
self.assertIn('already created', str(exc))
|
|
|
|
def test_cannot_create_requires_name(self):
|
|
rc = resource_provider.ResourceClass(self.context)
|
|
exc = self.assertRaises(exception.ObjectActionError, rc.create)
|
|
self.assertIn('name is required', str(exc))
|
|
|
|
|
|
class TestTraits(test.NoDBTestCase):
|
|
|
|
def setUp(self):
|
|
super(TestTraits, self).setUp()
|
|
self.user_id = 'fake-user'
|
|
self.project_id = 'fake-project'
|
|
self.context = context.RequestContext(self.user_id, self.project_id)
|
|
|
|
@mock.patch("nova.objects.resource_provider._trait_sync")
|
|
def test_sync_flag(self, mock_sync):
|
|
synced = nova.objects.resource_provider._TRAITS_SYNCED
|
|
self.assertFalse(synced)
|
|
# Sync the traits
|
|
nova.objects.resource_provider._ensure_trait_sync(self.context)
|
|
synced = nova.objects.resource_provider._TRAITS_SYNCED
|
|
self.assertTrue(synced)
|
|
|
|
@mock.patch('nova.objects.resource_provider.ResourceProvider.'
|
|
'obj_reset_changes')
|
|
@mock.patch('nova.objects.resource_provider._set_traits')
|
|
def test_set_traits_resets_changes(self, mock_set_traits, mock_reset):
|
|
trait = resource_provider.Trait(name="HW_CPU_X86_AVX2")
|
|
traits = resource_provider.TraitList(objects=[trait])
|
|
|
|
rp = resource_provider.ResourceProvider(self.context, name='cn1',
|
|
uuid=uuids.cn1)
|
|
rp.set_traits(traits)
|
|
mock_set_traits.assert_called_once_with(self.context, rp, traits)
|
|
mock_reset.assert_called_once_with()
|