Volume manage/unmanage support to ZFSSA drivers

This change for ZFSSA iSCSI and NFS drivers will allow cinder to
manage existing volumes on a ZFSSA backend and also unmanage them.

Change-Id: I2f280a2dfdb9cc6ce739b340eefa562f98d4038f
Implements: blueprint oracle-zfssa-volume-manage-unmanage
DocImpact
This commit is contained in:
Kedar Vidvans 2016-01-22 11:44:06 -05:00 committed by kedar-vidvans
parent c4770f7466
commit a75a528fd2
5 changed files with 529 additions and 12 deletions

View File

@ -1,4 +1,4 @@
# Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -169,6 +169,7 @@ class TestZFSSAISCSIDriver(test.TestCase):
self.configuration.zfssa_cache_project = zfssa_cache_dir self.configuration.zfssa_cache_project = zfssa_cache_dir
self.configuration.safe_get = self.fake_safe_get self.configuration.safe_get = self.fake_safe_get
self.configuration.zfssa_replication_ip = '1.1.1.1' self.configuration.zfssa_replication_ip = '1.1.1.1'
self.configuration.zfssa_manage_policy = 'loose'
def _util_migrate_volume_exceptions(self): def _util_migrate_volume_exceptions(self):
self.drv.zfssa.get_lun.return_value = ( self.drv.zfssa.get_lun.return_value = (
@ -739,6 +740,120 @@ class TestZFSSAISCSIDriver(test.TestCase):
lcfg.zfssa_cache_project, lcfg.zfssa_cache_project,
volname) volname)
@mock.patch.object(iscsi.ZFSSAISCSIDriver, '_get_existing_vol')
@mock.patch.object(iscsi.ZFSSAISCSIDriver, '_verify_volume_to_manage')
def test_volume_manage(self, _get_existing_vol, _verify_volume_to_manage):
lcfg = self.configuration
lcfg.zfssa_manage_policy = 'loose'
test_vol = self.test_vol
self.drv._get_existing_vol.return_value = test_vol
self.drv._verify_volume_to_manage.return_value = None
self.drv.zfssa.set_lun_props.return_value = True
self.assertIsNone(self.drv.manage_existing({'name': 'volume-123'},
{'source-name':
'volume-567'}))
self.drv._get_existing_vol.assert_called_once_with({'source-name':
'volume-567'})
self.drv._verify_volume_to_manage.assert_called_once_with(test_vol)
self.drv.zfssa.set_lun_props.assert_called_once_with(
lcfg.zfssa_pool,
lcfg.zfssa_project,
test_vol['name'],
name='volume-123',
schema={"custom:cinder_managed": True})
# Case when zfssa_manage_policy is 'loose' and 'cinder_managed' is
# set to true.
test_vol.update({'cinder_managed': False})
self.assertIsNone(self.drv.manage_existing({'name': 'volume-123'},
{'source-name':
'volume-567'}))
# Another case is when the zfssa_manage_policy is set to 'strict'
lcfg.zfssa_manage_policy = 'strict'
test_vol.update({'cinder_managed': False})
self.assertIsNone(self.drv.manage_existing({'name': 'volume-123'},
{'source-name':
'volume-567'}))
def test_volume_manage_negative(self):
lcfg = self.configuration
lcfg.zfssa_manage_policy = 'strict'
test_vol = self.test_vol
if 'cinder_managed' in test_vol:
del test_vol['cinder_managed']
self.drv.zfssa.get_lun.return_value = test_vol
self.assertRaises(exception.InvalidInput,
self.drv.manage_existing, {'name': 'cindervol'},
{'source-name': 'volume-567'})
test_vol.update({'cinder_managed': True})
self.drv.zfssa.get_lun.return_value = test_vol
self.assertRaises(exception.ManageExistingAlreadyManaged,
self.drv.manage_existing, {'name': 'cindervol'},
{'source-name': 'volume-567'})
test_vol.update({'cinder_managed': False})
self.drv.zfssa.get_lun.return_value = test_vol
self.assertRaises(exception.ManageExistingInvalidReference,
self.drv.manage_existing, {'name': 'cindervol'},
{'source-id': 'volume-567'})
lcfg.zfssa_manage_policy = 'loose'
self.assertRaises(exception.ManageExistingInvalidReference,
self.drv.manage_existing, {'name': 'cindervol'},
{'source-id': 'volume-567'})
@mock.patch.object(iscsi.ZFSSAISCSIDriver, '_verify_volume_to_manage')
def test_volume_manage_negative_api_exception(self,
_verify_volume_to_manage):
lcfg = self.configuration
lcfg.zfssa_manage_policy = 'loose'
self.drv.zfssa.get_lun.return_value = self.test_vol
self.drv._verify_volume_to_manage.return_value = None
self.drv.zfssa.set_lun_props.side_effect = \
exception.VolumeBackendAPIException(data='fake exception')
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.manage_existing, {'name': 'volume-123'},
{'source-name': 'volume-567'})
def test_volume_unmanage(self):
lcfg = self.configuration
self.drv.zfssa.set_lun_props.return_value = True
self.assertIsNone(self.drv.unmanage({'name': 'volume-123'}))
self.drv.zfssa.set_lun_props.assert_called_once_with(
lcfg.zfssa_pool,
lcfg.zfssa_project,
'volume-123',
name='unmanaged-volume-123',
schema={"custom:cinder_managed": False})
def test_volume_unmanage_negative(self):
self.drv.zfssa.set_lun_props.side_effect = \
exception.VolumeBackendAPIException(data='fake exception')
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.unmanage, {'name': 'volume-123'})
@mock.patch.object(iscsi.ZFSSAISCSIDriver, '_get_existing_vol')
def test_manage_existing_get_size(self, _get_existing_vol):
test_vol = self.test_vol
test_vol['size'] = 3 * units.Gi
self.drv._get_existing_vol.return_value = test_vol
self.assertEqual(3, self.drv.manage_existing_get_size(
{'name': 'volume-123'},
{'source-name': 'volume-567'}))
@mock.patch.object(iscsi.ZFSSAISCSIDriver, '_get_existing_vol')
def test_manage_existing_get_size_negative(self, _get_existing_vol):
self.drv._get_existing_vol.side_effect = \
exception.VolumeNotFound(volume_id='123')
self.assertRaises(exception.VolumeNotFound,
self.drv.manage_existing_get_size,
{'name': 'volume-123'},
{'source-name': 'volume-567'})
class TestZFSSANFSDriver(test.TestCase): class TestZFSSANFSDriver(test.TestCase):
@ -791,6 +906,7 @@ class TestZFSSANFSDriver(test.TestCase):
self.configuration.zfssa_enable_local_cache = True self.configuration.zfssa_enable_local_cache = True
self.configuration.zfssa_cache_directory = zfssa_cache_dir self.configuration.zfssa_cache_directory = zfssa_cache_dir
self.configuration.nfs_sparsed_volumes = 'true' self.configuration.nfs_sparsed_volumes = 'true'
self.configuration.zfssa_manage_policy = 'strict'
def test_migrate_volume(self): def test_migrate_volume(self):
self.drv.zfssa.get_asn.return_value = ( self.drv.zfssa.get_asn.return_value = (
@ -1082,6 +1198,152 @@ class TestZFSSANFSDriver(test.TestCase):
self.drv.zfssa.delete_file.assert_called_once_with( self.drv.zfssa.delete_file.assert_called_once_with(
img_props_nfs['name']) img_props_nfs['name'])
def test_volume_manage(self):
lcfg = self.configuration
lcfg.zfssa_manage_policy = 'loose'
test_vol = self.test_vol
self.drv.zfssa.get_volume.return_value = test_vol
self.drv.zfssa.rename_volume.return_value = None
self.drv.zfssa.set_file_props.return_value = None
self.drv.mount_path = lcfg.zfssa_data_ip + ':' + 'fake_mountpoint'
self.assertEqual({'provider_location': self.drv.mount_path},
self.drv.manage_existing({'name': 'volume-123'},
{'source-name':
'volume-567'}))
self.drv.zfssa.get_volume.assert_called_once_with('volume-567')
self.drv.zfssa.rename_volume.assert_called_once_with('volume-567',
'volume-123')
self.drv.zfssa.set_file_props.assert_called_once_with(
'volume-123', {'cinder_managed': 'True'})
# Test when 'zfssa_manage_policy' is set to 'strict'.
lcfg.zfssa_manage_policy = 'strict'
test_vol.update({'cinder_managed': 'False'})
self.drv.zfssa.get_volume.return_value = test_vol
self.assertEqual({'provider_location': self.drv.mount_path},
self.drv.manage_existing({'name': 'volume-123'},
{'source-name':
'volume-567'}))
def test_volume_manage_negative_no_source_name(self):
self.assertRaises(exception.ManageExistingInvalidReference,
self.drv.manage_existing,
{'name': 'volume-123'},
{'source-id': 'volume-567'})
def test_volume_manage_negative_backend_exception(self):
self.drv.zfssa.get_volume.side_effect = \
exception.VolumeNotFound(volume_id='volume-567')
self.assertRaises(exception.InvalidInput,
self.drv.manage_existing,
{'name': 'volume-123'},
{'source-name': 'volume-567'})
def test_volume_manage_negative_verify_fail(self):
lcfg = self.configuration
lcfg.zfssa_manage_policy = 'strict'
test_vol = self.test_vol
test_vol['cinder_managed'] = ''
self.drv.zfssa.get_volume.return_value = test_vol
self.assertRaises(exception.InvalidInput,
self.drv.manage_existing,
{'name': 'volume-123'},
{'source-name': 'volume-567'})
test_vol.update({'cinder_managed': 'True'})
self.drv.zfssa.get_volume.return_value = test_vol
self.assertRaises(exception.ManageExistingAlreadyManaged,
self.drv.manage_existing,
{'name': 'volume-123'},
{'source-name': 'volume-567'})
@mock.patch.object(zfssanfs.ZFSSANFSDriver, '_verify_volume_to_manage')
def test_volume_manage_negative_rename_fail(self,
_verify_volume_to_manage):
test_vol = self.test_vol
test_vol.update({'cinder_managed': 'False'})
self.drv.zfssa.get_volume.return_value = test_vol
self.drv._verify_volume_to_manage.return_value = None
self.drv.zfssa.rename_volume.side_effect = \
exception.VolumeBackendAPIException(data="fake exception")
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.manage_existing, {'name': 'volume-123'},
{'source-name': 'volume-567'})
@mock.patch.object(zfssanfs.ZFSSANFSDriver, '_verify_volume_to_manage')
def test_volume_manage_negative_set_prop_fail(self,
_verify_volume_to_manage):
test_vol = self.test_vol
test_vol.update({'cinder_managed': 'False'})
self.drv.zfssa.get_volume.return_value = test_vol
self.drv._verify_volume_to_manage.return_value = None
self.drv.zfssa.rename_volume.return_value = None
self.drv.zfssa.set_file_props.side_effect = \
exception.VolumeBackendAPIException(data="fake exception")
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.manage_existing, {'name': 'volume-123'},
{'source-name': 'volume-567'})
def test_volume_unmanage(self):
test_vol = self.test_vol
test_vol.update({'cinder_managed': 'True'})
self.drv.zfssa.rename_volume.return_value = None
self.drv.zfssa.set_file_props.return_value = None
self.assertIsNone(self.drv.unmanage(test_vol))
new_vol_name = 'unmanaged-' + test_vol['name']
self.drv.zfssa.rename_volume.assert_called_once_with(test_vol['name'],
new_vol_name)
self.drv.zfssa.set_file_props.assert_called_once_with(
new_vol_name, {'cinder_managed': 'False'})
def test_volume_unmanage_negative_rename_fail(self):
test_vol = self.test_vol
test_vol.update({'cinder_managed': 'True'})
self.drv.zfssa.rename_volume.side_effect = \
exception.VolumeBackendAPIException(data="fake exception")
self.drv.zfssa.set_file_props.return_value = None
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.unmanage, test_vol)
def test_volume_unmanage_negative_set_prop_fail(self):
test_vol = self.test_vol
test_vol.update({'cinder_managed': 'True'})
self.drv.zfssa.rename_volume.return_value = None
self.drv.zfssa.set_file_props.side_effect = \
exception.VolumeBackendAPIException(data="fake exception")
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.unmanage, test_vol)
@mock.patch.object(zfssanfs.ZFSSANFSDriver, '_get_mount_point_for_share')
def test_manage_existing_get_size(self, _get_mount_point_for_share):
self.drv._get_mount_point_for_share.return_value = \
'/fake/mnt/fake_share/'
self.drv._mounted_shares = []
self.drv._mounted_shares.append('fake_share')
file = mock.Mock(st_size=123 * units.Gi)
with mock.patch('os.path.isfile', return_value=True):
with mock.patch('os.stat', return_value=file):
self.assertEqual(float(file.st_size / units.Gi),
self.drv.manage_existing_get_size(
{'name': 'volume-123'},
{'source-name': 'volume-567'}))
@mock.patch.object(zfssanfs.ZFSSANFSDriver, '_get_mount_point_for_share')
def test_manage_existing_get_size_negative(self,
_get_mount_point_for_share):
self.drv._get_mount_point_for_share.return_value = \
'/fake/mnt/fake_share/'
self.drv._mounted_shares = []
self.drv._mounted_shares.append('fake_share')
with mock.patch('os.path.isfile', return_value=True):
with mock.patch('os.stat', side_effect=OSError):
self.assertRaises(exception.VolumeBackendAPIException,
self.drv.manage_existing_get_size,
{'name': 'volume-123'},
{'source-name': 'volume-567'})
class TestZFSSAApi(test.TestCase): class TestZFSSAApi(test.TestCase):

View File

@ -1,4 +1,4 @@
# Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -20,6 +20,7 @@ import math
from oslo_config import cfg from oslo_config import cfg
from oslo_log import log from oslo_log import log
from oslo_serialization import base64 from oslo_serialization import base64
from oslo_utils import excutils
from oslo_utils import units from oslo_utils import units
import six import six
@ -84,8 +85,10 @@ ZFSSA_OPTS = [
cfg.BoolOpt('zfssa_enable_local_cache', default=True, cfg.BoolOpt('zfssa_enable_local_cache', default=True,
help='Flag to enable local caching: True, False.'), help='Flag to enable local caching: True, False.'),
cfg.StrOpt('zfssa_cache_project', default='os-cinder-cache', cfg.StrOpt('zfssa_cache_project', default='os-cinder-cache',
help='Name of ZFSSA project where cache volumes are stored.') help='Name of ZFSSA project where cache volumes are stored.'),
cfg.StrOpt('zfssa_manage_policy', default='loose',
choices=['loose', 'strict'],
help='Driver policy for volume manage.')
] ]
CONF.register_opts(ZFSSA_OPTS) CONF.register_opts(ZFSSA_OPTS)
@ -109,8 +112,10 @@ class ZFSSAISCSIDriver(driver.ISCSIDriver):
1.0.1: 1.0.1:
Backend enabled volume migration. Backend enabled volume migration.
Local cache feature. Local cache feature.
1.0.2:
Volume manage/unmanage support.
""" """
VERSION = '1.0.1' VERSION = '1.0.2'
protocol = 'iSCSI' protocol = 'iSCSI'
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
@ -144,19 +149,25 @@ class ZFSSAISCSIDriver(driver.ISCSIDriver):
compression=lcfg.zfssa_lun_compression, compression=lcfg.zfssa_lun_compression,
logbias=lcfg.zfssa_lun_logbias) logbias=lcfg.zfssa_lun_logbias)
schemas = [
{'property': 'cinder_managed',
'description': 'Managed by Cinder',
'type': 'Boolean'}]
if lcfg.zfssa_enable_local_cache: if lcfg.zfssa_enable_local_cache:
self.zfssa.create_project(lcfg.zfssa_pool, self.zfssa.create_project(lcfg.zfssa_pool,
lcfg.zfssa_cache_project, lcfg.zfssa_cache_project,
compression=lcfg.zfssa_lun_compression, compression=lcfg.zfssa_lun_compression,
logbias=lcfg.zfssa_lun_logbias) logbias=lcfg.zfssa_lun_logbias)
schemas = [ schemas.extend([
{'property': 'image_id', {'property': 'image_id',
'description': 'OpenStack image ID', 'description': 'OpenStack image ID',
'type': 'String'}, 'type': 'String'},
{'property': 'updated_at', {'property': 'updated_at',
'description': 'Most recent updated time of image', 'description': 'Most recent updated time of image',
'type': 'String'}] 'type': 'String'}])
self.zfssa.create_schemas(schemas)
self.zfssa.create_schemas(schemas)
if (lcfg.zfssa_initiator_config != ''): if (lcfg.zfssa_initiator_config != ''):
initiator_config = ast.literal_eval(lcfg.zfssa_initiator_config) initiator_config = ast.literal_eval(lcfg.zfssa_initiator_config)
@ -224,6 +235,13 @@ class ZFSSAISCSIDriver(driver.ISCSIDriver):
self.zfssa.add_to_targetgroup(iqn, lcfg.zfssa_target_group) self.zfssa.add_to_targetgroup(iqn, lcfg.zfssa_target_group)
if lcfg.zfssa_manage_policy not in ("loose", "strict"):
err_msg = (_("zfssa_manage_policy property needs to be set to"
" 'strict' or 'loose'. Current value is: %s.") %
lcfg.zfssa_manage_policy)
LOG.error(err_msg)
raise exception.InvalidInput(reason=err_msg)
def check_for_setup_error(self): def check_for_setup_error(self):
"""Check that driver can login. """Check that driver can login.
@ -286,6 +304,7 @@ class ZFSSAISCSIDriver(driver.ISCSIDriver):
lcfg = self.configuration lcfg = self.configuration
volsize = str(volume['size']) + 'g' volsize = str(volume['size']) + 'g'
specs = self._get_voltype_specs(volume) specs = self._get_voltype_specs(volume)
specs.update({'custom:cinder_managed': True})
self.zfssa.create_lun(lcfg.zfssa_pool, self.zfssa.create_lun(lcfg.zfssa_pool,
lcfg.zfssa_project, lcfg.zfssa_project,
volume['name'], volume['name'],
@ -995,6 +1014,103 @@ class ZFSSAISCSIDriver(driver.ISCSIDriver):
LOG.warning(_LW("Volume %s exists but can't be deleted"), LOG.warning(_LW("Volume %s exists but can't be deleted"),
cache['share']) cache['share'])
def manage_existing(self, volume, existing_ref):
"""Manage an existing volume in the ZFSSA backend.
:param volume: Reference to the new volume.
:param existing_ref: Reference to the existing volume to be managed.
"""
lcfg = self.configuration
existing_vol = self._get_existing_vol(existing_ref)
self._verify_volume_to_manage(existing_vol)
new_vol_name = volume['name']
try:
self.zfssa.set_lun_props(lcfg.zfssa_pool,
lcfg.zfssa_project,
existing_vol['name'],
name=new_vol_name,
schema={"custom:cinder_managed": True})
except exception.VolumeBackendAPIException:
with excutils.save_and_reraise_exception():
LOG.error(_LE("Failed to rename volume %(existing)s to "
"%(new)s. Volume manage failed."),
{'existing': existing_vol['name'],
'new': new_vol_name})
return None
def manage_existing_get_size(self, volume, existing_ref):
"""Return size of the volume to be managed by manage_existing."""
existing_vol = self._get_existing_vol(existing_ref)
size = existing_vol['size']
return int(math.ceil(float(size) / units.Gi))
def unmanage(self, volume):
"""Remove an existing volume from cinder management.
:param volume: Reference to the volume to be unmanaged.
"""
lcfg = self.configuration
new_name = 'unmanaged-' + volume['name']
try:
self.zfssa.set_lun_props(lcfg.zfssa_pool,
lcfg.zfssa_project,
volume['name'],
name=new_name,
schema={"custom:cinder_managed": False})
except exception.VolumeBackendAPIException:
with excutils.save_and_reraise_exception():
LOG.error(_LE("Failed to rename volume %(existing)s to"
" %(new)s. Volume unmanage failed."),
{'existing': volume['name'],
'new': new_name})
return None
def _verify_volume_to_manage(self, volume):
lcfg = self.configuration
if lcfg.zfssa_manage_policy == 'loose':
return
vol_name = volume['name']
if 'cinder_managed' not in volume:
err_msg = (_("Unknown if the volume: %s to be managed is "
"already being managed by Cinder. Aborting manage "
"volume. Please add 'cinder_managed' custom schema "
"property to the volume and set its value to False."
" Alternatively, set the value of cinder config "
"policy 'zfssa_manage_policy' to 'loose' to "
"remove this restriction.") % vol_name)
LOG.error(err_msg)
raise exception.InvalidInput(reason=err_msg)
if volume['cinder_managed'] is True:
msg = (_("Volume: %s is already being managed by Cinder.")
% vol_name)
LOG.error(msg)
raise exception.ManageExistingAlreadyManaged(volume_ref=vol_name)
def _get_existing_vol(self, existing_ref):
lcfg = self.configuration
if 'source-name' not in existing_ref:
msg = (_("Reference to volume: %s to be managed must contain "
"source-name.") % existing_ref)
raise exception.ManageExistingInvalidReference(
existing_ref=existing_ref, reason=msg)
try:
existing_vol = self.zfssa.get_lun(lcfg.zfssa_pool,
lcfg.zfssa_project,
existing_ref['source-name'])
except exception.VolumeNotFound:
err_msg = (_("Volume %s doesn't exist on the ZFSSA "
"backend.") % existing_vol['name'])
LOG.error(err_msg)
raise exception.InvalidInput(reason=err_msg)
return existing_vol
class MigrateVolumeInit(task.Task): class MigrateVolumeInit(task.Task):
def execute(self, src_zfssa, volume, src_pool, src_project): def execute(self, src_zfssa, volume, src_pool, src_project):

View File

@ -1,4 +1,4 @@
# Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -17,6 +17,7 @@ ZFS Storage Appliance NFS Cinder Volume Driver
import datetime as dt import datetime as dt
import errno import errno
import math import math
import os
from oslo_config import cfg from oslo_config import cfg
from oslo_log import log from oslo_log import log
@ -59,7 +60,10 @@ ZFSSA_OPTS = [
help='Flag to enable local caching: True, False.'), help='Flag to enable local caching: True, False.'),
cfg.StrOpt('zfssa_cache_directory', default='os-cinder-cache', cfg.StrOpt('zfssa_cache_directory', default='os-cinder-cache',
help='Name of directory inside zfssa_nfs_share where cache ' help='Name of directory inside zfssa_nfs_share where cache '
'volumes are stored.') 'volumes are stored.'),
cfg.StrOpt('zfssa_manage_policy', default='loose',
choices=['loose', 'strict'],
help='Driver policy for volume manage.')
] ]
LOG = log.getLogger(__name__) LOG = log.getLogger(__name__)
@ -79,8 +83,10 @@ class ZFSSANFSDriver(nfs.NfsDriver):
1.0.1: 1.0.1:
Backend enabled volume migration. Backend enabled volume migration.
Local cache feature. Local cache feature.
1.0.2:
Volume manage/unmanage support.
""" """
VERSION = '1.0.1' VERSION = '1.0.2'
volume_backend_name = 'ZFSSA_NFS' volume_backend_name = 'ZFSSA_NFS'
protocol = driver_prefix = driver_volume_type = 'nfs' protocol = driver_prefix = driver_volume_type = 'nfs'
@ -201,6 +207,10 @@ class ZFSSANFSDriver(nfs.NfsDriver):
self.zfssa.verify_service('http') self.zfssa.verify_service('http')
self.zfssa.verify_service('nfs') self.zfssa.verify_service('nfs')
def create_volume(self, volume):
super(ZFSSANFSDriver, self).create_volume(volume)
self.zfssa.set_file_props(volume['name'], {'cinder_managed': 'True'})
def create_snapshot(self, snapshot): def create_snapshot(self, snapshot):
"""Creates a snapshot of a volume.""" """Creates a snapshot of a volume."""
LOG.info(_LI('Creating snapshot: %s'), snapshot['name']) LOG.info(_LI('Creating snapshot: %s'), snapshot['name'])
@ -635,3 +645,119 @@ class ZFSSANFSDriver(nfs.NfsDriver):
method='MOVE') method='MOVE')
provider_location = new_volume['provider_location'] provider_location = new_volume['provider_location']
return {'_name_id': None, 'provider_location': provider_location} return {'_name_id': None, 'provider_location': provider_location}
def manage_existing(self, volume, existing_ref):
"""Manage an existing volume in the ZFSSA backend.
:param volume: Reference to the new volume.
:param existing_ref: Reference to the existing volume to be managed.
"""
existing_vol_name = self._get_existing_vol_name(existing_ref)
try:
vol_props = self.zfssa.get_volume(existing_vol_name)
except exception.VolumeNotFound:
err_msg = (_("Volume %s doesn't exist on the ZFSSA backend.") %
existing_vol_name)
LOG.error(err_msg)
raise exception.InvalidInput(reason=err_msg)
self._verify_volume_to_manage(existing_vol_name, vol_props)
try:
self.zfssa.rename_volume(existing_vol_name, volume['name'])
except Exception:
LOG.error(_LE("Failed to rename volume %(existing)s to %(new)s. "
"Volume manage failed."),
{'existing': existing_vol_name,
'new': volume['name']})
raise
try:
self.zfssa.set_file_props(volume['name'],
{'cinder_managed': 'True'})
except Exception:
self.zfssa.rename_volume(volume['name'], existing_vol_name)
LOG.error(_LE("Failed to set properties for volume %(existing)s. "
"Volume manage failed."),
{'existing': volume['name']})
raise
return {'provider_location': self.mount_path}
def manage_existing_get_size(self, volume, existing_ref):
"""Return size of the volume to be managed by manage_existing."""
existing_vol_name = self._get_existing_vol_name(existing_ref)
# The ZFSSA NFS driver only has one mounted share.
local_share_mount = self._get_mount_point_for_share(
self._mounted_shares[0])
local_vol_path = os.path.join(local_share_mount, existing_vol_name)
try:
if os.path.isfile(local_vol_path):
size = int(math.ceil(float(
utils.get_file_size(local_vol_path)) / units.Gi))
except (OSError, ValueError):
err_msg = (_("Failed to get size of existing volume: %(vol). "
"Volume Manage failed."), {'vol': existing_vol_name})
LOG.error(err_msg)
raise exception.VolumeBackendAPIException(data=err_msg)
LOG.debug("Size volume: %(vol)s to be migrated is: %(size)s.",
{'vol': existing_vol_name, 'size': size})
return size
def _verify_volume_to_manage(self, name, vol_props):
lcfg = self.configuration
if lcfg.zfssa_manage_policy != 'strict':
return
if vol_props['cinder_managed'] == "":
err_msg = (_("Unknown if the volume: %s to be managed is "
"already being managed by Cinder. Aborting manage "
"volume. Please add 'cinder_managed' custom schema "
"property to the volume and set its value to False. "
"Alternatively, Set the value of cinder config "
"policy 'zfssa_manage_policy' to 'loose' to "
"remove this restriction.") % name)
LOG.error(err_msg)
raise exception.InvalidInput(reason=err_msg)
if vol_props['cinder_managed'] == 'True':
msg = (_("Volume: %s is already being managed by Cinder.") % name)
LOG.error(msg)
raise exception.ManageExistingAlreadyManaged(volume_ref=name)
def unmanage(self, volume):
"""Remove an existing volume from cinder management.
:param volume: Reference to the volume to be unmanaged.
"""
new_name = 'unmanaged-' + volume['name']
try:
self.zfssa.rename_volume(volume['name'], new_name)
except Exception:
LOG.error(_LE("Failed to rename volume %(existing)s to %(new)s. "
"Volume unmanage failed."),
{'existing': volume['name'],
'new': new_name})
raise
try:
self.zfssa.set_file_props(new_name, {'cinder_managed': 'False'})
except Exception:
self.zfssa.rename_volume(new_name, volume['name'])
LOG.error(_LE("Failed to set properties for volume %(existing)s. "
"Volume unmanage failed."),
{'existing': volume['name']})
raise
def _get_existing_vol_name(self, existing_ref):
if 'source-name' not in existing_ref:
msg = _("Reference to volume to be managed must contain "
"source-name.")
raise exception.ManageExistingInvalidReference(
existing_ref=existing_ref, reason=msg)
return existing_ref['source-name']

View File

@ -1,4 +1,4 @@
# Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. # Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -747,6 +747,7 @@ class ZFSSAApi(object):
val = json.loads(ret.data) val = json.loads(ret.data)
ret = { ret = {
'name': val['lun']['name'],
'guid': val['lun']['lunguid'], 'guid': val['lun']['lunguid'],
'number': val['lun']['assignednumber'], 'number': val['lun']['assignednumber'],
'initiatorgroup': val['lun']['initiatorgroup'], 'initiatorgroup': val['lun']['initiatorgroup'],
@ -759,6 +760,8 @@ class ZFSSAApi(object):
if 'custom:image_id' in val['lun']: if 'custom:image_id' in val['lun']:
ret.update({'image_id': val['lun']['custom:image_id']}) ret.update({'image_id': val['lun']['custom:image_id']})
ret.update({'updated_at': val['lun']['custom:updated_at']}) ret.update({'updated_at': val['lun']['custom:updated_at']})
if 'custom:cinder_managed' in val['lun']:
ret.update({'cinder_managed': val['lun']['custom:cinder_managed']})
return ret return ret
@ -923,6 +926,9 @@ class ZFSSAApi(object):
if kargs is None: if kargs is None:
return return
if 'schema' in kargs:
kargs.update(kargs.pop('schema'))
ret = self.rclient.put(svc, kargs) ret = self.rclient.put(svc, kargs)
if ret.status != restclient.Status.ACCEPTED: if ret.status != restclient.Status.ACCEPTED:
exception_msg = (_('Error Setting props ' exception_msg = (_('Error Setting props '
@ -1251,6 +1257,7 @@ class ZFSSANfsApi(ZFSSAApi):
'updated_at': self._parse_prop(resp, 'updated_at'), 'updated_at': self._parse_prop(resp, 'updated_at'),
'image_id': self._parse_prop(resp, 'image_id'), 'image_id': self._parse_prop(resp, 'image_id'),
'origin': self._parse_prop(resp, 'origin'), 'origin': self._parse_prop(resp, 'origin'),
'cinder_managed': self._parse_prop(resp, 'cinder_managed'),
} }
return result return result
@ -1288,3 +1295,7 @@ class ZFSSANfsApi(ZFSSAApi):
except Exception: except Exception:
exception_msg = (_('Cannot create directory %s.'), dirname) exception_msg = (_('Cannot create directory %s.'), dirname)
raise exception.VolumeBackendAPIException(data=exception_msg) raise exception.VolumeBackendAPIException(data=exception_msg)
def rename_volume(self, src, dst):
return self.webdavclient.request(src_file=src, dst_file=dst,
method='MOVE')

View File

@ -0,0 +1,2 @@
features:
- Volume manage/unmanage support for Oracle ZFSSA iSCSI and NFS drivers.