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:
@@ -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
|
||||
|
||||
@@ -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):
|
||||
|
||||
|
||||
104
watcher/tests/api/v1/test_microversions.py
Normal file
104
watcher/tests/api/v1/test_microversions.py
Normal 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'])
|
||||
Reference in New Issue
Block a user