
These bugs creates a lot of logs/traces/warnings during unit testing Release-notes-impact: CinderVolumes.create_nested_snapshots_and_attach_volume removed abbility to pass nested_level as a dict, use always int now. Change-Id: Ie0cd81da00a2a5d9129a9df38d9d5abe5647b695
456 lines
19 KiB
Python
456 lines
19 KiB
Python
# Copyright 2014: Mirantis Inc.
|
|
# 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.
|
|
|
|
import mock
|
|
import six
|
|
|
|
from rally.plugins.openstack.cleanup import base
|
|
from rally.plugins.openstack.cleanup import manager
|
|
from tests.unit import test
|
|
|
|
|
|
BASE = "rally.plugins.openstack.cleanup.manager"
|
|
|
|
|
|
class SeekAndDestroyTestCase(test.TestCase):
|
|
|
|
def setUp(self):
|
|
super(SeekAndDestroyTestCase, self).setUp()
|
|
# clear out the client cache
|
|
manager.SeekAndDestroy.cache = {}
|
|
|
|
@mock.patch("%s.osclients.Clients" % BASE,
|
|
side_effect=[mock.MagicMock(), mock.MagicMock()])
|
|
def test__get_cached_client(self, mock_clients):
|
|
destroyer = manager.SeekAndDestroy(None, None, None)
|
|
|
|
self.assertIsNone(destroyer._get_cached_client(None))
|
|
|
|
users = [{"credential": "a"}, {"credential": "b"}]
|
|
|
|
self.assertEqual(destroyer._get_cached_client(users[0]),
|
|
destroyer._get_cached_client(users[0]))
|
|
# ensure that cache is used
|
|
self.assertItemsEqual(mock_clients.call_args_list,
|
|
[mock.call("a", api_info=None)])
|
|
|
|
mock_clients.reset_mock()
|
|
self.assertEqual(destroyer._get_cached_client(users[1]),
|
|
destroyer._get_cached_client(users[1]))
|
|
self.assertItemsEqual(mock_clients.call_args_list,
|
|
[mock.call("b", api_info=None)])
|
|
|
|
mock_clients.reset_mock()
|
|
self.assertNotEqual(destroyer._get_cached_client(users[0]),
|
|
destroyer._get_cached_client(users[1]))
|
|
self.assertFalse(mock_clients.called)
|
|
|
|
@mock.patch("%s.osclients.Clients" % BASE,
|
|
side_effect=[mock.MagicMock(), mock.MagicMock()])
|
|
def test__get_cached_client_shared_cache(self, mock_clients):
|
|
# ensure that cache is shared between SeekAndDestroy objects
|
|
destroyer1 = manager.SeekAndDestroy(None, None, None)
|
|
destroyer2 = manager.SeekAndDestroy(None, None, None)
|
|
|
|
user = {"credential": "a"}
|
|
|
|
self.assertEqual(destroyer1._get_cached_client(user),
|
|
destroyer2._get_cached_client(user))
|
|
self.assertItemsEqual(mock_clients.call_args_list,
|
|
[mock.call("a", api_info=None)])
|
|
|
|
@mock.patch("%s.osclients.Clients" % BASE,
|
|
side_effect=[mock.MagicMock(), mock.MagicMock()])
|
|
def test__get_cached_client_shared_cache_api_versions(self, mock_clients):
|
|
# ensure that cache is shared between SeekAndDestroy objects
|
|
# with matching api_versions dicts
|
|
api_versions = {"cinder": {"version": "1", "service_type": "volume"}}
|
|
|
|
destroyer1 = manager.SeekAndDestroy(None, None, None,
|
|
api_versions=api_versions)
|
|
destroyer2 = manager.SeekAndDestroy(None, None, None,
|
|
api_versions=api_versions)
|
|
|
|
user = {"credential": "a"}
|
|
|
|
self.assertEqual(destroyer1._get_cached_client(user),
|
|
destroyer2._get_cached_client(user))
|
|
self.assertItemsEqual(mock_clients.call_args_list,
|
|
[mock.call("a", api_info=api_versions)])
|
|
|
|
@mock.patch("%s.osclients.Clients" % BASE,
|
|
side_effect=[mock.MagicMock(), mock.MagicMock()])
|
|
def test__get_cached_client_no_cache_api_versions(self, mock_clients):
|
|
# ensure that cache is not shared between SeekAndDestroy
|
|
# objects with different api_versions dicts
|
|
api_versions = [
|
|
{"cinder": {"version": "1", "service_type": "volume"}},
|
|
{"cinder": {"version": "2", "service_type": "volumev2"}}
|
|
]
|
|
|
|
destroyer1 = manager.SeekAndDestroy(None, None, None,
|
|
api_versions=api_versions[0])
|
|
destroyer2 = manager.SeekAndDestroy(None, None, None,
|
|
api_versions=api_versions[1])
|
|
user = {"credential": "a"}
|
|
|
|
self.assertNotEqual(destroyer1._get_cached_client(user),
|
|
destroyer2._get_cached_client(user))
|
|
self.assertItemsEqual(mock_clients.call_args_list,
|
|
[mock.call("a", api_info=api_versions[0]),
|
|
mock.call("a", api_info=api_versions[1])])
|
|
|
|
@mock.patch("%s.LOG" % BASE)
|
|
def test__delete_single_resource(self, mock_log):
|
|
mock_resource = mock.MagicMock(_max_attempts=3, _timeout=10,
|
|
_interval=0.01)
|
|
mock_resource.delete.side_effect = [Exception, Exception, True]
|
|
mock_resource.is_deleted.side_effect = [False, False, True]
|
|
|
|
manager.SeekAndDestroy(None, None, None)._delete_single_resource(
|
|
mock_resource)
|
|
|
|
mock_resource.delete.assert_has_calls([mock.call()] * 3)
|
|
self.assertEqual(mock_resource.delete.call_count, 3)
|
|
mock_resource.is_deleted.assert_has_calls([mock.call()] * 3)
|
|
self.assertEqual(mock_resource.is_deleted.call_count, 3)
|
|
|
|
# NOTE(boris-42): No logs and no exceptions means no bugs!
|
|
self.assertEqual(0, mock_log.call_count)
|
|
|
|
@mock.patch("%s.LOG" % BASE)
|
|
def test__delete_single_resource_timeout(self, mock_log):
|
|
|
|
mock_resource = mock.MagicMock(_max_attempts=1, _timeout=0.02,
|
|
_interval=0.025)
|
|
|
|
mock_resource.delete.return_value = True
|
|
mock_resource.is_deleted.side_effect = [False, False, True]
|
|
|
|
manager.SeekAndDestroy(None, None, None)._delete_single_resource(
|
|
mock_resource)
|
|
|
|
mock_resource.delete.assert_called_once_with()
|
|
mock_resource.is_deleted.assert_called_once_with()
|
|
|
|
self.assertEqual(1, mock_log.warning.call_count)
|
|
|
|
@mock.patch("%s.LOG" % BASE)
|
|
def test__delete_single_resource_excpetion_in_is_deleted(self, mock_log):
|
|
mock_resource = mock.MagicMock(_max_attempts=3, _timeout=10,
|
|
_interval=0)
|
|
mock_resource.delete.return_value = True
|
|
mock_resource.is_deleted.side_effect = [Exception] * 4
|
|
manager.SeekAndDestroy(None, None, None)._delete_single_resource(
|
|
mock_resource)
|
|
|
|
mock_resource.delete.assert_called_once_with()
|
|
self.assertEqual(4, mock_resource.is_deleted.call_count)
|
|
|
|
self.assertEqual(5, mock_log.warning.call_count)
|
|
self.assertEqual(4, mock_log.exception.call_count)
|
|
|
|
def _manager(self, list_side_effect, **kw):
|
|
mock_mgr = mock.MagicMock()
|
|
mock_mgr().list.side_effect = list_side_effect
|
|
mock_mgr.reset_mock()
|
|
|
|
for k, v in six.iteritems(kw):
|
|
setattr(mock_mgr, k, v)
|
|
|
|
return mock_mgr
|
|
|
|
@mock.patch("%s.SeekAndDestroy._get_cached_client" % BASE)
|
|
def test__gen_publisher_admin(self, mock__get_cached_client):
|
|
mock_mgr = self._manager([Exception, Exception, [1, 2, 3]],
|
|
_perform_for_admin_only=False)
|
|
admin = mock.MagicMock()
|
|
publish = manager.SeekAndDestroy(
|
|
mock_mgr, admin, None)._gen_publisher()
|
|
|
|
queue = []
|
|
publish(queue)
|
|
mock__get_cached_client.assert_called_once_with(admin)
|
|
mock_mgr.assert_called_once_with(
|
|
admin=mock__get_cached_client.return_value)
|
|
self.assertEqual(queue, [(admin, None, x) for x in range(1, 4)])
|
|
|
|
@mock.patch("%s.SeekAndDestroy._get_cached_client" % BASE)
|
|
def test__gen_publisher_admin_only(self, mock__get_cached_client):
|
|
mock_mgr = self._manager([Exception, Exception, [1, 2, 3]],
|
|
_perform_for_admin_only=True)
|
|
admin = mock.MagicMock()
|
|
publish = manager.SeekAndDestroy(
|
|
mock_mgr, admin, ["u1", "u2"])._gen_publisher()
|
|
|
|
queue = []
|
|
publish(queue)
|
|
mock__get_cached_client.assert_called_once_with(admin)
|
|
mock_mgr.assert_called_once_with(
|
|
admin=mock__get_cached_client.return_value)
|
|
self.assertEqual(queue, [(admin, None, x) for x in range(1, 4)])
|
|
|
|
@mock.patch("%s.SeekAndDestroy._get_cached_client" % BASE)
|
|
def test__gen_publisher_user_resource(self, mock__get_cached_client):
|
|
mock_mgr = self._manager([Exception, Exception, [1, 2, 3],
|
|
Exception, Exception, [4, 5]],
|
|
_perform_for_admin_only=False,
|
|
_tenant_resource=True)
|
|
|
|
admin = mock.MagicMock()
|
|
users = [{"tenant_id": 1, "id": 1}, {"tenant_id": 2, "id": 2}]
|
|
publish = manager.SeekAndDestroy(
|
|
mock_mgr, admin, users)._gen_publisher()
|
|
|
|
queue = []
|
|
publish(queue)
|
|
|
|
mock_client = mock__get_cached_client.return_value
|
|
mock_mgr.assert_has_calls([
|
|
mock.call(admin=mock_client, user=mock_client,
|
|
tenant_uuid=users[0]["tenant_id"]),
|
|
mock.call().list(),
|
|
mock.call().list(),
|
|
mock.call().list(),
|
|
mock.call(admin=mock_client, user=mock_client,
|
|
tenant_uuid=users[1]["tenant_id"]),
|
|
mock.call().list(),
|
|
mock.call().list()
|
|
])
|
|
mock__get_cached_client.assert_has_calls([
|
|
mock.call(admin),
|
|
mock.call(users[0]),
|
|
mock.call(users[1])
|
|
])
|
|
expected_queue = [(admin, users[0], x) for x in range(1, 4)]
|
|
expected_queue += [(admin, users[1], x) for x in range(4, 6)]
|
|
self.assertEqual(queue, expected_queue)
|
|
|
|
@mock.patch("%s.LOG" % BASE)
|
|
@mock.patch("%s.SeekAndDestroy._get_cached_client" % BASE)
|
|
def test__gen_publisher_tenant_resource(self, mock__get_cached_client,
|
|
mock_log):
|
|
mock_mgr = self._manager([Exception, [1, 2, 3],
|
|
Exception, Exception, Exception,
|
|
["this shouldn't be in results"]],
|
|
_perform_for_admin_only=False,
|
|
_tenant_resource=True)
|
|
users = [{"tenant_id": 1, "id": 1},
|
|
{"tenant_id": 1, "id": 2},
|
|
{"tenant_id": 2, "id": 3}]
|
|
|
|
publish = manager.SeekAndDestroy(
|
|
mock_mgr, None, users)._gen_publisher()
|
|
|
|
queue = []
|
|
publish(queue)
|
|
|
|
mock_client = mock__get_cached_client.return_value
|
|
mock_mgr.assert_has_calls([
|
|
mock.call(admin=mock_client, user=mock_client,
|
|
tenant_uuid=users[0]["tenant_id"]),
|
|
mock.call().list(),
|
|
mock.call().list(),
|
|
mock.call(admin=mock_client, user=mock_client,
|
|
tenant_uuid=users[2]["tenant_id"]),
|
|
mock.call().list(),
|
|
mock.call().list(),
|
|
mock.call().list()
|
|
])
|
|
mock__get_cached_client.assert_has_calls([
|
|
mock.call(None),
|
|
mock.call(users[0]),
|
|
mock.call(users[2])
|
|
])
|
|
self.assertEqual(queue, [(None, users[0], x) for x in range(1, 4)])
|
|
self.assertTrue(mock_log.warning.mock_called)
|
|
self.assertTrue(mock_log.exception.mock_called)
|
|
|
|
@mock.patch("%s.SeekAndDestroy._get_cached_client" % BASE)
|
|
@mock.patch("%s.SeekAndDestroy._delete_single_resource" % BASE)
|
|
def test__gen_consumer(self, mock__delete_single_resource,
|
|
mock__get_cached_client):
|
|
mock_mgr = mock.MagicMock(__name__="Test")
|
|
|
|
consumer = manager.SeekAndDestroy(mock_mgr, None, None)._gen_consumer()
|
|
|
|
admin = mock.MagicMock()
|
|
user1 = {"id": "a", "tenant_id": "uuid1"}
|
|
cache = {}
|
|
|
|
consumer(cache, (admin, user1, "res"))
|
|
mock_mgr.assert_called_once_with(
|
|
resource="res",
|
|
admin=mock__get_cached_client.return_value,
|
|
user=mock__get_cached_client.return_value,
|
|
tenant_uuid=user1["tenant_id"])
|
|
mock__get_cached_client.assert_has_calls([
|
|
mock.call(admin),
|
|
mock.call(user1)
|
|
])
|
|
mock__delete_single_resource.assert_called_once_with(
|
|
mock_mgr.return_value)
|
|
|
|
mock_mgr.reset_mock()
|
|
mock__get_cached_client.reset_mock()
|
|
mock__delete_single_resource.reset_mock()
|
|
|
|
consumer(cache, (admin, None, "res2"))
|
|
mock_mgr.assert_called_once_with(
|
|
resource="res2",
|
|
admin=mock__get_cached_client.return_value,
|
|
user=mock__get_cached_client.return_value,
|
|
tenant_uuid=None)
|
|
|
|
mock__get_cached_client.assert_has_calls([
|
|
mock.call(admin),
|
|
mock.call(None)
|
|
])
|
|
mock__delete_single_resource.assert_called_once_with(
|
|
mock_mgr.return_value)
|
|
|
|
@mock.patch("%s.SeekAndDestroy._gen_consumer" % BASE)
|
|
@mock.patch("%s.SeekAndDestroy._gen_publisher" % BASE)
|
|
@mock.patch("%s.broker.run" % BASE)
|
|
def test_exterminate(self, mock_broker_run, mock__gen_publisher,
|
|
mock__gen_consumer):
|
|
|
|
manager_cls = mock.MagicMock(_threads=5)
|
|
manager.SeekAndDestroy(manager_cls, None, None).exterminate()
|
|
|
|
mock__gen_publisher.assert_called_once_with()
|
|
mock__gen_consumer.assert_called_once_with()
|
|
mock_broker_run.assert_called_once_with(
|
|
mock__gen_publisher.return_value,
|
|
mock__gen_consumer.return_value,
|
|
consumers_count=5)
|
|
|
|
|
|
class ResourceManagerTestCase(test.TestCase):
|
|
|
|
def _get_res_mock(self, **kw):
|
|
_mock = mock.MagicMock()
|
|
for k, v in six.iteritems(kw):
|
|
setattr(_mock, k, v)
|
|
return _mock
|
|
|
|
def _list_res_names_helper(self, names, admin_required, mock_iter):
|
|
self.assertEqual(set(names),
|
|
manager.list_resource_names(admin_required))
|
|
mock_iter.assert_called_once_with(base.ResourceManager)
|
|
mock_iter.reset_mock()
|
|
|
|
@mock.patch("%s.discover.itersubclasses" % BASE)
|
|
def test_list_resource_names(self, mock_itersubclasses):
|
|
mock_itersubclasses.return_value = [
|
|
self._get_res_mock(_service="fake", _resource="1",
|
|
_admin_required=True),
|
|
self._get_res_mock(_service="fake", _resource="2",
|
|
_admin_required=False),
|
|
self._get_res_mock(_service="other", _resource="2",
|
|
_admin_required=False)
|
|
]
|
|
|
|
self._list_res_names_helper(
|
|
["fake", "other", "fake.1", "fake.2", "other.2"],
|
|
None, mock_itersubclasses)
|
|
self._list_res_names_helper(
|
|
["fake", "fake.1"],
|
|
True, mock_itersubclasses)
|
|
self._list_res_names_helper(
|
|
["fake", "other", "fake.2", "other.2"],
|
|
False, mock_itersubclasses)
|
|
|
|
@mock.patch("%s.discover.itersubclasses" % BASE)
|
|
def test_find_resource_managers(self, mock_itersubclasses):
|
|
mock_itersubclasses.return_value = [
|
|
self._get_res_mock(_service="fake", _resource="1", _order=1,
|
|
_admin_required=True),
|
|
self._get_res_mock(_service="fake", _resource="2", _order=3,
|
|
_admin_required=False),
|
|
self._get_res_mock(_service="other", _resource="2", _order=2,
|
|
_admin_required=False)
|
|
]
|
|
|
|
self.assertEqual(mock_itersubclasses.return_value[0:2],
|
|
manager.find_resource_managers(names=["fake"]))
|
|
|
|
self.assertEqual(mock_itersubclasses.return_value[0:1],
|
|
manager.find_resource_managers(names=["fake.1"]))
|
|
|
|
self.assertEqual(
|
|
[mock_itersubclasses.return_value[0],
|
|
mock_itersubclasses.return_value[2],
|
|
mock_itersubclasses.return_value[1]],
|
|
manager.find_resource_managers(names=["fake", "other"]))
|
|
|
|
self.assertEqual(mock_itersubclasses.return_value[0:1],
|
|
manager.find_resource_managers(names=["fake"],
|
|
admin_required=True))
|
|
self.assertEqual(mock_itersubclasses.return_value[1:2],
|
|
manager.find_resource_managers(names=["fake"],
|
|
admin_required=False))
|
|
|
|
@mock.patch("%s.SeekAndDestroy" % BASE)
|
|
@mock.patch("%s.find_resource_managers" % BASE,
|
|
return_value=[mock.MagicMock(), mock.MagicMock()])
|
|
def test_cleanup(self, mock_find_resource_managers, mock_seek_and_destroy):
|
|
manager.cleanup(names=["a", "b"], admin_required=True,
|
|
admin="admin", users=["user"])
|
|
|
|
mock_find_resource_managers.assert_called_once_with(["a", "b"], True)
|
|
|
|
mock_seek_and_destroy.assert_has_calls([
|
|
mock.call(
|
|
mock_find_resource_managers.return_value[0], "admin",
|
|
["user"], None
|
|
),
|
|
mock.call().exterminate(),
|
|
mock.call(
|
|
mock_find_resource_managers.return_value[1], "admin",
|
|
["user"], None
|
|
),
|
|
mock.call().exterminate()
|
|
])
|
|
|
|
@mock.patch("%s.SeekAndDestroy" % BASE)
|
|
@mock.patch("%s.find_resource_managers" % BASE,
|
|
return_value=[mock.MagicMock(), mock.MagicMock()])
|
|
def test_cleanup_with_api_versions(self,
|
|
mock_find_resource_managers,
|
|
mock_seek_and_destroy):
|
|
manager.cleanup(names=["a", "b"], admin_required=True,
|
|
admin="admin", users=["user"],
|
|
api_versions={"cinder": {
|
|
"version": "1", "service_type": "volume"
|
|
}})
|
|
|
|
mock_find_resource_managers.assert_called_once_with(["a", "b"], True)
|
|
|
|
mock_seek_and_destroy.assert_has_calls([
|
|
mock.call(
|
|
mock_find_resource_managers.return_value[0], "admin",
|
|
["user"],
|
|
{"cinder": {"service_type": "volume", "version": "1"}}
|
|
),
|
|
mock.call().exterminate(),
|
|
mock.call(
|
|
mock_find_resource_managers.return_value[1], "admin",
|
|
["user"],
|
|
{"cinder": {"service_type": "volume", "version": "1"}}
|
|
),
|
|
mock.call().exterminate()
|
|
])
|