Transform keypair.delete notification

The keypair.delete.start and keypair.delete.end notifications
has been transformed to the versioned notification framework.

Change-Id: I48f28a56f9232faf52f458588500a9d77eb6b679
Implements: bp versioned-notification-transformation-queens
This commit is contained in:
Béla Vancsics 2017-05-11 09:40:55 +02:00 committed by Matt Riedemann
parent 008bc0b971
commit a489dfa7da
8 changed files with 86 additions and 5 deletions

View File

@ -0,0 +1,17 @@
{
"priority": "INFO",
"payload": {
"nova_object.version": "1.0",
"nova_object.namespace": "nova",
"nova_object.name": "KeypairPayload",
"nova_object.data": {
"user_id": "fake",
"name": "my-key",
"fingerprint": "1e:2c:9b:56:79:4b:45:77:f9:ca:7a:98:2c:b0:d5:3c",
"public_key": "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDx8nkQv/zgGgB4rMYmIf+6A4l6Rr+o/6lHBQdW5aYd44bd8JttDCE/F/pNRr0lRE+PiqSPO8nDPHw0010JeMH9gYgnnFlyY3/OcJ02RhIPyyxYpv9FhY+2YiUkpwFOcLImyrxEsYXpD/0d3ac30bNH6Sw9JD9UZHYcpSxsIbECHw== Generated-by-Nova",
"type": "ssh"
}
},
"event_type": "keypair.delete.end",
"publisher_id": "nova-api:fake-mini"
}

View File

@ -0,0 +1,17 @@
{
"priority": "INFO",
"payload": {
"nova_object.version": "1.0",
"nova_object.namespace": "nova",
"nova_object.name": "KeypairPayload",
"nova_object.data": {
"user_id": "fake",
"name": "my-key",
"fingerprint": "1e:2c:9b:56:79:4b:45:77:f9:ca:7a:98:2c:b0:d5:3c",
"public_key": "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDx8nkQv/zgGgB4rMYmIf+6A4l6Rr+o/6lHBQdW5aYd44bd8JttDCE/F/pNRr0lRE+PiqSPO8nDPHw0010JeMH9gYgnnFlyY3/OcJ02RhIPyyxYpv9FhY+2YiUkpwFOcLImyrxEsYXpD/0d3ac30bNH6Sw9JD9UZHYcpSxsIbECHw== Generated-by-Nova",
"type": "ssh"
}
},
"event_type": "keypair.delete.start",
"publisher_id": "nova-api:fake-mini"
}

View File

@ -4904,7 +4904,18 @@ class KeypairAPI(base.Base):
def delete_key_pair(self, context, user_id, key_name): def delete_key_pair(self, context, user_id, key_name):
"""Delete a keypair by name.""" """Delete a keypair by name."""
self._notify(context, 'delete.start', key_name) self._notify(context, 'delete.start', key_name)
keypair = self.get_key_pair(context, user_id, key_name)
compute_utils.notify_about_keypair_action(
context=context,
keypair=keypair,
action=fields_obj.NotificationAction.DELETE,
phase=fields_obj.NotificationPhase.START)
objects.KeyPair.destroy_by_name(context, user_id, key_name) objects.KeyPair.destroy_by_name(context, user_id, key_name)
compute_utils.notify_about_keypair_action(
context=context,
keypair=keypair,
action=fields_obj.NotificationAction.DELETE,
phase=fields_obj.NotificationPhase.END)
self._notify(context, 'delete.end', key_name) self._notify(context, 'delete.end', key_name)
def get_key_pairs(self, context, user_id, limit=None, marker=None): def get_key_pairs(self, context, user_id, limit=None, marker=None):

View File

@ -41,6 +41,8 @@ class KeypairPayload(base.NotificationPayloadBase):
@base.notification_sample('keypair-create-start.json') @base.notification_sample('keypair-create-start.json')
@base.notification_sample('keypair-create-end.json') @base.notification_sample('keypair-create-end.json')
@base.notification_sample('keypair-delete-start.json')
@base.notification_sample('keypair-delete-end.json')
@nova_base.NovaObjectRegistry.register_notification @nova_base.NovaObjectRegistry.register_notification
class KeypairNotification(base.NotificationBase): class KeypairNotification(base.NotificationBase):
# Version 1.0: Initial version # Version 1.0: Initial version

View File

@ -430,6 +430,9 @@ class TestOpenStackClient(object):
def post_keypair(self, keypair): def post_keypair(self, keypair):
return self.api_post('/os-keypairs', keypair).body['keypair'] return self.api_post('/os-keypairs', keypair).body['keypair']
def delete_keypair(self, keypair_name):
self.api_delete('/os-keypairs/%s' % keypair_name)
def get_active_migrations(self, server_id): def get_active_migrations(self, server_id):
return self.api_get('/servers/%s/migrations' % return self.api_get('/servers/%s/migrations' %
server_id).body['migrations'] server_id).body['migrations']

View File

@ -17,7 +17,7 @@ from nova.tests.unit import fake_notifier
class TestKeypairNotificationSample( class TestKeypairNotificationSample(
notification_sample_base.NotificationSampleTestBase): notification_sample_base.NotificationSampleTestBase):
def test_keypair_create(self): def test_keypair_create_delete(self):
keypair_req = { keypair_req = {
"keypair": { "keypair": {
"name": "my-key", "name": "my-key",
@ -38,3 +38,20 @@ class TestKeypairNotificationSample(
"public_key": keypair['public_key'] "public_key": keypair['public_key']
}, },
actual=fake_notifier.VERSIONED_NOTIFICATIONS[1]) actual=fake_notifier.VERSIONED_NOTIFICATIONS[1])
self.api.delete_keypair(keypair['name'])
self.assertEqual(4, len(fake_notifier.VERSIONED_NOTIFICATIONS))
self._verify_notification(
'keypair-delete-start',
replacements={
"fingerprint": keypair['fingerprint'],
"public_key": keypair['public_key']
},
actual=fake_notifier.VERSIONED_NOTIFICATIONS[2])
self._verify_notification(
'keypair-delete-end',
replacements={
"fingerprint": keypair['fingerprint'],
"public_key": keypair['public_key']
},
actual=fake_notifier.VERSIONED_NOTIFICATIONS[3])

View File

@ -21,6 +21,7 @@ import webob
from nova.api.openstack.compute import keypairs as keypairs_v21 from nova.api.openstack.compute import keypairs as keypairs_v21
from nova.api.openstack import wsgi as os_wsgi from nova.api.openstack import wsgi as os_wsgi
from nova.compute import api as compute_api from nova.compute import api as compute_api
from nova import context as nova_context
from nova import exception from nova import exception
from nova import objects from nova import objects
from nova import policy from nova import policy
@ -271,7 +272,10 @@ class KeypairsTestV21(test.TestCase):
self.assertIn("Key pair 'create_duplicate' already exists.", self.assertIn("Key pair 'create_duplicate' already exists.",
ex.explanation) ex.explanation)
def test_keypair_delete(self): @mock.patch('nova.objects.KeyPair.get_by_name')
def test_keypair_delete(self, mock_get_by_name):
mock_get_by_name.return_value = objects.KeyPair(
nova_context.get_admin_context(), **fake_keypair('FAKE'))
self.controller.delete(self.req, 'FAKE') self.controller.delete(self.req, 'FAKE')
def test_keypair_get_keypair_not_found(self): def test_keypair_get_keypair_not_found(self):

View File

@ -28,6 +28,7 @@ from nova.tests.unit.compute import test_compute
from nova.tests.unit import fake_crypto from nova.tests.unit import fake_crypto
from nova.tests.unit import fake_notifier from nova.tests.unit import fake_notifier
from nova.tests.unit.objects import test_keypair from nova.tests.unit.objects import test_keypair
from nova.tests.unit import utils as test_utils
CONF = cfg.CONF CONF = cfg.CONF
@ -243,14 +244,23 @@ class GetKeypairsTestCase(KeypairAPITestCase):
class DeleteKeypairTestCase(KeypairAPITestCase): class DeleteKeypairTestCase(KeypairAPITestCase):
def test_success(self): @mock.patch('nova.compute.utils.notify_about_keypair_action')
self.keypair_api.get_key_pair(self.ctxt, self.ctxt.user_id, def test_success(self, mock_notify):
self.existing_key_name)
self.keypair_api.delete_key_pair(self.ctxt, self.ctxt.user_id, self.keypair_api.delete_key_pair(self.ctxt, self.ctxt.user_id,
self.existing_key_name) self.existing_key_name)
self.assertRaises(exception.KeypairNotFound, self.assertRaises(exception.KeypairNotFound,
self.keypair_api.get_key_pair, self.ctxt, self.ctxt.user_id, self.keypair_api.get_key_pair, self.ctxt, self.ctxt.user_id,
self.existing_key_name) self.existing_key_name)
match_by_name = test_utils.CustomMockCallMatcher(
lambda keypair: keypair['name'] == self.existing_key_name)
mock_notify.assert_has_calls([
mock.call(context=self.ctxt,
keypair=match_by_name,
action='delete', phase='start'),
mock.call(context=self.ctxt,
keypair=match_by_name,
action='delete', phase='end')])
self._check_notifications(action='delete', self._check_notifications(action='delete',
key_name=self.existing_key_name) key_name=self.existing_key_name)