Fix os-simple-tenant-usage result order

nova usage-list can return incorrect results, having resources counted
twice. This only occurs when using the 2.40 microversion or later.

This microversion introduced pagination, which doesn't work properly.
Nova API will sort the instances using the tenant id and instance uuid,
but 'os-simple-tenant-usage' will not preserve the order when returning
the results.

For this reason, subsequent API calls made by the client will use the
wrong marker (which is supposed to be the last instance id), ending
up counting the same instances twice.

Change-Id: I6c7a67b23ec49aa207c33c38580acd834bb27e3c
Closes-Bug: #1796689
This commit is contained in:
Lucian Petrut 2018-10-08 17:13:55 +03:00 committed by Matt Riedemann
parent 88951ca98e
commit afc3a16ce3
8 changed files with 204 additions and 16 deletions
doc/api_samples/os-simple-tenant-usage/v2.40
nova
api/openstack/compute
tests
functional
unit/api_samples_test_base
releasenotes/notes

@ -0,0 +1,68 @@
{
"tenant_usages": [
{
"server_usages": [
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f06",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-3",
"started_at": "2018-10-09T11:29:04.166194",
"state": "active",
"tenant_id": "0000000e737461636b20342065000000",
"uptime": 3600,
"vcpus": 1
}
],
"start": "2018-10-09T11:29:04.166194",
"stop": "2018-10-09T12:29:04.166194",
"tenant_id": "0000000e737461636b20342065000000",
"total_hours": 1.0,
"total_local_gb_usage": 1.0,
"total_memory_mb_usage": 512.0,
"total_vcpus_usage": 1.0
},
{
"server_usages": [
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f00",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-1",
"started_at": "2018-10-09T11:29:04.166194",
"state": "active",
"tenant_id": "6f70656e737461636b20342065766572",
"uptime": 3600,
"vcpus": 1
},
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f03",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-2",
"started_at": "2018-10-09T11:29:04.166194",
"state": "active",
"tenant_id": "6f70656e737461636b20342065766572",
"uptime": 3600,
"vcpus": 1
}
],
"start": "2018-10-09T11:29:04.166194",
"stop": "2018-10-09T12:29:04.166194",
"tenant_id": "6f70656e737461636b20342065766572",
"total_hours": 2.0,
"total_local_gb_usage": 2.0,
"total_memory_mb_usage": 1024.0,
"total_vcpus_usage": 2.0
}
]
}

@ -13,6 +13,7 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import collections
import datetime import datetime
import iso8601 import iso8601
@ -142,7 +143,7 @@ class SimpleTenantUsageController(wsgi.Controller):
instances = self._get_instances_all_cells(context, period_start, instances = self._get_instances_all_cells(context, period_start,
period_stop, tenant_id, period_stop, tenant_id,
limit, marker) limit, marker)
rval = {} rval = collections.OrderedDict()
flavors = {} flavors = {}
all_server_usages = [] all_server_usages = []

@ -62,7 +62,7 @@ class ApiSampleTestBaseV21(testscenarios.WithScenarios,
# any additional fixtures needed for this scenario # any additional fixtures needed for this scenario
_additional_fixtures = [] _additional_fixtures = []
sample_dir = None sample_dir = None
_project_id = True _use_project_id = True
scenarios = [ scenarios = [
# test v2 with the v2.1 compatibility stack # test v2 with the v2.1 compatibility stack
@ -74,7 +74,7 @@ class ApiSampleTestBaseV21(testscenarios.WithScenarios,
# test v2.18 code without project id # test v2.18 code without project id
('v2_1_noproject_id', { ('v2_1_noproject_id', {
'api_major_version': 'v2.1', 'api_major_version': 'v2.1',
'_project_id': False, '_use_project_id': False,
'_additional_fixtures': [ '_additional_fixtures': [
api_paste_fixture.ApiPasteNoProjectId]}) api_paste_fixture.ApiPasteNoProjectId]})
] ]

@ -0,0 +1,68 @@
{
"tenant_usages": [
{
"server_usages": [
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f06",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-3",
"started_at": "%(strtime)s",
"state": "active",
"tenant_id": "0000000e737461636b20342065000000",
"uptime": 3600,
"vcpus": 1
}
],
"start": "%(strtime)s",
"stop": "%(strtime)s",
"tenant_id": "0000000e737461636b20342065000000",
"total_hours": 1.0,
"total_local_gb_usage": 1.0,
"total_memory_mb_usage": 512.0,
"total_vcpus_usage": 1.0
},
{
"server_usages": [
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f00",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-1",
"started_at": "%(strtime)s",
"state": "active",
"tenant_id": "6f70656e737461636b20342065766572",
"uptime": 3600,
"vcpus": 1
},
{
"ended_at": null,
"flavor": "m1.tiny",
"hours": 1.0,
"instance_id": "1f1deceb-17b5-4c04-84c7-e0d4499c8f03",
"local_gb": 1,
"memory_mb": 512,
"name": "instance-2",
"started_at": "%(strtime)s",
"state": "active",
"tenant_id": "6f70656e737461636b20342065766572",
"uptime": 3600,
"vcpus": 1
}
],
"start": "%(strtime)s",
"stop": "%(strtime)s",
"tenant_id": "6f70656e737461636b20342065766572",
"total_hours": 2.0,
"total_local_gb_usage": 2.0,
"total_memory_mb_usage": 1024.0,
"total_vcpus_usage": 2.0
}
]
}

@ -79,6 +79,9 @@ class SimpleTenantUsageV240Test(test_servers.ServersSampleBase):
super(SimpleTenantUsageV240Test, self).setUp() super(SimpleTenantUsageV240Test, self).setUp()
self.api.microversion = self.microversion self.api.microversion = self.microversion
self.project_id_0 = astb.PROJECT_ID
self.project_id_1 = '0000000e737461636b20342065000000'
started = timeutils.utcnow() started = timeutils.utcnow()
now = started + datetime.timedelta(hours=1) now = started + datetime.timedelta(hours=1)
@ -87,8 +90,11 @@ class SimpleTenantUsageV240Test(test_servers.ServersSampleBase):
# make uuids incrementing, so that sort order is deterministic # make uuids incrementing, so that sort order is deterministic
uuid_format = '1f1deceb-17b5-4c04-84c7-e0d4499c8f%02d' uuid_format = '1f1deceb-17b5-4c04-84c7-e0d4499c8f%02d'
mock_uuids.side_effect = [uuid_format % x for x in range(100)] mock_uuids.side_effect = [uuid_format % x for x in range(100)]
self.project_id = self.project_id_0
self.instance1_uuid = self._post_server(name='instance-1') self.instance1_uuid = self._post_server(name='instance-1')
self.instance2_uuid = self._post_server(name='instance-2') self.instance2_uuid = self._post_server(name='instance-2')
self.project_id = self.project_id_1
self.instance3_uuid = self._post_server(name='instance-3') self.instance3_uuid = self._post_server(name='instance-3')
timeutils.set_time_override(now) timeutils.set_time_override(now)
@ -118,9 +124,27 @@ class SimpleTenantUsageV240Test(test_servers.ServersSampleBase):
self._verify_response(template_name, {}, response, 200) self._verify_response(template_name, {}, response, 200)
def test_get_tenant_usage_details(self): def test_get_tenant_usage_details(self):
tenant_id = astb.PROJECT_ID tenant_id = self.project_id_0
url = 'os-simple-tenant-usage/{tenant}?%s'.format(tenant=tenant_id) url = 'os-simple-tenant-usage/{tenant}?%s'.format(tenant=tenant_id)
response = self._do_get(url % (parse.urlencode(self.query))) response = self._do_get(url % (parse.urlencode(self.query)))
template_name = 'simple-tenant-usage-get-specific' template_name = 'simple-tenant-usage-get-specific'
subs = {'tenant_id': self.api.project_id} subs = {'tenant_id': self.project_id_0}
self._verify_response(template_name, subs, response, 200) self._verify_response(template_name, subs, response, 200)
def test_get_tenants_usage_end_marker(self):
# When using the last server retrieved as a marker,
# the subsequent usages list should be empty (bug #1796689).
url = 'os-simple-tenant-usage?%s'
query = dict(detailed=1,
start=self.query['start'],
end=self.query['end'])
response = self._do_get(url % (parse.urlencode(query)))
template_name = 'simple-tenant-usage-get-all'
self._verify_response(template_name, {}, response, 200)
last_server = response.json()['tenant_usages'][-1]['server_usages'][-1]
query['marker'] = last_server['instance_id']
response = self._do_get(url % (parse.urlencode(query)))
self.assertEqual(0, len(response.json()['tenant_usages']))

@ -303,6 +303,24 @@ class ApiSampleTestBase(integrated_helpers._IntegratedTestBase):
return matched_value return matched_value
@property
def project_id(self):
# We'll allow test cases to override the default project id. This is
# useful when using multiple tenants.
project_id = None
try:
project_id = self.api.project_id
except AttributeError:
pass
return project_id or PROJECT_ID
@project_id.setter
def project_id(self, project_id):
self.api.project_id = project_id
# Reset cached credentials
self.api.auth_result = None
def generalize_subs(self, subs, vanilla_regexes): def generalize_subs(self, subs, vanilla_regexes):
"""Give the test a chance to modify subs after the server response """Give the test a chance to modify subs after the server response
was verified, and before the on-disk doc/api_samples file is checked. was verified, and before the on-disk doc/api_samples file is checked.
@ -317,17 +335,20 @@ class ApiSampleTestBase(integrated_helpers._IntegratedTestBase):
def _update_links(self, sample_data): def _update_links(self, sample_data):
"""Process sample data and update version specific links.""" """Process sample data and update version specific links."""
# replace version urls # replace version urls
url_re = self._get_host() + "/v(2|2\.1)/" + PROJECT_ID
project_id_exp = '(%s|%s)' % (PROJECT_ID, self.project_id)
url_re = self._get_host() + "/v(2|2\.1)/" + project_id_exp
new_url = self._get_host() + "/" + self.api_major_version new_url = self._get_host() + "/" + self.api_major_version
if self._project_id: if self._use_project_id:
new_url += "/" + PROJECT_ID new_url += "/" + self.project_id
updated_data = re.sub(url_re, new_url, sample_data) updated_data = re.sub(url_re, new_url, sample_data)
# replace unversioned urls # replace unversioned urls
url_re = self._get_host() + "/" + PROJECT_ID url_re = self._get_host() + "/" + project_id_exp
new_url = self._get_host() new_url = self._get_host()
if self._project_id: if self._use_project_id:
new_url += "/" + PROJECT_ID new_url += "/" + self.project_id
updated_data = re.sub(url_re, new_url, updated_data) updated_data = re.sub(url_re, new_url, updated_data)
return updated_data return updated_data
@ -461,17 +482,17 @@ class ApiSampleTestBase(integrated_helpers._IntegratedTestBase):
def _get_compute_endpoint(self): def _get_compute_endpoint(self):
# NOTE(sdague): "openstack" is stand in for project_id, it # NOTE(sdague): "openstack" is stand in for project_id, it
# should be more generic in future. # should be more generic in future.
if self._project_id: if self._use_project_id:
return '%s/%s' % (self._get_host(), PROJECT_ID) return '%s/%s' % (self._get_host(), self.project_id)
else: else:
return self._get_host() return self._get_host()
def _get_vers_compute_endpoint(self): def _get_vers_compute_endpoint(self):
# NOTE(sdague): "openstack" is stand in for project_id, it # NOTE(sdague): "openstack" is stand in for project_id, it
# should be more generic in future. # should be more generic in future.
if self._project_id: if self._use_project_id:
return '%s/%s/%s' % (self._get_host(), self.api_major_version, return '%s/%s/%s' % (self._get_host(), self.api_major_version,
PROJECT_ID) self.project_id)
else: else:
return '%s/%s' % (self._get_host(), self.api_major_version) return '%s/%s' % (self._get_host(), self.api_major_version)

@ -43,7 +43,7 @@ class TestCompareResult(test.NoDBTestCase):
# required by ApiSampleTestBase # required by ApiSampleTestBase
ast_instance.api_major_version = 'v2' ast_instance.api_major_version = 'v2'
ast_instance._project_id = 'True' ast_instance._use_project_id = 'True'
# automagically create magic methods usually handled by test classes # automagically create magic methods usually handled by test classes
ast_instance.compute = mock.MagicMock() ast_instance.compute = mock.MagicMock()

@ -0,0 +1,6 @@
---
fixes:
- |
The ``os-simple-tenant-usage`` pagination has been fixed. In some cases,
nova usage-list would have returned incorrect results because of this.
See bug https://launchpad.net/bugs/1796689 for details.