Update nova os-server-external-events response logic

When the power update was written [1] to use nova's
os-server-external-events API, the internals of that method had a bug
(see the related bug). Fortunately, the code was written so it works
against the nova side with or without the fix. However, for the sake of
propriety, this commit refactors the code to reflect nova's behavior
more accurately.

Specifically: Previously, it was impossible for nova to respond 207 when
the client sent a single event (as ironic does). The code path
accounting for that 207 existed, but always returned True ("success")
rather than returning False ("failure") if the event code was >=400.
Fortunately, the return value is only ever used in unit test, not
production code, so it didn't matter.

With this commit, the 207 path is handled correctly, such that the
method "succeeds" if the event code is <400 (which would never happen in
real life) and "fails" if the event code is >=400.

[1] I6d105524e1645d9a40dfeae2850c33cf2d110826
Related-Bug: #1855752

Change-Id: I13744175127e9956fb785a9efc82193c333b2bdc
This commit is contained in:
Eric Fried 2019-12-10 09:19:18 -06:00
parent 8a0b1a39f2
commit 950acaaf17
2 changed files with 28 additions and 26 deletions

View File

@ -82,7 +82,8 @@ def _send_event(context, event, api_version=None):
if code >= 400: if code >= 400:
LOG.warning('Nova event: %s returned with failed status.', resp_event) LOG.warning('Nova event: %s returned with failed status.', resp_event)
else: return False
LOG.debug('Nova event response: %s.', resp_event) LOG.debug('Nova event response: %s.', resp_event)
return True return True

View File

@ -53,15 +53,25 @@ class NovaApiTestCase(base.TestCase):
self.api = nova self.api = nova
self.ctx = context.get_admin_context() self.ctx = context.get_admin_context()
@ddt.data({'events': [{'status': 'completed', @ddt.unpack
# one @ddt.data element comprises:
# - nova_result: POST response JSON dict
# - resp_status: POST response status_code
# - exp_ret: Expected bool return value from power_update()
@ddt.data([{'events': [{'status': 'completed',
'tag': 'POWER_OFF', 'tag': 'POWER_OFF',
'name': 'power-update', 'name': 'power-update',
'server_uuid': '1234', 'server_uuid': '1234',
'code': 200}]}, 'code': 200}]},
{'events': [{'code': 404}]}, 200, True],
{'events': [{'code': 400}]}) [{'events': [{'code': 422}]}, 207, False],
[{'events': [{'code': 404}]}, 207, False],
[{'events': [{'code': 400}]}, 207, False],
# This (response 207, event code 200) will never happen IRL
[{'events': [{'code': 200}]}, 207, True])
@mock.patch.object(nova, '_get_nova_adapter') @mock.patch.object(nova, '_get_nova_adapter')
def test_power_update(self, nova_result, mock_adapter, mock_log): def test_power_update(self, nova_result, resp_status, exp_ret,
mock_adapter, mock_log):
server_ids = ['server-id-1', 'server-id-2'] server_ids = ['server-id-1', 'server-id-2']
nova_adapter = mock.Mock() nova_adapter = mock.Mock()
with mock.patch.object(nova_adapter, 'post') as mock_post_event: with mock.patch.object(nova_adapter, 'post') as mock_post_event:
@ -70,12 +80,12 @@ class NovaApiTestCase(base.TestCase):
def json_func(): def json_func():
return nova_result return nova_result
post_resp_mock.json = json_func post_resp_mock.json = json_func
post_resp_mock.status_code = 200 post_resp_mock.status_code = resp_status
mock_adapter.return_value = nova_adapter mock_adapter.return_value = nova_adapter
mock_post_event.return_value = post_resp_mock mock_post_event.return_value = post_resp_mock
for server in server_ids: for server in server_ids:
result = self.api.power_update(self.ctx, server, 'power on') result = self.api.power_update(self.ctx, server, 'power on')
self.assertTrue(result) self.assertEqual(exp_ret, result)
mock_adapter.assert_has_calls([mock.call(), mock.call()]) mock_adapter.assert_has_calls([mock.call(), mock.call()])
req_url = '/os-server-external-events' req_url = '/os-server-external-events'
@ -95,7 +105,7 @@ class NovaApiTestCase(base.TestCase):
global_request_id=self.ctx.global_id, global_request_id=self.ctx.global_id,
raise_exc=False) raise_exc=False)
]) ])
if nova_result['events'][0]['code'] != 200: if not exp_ret:
expected = ('Nova event: %s returned with failed status.', expected = ('Nova event: %s returned with failed status.',
nova_result['events'][0]) nova_result['events'][0])
mock_log.warning.assert_called_with(*expected) mock_log.warning.assert_called_with(*expected)
@ -122,23 +132,14 @@ class NovaApiTestCase(base.TestCase):
'tag': 'POWER_OFF'}] 'tag': 'POWER_OFF'}]
nova_result = requests.Response() nova_result = requests.Response()
with mock.patch.object(nova_adapter, 'post') as mock_post_event: with mock.patch.object(nova_adapter, 'post') as mock_post_event:
for stat_code in (500, 404, 207): for stat_code in (500, 404, 400):
mock_log.reset_mock() mock_log.reset_mock()
nova_result.status_code = stat_code nova_result.status_code = stat_code
type(nova_result).text = mock.PropertyMock(return_value="blah") type(nova_result).text = mock.PropertyMock(return_value="blah")
if stat_code == 207:
def json_func():
return {'events': [{}]}
nova_result.json = json_func
mock_post_event.return_value = nova_result mock_post_event.return_value = nova_result
result = self.api.power_update( result = self.api.power_update(
self.ctx, 'server-id-1', 'power off') self.ctx, 'server-id-1', 'power off')
self.assertFalse(result) self.assertFalse(result)
if stat_code == 207:
expected = ('Invalid response %s returned from nova for '
'power-update event %s. %s.')
self.assertIn(expected, mock_log.error.call_args[0][0])
else:
expected = ("Failed to notify nova on event: %s. %s.", expected = ("Failed to notify nova on event: %s. %s.",
event[0], "blah") event[0], "blah")
mock_log.warning.assert_called_once_with(*expected) mock_log.warning.assert_called_once_with(*expected)