578 lines
26 KiB
Python
578 lines
26 KiB
Python
#
|
|
# Copyright 2013 OpenStack Foundation
|
|
# All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
"""
|
|
Tests for cinder.api.contrib.quotas.py
|
|
"""
|
|
|
|
|
|
import mock
|
|
|
|
from lxml import etree
|
|
|
|
import uuid
|
|
import webob.exc
|
|
|
|
from cinder.api.contrib import quotas
|
|
from cinder import context
|
|
from cinder import db
|
|
from cinder import test
|
|
from cinder.tests.unit import test_db_api
|
|
|
|
from keystonemiddleware import auth_token
|
|
from oslo_config import cfg
|
|
from oslo_config import fixture as config_fixture
|
|
|
|
|
|
CONF = cfg.CONF
|
|
|
|
|
|
def make_body(root=True, gigabytes=1000, snapshots=10,
|
|
volumes=10, backups=10, backup_gigabytes=1000,
|
|
tenant_id='foo', per_volume_gigabytes=-1):
|
|
resources = {'gigabytes': gigabytes,
|
|
'snapshots': snapshots,
|
|
'volumes': volumes,
|
|
'backups': backups,
|
|
'backup_gigabytes': backup_gigabytes,
|
|
'per_volume_gigabytes': per_volume_gigabytes, }
|
|
# need to consider preexisting volume types as well
|
|
volume_types = db.volume_type_get_all(context.get_admin_context())
|
|
for volume_type in volume_types:
|
|
resources['gigabytes_' + volume_type] = -1
|
|
resources['snapshots_' + volume_type] = -1
|
|
resources['volumes_' + volume_type] = -1
|
|
|
|
if tenant_id:
|
|
resources['id'] = tenant_id
|
|
if root:
|
|
result = {'quota_set': resources}
|
|
else:
|
|
result = resources
|
|
return result
|
|
|
|
|
|
def make_subproject_body(root=True, gigabytes=0, snapshots=0,
|
|
volumes=0, backups=0, backup_gigabytes=0,
|
|
tenant_id='foo', per_volume_gigabytes=0):
|
|
return make_body(root=root, gigabytes=gigabytes, snapshots=snapshots,
|
|
volumes=volumes, backups=backups,
|
|
backup_gigabytes=backup_gigabytes, tenant_id=tenant_id,
|
|
per_volume_gigabytes=per_volume_gigabytes)
|
|
|
|
|
|
class QuotaSetsControllerTest(test.TestCase):
|
|
|
|
class FakeProject(object):
|
|
|
|
def __init__(self, id='foo', parent_id=None):
|
|
self.id = id
|
|
self.parent_id = parent_id
|
|
self.subtree = None
|
|
|
|
def setUp(self):
|
|
super(QuotaSetsControllerTest, self).setUp()
|
|
self.controller = quotas.QuotaSetsController()
|
|
|
|
self.req = self.mox.CreateMockAnything()
|
|
self.req.environ = {'cinder.context': context.get_admin_context()}
|
|
self.req.environ['cinder.context'].is_admin = True
|
|
self.req.environ['cinder.context'].auth_token = uuid.uuid4().hex
|
|
self.req.environ['cinder.context'].project_id = 'foo'
|
|
|
|
self._create_project_hierarchy()
|
|
|
|
self.auth_url = 'http://localhost:5000'
|
|
self.fixture = self.useFixture(config_fixture.Config(auth_token.CONF))
|
|
self.fixture.config(auth_uri=self.auth_url, group='keystone_authtoken')
|
|
|
|
def _create_project_hierarchy(self):
|
|
"""Sets an environment used for nested quotas tests.
|
|
|
|
Create a project hierarchy such as follows:
|
|
+-----------+
|
|
| |
|
|
| A |
|
|
| / \ |
|
|
| B C |
|
|
| / |
|
|
| D |
|
|
+-----------+
|
|
"""
|
|
self.A = self.FakeProject(id=uuid.uuid4().hex, parent_id=None)
|
|
self.B = self.FakeProject(id=uuid.uuid4().hex, parent_id=self.A.id)
|
|
self.C = self.FakeProject(id=uuid.uuid4().hex, parent_id=self.A.id)
|
|
self.D = self.FakeProject(id=uuid.uuid4().hex, parent_id=self.B.id)
|
|
|
|
# update projects subtrees
|
|
self.B.subtree = {self.D.id: self.D.subtree}
|
|
self.A.subtree = {self.B.id: self.B.subtree, self.C.id: self.C.subtree}
|
|
|
|
# project_by_id attribute is used to recover a project based on its id.
|
|
self.project_by_id = {self.A.id: self.A, self.B.id: self.B,
|
|
self.C.id: self.C, self.D.id: self.D}
|
|
|
|
def _get_project(self, context, id, subtree_as_ids=False):
|
|
return self.project_by_id.get(id, self.FakeProject())
|
|
|
|
@mock.patch('keystoneclient.client.Client')
|
|
@mock.patch('keystoneclient.session.Session')
|
|
def test_keystone_client_instantiation(self, ksclient_session,
|
|
ksclient_class):
|
|
context = self.req.environ['cinder.context']
|
|
self.controller._keystone_client(context)
|
|
ksclient_class.assert_called_once_with(auth_url=self.auth_url,
|
|
session=ksclient_session())
|
|
|
|
@mock.patch('keystoneclient.client.Client')
|
|
def test_get_project_keystoneclient_v2(self, ksclient_class):
|
|
context = self.req.environ['cinder.context']
|
|
keystoneclient = ksclient_class.return_value
|
|
keystoneclient.version = 'v2.0'
|
|
expected_project = self.controller.GenericProjectInfo(
|
|
context.project_id, 'v2.0')
|
|
project = self.controller._get_project(context, context.project_id)
|
|
self.assertEqual(expected_project.__dict__, project.__dict__)
|
|
|
|
@mock.patch('keystoneclient.client.Client')
|
|
def test_get_project_keystoneclient_v3(self, ksclient_class):
|
|
context = self.req.environ['cinder.context']
|
|
keystoneclient = ksclient_class.return_value
|
|
keystoneclient.version = 'v3'
|
|
returned_project = self.FakeProject(context.project_id, 'bar')
|
|
del returned_project.subtree
|
|
keystoneclient.projects.get.return_value = returned_project
|
|
expected_project = self.controller.GenericProjectInfo(
|
|
context.project_id, 'v3', 'bar')
|
|
project = self.controller._get_project(context, context.project_id)
|
|
self.assertEqual(expected_project.__dict__, project.__dict__)
|
|
|
|
@mock.patch('keystoneclient.client.Client')
|
|
def test_get_project_keystoneclient_v3_with_subtree(self, ksclient_class):
|
|
context = self.req.environ['cinder.context']
|
|
keystoneclient = ksclient_class.return_value
|
|
keystoneclient.version = 'v3'
|
|
returned_project = self.FakeProject(context.project_id, 'bar')
|
|
subtree_dict = {'baz': {'quux': None}}
|
|
returned_project.subtree = subtree_dict
|
|
keystoneclient.projects.get.return_value = returned_project
|
|
expected_project = self.controller.GenericProjectInfo(
|
|
context.project_id, 'v3', 'bar', subtree_dict)
|
|
project = self.controller._get_project(context, context.project_id,
|
|
subtree_as_ids=True)
|
|
keystoneclient.projects.get.assert_called_once_with(
|
|
context.project_id, subtree_as_ids=True)
|
|
self.assertEqual(expected_project.__dict__, project.__dict__)
|
|
|
|
def test_defaults(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
result = self.controller.defaults(self.req, 'foo')
|
|
self.assertDictMatch(make_body(), result)
|
|
|
|
def test_subproject_defaults(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
context = self.req.environ['cinder.context']
|
|
context.project_id = self.B.id
|
|
result = self.controller.defaults(self.req, self.B.id)
|
|
expected = make_subproject_body(tenant_id=self.B.id)
|
|
self.assertDictMatch(expected, result)
|
|
|
|
def test_show(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
result = self.controller.show(self.req, 'foo')
|
|
self.assertDictMatch(make_body(), result)
|
|
|
|
def test_subproject_show(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
result = self.controller.show(self.req, self.B.id)
|
|
expected = make_subproject_body(tenant_id=self.B.id)
|
|
self.assertDictMatch(expected, result)
|
|
|
|
def test_subproject_show_in_hierarchy(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
# An user scoped to a root project in an hierarchy can see its children
|
|
# quotas.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
result = self.controller.show(self.req, self.D.id)
|
|
expected = make_subproject_body(tenant_id=self.D.id)
|
|
self.assertDictMatch(result, expected)
|
|
# An user scoped to a parent project can see its immediate children
|
|
# quotas.
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
result = self.controller.show(self.req, self.D.id)
|
|
expected = make_subproject_body(tenant_id=self.D.id)
|
|
self.assertDictMatch(result, expected)
|
|
|
|
def test_subproject_show_target_project_equals_to_context_project(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
result = self.controller.show(self.req, self.B.id)
|
|
expected = make_subproject_body(tenant_id=self.B.id)
|
|
self.assertDictMatch(result, expected)
|
|
|
|
def test_show_not_authorized(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].is_admin = False
|
|
self.req.environ['cinder.context'].user_id = 'bad_user'
|
|
self.req.environ['cinder.context'].project_id = 'bad_project'
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.show,
|
|
self.req, 'foo')
|
|
|
|
def test_subproject_show_not_authorized(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.show,
|
|
self.req, self.C.id)
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.show,
|
|
self.req, self.A.id)
|
|
|
|
def test_update(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
result = self.controller.update(self.req, 'foo', body)
|
|
self.assertDictMatch(body, result)
|
|
|
|
body = make_body(gigabytes=db.MAX_INT, tenant_id=None)
|
|
result = self.controller.update(self.req, 'foo', body)
|
|
self.assertDictMatch(body, result)
|
|
|
|
def test_update_subproject(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
# Update the project A quota.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
result = self.controller.update(self.req, self.A.id, body)
|
|
self.assertDictMatch(result, body)
|
|
# Update the quota of B to be equal to its parent quota
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
result = self.controller.update(self.req, self.B.id, body)
|
|
self.assertDictMatch(result, body)
|
|
# Try to update the quota of C, it will not be allowed, since the
|
|
# project A doesn't have free quota available.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, self.C.id, body)
|
|
# Successfully update the quota of D.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=1000, snapshots=7,
|
|
volumes=3, backups=3, tenant_id=None)
|
|
result = self.controller.update(self.req, self.D.id, body)
|
|
self.assertDictMatch(result, body)
|
|
# An admin of B can also update the quota of D, since D is its an
|
|
# immediate child.
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
body = make_body(gigabytes=1500, snapshots=10,
|
|
volumes=4, backups=4, tenant_id=None)
|
|
result = self.controller.update(self.req, self.D.id, body)
|
|
|
|
def test_update_subproject_not_in_hierarchy(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
|
|
# Create another project hierarchy
|
|
E = self.FakeProject(id=uuid.uuid4().hex, parent_id=None)
|
|
F = self.FakeProject(id=uuid.uuid4().hex, parent_id=E.id)
|
|
E.subtree = {F.id: F.subtree}
|
|
self.project_by_id[E.id] = E
|
|
self.project_by_id[F.id] = F
|
|
|
|
# Update the project A quota.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
result = self.controller.update(self.req, self.A.id, body)
|
|
self.assertDictMatch(result, body)
|
|
# Try to update the quota of F, it will not be allowed, since the
|
|
# project E doesn't belongs to the project hierarchy of A.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.update,
|
|
self.req, F.id, body)
|
|
|
|
def test_update_subproject_with_not_root_context_project(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
# Update the project A quota.
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
result = self.controller.update(self.req, self.A.id, body)
|
|
self.assertDictMatch(result, body)
|
|
# Try to update the quota of B, it will not be allowed, since the
|
|
# project in the context (B) is not a root project.
|
|
self.req.environ['cinder.context'].project_id = self.B.id
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5, tenant_id=None)
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.update,
|
|
self.req, self.B.id, body)
|
|
|
|
@mock.patch(
|
|
'cinder.api.openstack.wsgi.Controller.validate_string_length')
|
|
@mock.patch(
|
|
'cinder.api.openstack.wsgi.Controller.validate_integer')
|
|
def test_update_limit(self, mock_validate_integer, mock_validate):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
mock_validate_integer.return_value = 10
|
|
|
|
body = {'quota_set': {'volumes': 10}}
|
|
result = self.controller.update(self.req, 'foo', body)
|
|
|
|
self.assertEqual(10, result['quota_set']['volumes'])
|
|
self.assertTrue(mock_validate.called)
|
|
self.assertTrue(mock_validate_integer.called)
|
|
|
|
def test_update_wrong_key(self):
|
|
body = {'quota_set': {'bad': 'bad'}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_invalid_value_key_value(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
body = {'quota_set': {'gigabytes': "should_be_int"}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_invalid_type_key_value(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
body = {'quota_set': {'gigabytes': None}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_multi_value_with_bad_data(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
orig_quota = self.controller.show(self.req, 'foo')
|
|
body = make_body(gigabytes=2000, snapshots=15, volumes="should_be_int",
|
|
backups=5, tenant_id=None)
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
# Verify that quota values are not updated in db
|
|
new_quota = self.controller.show(self.req, 'foo')
|
|
self.assertDictMatch(orig_quota, new_quota)
|
|
|
|
def test_update_bad_quota_limit(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
body = {'quota_set': {'gigabytes': -1000}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
body = {'quota_set': {'gigabytes': db.MAX_INT + 1}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_no_admin(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].is_admin = False
|
|
self.req.environ['cinder.context'].project_id = 'foo'
|
|
self.req.environ['cinder.context'].user_id = 'foo_user'
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.update,
|
|
self.req, 'foo', make_body(tenant_id=None))
|
|
|
|
def test_update_without_quota_set_field(self):
|
|
body = {'fake_quota_set': {'gigabytes': 100}}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_empty_body(self):
|
|
body = {}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def _commit_quota_reservation(self):
|
|
# Create simple quota and quota usage.
|
|
ctxt = context.get_admin_context()
|
|
res = test_db_api._quota_reserve(ctxt, 'foo')
|
|
db.reservation_commit(ctxt, res, 'foo')
|
|
expected = {'project_id': 'foo',
|
|
'volumes': {'reserved': 0, 'in_use': 1},
|
|
'gigabytes': {'reserved': 0, 'in_use': 2},
|
|
}
|
|
self.assertEqual(expected,
|
|
db.quota_usage_get_all_by_project(ctxt, 'foo'))
|
|
|
|
def test_update_lower_than_existing_resources_when_skip_false(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self._commit_quota_reservation()
|
|
body = {'quota_set': {'volumes': 0},
|
|
'skip_validation': 'false'}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
body = {'quota_set': {'gigabytes': 1},
|
|
'skip_validation': 'false'}
|
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.update,
|
|
self.req, 'foo', body)
|
|
|
|
def test_update_lower_than_existing_resources_when_skip_true(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self._commit_quota_reservation()
|
|
body = {'quota_set': {'volumes': 0},
|
|
'skip_validation': 'true'}
|
|
result = self.controller.update(self.req, 'foo', body)
|
|
self.assertEqual(body['quota_set']['volumes'],
|
|
result['quota_set']['volumes'])
|
|
|
|
def test_update_lower_than_existing_resources_without_skip_argument(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self._commit_quota_reservation()
|
|
body = {'quota_set': {'volumes': 0}}
|
|
result = self.controller.update(self.req, 'foo', body)
|
|
self.assertEqual(body['quota_set']['volumes'],
|
|
result['quota_set']['volumes'])
|
|
|
|
def test_delete(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
result_show = self.controller.show(self.req, 'foo')
|
|
self.assertDictMatch(result_show, make_body())
|
|
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5,
|
|
backup_gigabytes=1000, tenant_id=None)
|
|
result_update = self.controller.update(self.req, 'foo', body)
|
|
self.assertDictMatch(result_update, body)
|
|
|
|
self.controller.delete(self.req, 'foo')
|
|
|
|
result_show_after = self.controller.show(self.req, 'foo')
|
|
self.assertDictMatch(result_show, result_show_after)
|
|
|
|
def test_subproject_delete(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5,
|
|
backup_gigabytes=1000, tenant_id=None)
|
|
result_update = self.controller.update(self.req, self.A.id, body)
|
|
self.assertDictMatch(result_update, body)
|
|
|
|
# Set usage param to True in order to see get allocated values.
|
|
self.req.params = {'usage': 'True'}
|
|
result_show = self.controller.show(self.req, self.A.id)
|
|
|
|
result_update = self.controller.update(self.req, self.B.id, body)
|
|
self.assertDictMatch(result_update, body)
|
|
|
|
self.controller.delete(self.req, self.B.id)
|
|
|
|
result_show_after = self.controller.show(self.req, self.A.id)
|
|
self.assertDictMatch(result_show, result_show_after)
|
|
|
|
def test_subproject_delete_not_considering_default_quotas(self):
|
|
"""Test delete subprojects' quotas won't consider default quotas.
|
|
|
|
Test plan:
|
|
- Update the volume quotas of project A
|
|
- Update the volume quotas of project B
|
|
- Delete the quotas of project B
|
|
|
|
Resources with default quotas aren't expected to be considered when
|
|
updating the allocated values of the parent project. Thus, the delete
|
|
operation should succeed.
|
|
"""
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
|
|
body = {'quota_set': {'volumes': 5}}
|
|
result = self.controller.update(self.req, self.A.id, body)
|
|
self.assertEqual(body['quota_set']['volumes'],
|
|
result['quota_set']['volumes'])
|
|
|
|
body = {'quota_set': {'volumes': 2}}
|
|
result = self.controller.update(self.req, self.B.id, body)
|
|
self.assertEqual(body['quota_set']['volumes'],
|
|
result['quota_set']['volumes'])
|
|
|
|
self.controller.delete(self.req, self.B.id)
|
|
|
|
def test_delete_with_allocated_quota_different_from_zero(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].project_id = self.A.id
|
|
|
|
body = make_body(gigabytes=2000, snapshots=15,
|
|
volumes=5, backups=5,
|
|
backup_gigabytes=1000, tenant_id=None)
|
|
result_update = self.controller.update(self.req, self.A.id, body)
|
|
self.assertDictMatch(result_update, body)
|
|
|
|
# Set usage param to True in order to see get allocated values.
|
|
self.req.params = {'usage': 'True'}
|
|
result_show = self.controller.show(self.req, self.A.id)
|
|
|
|
result_update = self.controller.update(self.req, self.B.id, body)
|
|
self.assertDictMatch(result_update, body)
|
|
|
|
self.controller.delete(self.req, self.B.id)
|
|
|
|
result_show_after = self.controller.show(self.req, self.A.id)
|
|
self.assertDictMatch(result_show, result_show_after)
|
|
|
|
def test_delete_no_admin(self):
|
|
self.controller._get_project = mock.Mock()
|
|
self.controller._get_project.side_effect = self._get_project
|
|
self.req.environ['cinder.context'].is_admin = False
|
|
self.assertRaises(webob.exc.HTTPForbidden, self.controller.delete,
|
|
self.req, 'foo')
|
|
|
|
|
|
class QuotaSerializerTest(test.TestCase):
|
|
|
|
def setUp(self):
|
|
super(QuotaSerializerTest, self).setUp()
|
|
self.req = self.mox.CreateMockAnything()
|
|
self.req.environ = {'cinder.context': context.get_admin_context()}
|
|
|
|
def test_update_serializer(self):
|
|
serializer = quotas.QuotaTemplate()
|
|
quota_set = make_body(root=False)
|
|
text = serializer.serialize({'quota_set': quota_set})
|
|
tree = etree.fromstring(text)
|
|
self.assertEqual('quota_set', tree.tag)
|
|
self.assertEqual(quota_set['id'], tree.get('id'))
|
|
body = make_body(root=False, tenant_id=None)
|
|
for node in tree:
|
|
self.assertIn(node.tag, body)
|
|
self.assertEqual(str(body[node.tag]), node.text)
|