Add delete action for key pair
This patch adds delete actions for angularized key pair panel. Change-Id: Iccb5014add0e19d6154bd6261d97a83b2ecdf32f Partial-Implements: blueprint ng-keypairs
This commit is contained in:
parent
f911d0dd40
commit
e7f22178b2
openstack_dashboard
api
static/app/core
keypairs
openstack-service-api
test/api_tests
@ -459,8 +459,8 @@ def keypair_import(request, name, public_key):
|
|||||||
|
|
||||||
|
|
||||||
@profiler.trace
|
@profiler.trace
|
||||||
def keypair_delete(request, keypair_id):
|
def keypair_delete(request, name):
|
||||||
novaclient(request).keypairs.delete(keypair_id)
|
novaclient(request).keypairs.delete(name)
|
||||||
|
|
||||||
|
|
||||||
@profiler.trace
|
@profiler.trace
|
||||||
@ -469,8 +469,8 @@ def keypair_list(request):
|
|||||||
|
|
||||||
|
|
||||||
@profiler.trace
|
@profiler.trace
|
||||||
def keypair_get(request, keypair_id):
|
def keypair_get(request, name):
|
||||||
return novaclient(request).keypairs.get(keypair_id)
|
return novaclient(request).keypairs.get(name)
|
||||||
|
|
||||||
|
|
||||||
@profiler.trace
|
@profiler.trace
|
||||||
|
@ -101,6 +101,10 @@ class Keypair(generic.View):
|
|||||||
"""Get a specific keypair."""
|
"""Get a specific keypair."""
|
||||||
return api.nova.keypair_get(request, name).to_dict()
|
return api.nova.keypair_get(request, name).to_dict()
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def delete(self, request, name):
|
||||||
|
api.nova.keypair_delete(request, name)
|
||||||
|
|
||||||
|
|
||||||
@urls.register
|
@urls.register
|
||||||
class Services(generic.View):
|
class Services(generic.View):
|
||||||
|
@ -0,0 +1,62 @@
|
|||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @ngdoc overview
|
||||||
|
* @ngname horizon.app.core.keypairs.actions
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* Provides all of the actions for keypairs.
|
||||||
|
*/
|
||||||
|
angular.module('horizon.app.core.keypairs.actions', [
|
||||||
|
'horizon.framework.conf',
|
||||||
|
'horizon.app.core.keypairs'
|
||||||
|
])
|
||||||
|
.run(registerKeypairActions);
|
||||||
|
|
||||||
|
registerKeypairActions.$inject = [
|
||||||
|
'horizon.framework.conf.resource-type-registry.service',
|
||||||
|
'horizon.app.core.keypairs.actions.delete.service',
|
||||||
|
'horizon.app.core.keypairs.resourceType'
|
||||||
|
];
|
||||||
|
|
||||||
|
function registerKeypairActions(
|
||||||
|
registry,
|
||||||
|
deleteKeypairService,
|
||||||
|
resourceType
|
||||||
|
) {
|
||||||
|
var keypairResourceType = registry.getResourceType(resourceType);
|
||||||
|
keypairResourceType.batchActions
|
||||||
|
.append({
|
||||||
|
id: 'batchDeleteKeypairAction',
|
||||||
|
service: deleteKeypairService,
|
||||||
|
template: {
|
||||||
|
type: 'delete-selected',
|
||||||
|
text: gettext('Delete Key Pairs')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
keypairResourceType.itemActions
|
||||||
|
.append({
|
||||||
|
id: 'deleteKeypairAction',
|
||||||
|
service: deleteKeypairService,
|
||||||
|
template: {
|
||||||
|
type: 'delete',
|
||||||
|
text: gettext('Delete Key Pair')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})();
|
@ -0,0 +1,121 @@
|
|||||||
|
/**
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
* not use self 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
angular
|
||||||
|
.module('horizon.app.core.keypairs')
|
||||||
|
.factory('horizon.app.core.keypairs.actions.delete.service', deleteService);
|
||||||
|
|
||||||
|
deleteService.$inject = [
|
||||||
|
'$location',
|
||||||
|
'horizon.app.core.keypairs.resourceType',
|
||||||
|
'horizon.app.core.openstack-service-api.nova',
|
||||||
|
'horizon.app.core.openstack-service-api.policy',
|
||||||
|
'horizon.framework.util.actions.action-result.service',
|
||||||
|
'horizon.framework.util.i18n.gettext',
|
||||||
|
'horizon.framework.widgets.modal.deleteModalService'
|
||||||
|
];
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @ngdoc factory
|
||||||
|
* @name horizon.app.core.keypairs.actions.delete.service
|
||||||
|
*
|
||||||
|
* @Description
|
||||||
|
* Brings up the delete keypairs confirmation modal dialog.
|
||||||
|
|
||||||
|
* On submit, delete given keypairs.
|
||||||
|
* On cancel, do nothing.
|
||||||
|
*/
|
||||||
|
function deleteService(
|
||||||
|
$location,
|
||||||
|
resourceType,
|
||||||
|
nova,
|
||||||
|
policy,
|
||||||
|
actionResultService,
|
||||||
|
gettext,
|
||||||
|
deleteModal
|
||||||
|
) {
|
||||||
|
|
||||||
|
var service = {
|
||||||
|
allowed: allowed,
|
||||||
|
perform: perform
|
||||||
|
};
|
||||||
|
|
||||||
|
return service;
|
||||||
|
|
||||||
|
//////////////
|
||||||
|
|
||||||
|
function allowed() {
|
||||||
|
return policy.ifAllowed({ rules: [['compute', 'os_compute_api:os-keypairs:delete']] });
|
||||||
|
}
|
||||||
|
|
||||||
|
function perform(items, scope) {
|
||||||
|
var keypairs = angular.isArray(items) ? items : [items];
|
||||||
|
var context = {
|
||||||
|
labels: labelize(keypairs.length),
|
||||||
|
deleteEntity: deleteKeypair
|
||||||
|
};
|
||||||
|
return deleteModal.open(scope, keypairs, context).then(deleteResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
function deleteResult(deleteModalResult) {
|
||||||
|
// To make the result of this action generically useful, reformat the return
|
||||||
|
// from the deleteModal into a standard form
|
||||||
|
var actionResult = actionResultService.getActionResult();
|
||||||
|
deleteModalResult.pass.forEach(function markDeleted(item) {
|
||||||
|
actionResult.deleted(resourceType, item.context.id);
|
||||||
|
});
|
||||||
|
deleteModalResult.fail.forEach(function markFailed(item) {
|
||||||
|
actionResult.failed(resourceType, item.context.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (actionResult.result.failed.length === 0 && actionResult.result.deleted.length > 0) {
|
||||||
|
$location.path("/project/key_pairs");
|
||||||
|
} else {
|
||||||
|
return actionResult.result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function labelize(count) {
|
||||||
|
return {
|
||||||
|
|
||||||
|
title: ngettext(
|
||||||
|
'Confirm Delete Key Pair',
|
||||||
|
'Confirm Delete Key Pairs', count),
|
||||||
|
|
||||||
|
message: ngettext(
|
||||||
|
'You have selected "%s". Deleted key pair is not recoverable.',
|
||||||
|
'You have selected "%s". Deleted key pairs are not recoverable.', count),
|
||||||
|
|
||||||
|
submit: ngettext(
|
||||||
|
'Delete Key Pair',
|
||||||
|
'Delete Key Pairs', count),
|
||||||
|
|
||||||
|
success: ngettext(
|
||||||
|
'Deleted Key Pair: %s.',
|
||||||
|
'Deleted Key Pairs: %s.', count),
|
||||||
|
|
||||||
|
error: ngettext(
|
||||||
|
'Unable to delete Key Pair: %s.',
|
||||||
|
'Unable to delete Key Pairs: %s.', count)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function deleteKeypair(keypair) {
|
||||||
|
return nova.deleteKeypair(keypair, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})();
|
@ -0,0 +1,135 @@
|
|||||||
|
/*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
* not use self 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
describe('horizon.app.core.keypairs.actions.delete.service', function() {
|
||||||
|
|
||||||
|
var service, novaAPI, $scope, deferredModal;
|
||||||
|
var deleteModalService = {
|
||||||
|
open: function () {
|
||||||
|
deferredModal.resolve({
|
||||||
|
pass: [{context: {id: 'a'}}],
|
||||||
|
fail: [{context: {id: 'b'}}]
|
||||||
|
});
|
||||||
|
return deferredModal.promise;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////
|
||||||
|
|
||||||
|
beforeEach(module('horizon.app.core'));
|
||||||
|
beforeEach(module('horizon.app.core.keypairs'));
|
||||||
|
beforeEach(module('horizon.framework'));
|
||||||
|
beforeEach(module('horizon.framework.widgets.modal', function($provide) {
|
||||||
|
$provide.value('horizon.framework.widgets.modal.deleteModalService', deleteModalService);
|
||||||
|
}));
|
||||||
|
beforeEach(inject(function($injector, _$rootScope_, $q) {
|
||||||
|
$scope = _$rootScope_.$new();
|
||||||
|
deferredModal = $q.defer();
|
||||||
|
service = $injector.get('horizon.app.core.keypairs.actions.delete.service');
|
||||||
|
novaAPI = $injector.get('horizon.app.core.openstack-service-api.nova');
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe('perform method', function() {
|
||||||
|
|
||||||
|
beforeEach(function() {
|
||||||
|
spyOn(deleteModalService, 'open').and.callThrough();
|
||||||
|
});
|
||||||
|
|
||||||
|
////////////
|
||||||
|
|
||||||
|
it('should open the delete modal and show correct labels, single object', testSingleLabels);
|
||||||
|
it('should open the delete modal and show correct labels, plural objects', testPluralLabels);
|
||||||
|
it('should open the delete modal with correct entities', testEntities);
|
||||||
|
it('should only delete keypairs that are valid', testValids);
|
||||||
|
it('should pass in a function that deletes an keypair', testNova);
|
||||||
|
it('should check the policy if the user is allowed to delete key pair', testAllowed);
|
||||||
|
|
||||||
|
////////////
|
||||||
|
|
||||||
|
function testSingleLabels() {
|
||||||
|
var keypairs = {name: 'Hokusai'};
|
||||||
|
service.perform(keypairs);
|
||||||
|
|
||||||
|
$scope.$apply();
|
||||||
|
|
||||||
|
var labels = deleteModalService.open.calls.argsFor(0)[2].labels;
|
||||||
|
expect(deleteModalService.open).toHaveBeenCalled();
|
||||||
|
angular.forEach(labels, function eachLabel(label) {
|
||||||
|
expect(label.toLowerCase()).toContain('key pair');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function testPluralLabels() {
|
||||||
|
var keypairs = [{name: 'Hokusai'}, {name: 'Utamaro'}];
|
||||||
|
service.perform(keypairs);
|
||||||
|
|
||||||
|
$scope.$apply();
|
||||||
|
|
||||||
|
var labels = deleteModalService.open.calls.argsFor(0)[2].labels;
|
||||||
|
expect(deleteModalService.open).toHaveBeenCalled();
|
||||||
|
angular.forEach(labels, function eachLabel(label) {
|
||||||
|
expect(label.toLowerCase()).toContain('key pairs');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function testEntities() {
|
||||||
|
var keypairs = [{name: 'Hokusai'}, {name: 'Utamaro'}, {name: 'Hiroshige'}];
|
||||||
|
service.perform(keypairs);
|
||||||
|
|
||||||
|
$scope.$apply();
|
||||||
|
|
||||||
|
var entities = deleteModalService.open.calls.argsFor(0)[1];
|
||||||
|
expect(deleteModalService.open).toHaveBeenCalled();
|
||||||
|
expect(entities.length).toEqual(keypairs.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
function testValids() {
|
||||||
|
var keypairs = [{name: 'Hokusai'}, {name: 'Utamaro'}, {name: 'Hiroshige'}];
|
||||||
|
service.perform(keypairs);
|
||||||
|
|
||||||
|
$scope.$apply();
|
||||||
|
|
||||||
|
var entities = deleteModalService.open.calls.argsFor(0)[1];
|
||||||
|
expect(deleteModalService.open).toHaveBeenCalled();
|
||||||
|
expect(entities.length).toBe(keypairs.length);
|
||||||
|
expect(entities[0].name).toEqual('Hokusai');
|
||||||
|
expect(entities[1].name).toEqual('Utamaro');
|
||||||
|
expect(entities[2].name).toEqual('Hiroshige');
|
||||||
|
}
|
||||||
|
|
||||||
|
function testNova() {
|
||||||
|
spyOn(novaAPI, 'deleteKeypair').and.callFake(angular.noop);
|
||||||
|
var keypairs = [{id: 1760, name: 'Hokusai'}, {id: 1753, name: 'Utamaro'}];
|
||||||
|
service.perform(keypairs);
|
||||||
|
|
||||||
|
$scope.$apply();
|
||||||
|
|
||||||
|
var contextArg = deleteModalService.open.calls.argsFor(0)[2];
|
||||||
|
var deleteFunction = contextArg.deleteEntity;
|
||||||
|
deleteFunction(keypairs[0].id);
|
||||||
|
expect(novaAPI.deleteKeypair).toHaveBeenCalledWith(keypairs[0].id, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
function testAllowed() {
|
||||||
|
var allowed = service.allowed();
|
||||||
|
expect(allowed).toBeTruthy();
|
||||||
|
}
|
||||||
|
}); // end of delete modal
|
||||||
|
|
||||||
|
}); // end of delete
|
||||||
|
|
||||||
|
})();
|
@ -28,6 +28,8 @@
|
|||||||
|
|
||||||
function onGetKeypair(response) {
|
function onGetKeypair(response) {
|
||||||
ctrl.keypair = response.data;
|
ctrl.keypair = response.data;
|
||||||
|
ctrl.keypair.keypair_id = ctrl.keypair.id;
|
||||||
|
ctrl.keypair.id = ctrl.keypair.name;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
resource-type-name="OS::Nova::Keypair"
|
resource-type-name="OS::Nova::Keypair"
|
||||||
cls="dl-horizontal"
|
cls="dl-horizontal"
|
||||||
item="ctrl.keypair"
|
item="ctrl.keypair"
|
||||||
property-groups="[['id', 'name', 'fingerprint', 'created_at', 'user_id', 'public_key']]">
|
property-groups="[['keypair_id', 'name', 'fingerprint', 'created_at', 'user_id', 'public_key']]">
|
||||||
</hz-resource-property-list>
|
</hz-resource-property-list>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
angular
|
angular
|
||||||
.module('horizon.app.core.keypairs', [
|
.module('horizon.app.core.keypairs', [
|
||||||
'ngRoute',
|
'ngRoute',
|
||||||
|
'horizon.app.core.keypairs.actions',
|
||||||
'horizon.app.core.keypairs.details'
|
'horizon.app.core.keypairs.details'
|
||||||
])
|
])
|
||||||
.constant('horizon.app.core.keypairs.resourceType', 'OS::Nova::Keypair')
|
.constant('horizon.app.core.keypairs.resourceType', 'OS::Nova::Keypair')
|
||||||
@ -72,7 +73,8 @@
|
|||||||
|
|
||||||
function keypairProperties() {
|
function keypairProperties() {
|
||||||
return {
|
return {
|
||||||
'id': {label: gettext('ID'), filters: ['noValue'] },
|
'id': {},
|
||||||
|
'keypair_id': {label: gettext('ID'), filters: ['noValue'] },
|
||||||
'name': {label: gettext('Name'), filters: ['noName'] },
|
'name': {label: gettext('Name'), filters: ['noName'] },
|
||||||
'fingerprint': {label: gettext('Fingerprint'), filters: ['noValue'] },
|
'fingerprint': {label: gettext('Fingerprint'), filters: ['noValue'] },
|
||||||
'created_at': {label: gettext('Created'), filters: ['mediumDate'] },
|
'created_at': {label: gettext('Created'), filters: ['mediumDate'] },
|
||||||
|
@ -19,7 +19,6 @@
|
|||||||
.factory('horizon.app.core.keypairs.service', keypairsService);
|
.factory('horizon.app.core.keypairs.service', keypairsService);
|
||||||
|
|
||||||
keypairsService.$inject = [
|
keypairsService.$inject = [
|
||||||
'$filter',
|
|
||||||
'horizon.app.core.detailRoute',
|
'horizon.app.core.detailRoute',
|
||||||
'horizon.app.core.openstack-service-api.nova'
|
'horizon.app.core.openstack-service-api.nova'
|
||||||
];
|
];
|
||||||
@ -34,7 +33,7 @@
|
|||||||
* but do not need to be restricted to such use. Each exposed function
|
* but do not need to be restricted to such use. Each exposed function
|
||||||
* is documented below.
|
* is documented below.
|
||||||
*/
|
*/
|
||||||
function keypairsService($filter, detailRoute, nova) {
|
function keypairsService(detailRoute, nova) {
|
||||||
return {
|
return {
|
||||||
getKeypairsPromise: getKeypairsPromise,
|
getKeypairsPromise: getKeypairsPromise,
|
||||||
getKeypairPromise: getKeypairPromise,
|
getKeypairPromise: getKeypairPromise,
|
||||||
@ -54,11 +53,12 @@
|
|||||||
return nova.getKeypairs(params).then(modifyResponse);
|
return nova.getKeypairs(params).then(modifyResponse);
|
||||||
|
|
||||||
function modifyResponse(response) {
|
function modifyResponse(response) {
|
||||||
return {data: {items: response.data.items.map(modifyItems)}};
|
return {data: {items: response.data.items.map(modifyItem)}};
|
||||||
|
|
||||||
function modifyItems(item) {
|
function modifyItem(item) {
|
||||||
item = item.keypair;
|
item = item.keypair;
|
||||||
item.trackBy = item.name;
|
item.id = item.name;
|
||||||
|
item.trackBy = item.name + item.fingerprint;
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -40,13 +40,13 @@
|
|||||||
deferredSession.resolve({});
|
deferredSession.resolve({});
|
||||||
deferred.resolve({
|
deferred.resolve({
|
||||||
data: {
|
data: {
|
||||||
items: [{keypair: {name: 'keypair1'}}]
|
items: [{keypair: {name: 'keypair1', fingerprint: 'fp'}}]
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
$timeout.flush();
|
$timeout.flush();
|
||||||
expect(nova.getKeypairs).toHaveBeenCalled();
|
expect(nova.getKeypairs).toHaveBeenCalled();
|
||||||
expect(result.$$state.value.data.items[0].name).toBe('keypair1');
|
expect(result.$$state.value.data.items[0].name).toBe('keypair1');
|
||||||
expect(result.$$state.value.data.items[0].trackBy).toBe('keypair1');
|
expect(result.$$state.value.data.items[0].trackBy).toBe('keypair1fp');
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -46,6 +46,7 @@
|
|||||||
getKeypairs: getKeypairs,
|
getKeypairs: getKeypairs,
|
||||||
createKeypair: createKeypair,
|
createKeypair: createKeypair,
|
||||||
getKeypair: getKeypair,
|
getKeypair: getKeypair,
|
||||||
|
deleteKeypair: deleteKeypair,
|
||||||
getAvailabilityZones: getAvailabilityZones,
|
getAvailabilityZones: getAvailabilityZones,
|
||||||
getLimits: getLimits,
|
getLimits: getLimits,
|
||||||
createServer: createServer,
|
createServer: createServer,
|
||||||
@ -175,6 +176,28 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name deleteKeypair
|
||||||
|
* @description
|
||||||
|
* Delete a single keypair by name.
|
||||||
|
*
|
||||||
|
* @param {String} name
|
||||||
|
* Keypair to delete
|
||||||
|
*
|
||||||
|
* @param {boolean} suppressError
|
||||||
|
* If passed in, this will not show the default error handling
|
||||||
|
* (horizon alert).
|
||||||
|
*
|
||||||
|
* @returns {Object} The result of the API call
|
||||||
|
*/
|
||||||
|
function deleteKeypair(name, suppressError) {
|
||||||
|
var promise = apiService.delete('/api/nova/keypairs/' + name);
|
||||||
|
return suppressError ? promise : promise.error(function() {
|
||||||
|
var msg = gettext('Unable to delete the keypair with name: %(name)s');
|
||||||
|
toastService.add('error', interpolate(msg, { name: name }, true));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Availability Zones
|
// Availability Zones
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -142,6 +142,19 @@
|
|||||||
{}
|
{}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"func": "deleteKeypair",
|
||||||
|
"method": "delete",
|
||||||
|
"path": "/api/nova/keypairs/19",
|
||||||
|
"error": "Unable to delete the keypair with name: 19",
|
||||||
|
"testInput": [19]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"func": "deleteKeypair",
|
||||||
|
"method": "delete",
|
||||||
|
"path": "/api/nova/keypairs/19",
|
||||||
|
"testInput": [19, true]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"func": "getKeypair",
|
"func": "getKeypair",
|
||||||
"method": "get",
|
"method": "get",
|
||||||
|
@ -224,6 +224,12 @@ class NovaRestTestCase(test.TestCase):
|
|||||||
response.json)
|
response.json)
|
||||||
nc.keypair_get.assert_called_once_with(request, "1")
|
nc.keypair_get.assert_called_once_with(request, "1")
|
||||||
|
|
||||||
|
@mock.patch.object(nova.api, 'nova')
|
||||||
|
def test_keypair_delete(self, nc):
|
||||||
|
request = self.mock_rest_request()
|
||||||
|
nova.Keypair().delete(request, "1")
|
||||||
|
nc.keypair_delete.assert_called_once_with(request, "1")
|
||||||
|
|
||||||
#
|
#
|
||||||
# Availability Zones
|
# Availability Zones
|
||||||
#
|
#
|
||||||
|
Loading…
x
Reference in New Issue
Block a user