API Microversioning

This patch set adds API microversion support along
with the first API microversion: start/end time for
CONTINUOUS audits.

APIImpact

Implements: blueprint api-microversioning
Depends-On: I6bb838d777b2c7aa799a70485980e5dc87838456
Change-Id: I17309d80b637f02bc5e6d33294472e02add88f86
This commit is contained in:
Alexander Chadin
2018-10-04 12:33:28 +03:00
parent c2550e534e
commit c4a30153f1
29 changed files with 535 additions and 18 deletions

View File

@@ -203,7 +203,8 @@ class FunctionalTest(base.DbTestCase):
return response
def get_json(self, path, expect_errors=False, headers=None,
extra_environ=None, q=[], path_prefix=PATH_PREFIX, **params):
extra_environ=None, q=[], path_prefix=PATH_PREFIX,
return_json=True, **params):
"""Sends simulated HTTP GET request to Pecan test app.
:param path: url path of target service
@@ -236,7 +237,7 @@ class FunctionalTest(base.DbTestCase):
headers=headers,
extra_environ=extra_environ,
expect_errors=expect_errors)
if not expect_errors:
if return_json and not expect_errors:
response = response.json
print('GOT:%s' % response)
return response

View File

@@ -900,7 +900,10 @@ class TestPost(api_base.FunctionalTest):
audit_dict['start_time'] = str(start_time)
audit_dict['end_time'] = str(end_time)
response = self.post_json('/audits', audit_dict)
response = self.post_json(
'/audits',
audit_dict,
headers={'OpenStack-API-Version': 'infra-optim 1.1'})
self.assertEqual('application/json', response.content_type)
self.assertEqual(201, response.status_int)
self.assertEqual(objects.audit.State.PENDING,
@@ -919,6 +922,34 @@ class TestPost(api_base.FunctionalTest):
self.assertEqual(iso_start_time, return_start_time)
self.assertEqual(iso_end_time, return_end_time)
@mock.patch.object(deapi.DecisionEngineAPI, 'trigger_audit')
def test_create_continuous_audit_with_start_end_time_incompatible_version(
self, mock_trigger_audit):
mock_trigger_audit.return_value = mock.ANY
start_time = datetime.datetime(2018, 3, 1, 0, 0)
end_time = datetime.datetime(2018, 4, 1, 0, 0)
audit_dict = post_get_test_audit(
params_to_exclude=['uuid', 'state', 'scope',
'next_run_time', 'hostname', 'goal']
)
audit_dict['audit_type'] = objects.audit.AuditType.CONTINUOUS.value
audit_dict['interval'] = '1200'
audit_dict['start_time'] = str(start_time)
audit_dict['end_time'] = str(end_time)
response = self.post_json(
'/audits',
audit_dict,
headers={'X-OpenStack-Watcher-API-Version': '1.0'},
expect_errors=True)
self.assertEqual('application/json', response.content_type)
self.assertEqual(406, response.status_int)
expected_error_msg = 'Request not acceptable.'
self.assertTrue(response.json['error_message'])
self.assertIn(expected_error_msg, response.json['error_message'])
assert not mock_trigger_audit.called
class TestDelete(api_base.FunctionalTest):

View File

@@ -0,0 +1,104 @@
# 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.
from watcher.api.controllers.v1 import versions
from watcher.tests.api import base as api_base
SERVICE_TYPE = 'infra-optim'
H_MIN_VER = 'openstack-api-minimum-version'
H_MAX_VER = 'openstack-api-maximum-version'
H_RESP_VER = 'openstack-api-version'
MIN_VER = versions.min_version_string()
MAX_VER = versions.max_version_string()
class TestMicroversions(api_base.FunctionalTest):
controller_list_response = [
'scoring_engines', 'audit_templates', 'audits', 'actions',
'action_plans', 'services']
def setUp(self):
super(TestMicroversions, self).setUp()
def test_wrong_major_version(self):
response = self.get_json(
'/',
headers={'OpenStack-API-Version': ' '.join([SERVICE_TYPE,
'10'])},
expect_errors=True, return_json=False)
self.assertEqual('application/json', response.content_type)
self.assertEqual(406, response.status_int)
expected_error_msg = ('Invalid value for'
' OpenStack-API-Version header')
self.assertTrue(response.json['error_message'])
self.assertIn(expected_error_msg, response.json['error_message'])
def test_extend_initial_version_with_micro(self):
response = self.get_json(
'/',
headers={'OpenStack-API-Version': ' '.join([SERVICE_TYPE,
'1'])},
return_json=False)
self.assertEqual(response.headers[H_MIN_VER], MIN_VER)
self.assertEqual(response.headers[H_MAX_VER], MAX_VER)
self.assertEqual(response.headers[H_RESP_VER], MIN_VER)
self.assertTrue(all(x in response.json.keys() for x in
self.controller_list_response))
def test_without_microversion(self):
response = self.get_json('/', return_json=False)
self.assertEqual(response.headers[H_MIN_VER], MIN_VER)
self.assertEqual(response.headers[H_MAX_VER], MAX_VER)
self.assertEqual(response.headers[H_RESP_VER], MIN_VER)
self.assertTrue(all(x in response.json.keys() for x in
self.controller_list_response))
def test_new_client_new_api(self):
response = self.get_json(
'/',
headers={'OpenStack-API-Version': ' '.join([SERVICE_TYPE,
'1.1'])},
return_json=False)
self.assertEqual(response.headers[H_MIN_VER], MIN_VER)
self.assertEqual(response.headers[H_MAX_VER], MAX_VER)
self.assertEqual(response.headers[H_RESP_VER], '1.1')
self.assertTrue(all(x in response.json.keys() for x in
self.controller_list_response))
def test_latest_microversion(self):
response = self.get_json(
'/',
headers={'OpenStack-API-Version': ' '.join([SERVICE_TYPE,
'latest'])},
return_json=False)
self.assertEqual(response.headers[H_MIN_VER], MIN_VER)
self.assertEqual(response.headers[H_MAX_VER], MAX_VER)
self.assertEqual(response.headers[H_RESP_VER], MAX_VER)
self.assertTrue(all(x in response.json.keys() for x in
self.controller_list_response))
def test_unsupported_version(self):
response = self.get_json(
'/',
headers={'OpenStack-API-Version': ' '.join([SERVICE_TYPE,
'1.999'])},
expect_errors=True)
self.assertEqual(406, response.status_int)
self.assertEqual(response.headers[H_MIN_VER], MIN_VER)
self.assertEqual(response.headers[H_MAX_VER], MAX_VER)
expected_error_msg = ('Version 1.999 was requested but the minor '
'version is not supported by this service. '
'The supported version range is')
self.assertTrue(response.json['error_message'])
self.assertIn(expected_error_msg, response.json['error_message'])