Update quota of proper user on resource delete
When deleting a resource (share, snapshot, share-network) the quota usage of the current user gets updated, which might not be the same user, who created that resource. That means the quota usage will never be reduced for initial user. This fix adds the resources user_id to the quota update statements, to update the quota_usage for the user, that created this resource, irregardless who deletes it. Change-Id: Iefe8f4d0e7c526e3ed94c1994ba62f1a2a929ba2 Closes-Bug: #1542598
This commit is contained in:
parent
83169e0f7d
commit
8d3e72f8b2
@ -97,13 +97,14 @@ class ShareNetworkController(wsgi.Controller):
|
|||||||
try:
|
try:
|
||||||
reservations = QUOTAS.reserve(
|
reservations = QUOTAS.reserve(
|
||||||
context, project_id=share_network['project_id'],
|
context, project_id=share_network['project_id'],
|
||||||
share_networks=-1)
|
share_networks=-1, user_id=share_network['user_id'])
|
||||||
except Exception:
|
except Exception:
|
||||||
LOG.exception(_LE("Failed to update usages deleting "
|
LOG.exception(_LE("Failed to update usages deleting "
|
||||||
"share-network."))
|
"share-network."))
|
||||||
else:
|
else:
|
||||||
QUOTAS.commit(context, reservations,
|
QUOTAS.commit(context, reservations,
|
||||||
project_id=share_network['project_id'])
|
project_id=share_network['project_id'],
|
||||||
|
user_id=share_network['user_id'])
|
||||||
return webob.Response(status_int=202)
|
return webob.Response(status_int=202)
|
||||||
|
|
||||||
def _get_share_networks(self, req, is_detail=True):
|
def _get_share_networks(self, req, is_detail=True):
|
||||||
@ -247,6 +248,7 @@ class ShareNetworkController(wsgi.Controller):
|
|||||||
|
|
||||||
values = body[RESOURCE_NAME]
|
values = body[RESOURCE_NAME]
|
||||||
values['project_id'] = context.project_id
|
values['project_id'] = context.project_id
|
||||||
|
values['user_id'] = context.user_id
|
||||||
self._verify_no_mutually_exclusive_data(values)
|
self._verify_no_mutually_exclusive_data(values)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
@ -301,7 +301,8 @@ class DbQuotaDriver(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Filter resources
|
# Filter resources
|
||||||
@ -358,7 +359,8 @@ class DbQuotaDriver(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Ensure no value is less than zero
|
# Ensure no value is less than zero
|
||||||
@ -425,7 +427,8 @@ class DbQuotaDriver(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# Set up the reservation expiration
|
# Set up the reservation expiration
|
||||||
@ -441,7 +444,7 @@ class DbQuotaDriver(object):
|
|||||||
# If project_id is None, then we use the project_id in context
|
# If project_id is None, then we use the project_id in context
|
||||||
if project_id is None:
|
if project_id is None:
|
||||||
project_id = context.project_id
|
project_id = context.project_id
|
||||||
# If user_id is None, then we use the project_id in context
|
# If user_id is None, then we use the user_id in context
|
||||||
if user_id is None:
|
if user_id is None:
|
||||||
user_id = context.user_id
|
user_id = context.user_id
|
||||||
|
|
||||||
@ -476,7 +479,8 @@ class DbQuotaDriver(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
# If project_id is None, then we use the project_id in context
|
# If project_id is None, then we use the project_id in context
|
||||||
if project_id is None:
|
if project_id is None:
|
||||||
@ -499,7 +503,8 @@ class DbQuotaDriver(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
# If project_id is None, then we use the project_id in context
|
# If project_id is None, then we use the project_id in context
|
||||||
if project_id is None:
|
if project_id is None:
|
||||||
@ -912,7 +917,8 @@ class QuotaEngine(object):
|
|||||||
common user's tenant.
|
common user's tenant.
|
||||||
:param user_id: Specify the user_id if current context
|
:param user_id: Specify the user_id if current context
|
||||||
is admin and admin wants to impact on
|
is admin and admin wants to impact on
|
||||||
common user.
|
common user. (Special case: user operates on
|
||||||
|
resource, owned/created by different user)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
return self._driver.limit_check(context, self._resources, values,
|
return self._driver.limit_check(context, self._resources, values,
|
||||||
|
@ -605,10 +605,13 @@ class API(base.Base):
|
|||||||
self._check_is_share_busy(share)
|
self._check_is_share_busy(share)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
# we give the user_id of the share, to update the quota usage
|
||||||
|
# for the user, who created the share
|
||||||
reservations = QUOTAS.reserve(context,
|
reservations = QUOTAS.reserve(context,
|
||||||
project_id=project_id,
|
project_id=project_id,
|
||||||
shares=-1,
|
shares=-1,
|
||||||
gigabytes=-share['size'])
|
gigabytes=-share['size'],
|
||||||
|
user_id=share['user_id'])
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
reservations = None
|
reservations = None
|
||||||
LOG.exception(
|
LOG.exception(
|
||||||
@ -623,7 +626,10 @@ class API(base.Base):
|
|||||||
self.db.share_instance_delete(context, share_instance['id'])
|
self.db.share_instance_delete(context, share_instance['id'])
|
||||||
|
|
||||||
if reservations:
|
if reservations:
|
||||||
QUOTAS.commit(context, reservations, project_id=project_id)
|
# we give the user_id of the share, to update the quota usage
|
||||||
|
# for the user, who created the share
|
||||||
|
QUOTAS.commit(context, reservations, project_id=project_id,
|
||||||
|
user_id=share['user_id'])
|
||||||
|
|
||||||
def delete_instance(self, context, share_instance, force=False):
|
def delete_instance(self, context, share_instance, force=False):
|
||||||
policy.check_policy(context, 'share', 'delete')
|
policy.check_policy(context, 'share', 'delete')
|
||||||
|
@ -1808,13 +1808,15 @@ class ShareManager(manager.SchedulerDependentManager):
|
|||||||
try:
|
try:
|
||||||
reservations = QUOTAS.reserve(
|
reservations = QUOTAS.reserve(
|
||||||
context, project_id=project_id, snapshots=-1,
|
context, project_id=project_id, snapshots=-1,
|
||||||
snapshot_gigabytes=-snapshot_ref['size'])
|
snapshot_gigabytes=-snapshot_ref['size'],
|
||||||
|
user_id=snapshot_ref['user_id'])
|
||||||
except Exception:
|
except Exception:
|
||||||
reservations = None
|
reservations = None
|
||||||
LOG.exception(_LE("Failed to update usages deleting snapshot"))
|
LOG.exception(_LE("Failed to update usages deleting snapshot"))
|
||||||
|
|
||||||
if reservations:
|
if reservations:
|
||||||
QUOTAS.commit(context, reservations, project_id=project_id)
|
QUOTAS.commit(context, reservations, project_id=project_id,
|
||||||
|
user_id=snapshot_ref['user_id'])
|
||||||
|
|
||||||
@add_hooks
|
@add_hooks
|
||||||
@utils.require_driver_initialized
|
@utils.require_driver_initialized
|
||||||
|
@ -238,6 +238,8 @@ class ShareNetworkAPITest(test.TestCase):
|
|||||||
def test_quota_delete_reservation_failed(self):
|
def test_quota_delete_reservation_failed(self):
|
||||||
share_nw = fake_share_network.copy()
|
share_nw = fake_share_network.copy()
|
||||||
share_nw['share_servers'] = ['foo', 'bar']
|
share_nw['share_servers'] = ['foo', 'bar']
|
||||||
|
share_nw['user_id'] = 'fake_user_id'
|
||||||
|
|
||||||
self.mock_object(db_api, 'share_network_get',
|
self.mock_object(db_api, 'share_network_get',
|
||||||
mock.Mock(return_value=share_nw))
|
mock.Mock(return_value=share_nw))
|
||||||
self.mock_object(db_api, 'share_instances_get_all_by_share_network',
|
self.mock_object(db_api, 'share_instances_get_all_by_share_network',
|
||||||
@ -262,7 +264,12 @@ class ShareNetworkAPITest(test.TestCase):
|
|||||||
mock.call(self.req.environ['manila.context'], 'bar')])
|
mock.call(self.req.environ['manila.context'], 'bar')])
|
||||||
db_api.share_network_delete.assert_called_once_with(
|
db_api.share_network_delete.assert_called_once_with(
|
||||||
self.req.environ['manila.context'], share_nw['id'])
|
self.req.environ['manila.context'], share_nw['id'])
|
||||||
self.assertTrue(share_networks.QUOTAS.reserve.called)
|
share_networks.QUOTAS.reserve.assert_called_once_with(
|
||||||
|
self.req.environ['manila.context'],
|
||||||
|
project_id=share_nw['project_id'],
|
||||||
|
share_networks=-1,
|
||||||
|
user_id=share_nw['user_id']
|
||||||
|
)
|
||||||
self.assertFalse(share_networks.QUOTAS.commit.called)
|
self.assertFalse(share_networks.QUOTAS.commit.called)
|
||||||
|
|
||||||
def test_delete_in_use_by_share(self):
|
def test_delete_in_use_by_share(self):
|
||||||
|
@ -1082,6 +1082,30 @@ class ShareAPITestCase(test.TestCase):
|
|||||||
db_api.share_snapshot_get_all_for_share.assert_called_once_with(
|
db_api.share_snapshot_get_all_for_share.assert_called_once_with(
|
||||||
utils.IsAMatcher(context.RequestContext), share['id'])
|
utils.IsAMatcher(context.RequestContext), share['id'])
|
||||||
|
|
||||||
|
def test_delete_quota_with_different_user(self):
|
||||||
|
share = self._setup_delete_mocks(constants.STATUS_AVAILABLE)
|
||||||
|
diff_user_context = context.RequestContext(
|
||||||
|
user_id='fake2',
|
||||||
|
project_id='fake',
|
||||||
|
is_admin=False
|
||||||
|
)
|
||||||
|
|
||||||
|
self.api.delete(diff_user_context, share)
|
||||||
|
|
||||||
|
quota.QUOTAS.reserve.assert_called_once_with(
|
||||||
|
diff_user_context,
|
||||||
|
project_id=share['project_id'],
|
||||||
|
shares=-1,
|
||||||
|
gigabytes=-share['size'],
|
||||||
|
user_id=share['user_id']
|
||||||
|
)
|
||||||
|
quota.QUOTAS.commit.assert_called_once_with(
|
||||||
|
diff_user_context,
|
||||||
|
mock.ANY,
|
||||||
|
project_id=share['project_id'],
|
||||||
|
user_id=share['user_id']
|
||||||
|
)
|
||||||
|
|
||||||
def test_delete_wrong_status(self):
|
def test_delete_wrong_status(self):
|
||||||
share = fake_share('fakeid')
|
share = fake_share('fakeid')
|
||||||
self.mock_object(db_api, 'share_get', mock.Mock(return_value=share))
|
self.mock_object(db_api, 'share_get', mock.Mock(return_value=share))
|
||||||
@ -1148,7 +1172,8 @@ class ShareAPITestCase(test.TestCase):
|
|||||||
self.context,
|
self.context,
|
||||||
project_id=share['project_id'],
|
project_id=share['project_id'],
|
||||||
shares=-1,
|
shares=-1,
|
||||||
gigabytes=-share['size']
|
gigabytes=-share['size'],
|
||||||
|
user_id=share['user_id']
|
||||||
)
|
)
|
||||||
self.assertFalse(quota.QUOTAS.commit.called)
|
self.assertFalse(quota.QUOTAS.commit.called)
|
||||||
|
|
||||||
|
@ -1308,6 +1308,29 @@ class ShareManagerTestCase(test.TestCase):
|
|||||||
utils.IsAMatcher(models.ShareSnapshotInstance),
|
utils.IsAMatcher(models.ShareSnapshotInstance),
|
||||||
share_server=None)
|
share_server=None)
|
||||||
|
|
||||||
|
def test_delete_snapshot_quota_error(self):
|
||||||
|
share = db_utils.create_share()
|
||||||
|
share_id = share['id']
|
||||||
|
snapshot = db_utils.create_snapshot(share_id=share_id)
|
||||||
|
snapshot_id = snapshot['id']
|
||||||
|
snapshot = db_utils.create_snapshot(
|
||||||
|
with_share=True, status=constants.STATUS_AVAILABLE)
|
||||||
|
|
||||||
|
self.mock_object(quota.QUOTAS, 'reserve',
|
||||||
|
mock.Mock(side_effect=exception.QuotaError('fake')))
|
||||||
|
self.mock_object(quota.QUOTAS, 'commit')
|
||||||
|
|
||||||
|
self.share_manager.delete_snapshot(self.context, snapshot_id)
|
||||||
|
|
||||||
|
quota.QUOTAS.reserve.assert_called_once_with(
|
||||||
|
mock.ANY,
|
||||||
|
project_id=six.text_type(snapshot['project_id']),
|
||||||
|
snapshots=-1,
|
||||||
|
snapshot_gigabytes=-snapshot['size'],
|
||||||
|
user_id=six.text_type(snapshot['user_id'])
|
||||||
|
)
|
||||||
|
self.assertFalse(quota.QUOTAS.commit.called)
|
||||||
|
|
||||||
def test_delete_share_instance_if_busy(self):
|
def test_delete_share_instance_if_busy(self):
|
||||||
"""Test snapshot could not be deleted if busy."""
|
"""Test snapshot could not be deleted if busy."""
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user