Merge "Enable volumes metadata update"
This commit is contained in:
commit
1d14504de9
@ -355,6 +355,14 @@ def volume_update(request, volume_id, name, description):
|
|||||||
**vol_data)
|
**vol_data)
|
||||||
|
|
||||||
|
|
||||||
|
def volume_set_metadata(request, volume_id, metadata):
|
||||||
|
return cinderclient(request).volumes.set_metadata(volume_id, metadata)
|
||||||
|
|
||||||
|
|
||||||
|
def volume_delete_metadata(request, volume_id, keys):
|
||||||
|
return cinderclient(request).volumes.delete_metadata(volume_id, keys)
|
||||||
|
|
||||||
|
|
||||||
def volume_reset_state(request, volume_id, state):
|
def volume_reset_state(request, volume_id, state):
|
||||||
return cinderclient(request).volumes.reset_state(volume_id, state)
|
return cinderclient(request).volumes.reset_state(volume_id, state)
|
||||||
|
|
||||||
@ -445,6 +453,16 @@ def volume_snapshot_update(request, snapshot_id, name, description):
|
|||||||
**snapshot_data)
|
**snapshot_data)
|
||||||
|
|
||||||
|
|
||||||
|
def volume_snapshot_set_metadata(request, snapshot_id, metadata):
|
||||||
|
return cinderclient(request).volume_snapshots.set_metadata(
|
||||||
|
snapshot_id, metadata)
|
||||||
|
|
||||||
|
|
||||||
|
def volume_snapshot_delete_metadata(request, snapshot_id, keys):
|
||||||
|
return cinderclient(request).volume_snapshots.delete_metadata(
|
||||||
|
snapshot_id, keys)
|
||||||
|
|
||||||
|
|
||||||
def volume_snapshot_reset_state(request, snapshot_id, state):
|
def volume_snapshot_reset_state(request, snapshot_id, state):
|
||||||
return cinderclient(request).volume_snapshots.reset_state(
|
return cinderclient(request).volume_snapshots.reset_state(
|
||||||
snapshot_id, state)
|
snapshot_id, state)
|
||||||
@ -792,7 +810,7 @@ def volume_type_extra_set(request, type_id, metadata):
|
|||||||
|
|
||||||
def volume_type_extra_delete(request, type_id, keys):
|
def volume_type_extra_delete(request, type_id, keys):
|
||||||
vol_type = volume_type_get(request, type_id)
|
vol_type = volume_type_get(request, type_id)
|
||||||
return vol_type.unset_keys([keys])
|
return vol_type.unset_keys(keys)
|
||||||
|
|
||||||
|
|
||||||
def qos_spec_list(request):
|
def qos_spec_list(request):
|
||||||
|
@ -132,6 +132,34 @@ class VolumeTypes(generic.View):
|
|||||||
return {'items': [api.cinder.VolumeType(u).to_dict() for u in result]}
|
return {'items': [api.cinder.VolumeType(u).to_dict() for u in result]}
|
||||||
|
|
||||||
|
|
||||||
|
@urls.register
|
||||||
|
class VolumeMetadata(generic.View):
|
||||||
|
"""API for volume metadata"""
|
||||||
|
url_regex = r'cinder/volumes/(?P<volume_id>[^/]+)/metadata$'
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def get(self, request, volume_id):
|
||||||
|
"""Get a specific volume's metadata
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumes/1/metadata
|
||||||
|
"""
|
||||||
|
return api.cinder.volume_get(request,
|
||||||
|
volume_id).to_dict().get('metadata')
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def patch(self, request, volume_id):
|
||||||
|
"""Update metadata items for specific volume
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumes/1/metadata
|
||||||
|
"""
|
||||||
|
updated = request.DATA['updated']
|
||||||
|
removed = request.DATA['removed']
|
||||||
|
if updated:
|
||||||
|
api.cinder.volume_set_metadata(request, volume_id, updated)
|
||||||
|
if removed:
|
||||||
|
api.cinder.volume_delete_metadata(request, volume_id, removed)
|
||||||
|
|
||||||
|
|
||||||
@urls.register
|
@urls.register
|
||||||
class VolumeType(generic.View):
|
class VolumeType(generic.View):
|
||||||
"""API for getting a volume type.
|
"""API for getting a volume type.
|
||||||
@ -179,6 +207,74 @@ class VolumeSnapshots(generic.View):
|
|||||||
return {'items': [u.to_dict() for u in result]}
|
return {'items': [u.to_dict() for u in result]}
|
||||||
|
|
||||||
|
|
||||||
|
@urls.register
|
||||||
|
class VolumeSnapshotMetadata(generic.View):
|
||||||
|
"""API for getting snapshots metadata"""
|
||||||
|
url_regex = r'cinder/volumesnapshots/' \
|
||||||
|
r'(?P<volume_snapshot_id>[^/]+)/metadata$'
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def get(self, request, volume_snapshot_id):
|
||||||
|
"""Get a specific volumes snapshot metadata
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumesnapshots/1/metadata
|
||||||
|
"""
|
||||||
|
result = api.cinder.volume_snapshot_get(request,
|
||||||
|
volume_snapshot_id).\
|
||||||
|
to_dict().get('metadata')
|
||||||
|
return result
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def patch(self, request, volume_snapshot_id):
|
||||||
|
"""Update metadata for specific volume snapshot
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumesnapshots/1/metadata
|
||||||
|
"""
|
||||||
|
updated = request.DATA['updated']
|
||||||
|
removed = request.DATA['removed']
|
||||||
|
if updated:
|
||||||
|
api.cinder.volume_snapshot_set_metadata(request,
|
||||||
|
volume_snapshot_id,
|
||||||
|
updated)
|
||||||
|
if removed:
|
||||||
|
api.cinder.volume_snapshot_delete_metadata(request,
|
||||||
|
volume_snapshot_id,
|
||||||
|
removed)
|
||||||
|
|
||||||
|
|
||||||
|
@urls.register
|
||||||
|
class VolumeTypeMetadata(generic.View):
|
||||||
|
"""API for getting snapshots metadata"""
|
||||||
|
url_regex = r'cinder/volumetypes/(?P<volume_type_id>[^/]+)/metadata$'
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def get(self, request, volume_type_id):
|
||||||
|
"""Get a specific volume's metadata
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumetypes/1/metadata
|
||||||
|
"""
|
||||||
|
metadata = api.cinder.volume_type_extra_get(request, volume_type_id)
|
||||||
|
result = {x.key: x.value for x in metadata}
|
||||||
|
return result
|
||||||
|
|
||||||
|
@rest_utils.ajax()
|
||||||
|
def patch(self, request, volume_type_id):
|
||||||
|
"""Update metadata for specific volume
|
||||||
|
|
||||||
|
http://localhost/api/cinder/volumetypes/1/metadata
|
||||||
|
"""
|
||||||
|
updated = request.DATA['updated']
|
||||||
|
removed = request.DATA['removed']
|
||||||
|
if updated:
|
||||||
|
api.cinder.volume_type_extra_set(request,
|
||||||
|
volume_type_id,
|
||||||
|
updated)
|
||||||
|
if removed:
|
||||||
|
api.cinder.volume_type_extra_delete(request,
|
||||||
|
volume_type_id,
|
||||||
|
removed)
|
||||||
|
|
||||||
|
|
||||||
@urls.register
|
@urls.register
|
||||||
class Extensions(generic.View):
|
class Extensions(generic.View):
|
||||||
"""API for cinder extensions.
|
"""API for cinder extensions.
|
||||||
|
@ -72,7 +72,8 @@ class VolumeSnapshotsTable(volumes_tables.VolumesTableBase):
|
|||||||
table_actions = (snapshots_tables.VolumeSnapshotsFilterAction,
|
table_actions = (snapshots_tables.VolumeSnapshotsFilterAction,
|
||||||
snapshots_tables.DeleteVolumeSnapshot,)
|
snapshots_tables.DeleteVolumeSnapshot,)
|
||||||
row_actions = (snapshots_tables.DeleteVolumeSnapshot,
|
row_actions = (snapshots_tables.DeleteVolumeSnapshot,
|
||||||
UpdateVolumeSnapshotStatus,)
|
UpdateVolumeSnapshotStatus,
|
||||||
|
snapshots_tables.UpdateMetadata)
|
||||||
row_class = UpdateRow
|
row_class = UpdateRow
|
||||||
status_columns = ("status",)
|
status_columns = ("status",)
|
||||||
columns = ('tenant', 'host', 'name', 'description', 'size', 'status',
|
columns = ('tenant', 'host', 'name', 'description', 'size', 'status',
|
||||||
|
@ -19,7 +19,7 @@ from openstack_dashboard.test import helpers as test
|
|||||||
|
|
||||||
from openstack_dashboard.dashboards.admin.volumes.snapshots import forms
|
from openstack_dashboard.dashboards.admin.volumes.snapshots import forms
|
||||||
|
|
||||||
INDEX_URL = reverse('horizon:admin:volumes:index')
|
INDEX_URL = 'horizon:admin:volumes:index'
|
||||||
|
|
||||||
|
|
||||||
class VolumeSnapshotsViewTests(test.BaseAdminViewTests):
|
class VolumeSnapshotsViewTests(test.BaseAdminViewTests):
|
||||||
@ -79,7 +79,7 @@ class VolumeSnapshotsViewTests(test.BaseAdminViewTests):
|
|||||||
|
|
||||||
self.assertNoFormErrors(res)
|
self.assertNoFormErrors(res)
|
||||||
self.assertMessageCount(error=1)
|
self.assertMessageCount(error=1)
|
||||||
self.assertRedirectsNoFollow(res, INDEX_URL)
|
self.assertRedirectsNoFollow(res, reverse(INDEX_URL))
|
||||||
|
|
||||||
@test.create_stubs({cinder: ('volume_snapshot_get',
|
@test.create_stubs({cinder: ('volume_snapshot_get',
|
||||||
'volume_get')})
|
'volume_get')})
|
||||||
@ -101,7 +101,7 @@ class VolumeSnapshotsViewTests(test.BaseAdminViewTests):
|
|||||||
|
|
||||||
self.assertNoFormErrors(res)
|
self.assertNoFormErrors(res)
|
||||||
self.assertMessageCount(error=1)
|
self.assertMessageCount(error=1)
|
||||||
self.assertRedirectsNoFollow(res, INDEX_URL)
|
self.assertRedirectsNoFollow(res, reverse(INDEX_URL))
|
||||||
|
|
||||||
def test_get_snapshot_status_choices_without_current(self):
|
def test_get_snapshot_status_choices_without_current(self):
|
||||||
current_status = {'status': 'available'}
|
current_status = {'status': 'available'}
|
||||||
|
@ -195,6 +195,23 @@ class UpdateRow(tables.Row):
|
|||||||
return volume_type
|
return volume_type
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateMetadata(tables.LinkAction):
|
||||||
|
name = "update_metadata"
|
||||||
|
verbose_name = _("Update Metadata")
|
||||||
|
ajax = False
|
||||||
|
attrs = {"ng-controller": "MetadataModalHelperController as modal"}
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
kwargs['preempt'] = True
|
||||||
|
super(UpdateMetadata, self).__init__(**kwargs)
|
||||||
|
|
||||||
|
def get_link_url(self, datum):
|
||||||
|
obj_id = self.table.get_object_id(datum)
|
||||||
|
self.attrs['ng-click'] = (
|
||||||
|
"modal.openMetadataModal('volume_type', '%s', true)" % obj_id)
|
||||||
|
return "javascript:void(0);"
|
||||||
|
|
||||||
|
|
||||||
class VolumeTypesTable(tables.DataTable):
|
class VolumeTypesTable(tables.DataTable):
|
||||||
name = tables.WrappingColumn("name", verbose_name=_("Name"),
|
name = tables.WrappingColumn("name", verbose_name=_("Name"),
|
||||||
form_field=forms.CharField(max_length=64))
|
form_field=forms.CharField(max_length=64))
|
||||||
@ -233,7 +250,8 @@ class VolumeTypesTable(tables.DataTable):
|
|||||||
EditVolumeType,
|
EditVolumeType,
|
||||||
UpdateVolumeTypeEncryption,
|
UpdateVolumeTypeEncryption,
|
||||||
DeleteVolumeTypeEncryption,
|
DeleteVolumeTypeEncryption,
|
||||||
DeleteVolumeType,)
|
DeleteVolumeType,
|
||||||
|
UpdateMetadata)
|
||||||
row_class = UpdateRow
|
row_class = UpdateRow
|
||||||
|
|
||||||
|
|
||||||
|
@ -111,6 +111,7 @@ class VolumesTable(volumes_tables.VolumesTable):
|
|||||||
row_actions = (volumes_tables.DeleteVolume,
|
row_actions = (volumes_tables.DeleteVolume,
|
||||||
UpdateVolumeStatusAction,
|
UpdateVolumeStatusAction,
|
||||||
UnmanageVolumeAction,
|
UnmanageVolumeAction,
|
||||||
MigrateVolume)
|
MigrateVolume,
|
||||||
|
volumes_tables.UpdateMetadata)
|
||||||
columns = ('tenant', 'host', 'name', 'size', 'status', 'volume_type',
|
columns = ('tenant', 'host', 'name', 'size', 'status', 'volume_type',
|
||||||
'attachments', 'bootable', 'encryption',)
|
'attachments', 'bootable', 'encryption',)
|
||||||
|
@ -132,6 +132,24 @@ class CreateVolumeFromSnapshot(tables.LinkAction):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateMetadata(tables.LinkAction):
|
||||||
|
name = "update_metadata"
|
||||||
|
verbose_name = _("Update Metadata")
|
||||||
|
|
||||||
|
ajax = False
|
||||||
|
attrs = {"ng-controller": "MetadataModalHelperController as modal"}
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
kwargs['preempt'] = True
|
||||||
|
super(UpdateMetadata, self).__init__(**kwargs)
|
||||||
|
|
||||||
|
def get_link_url(self, datum):
|
||||||
|
obj_id = self.table.get_object_id(datum)
|
||||||
|
self.attrs['ng-click'] = (
|
||||||
|
"modal.openMetadataModal('volume_snapshot', '%s', true)" % obj_id)
|
||||||
|
return "javascript:void(0);"
|
||||||
|
|
||||||
|
|
||||||
class UpdateRow(tables.Row):
|
class UpdateRow(tables.Row):
|
||||||
ajax = True
|
ajax = True
|
||||||
|
|
||||||
@ -191,7 +209,8 @@ class VolumeSnapshotsTable(volume_tables.VolumesTableBase):
|
|||||||
launch_actions = (LaunchSnapshotNG,) + launch_actions
|
launch_actions = (LaunchSnapshotNG,) + launch_actions
|
||||||
|
|
||||||
row_actions = ((CreateVolumeFromSnapshot,) + launch_actions +
|
row_actions = ((CreateVolumeFromSnapshot,) + launch_actions +
|
||||||
(EditVolumeSnapshot, DeleteVolumeSnapshot))
|
(EditVolumeSnapshot, DeleteVolumeSnapshot,
|
||||||
|
UpdateMetadata))
|
||||||
row_class = UpdateRow
|
row_class = UpdateRow
|
||||||
status_columns = ("status",)
|
status_columns = ("status",)
|
||||||
permissions = [(
|
permissions = [(
|
||||||
|
@ -469,6 +469,23 @@ class VolumesFilterAction(tables.FilterAction):
|
|||||||
if q in volume.name.lower()]
|
if q in volume.name.lower()]
|
||||||
|
|
||||||
|
|
||||||
|
class UpdateMetadata(tables.LinkAction):
|
||||||
|
name = "update_metadata"
|
||||||
|
verbose_name = _("Update Metadata")
|
||||||
|
ajax = False
|
||||||
|
attrs = {"ng-controller": "MetadataModalHelperController as modal"}
|
||||||
|
|
||||||
|
def __init__(self, **kwargs):
|
||||||
|
kwargs['preempt'] = True
|
||||||
|
super(UpdateMetadata, self).__init__(**kwargs)
|
||||||
|
|
||||||
|
def get_link_url(self, datum):
|
||||||
|
obj_id = self.table.get_object_id(datum)
|
||||||
|
self.attrs['ng-click'] = (
|
||||||
|
"modal.openMetadataModal('volume', '%s', true)" % obj_id)
|
||||||
|
return "javascript:void(0);"
|
||||||
|
|
||||||
|
|
||||||
class VolumesTable(VolumesTableBase):
|
class VolumesTable(VolumesTableBase):
|
||||||
name = tables.WrappingColumn("name",
|
name = tables.WrappingColumn("name",
|
||||||
verbose_name=_("Name"),
|
verbose_name=_("Name"),
|
||||||
@ -504,7 +521,7 @@ class VolumesTable(VolumesTableBase):
|
|||||||
launch_actions +
|
launch_actions +
|
||||||
(EditAttachments, CreateSnapshot, CreateBackup,
|
(EditAttachments, CreateSnapshot, CreateBackup,
|
||||||
RetypeVolume, UploadToImage, CreateTransfer,
|
RetypeVolume, UploadToImage, CreateTransfer,
|
||||||
DeleteTransfer, DeleteVolume))
|
DeleteTransfer, DeleteVolume, UpdateMetadata))
|
||||||
|
|
||||||
|
|
||||||
class DetachVolume(tables.BatchAction):
|
class DetachVolume(tables.BatchAction):
|
||||||
|
@ -22,7 +22,8 @@
|
|||||||
|
|
||||||
metadataService.$inject = [
|
metadataService.$inject = [
|
||||||
'horizon.app.core.openstack-service-api.nova',
|
'horizon.app.core.openstack-service-api.nova',
|
||||||
'horizon.app.core.openstack-service-api.glance'
|
'horizon.app.core.openstack-service-api.glance',
|
||||||
|
'horizon.app.core.openstack-service-api.cinder'
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -32,7 +33,7 @@
|
|||||||
*
|
*
|
||||||
* Unified acquisition and modification of metadata.
|
* Unified acquisition and modification of metadata.
|
||||||
*/
|
*/
|
||||||
function metadataService(nova, glance) {
|
function metadataService(nova, glance, cinder) {
|
||||||
var service = {
|
var service = {
|
||||||
getMetadata: getMetadata,
|
getMetadata: getMetadata,
|
||||||
editMetadata: editMetadata,
|
editMetadata: editMetadata,
|
||||||
@ -52,7 +53,10 @@
|
|||||||
aggregate: nova.getAggregateExtraSpecs,
|
aggregate: nova.getAggregateExtraSpecs,
|
||||||
flavor: nova.getFlavorExtraSpecs,
|
flavor: nova.getFlavorExtraSpecs,
|
||||||
image: glance.getImageProps,
|
image: glance.getImageProps,
|
||||||
instance: nova.getInstanceMetadata
|
instance: nova.getInstanceMetadata,
|
||||||
|
volume: cinder.getVolumeMetadata,
|
||||||
|
volume_snapshot: cinder.getVolumeSnapshotMetadata,
|
||||||
|
volume_type: cinder.getVolumeTypeMetadata
|
||||||
}[resource](id);
|
}[resource](id);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,7 +73,10 @@
|
|||||||
aggregate: nova.editAggregateExtraSpecs,
|
aggregate: nova.editAggregateExtraSpecs,
|
||||||
flavor: nova.editFlavorExtraSpecs,
|
flavor: nova.editFlavorExtraSpecs,
|
||||||
image: glance.editImageProps,
|
image: glance.editImageProps,
|
||||||
instance: nova.editInstanceMetadata
|
instance: nova.editInstanceMetadata,
|
||||||
|
volume: cinder.editVolumeMetadata,
|
||||||
|
volume_snapshot: cinder.editVolumeSnapshotMetadata,
|
||||||
|
volume_type: cinder.editVolumeTypeMetadata
|
||||||
}[resource](id, updated, removed);
|
}[resource](id, updated, removed);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -86,7 +93,10 @@
|
|||||||
aggregate: 'OS::Nova::Aggregate',
|
aggregate: 'OS::Nova::Aggregate',
|
||||||
flavor: 'OS::Nova::Flavor',
|
flavor: 'OS::Nova::Flavor',
|
||||||
image: 'OS::Glance::Image',
|
image: 'OS::Glance::Image',
|
||||||
instance: 'OS::Nova::Server'
|
instance: 'OS::Nova::Server',
|
||||||
|
volume: 'OS::Cinder::Volume',
|
||||||
|
volume_snapshot: 'OS::Cinder::Snapshot',
|
||||||
|
volume_type: 'OS:Cinder::VolumeType'
|
||||||
}[resource]
|
}[resource]
|
||||||
};
|
};
|
||||||
if (propertiesTarget) {
|
if (propertiesTarget) {
|
||||||
|
@ -31,10 +31,17 @@
|
|||||||
editImageProps: function() {},
|
editImageProps: function() {},
|
||||||
getNamespaces: function() {}};
|
getNamespaces: function() {}};
|
||||||
|
|
||||||
|
var cinder = {getVolumeMetadata:function() {},
|
||||||
|
getVolumeSnapshotMetadata:function() {},
|
||||||
|
getVolumeTypeMetadata:function() {},
|
||||||
|
editVolumeMetadata: function() {},
|
||||||
|
editVolumeSnapshotMetadata: function() {}};
|
||||||
|
|
||||||
beforeEach(function() {
|
beforeEach(function() {
|
||||||
module(function($provide) {
|
module(function($provide) {
|
||||||
$provide.value('horizon.app.core.openstack-service-api.nova', nova);
|
$provide.value('horizon.app.core.openstack-service-api.nova', nova);
|
||||||
$provide.value('horizon.app.core.openstack-service-api.glance', glance);
|
$provide.value('horizon.app.core.openstack-service-api.glance', glance);
|
||||||
|
$provide.value('horizon.app.core.openstack-service-api.cinder', cinder);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -97,6 +104,18 @@
|
|||||||
expect(glance.editImageProps).toHaveBeenCalledWith('1', 'updated', ['removed']);
|
expect(glance.editImageProps).toHaveBeenCalledWith('1', 'updated', ['removed']);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should edit volume metadata', function() {
|
||||||
|
spyOn(cinder, 'editVolumeMetadata');
|
||||||
|
metadataService.editMetadata('volume', '1', 'updated', ['removed']);
|
||||||
|
expect(cinder.editVolumeMetadata).toHaveBeenCalledWith('1', 'updated', ['removed']);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should edit volume snapshot metadata', function() {
|
||||||
|
spyOn(cinder, 'editVolumeSnapshotMetadata');
|
||||||
|
metadataService.editMetadata('volume_snapshot', '1', 'updated', ['removed']);
|
||||||
|
expect(cinder.editVolumeSnapshotMetadata).toHaveBeenCalledWith('1', 'updated', ['removed']);
|
||||||
|
});
|
||||||
|
|
||||||
it('should get image namespace', function() {
|
it('should get image namespace', function() {
|
||||||
spyOn(glance, 'getNamespaces');
|
spyOn(glance, 'getNamespaces');
|
||||||
metadataService.getNamespaces('image');
|
metadataService.getNamespaces('image');
|
||||||
@ -111,6 +130,13 @@
|
|||||||
expect(actual).toBe(expected);
|
expect(actual).toBe(expected);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should get volume metadata', function() {
|
||||||
|
var expected = 'volume metadata';
|
||||||
|
spyOn(cinder, 'getVolumeMetadata').and.returnValue(expected);
|
||||||
|
var actual = metadataService.getMetadata('volume', '1');
|
||||||
|
expect(actual).toBe(expected);
|
||||||
|
});
|
||||||
|
|
||||||
it('should edit instance metadata', function() {
|
it('should edit instance metadata', function() {
|
||||||
spyOn(nova, 'editInstanceMetadata');
|
spyOn(nova, 'editInstanceMetadata');
|
||||||
metadataService.editMetadata('instance', '1', 'updated', ['removed']);
|
metadataService.editMetadata('instance', '1', 'updated', ['removed']);
|
||||||
|
@ -7,6 +7,9 @@
|
|||||||
<span translate ng-if="modal.resourceType==='flavor'">Update Flavor Metadata</span>
|
<span translate ng-if="modal.resourceType==='flavor'">Update Flavor Metadata</span>
|
||||||
<span translate ng-if="modal.resourceType==='image'">Update Image Metadata</span>
|
<span translate ng-if="modal.resourceType==='image'">Update Image Metadata</span>
|
||||||
<span translate ng-if="modal.resourceType==='instance'">Update Instance Metadata</span>
|
<span translate ng-if="modal.resourceType==='instance'">Update Instance Metadata</span>
|
||||||
|
<span translate ng-if="modal.resourceType==='volume'">Update Volume Metadata</span>
|
||||||
|
<span translate ng-if="modal.resourceType==='volume_snapshot'">Update Volume Snapshot Metadata</span>
|
||||||
|
<span translate ng-if="modal.resourceType==='volume_type'">Update Volume Type Metadata</span>
|
||||||
</h3>
|
</h3>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
|
@ -38,6 +38,9 @@
|
|||||||
getVolumes: getVolumes,
|
getVolumes: getVolumes,
|
||||||
getVolume: getVolume,
|
getVolume: getVolume,
|
||||||
getVolumeTypes: getVolumeTypes,
|
getVolumeTypes: getVolumeTypes,
|
||||||
|
getVolumeMetadata: getVolumeMetadata,
|
||||||
|
getVolumeSnapshotMetadata: getVolumeSnapshotMetadata,
|
||||||
|
getVolumeTypeMetadata: getVolumeTypeMetadata,
|
||||||
getVolumeType: getVolumeType,
|
getVolumeType: getVolumeType,
|
||||||
getDefaultVolumeType: getDefaultVolumeType,
|
getDefaultVolumeType: getDefaultVolumeType,
|
||||||
getVolumeSnapshots: getVolumeSnapshots,
|
getVolumeSnapshots: getVolumeSnapshots,
|
||||||
@ -48,7 +51,10 @@
|
|||||||
getServices: getServices,
|
getServices: getServices,
|
||||||
getDefaultQuotaSets: getDefaultQuotaSets,
|
getDefaultQuotaSets: getDefaultQuotaSets,
|
||||||
setDefaultQuotaSets: setDefaultQuotaSets,
|
setDefaultQuotaSets: setDefaultQuotaSets,
|
||||||
updateProjectQuota: updateProjectQuota
|
updateProjectQuota: updateProjectQuota,
|
||||||
|
editVolumeMetadata: editVolumeMetadata,
|
||||||
|
editVolumeSnapshotMetadata: editVolumeSnapshotMetadata,
|
||||||
|
editVolumeTypeMetadata:editVolumeTypeMetadata
|
||||||
};
|
};
|
||||||
|
|
||||||
return service;
|
return service;
|
||||||
@ -146,6 +152,63 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getVolumeMetadata(id) {
|
||||||
|
return apiService.get('/api/cinder/volumes/' + id + '/metadata')
|
||||||
|
.error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to retrieve the volume metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getVolumeSnapshotMetadata(id) {
|
||||||
|
return apiService.get('/api/cinder/volumesnapshots/' + id + '/metadata')
|
||||||
|
.error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to retrieve the snapshot metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getVolumeTypeMetadata(id) {
|
||||||
|
return apiService.get('/api/cinder/volumetypes/' + id + '/metadata')
|
||||||
|
.error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to retrieve the volume type metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function editVolumeMetadata(id, updated, removed) {
|
||||||
|
return apiService.patch(
|
||||||
|
'/api/cinder/volumes/' + id + '/metadata',
|
||||||
|
{
|
||||||
|
updated: updated,
|
||||||
|
removed: removed
|
||||||
|
}
|
||||||
|
).error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to edit volume metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function editVolumeSnapshotMetadata(id, updated, removed) {
|
||||||
|
return apiService.patch(
|
||||||
|
'/api/cinder/volumesnapshots/' + id + '/metadata',
|
||||||
|
{
|
||||||
|
updated: updated,
|
||||||
|
removed: removed
|
||||||
|
}
|
||||||
|
).error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to edit snapshot metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function editVolumeTypeMetadata(id, updated, removed) {
|
||||||
|
return apiService.patch(
|
||||||
|
'/api/cinder/volumetypes/' + id + '/metadata',
|
||||||
|
{
|
||||||
|
updated: updated,
|
||||||
|
removed: removed
|
||||||
|
}
|
||||||
|
).error(function () {
|
||||||
|
toastService.add('error', gettext('Unable to edit volume type metadata.'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name getVolumeType
|
* @name getVolumeType
|
||||||
* @description
|
* @description
|
||||||
|
@ -62,6 +62,13 @@
|
|||||||
error: 'Unable to retrieve the volume.',
|
error: 'Unable to retrieve the volume.',
|
||||||
testInput: [1]
|
testInput: [1]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
func: 'getVolumeMetadata',
|
||||||
|
method: 'get',
|
||||||
|
path: '/api/cinder/volumes/1/metadata',
|
||||||
|
error: 'Unable to retrieve the volume metadata.',
|
||||||
|
testInput: [1]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
func: 'getVolumeTypes',
|
func: 'getVolumeTypes',
|
||||||
method: 'get',
|
method: 'get',
|
||||||
@ -76,6 +83,13 @@
|
|||||||
error: 'Unable to retrieve the volume type.',
|
error: 'Unable to retrieve the volume type.',
|
||||||
testInput: [1]
|
testInput: [1]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
func: 'getVolumeTypeMetadata',
|
||||||
|
method: 'get',
|
||||||
|
path: '/api/cinder/volumetypes/1/metadata',
|
||||||
|
error: 'Unable to retrieve the volume type metadata.',
|
||||||
|
testInput: [1]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
func: 'getDefaultVolumeType',
|
func: 'getDefaultVolumeType',
|
||||||
method: 'get',
|
method: 'get',
|
||||||
@ -112,6 +126,13 @@
|
|||||||
error: 'Unable to retrieve the volume snapshots.',
|
error: 'Unable to retrieve the volume snapshots.',
|
||||||
testInput: [ 'config' ]
|
testInput: [ 'config' ]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
func: 'getVolumeSnapshotMetadata',
|
||||||
|
method: 'get',
|
||||||
|
path: '/api/cinder/volumesnapshots/1/metadata',
|
||||||
|
error: 'Unable to retrieve the snapshot metadata.',
|
||||||
|
testInput: [1]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
func: 'createVolume',
|
func: 'createVolume',
|
||||||
method: 'post',
|
method: 'post',
|
||||||
@ -165,6 +186,42 @@
|
|||||||
data: {'volumes': 42},
|
data: {'volumes': 42},
|
||||||
error: 'Unable to update project quota data.',
|
error: 'Unable to update project quota data.',
|
||||||
testInput: [{'volumes': 42}, 42]
|
testInput: [{'volumes': 42}, 42]
|
||||||
|
},
|
||||||
|
{ func: 'editVolumeMetadata',
|
||||||
|
method: 'patch',
|
||||||
|
path: '/api/cinder/volumes/42/metadata',
|
||||||
|
data: {
|
||||||
|
"updated": {a: '1', b: '2'},
|
||||||
|
"removed": ['c', 'd']
|
||||||
|
},
|
||||||
|
error: "Unable to edit volume metadata.",
|
||||||
|
testInput: [
|
||||||
|
42, {a: '1', b: '2'}, ['c', 'd']
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{ func: 'editVolumeSnapshotMetadata',
|
||||||
|
method: 'patch',
|
||||||
|
path: '/api/cinder/volumesnapshots/42/metadata',
|
||||||
|
data: {
|
||||||
|
"updated": {a: '1', b: '2'},
|
||||||
|
"removed": ['c', 'd']
|
||||||
|
},
|
||||||
|
error: "Unable to edit snapshot metadata.",
|
||||||
|
testInput: [
|
||||||
|
42, {a: '1', b: '2'}, ['c', 'd']
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{ func: 'editVolumeTypeMetadata',
|
||||||
|
method: 'patch',
|
||||||
|
path: '/api/cinder/volumetypes/42/metadata',
|
||||||
|
data: {
|
||||||
|
"updated": {a: '1', b: '2'},
|
||||||
|
"removed": ['c', 'd']
|
||||||
|
},
|
||||||
|
error: "Unable to edit volume type metadata.",
|
||||||
|
testInput: [
|
||||||
|
42, {a: '1', b: '2'}, ['c', 'd']
|
||||||
|
]
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
@ -18,6 +18,7 @@ from django.conf import settings
|
|||||||
|
|
||||||
from openstack_dashboard import api
|
from openstack_dashboard import api
|
||||||
from openstack_dashboard.api.base import Quota
|
from openstack_dashboard.api.base import Quota
|
||||||
|
from openstack_dashboard.api.cinder import VolTypeExtraSpec
|
||||||
from openstack_dashboard.api.rest import cinder
|
from openstack_dashboard.api.rest import cinder
|
||||||
from openstack_dashboard.test import helpers as test
|
from openstack_dashboard.test import helpers as test
|
||||||
|
|
||||||
@ -99,6 +100,33 @@ class CinderRestTestCase(test.TestCase):
|
|||||||
self.assertStatusCode(response, 201)
|
self.assertStatusCode(response, 201)
|
||||||
self.assertEqual(response.content.decode("utf-8"), mock_post_response)
|
self.assertEqual(response.content.decode("utf-8"), mock_post_response)
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_get_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(**{'GET': {}})
|
||||||
|
cc.volume_get.return_value = mock.Mock(
|
||||||
|
**{'to_dict.return_value': {'id': 'one',
|
||||||
|
'metadata': {'foo': 'bar'}}})
|
||||||
|
response = cinder.VolumeMetadata().get(request, '1')
|
||||||
|
self.assertStatusCode(response, 200)
|
||||||
|
self.assertEqual(response.json, {'foo': 'bar'})
|
||||||
|
cc.volume_get.assert_called_once_with(request, '1')
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_update_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(
|
||||||
|
body='{"updated": {"a": "1", "b": "2"}, '
|
||||||
|
'"removed": ["c", "d"]}'
|
||||||
|
)
|
||||||
|
response = cinder.VolumeMetadata().patch(request, '1')
|
||||||
|
self.assertStatusCode(response, 204)
|
||||||
|
self.assertEqual(b'', response.content)
|
||||||
|
cc.volume_set_metadata.assert_called_once_with(
|
||||||
|
request, '1', {'a': '1', 'b': '2'}
|
||||||
|
)
|
||||||
|
cc.volume_delete_metadata.assert_called_once_with(
|
||||||
|
request, '1', ['c', 'd']
|
||||||
|
)
|
||||||
|
|
||||||
#
|
#
|
||||||
# Volume Types
|
# Volume Types
|
||||||
#
|
#
|
||||||
@ -138,6 +166,35 @@ class CinderRestTestCase(test.TestCase):
|
|||||||
cc.volume_type_default.assert_called_once_with(request)
|
cc.volume_type_default.assert_called_once_with(request)
|
||||||
cc.VolumeType.assert_called_once_with({'name': 'one'})
|
cc.VolumeType.assert_called_once_with({'name': 'one'})
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_type_get_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(**{'GET': {}})
|
||||||
|
cc.volume_type_extra_get = mock.Mock()
|
||||||
|
|
||||||
|
cc.volume_type_extra_get.return_value = \
|
||||||
|
[VolTypeExtraSpec(1, 'foo', 'bar')]
|
||||||
|
# cc.volume_type_extra_get.side_effect = [{'foo': 'bar'}]
|
||||||
|
response = cinder.VolumeTypeMetadata().get(request, '1')
|
||||||
|
self.assertStatusCode(response, 200)
|
||||||
|
self.assertEqual(response.json, {'foo': 'bar'})
|
||||||
|
cc.volume_type_extra_get.assert_called_once_with(request, '1')
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_type_update_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(
|
||||||
|
body='{"updated": {"a": "1", "b": "2"}, '
|
||||||
|
'"removed": ["c", "d"]}'
|
||||||
|
)
|
||||||
|
response = cinder.VolumeTypeMetadata().patch(request, '1')
|
||||||
|
self.assertStatusCode(response, 204)
|
||||||
|
self.assertEqual(b'', response.content)
|
||||||
|
cc.volume_type_extra_set.assert_called_once_with(
|
||||||
|
request, '1', {'a': '1', 'b': '2'}
|
||||||
|
)
|
||||||
|
cc.volume_type_extra_delete.assert_called_once_with(
|
||||||
|
request, '1', ['c', 'd']
|
||||||
|
)
|
||||||
|
|
||||||
#
|
#
|
||||||
# Volume Snapshots
|
# Volume Snapshots
|
||||||
#
|
#
|
||||||
@ -170,6 +227,33 @@ class CinderRestTestCase(test.TestCase):
|
|||||||
cc.volume_snapshot_list.assert_called_once_with(request,
|
cc.volume_snapshot_list.assert_called_once_with(request,
|
||||||
search_opts=filters)
|
search_opts=filters)
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_snapshot_get_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(**{'GET': {}})
|
||||||
|
cc.volume_snapshot_get.return_value = mock.Mock(
|
||||||
|
**{'to_dict.return_value': {'id': 'one',
|
||||||
|
'metadata': {'foo': 'bar'}}})
|
||||||
|
response = cinder.VolumeSnapshotMetadata().get(request, '1')
|
||||||
|
self.assertStatusCode(response, 200)
|
||||||
|
self.assertEqual(response.json, {'foo': 'bar'})
|
||||||
|
cc.volume_snapshot_get.assert_called_once_with(request, '1')
|
||||||
|
|
||||||
|
@mock.patch.object(cinder.api, 'cinder')
|
||||||
|
def test_volume_snapshot_update_metadata(self, cc):
|
||||||
|
request = self.mock_rest_request(
|
||||||
|
body='{"updated": {"a": "1", "b": "2"}, '
|
||||||
|
'"removed": ["c", "d"]}'
|
||||||
|
)
|
||||||
|
response = cinder.VolumeSnapshotMetadata().patch(request, '1')
|
||||||
|
self.assertStatusCode(response, 204)
|
||||||
|
self.assertEqual(b'', response.content)
|
||||||
|
cc.volume_snapshot_set_metadata.assert_called_once_with(
|
||||||
|
request, '1', {'a': '1', 'b': '2'}
|
||||||
|
)
|
||||||
|
cc.volume_snapshot_delete_metadata.assert_called_once_with(
|
||||||
|
request, '1', ['c', 'd']
|
||||||
|
)
|
||||||
|
|
||||||
#
|
#
|
||||||
# Extensions
|
# Extensions
|
||||||
#
|
#
|
||||||
|
@ -286,8 +286,8 @@ class GlanceApiTests(test.APITestCase):
|
|||||||
def test_metadefs_namespace_list_with_properties_target(self):
|
def test_metadefs_namespace_list_with_properties_target(self):
|
||||||
metadata_defs = self.metadata_defs.list()
|
metadata_defs = self.metadata_defs.list()
|
||||||
limit = getattr(settings, 'API_RESULT_LIMIT', 1000)
|
limit = getattr(settings, 'API_RESULT_LIMIT', 1000)
|
||||||
filters = {'resource_types': ['mock name'],
|
filters = {'resource_types': ['OS::Cinder::Volume'],
|
||||||
'properties_target': 'mock properties target'}
|
'properties_target': 'user'}
|
||||||
|
|
||||||
glanceclient = self.stub_glanceclient()
|
glanceclient = self.stub_glanceclient()
|
||||||
glanceclient.metadefs_namespace = self.mox.CreateMockAnything()
|
glanceclient.metadefs_namespace = self.mox.CreateMockAnything()
|
||||||
@ -295,7 +295,7 @@ class GlanceApiTests(test.APITestCase):
|
|||||||
limit=limit,
|
limit=limit,
|
||||||
filters=filters,
|
filters=filters,
|
||||||
sort_dir='asc',
|
sort_dir='asc',
|
||||||
sort_key='namespace',) \
|
sort_key='namespace', ) \
|
||||||
.AndReturn(metadata_defs)
|
.AndReturn(metadata_defs)
|
||||||
|
|
||||||
self.mox.ReplayAll()
|
self.mox.ReplayAll()
|
||||||
|
@ -404,8 +404,9 @@ def data(TEST):
|
|||||||
{
|
{
|
||||||
'created_at': '2014-08-21T08:39:43Z',
|
'created_at': '2014-08-21T08:39:43Z',
|
||||||
'prefix': 'mock',
|
'prefix': 'mock',
|
||||||
'name': 'mock name',
|
'name': 'OS::Cinder::Volume',
|
||||||
'properties_target': 'mock properties target'
|
'properties_target': 'user'
|
||||||
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
'visibility': 'public',
|
'visibility': 'public',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user