Handle manage/unmanage for replicated shares

Managing a share with a share type that has replication_type
extra_spec must be allowed. Drivers are expected to fail
this operation if the share was part of a replication relationship
that Manila does not know about.

Unmanaging a share with replicas must not be permitted
until all replicas are removed.

Managing and unmanaging of snapshots must not
be permitted for a share that has replicas.

Modify the NetApp driver for manage_existing to check
for existing replicas.

Also fix issue with manage retry where the share
data was being altered inappropriately by a DB API.

Closes-Bug: #1561641
Closes-Bug: #1565903

Co-Authored-By: Goutham Pacha Ravi <gouthamr@netapp.com>

Change-Id: I82f1fef1e30114e017efd00fa7da70aceecab94c
This commit is contained in:
yogesh 2016-03-24 17:36:36 -04:00 committed by Goutham Pacha Ravi
parent f63fe255ec
commit 53364e07a3
18 changed files with 579 additions and 32 deletions

View File

@ -37,6 +37,10 @@ class ShareUnmanageMixin(object):
try: try:
share = self.share_api.get(context, id) share = self.share_api.get(context, id)
if share.get('has_replicas'):
msg = _("Share %s has replicas. It cannot be unmanaged "
"until all replicas are removed.") % share['id']
raise exc.HTTPConflict(explanation=msg)
if share['instance'].get('share_server_id'): if share['instance'].get('share_server_id'):
msg = _("Operation 'unmanage' is not supported for shares " msg = _("Operation 'unmanage' is not supported for shares "
"that are created on top of share servers " "that are created on top of share servers "

View File

@ -59,6 +59,11 @@ class ShareSnapshotsController(share_snapshots.ShareSnapshotMixin,
"snapshots of shares that are created with share" "snapshots of shares that are created with share"
" servers (created with share-networks).") " servers (created with share-networks).")
raise exc.HTTPForbidden(explanation=msg) raise exc.HTTPForbidden(explanation=msg)
elif share.get('has_replicas'):
msg = _("Share %s has replicas. Snapshots of this share "
"cannot currently be unmanaged until all replicas "
"are removed.") % share['id']
raise exc.HTTPConflict(explanation=msg)
elif snapshot['status'] in constants.TRANSITIONAL_STATUSES: elif snapshot['status'] in constants.TRANSITIONAL_STATUSES:
msg = _("Snapshot with transitional state cannot be " msg = _("Snapshot with transitional state cannot be "
"unmanaged. Snapshot '%(s_id)s' is in '%(state)s' " "unmanaged. Snapshot '%(s_id)s' is in '%(state)s' "
@ -118,7 +123,8 @@ class ShareSnapshotsController(share_snapshots.ShareSnapshotMixin,
driver_options) driver_options)
except (exception.ShareNotFound, exception.ShareSnapshotNotFound) as e: except (exception.ShareNotFound, exception.ShareSnapshotNotFound) as e:
raise exc.HTTPNotFound(explanation=six.text_type(e)) raise exc.HTTPNotFound(explanation=six.text_type(e))
except exception.ManageInvalidShareSnapshot as e: except (exception.InvalidShare,
exception.ManageInvalidShareSnapshot) as e:
raise exc.HTTPConflict(explanation=six.text_type(e)) raise exc.HTTPConflict(explanation=six.text_type(e))
return self._view_builder.detail(req, snapshot_ref) return self._view_builder.detail(req, snapshot_ref)

View File

@ -1109,27 +1109,35 @@ def reservation_expire(context):
################ ################
def extract_instance_values(values, fields): def extract_instance_values(values_to_extract, fields):
values = copy.deepcopy(values_to_extract)
instance_values = {} instance_values = {}
for field in fields: for field in fields:
field_value = values.pop(field, None) field_value = values.pop(field, None)
if field_value: if field_value:
instance_values.update({field: field_value}) instance_values.update({field: field_value})
return instance_values return instance_values, values
def extract_share_instance_values(values): def extract_share_instance_values(values):
share_instance_model_fields = [ share_instance_model_fields = [
'status', 'host', 'scheduled_at', 'launched_at', 'terminated_at', 'status', 'host', 'scheduled_at', 'launched_at', 'terminated_at',
'share_server_id', 'share_network_id', 'availability_zone' 'share_server_id', 'share_network_id', 'availability_zone',
'replica_state',
] ]
return extract_instance_values(values, share_instance_model_fields) share_instance_values, share_values = (
extract_instance_values(values, share_instance_model_fields)
)
return share_instance_values, share_values
def extract_snapshot_instance_values(values): def extract_snapshot_instance_values(values):
fields = ['status', 'progress', 'provider_location'] fields = ['status', 'progress', 'provider_location']
return extract_instance_values(values, fields) snapshot_instance_values, snapshot_values = (
extract_instance_values(values, fields)
)
return snapshot_instance_values, snapshot_values
################ ################
@ -1480,10 +1488,10 @@ def share_create(context, share_values, create_share_instance=True):
models.ShareMetadata) models.ShareMetadata)
session = get_session() session = get_session()
share_ref = models.Share() share_ref = models.Share()
share_instance_values = extract_share_instance_values(values) share_instance_values, share_values = extract_share_instance_values(values)
ensure_availability_zone_exists(context, share_instance_values, session, ensure_availability_zone_exists(context, share_instance_values, session,
strict=False) strict=False)
share_ref.update(values) share_ref.update(share_values)
with session.begin(): with session.begin():
share_ref.save(session=session) share_ref.save(session=session)
@ -1514,10 +1522,11 @@ def share_data_get_for_project(context, project_id, user_id, session=None):
@require_context @require_context
@oslo_db_api.wrap_db_retry(max_retries=5, retry_on_deadlock=True) @oslo_db_api.wrap_db_retry(max_retries=5, retry_on_deadlock=True)
def share_update(context, share_id, values): def share_update(context, share_id, update_values):
session = get_session() session = get_session()
values = copy.deepcopy(update_values)
share_instance_values = extract_share_instance_values(values) share_instance_values, share_values = extract_share_instance_values(values)
ensure_availability_zone_exists(context, share_instance_values, session, ensure_availability_zone_exists(context, share_instance_values, session,
strict=False) strict=False)
@ -1527,7 +1536,7 @@ def share_update(context, share_id, values):
_share_instance_update(context, share_ref.instance['id'], _share_instance_update(context, share_ref.instance['id'],
share_instance_values, session=session) share_instance_values, session=session)
share_ref.update(values) share_ref.update(share_values)
share_ref.save(session=session) share_ref.save(session=session)
return share_ref return share_ref
@ -2024,17 +2033,21 @@ def _set_share_snapshot_instance_data(context, snapshot_instances, session):
@require_context @require_context
def share_snapshot_create(context, values, create_snapshot_instance=True): def share_snapshot_create(context, create_values,
create_snapshot_instance=True):
values = copy.deepcopy(create_values)
values = ensure_model_dict_has_id(values) values = ensure_model_dict_has_id(values)
snapshot_ref = models.ShareSnapshot() snapshot_ref = models.ShareSnapshot()
snapshot_instance_values = extract_snapshot_instance_values(values) snapshot_instance_values, snapshot_values = (
share_ref = share_get(context, values.get('share_id')) extract_snapshot_instance_values(values)
)
share_ref = share_get(context, snapshot_values.get('share_id'))
snapshot_instance_values.update( snapshot_instance_values.update(
{'share_instance_id': share_ref.instance.id} {'share_instance_id': share_ref.instance.id}
) )
snapshot_ref.update(values) snapshot_ref.update(snapshot_values)
session = get_session() session = get_session()
with session.begin(): with session.begin():
snapshot_ref.save(session=session) snapshot_ref.save(session=session)
@ -2046,7 +2059,8 @@ def share_snapshot_create(context, values, create_snapshot_instance=True):
snapshot_instance_values, snapshot_instance_values,
session=session session=session
) )
return share_snapshot_get(context, values['id'], session=session) return share_snapshot_get(
context, snapshot_values['id'], session=session)
@require_admin_context @require_admin_context
@ -2198,10 +2212,12 @@ def share_snapshot_update(context, snapshot_id, values):
snapshot_ref = share_snapshot_get(context, snapshot_id, snapshot_ref = share_snapshot_get(context, snapshot_id,
session=session) session=session)
instance_values = extract_snapshot_instance_values(values) instance_values, snapshot_values = (
extract_snapshot_instance_values(values)
)
if values: if snapshot_values:
snapshot_ref.update(values) snapshot_ref.update(snapshot_values)
snapshot_ref.save(session=session) snapshot_ref.save(session=session)
if instance_values: if instance_values:

View File

@ -505,15 +505,15 @@ class API(base.Base):
'share_type_id': share_data['share_type_id'] 'share_type_id': share_data['share_type_id']
}) })
share_type = {}
share_type_id = share_data['share_type_id'] share_type_id = share_data['share_type_id']
if share_type_id:
share_type = share_types.get_share_type(context, share_type_id) share_type = share_types.get_share_type(context, share_type_id)
snapshot_support = strutils.bool_from_string( snapshot_support = strutils.bool_from_string(
share_type.get('extra_specs', {}).get( share_type.get('extra_specs', {}).get(
'snapshot_support', True) if share_type else True, 'snapshot_support', True) if share_type else True,
strict=True) strict=True)
replication_type = share_type.get('extra_specs', {}).get(
'replication_type')
share_data.update({ share_data.update({
'user_id': context.user_id, 'user_id': context.user_id,
@ -521,6 +521,7 @@ class API(base.Base):
'status': constants.STATUS_MANAGING, 'status': constants.STATUS_MANAGING,
'scheduled_at': timeutils.utcnow(), 'scheduled_at': timeutils.utcnow(),
'snapshot_support': snapshot_support, 'snapshot_support': snapshot_support,
'replication_type': replication_type,
}) })
LOG.debug("Manage: Found shares %s.", len(shares)) LOG.debug("Manage: Found shares %s.", len(shares))
@ -621,6 +622,12 @@ class API(base.Base):
except exception.NotFound: except exception.NotFound:
raise exception.ShareNotFound(share_id=snapshot_data['share_id']) raise exception.ShareNotFound(share_id=snapshot_data['share_id'])
if share['has_replicas']:
msg = (_("Share %s has replicas. Snapshots of this share cannot "
"currently be managed until all replicas are removed.")
% share['id'])
raise exception.InvalidShare(reason=msg)
existing_snapshots = self.db.share_snapshot_get_all_for_share( existing_snapshots = self.db.share_snapshot_get_all_for_share(
context, snapshot_data['share_id']) context, snapshot_data['share_id'])

View File

@ -637,9 +637,13 @@ class ShareDriver(object):
def manage_existing(self, share, driver_options): def manage_existing(self, share, driver_options):
"""Brings an existing share under Manila management. """Brings an existing share under Manila management.
If provided share is not valid, then raise a If the provided share is not valid, then raise a
ManageInvalidShare exception, specifying a reason for the failure. ManageInvalidShare exception, specifying a reason for the failure.
If the provided share is not in a state that can be managed, such as
being replicated on the backend, the driver *MUST* raise
ManageInvalidShare exception with an appropriate message.
The share has a share_type, and the driver can inspect that and The share has a share_type, and the driver can inspect that and
compare against the properties of the referenced backend share. compare against the properties of the referenced backend share.
If they are incompatible, raise a If they are incompatible, raise a

View File

@ -1579,6 +1579,7 @@ class NetAppCmodeClient(client_base.NetAppBaseClient):
'name': None, 'name': None,
'type': None, 'type': None,
'style': None, 'style': None,
'owning-vserver-name': None,
}, },
'volume-space-attributes': { 'volume-space-attributes': {
'size': None, 'size': None,
@ -1607,6 +1608,8 @@ class NetAppCmodeClient(client_base.NetAppBaseClient):
'name': volume_id_attributes.get_child_content('name'), 'name': volume_id_attributes.get_child_content('name'),
'type': volume_id_attributes.get_child_content('type'), 'type': volume_id_attributes.get_child_content('type'),
'style': volume_id_attributes.get_child_content('style'), 'style': volume_id_attributes.get_child_content('style'),
'owning-vserver-name': volume_id_attributes.get_child_content(
'owning-vserver-name'),
'size': volume_space_attributes.get_child_content('size'), 'size': volume_space_attributes.get_child_content('size'),
} }
return volume return volume
@ -2914,3 +2917,27 @@ class NetAppCmodeClient(client_base.NetAppBaseClient):
snapmirrors.append(snapmirror) snapmirrors.append(snapmirror)
return snapmirrors return snapmirrors
def volume_has_snapmirror_relationships(self, volume):
"""Return True if snapmirror relationships exist for a given volume.
If we have snapmirror control plane license, we can verify whether
the given volume is part of any snapmirror relationships.
"""
try:
# Check if volume is a source snapmirror volume
snapmirrors = self.get_snapmirrors(
volume['owning-vserver-name'], volume['name'], None, None)
# Check if volume is a destination snapmirror volume
if not snapmirrors:
snapmirrors = self.get_snapmirrors(
None, None, volume['owning-vserver-name'], volume['name'])
has_snapmirrors = len(snapmirrors) > 0
except netapp_api.NaApiError:
msg = _LE("Could not determine if volume %s is part of "
"existing snapmirror relationships.")
LOG.exception(msg, volume['name'])
has_snapmirrors = False
return has_snapmirrors

View File

@ -800,6 +800,7 @@ class NetAppCmodeFileStorageLibrary(object):
# Get existing volume info # Get existing volume info
volume = vserver_client.get_volume_to_manage(aggregate_name, volume = vserver_client.get_volume_to_manage(aggregate_name,
volume_name) volume_name)
if not volume: if not volume:
msg = _('Volume %(volume)s not found on aggregate %(aggr)s.') msg = _('Volume %(volume)s not found on aggregate %(aggr)s.')
msg_args = {'volume': volume_name, 'aggr': aggregate_name} msg_args = {'volume': volume_name, 'aggr': aggregate_name}
@ -866,6 +867,12 @@ class NetAppCmodeFileStorageLibrary(object):
msg_args = {'volume': volume['name']} msg_args = {'volume': volume['name']}
raise exception.ManageInvalidShare(reason=msg % msg_args) raise exception.ManageInvalidShare(reason=msg % msg_args)
if vserver_client.volume_has_snapmirror_relationships(volume):
msg = _('Volume %(volume)s must not be in any snapmirror '
'relationships.')
msg_args = {'volume': volume['name']}
raise exception.ManageInvalidShare(reason=msg % msg_args)
@na_utils.trace @na_utils.trace
def create_consistency_group(self, context, cg_dict, share_server=None): def create_consistency_group(self, context, cg_dict, share_server=None):
"""Creates a consistency group. """Creates a consistency group.

View File

@ -1566,6 +1566,11 @@ class ShareManager(manager.SchedulerDependentManager):
'availability_zone': CONF.storage_availability_zone, 'availability_zone': CONF.storage_availability_zone,
}) })
# If the share was managed with `replication_type` extra-spec, the
# instance becomes an `active` replica.
if share_ref.get('replication_type'):
share_update['replica_state'] = constants.REPLICA_STATE_ACTIVE
# NOTE(vponomaryov): we should keep only those export locations # NOTE(vponomaryov): we should keep only those export locations
# that driver has calculated to avoid incompatibilities with one # that driver has calculated to avoid incompatibilities with one
# provided by user. # provided by user.

View File

@ -98,6 +98,28 @@ class ShareUnmanageTest(test.TestCase):
self.mock_policy_check.assert_called_once_with( self.mock_policy_check.assert_called_once_with(
self.context, self.resource_name, 'unmanage') self.context, self.resource_name, 'unmanage')
def test_unmanage_share_that_has_replicas(self):
share = dict(status=constants.STATUS_AVAILABLE, id='foo_id',
instance={}, has_replicas=True)
mock_api_unmanage = self.mock_object(self.controller.share_api,
'unmanage')
mock_db_snapshots_get = self.mock_object(
self.controller.share_api.db, 'share_snapshot_get_all_for_share')
self.mock_object(
self.controller.share_api, 'get',
mock.Mock(return_value=share))
self.assertRaises(
webob.exc.HTTPConflict,
self.controller.unmanage, self.request, share['id'])
self.assertFalse(mock_api_unmanage.called)
self.assertFalse(mock_db_snapshots_get.called)
self.controller.share_api.get.assert_called_once_with(
self.request.environ['manila.context'], share['id'])
self.mock_policy_check.assert_called_once_with(
self.context, self.resource_name, 'unmanage')
def test_unmanage_share_based_on_share_server(self): def test_unmanage_share_based_on_share_server(self):
share = dict(instance=dict(share_server_id='foo_id'), id='bar_id') share = dict(instance=dict(share_server_id='foo_id'), id='bar_id')
self.mock_object( self.mock_object(

View File

@ -486,7 +486,8 @@ class ShareSnapshotAdminActionsAPITest(test.TestCase):
@ddt.data(exception.ShareNotFound(share_id='fake'), @ddt.data(exception.ShareNotFound(share_id='fake'),
exception.ShareSnapshotNotFound(snapshot_id='fake'), exception.ShareSnapshotNotFound(snapshot_id='fake'),
exception.ManageInvalidShareSnapshot(reason='error')) exception.ManageInvalidShareSnapshot(reason='error'),
exception.InvalidShare(reason='error'))
def test_manage_exception(self, exception_type): def test_manage_exception(self, exception_type):
self.mock_policy_check = self.mock_object( self.mock_policy_check = self.mock_object(
policy, 'check_policy', mock.Mock(return_value=True)) policy, 'check_policy', mock.Mock(return_value=True))
@ -497,10 +498,12 @@ class ShareSnapshotAdminActionsAPITest(test.TestCase):
share_api.API, 'manage_snapshot', mock.Mock( share_api.API, 'manage_snapshot', mock.Mock(
side_effect=exception_type)) side_effect=exception_type))
if isinstance(exception_type, exception.ManageInvalidShareSnapshot):
http_ex = webob.exc.HTTPConflict
else:
http_ex = webob.exc.HTTPNotFound http_ex = webob.exc.HTTPNotFound
if (isinstance(exception_type, exception.ManageInvalidShareSnapshot)
or isinstance(exception_type, exception.InvalidShare)):
http_ex = webob.exc.HTTPConflict
self.assertRaises(http_ex, self.assertRaises(http_ex,
self.controller.manage, self.controller.manage,
self.manage_request, body) self.manage_request, body)
@ -544,6 +547,29 @@ class ShareSnapshotAdminActionsAPITest(test.TestCase):
self.unmanage_request.environ['manila.context'], self.unmanage_request.environ['manila.context'],
self.resource_name, 'unmanage_snapshot') self.resource_name, 'unmanage_snapshot')
def test_snapshot_unmanage_replicated_snapshot(self):
self.mock_policy_check = self.mock_object(
policy, 'check_policy', mock.Mock(return_value=True))
share = {'status': constants.STATUS_AVAILABLE, 'id': 'bar_id',
'has_replicas': True}
self.mock_object(share_api.API, 'get', mock.Mock(return_value=share))
snapshot = {'status': constants.STATUS_AVAILABLE, 'id': 'foo_id',
'share_id': 'bar_id'}
self.mock_object(share_api.API, 'get_snapshot',
mock.Mock(return_value=snapshot))
self.assertRaises(webob.exc.HTTPConflict,
self.controller.unmanage,
self.unmanage_request,
snapshot['id'])
self.controller.share_api.get_snapshot.assert_called_once_with(
self.unmanage_request.environ['manila.context'], snapshot['id'])
self.controller.share_api.get.assert_called_once_with(
self.unmanage_request.environ['manila.context'], share['id'])
self.mock_policy_check.assert_called_once_with(
self.unmanage_request.environ['manila.context'],
self.resource_name, 'unmanage_snapshot')
@ddt.data(*constants.TRANSITIONAL_STATUSES) @ddt.data(*constants.TRANSITIONAL_STATUSES)
def test_snapshot_unmanage_with_transitional_state(self, status): def test_snapshot_unmanage_with_transitional_state(self, status):
self.mock_policy_check = self.mock_object( self.mock_policy_check = self.mock_object(

View File

@ -124,6 +124,7 @@ def fake_snapshot(create_instance=False, **kwargs):
snapshot['instance']['provider_location'] snapshot['instance']['provider_location']
) )
snapshot['progress'] = snapshot['instance']['progress'] snapshot['progress'] = snapshot['instance']['progress']
snapshot['instances'] = snapshot['instance'],
else: else:
snapshot['status'] = constants.STATUS_AVAILABLE snapshot['status'] = constants.STATUS_AVAILABLE
snapshot['progress'] = '0%' snapshot['progress'] = '0%'

View File

@ -1955,3 +1955,12 @@ FAKE_RESULT_SUCCESS = api.NaElement('result')
FAKE_RESULT_SUCCESS.add_attr('status', 'passed') FAKE_RESULT_SUCCESS.add_attr('status', 'passed')
FAKE_HTTP_OPENER = urllib.request.build_opener() FAKE_HTTP_OPENER = urllib.request.build_opener()
FAKE_MANAGE_VOLUME = {
'aggregate': SHARE_AGGREGATE_NAME,
'name': SHARE_NAME,
'owning-vserver-name': VSERVER_NAME,
'junction_path': VOLUME_JUNCTION_PATH,
'style': 'fake_style',
'size': SHARE_SIZE,
}

View File

@ -2840,6 +2840,7 @@ class NetAppClientCmodeTestCase(test.TestCase):
'name': None, 'name': None,
'type': None, 'type': None,
'style': None, 'style': None,
'owning-vserver-name': None,
}, },
'volume-space-attributes': { 'volume-space-attributes': {
'size': None, 'size': None,
@ -2854,6 +2855,7 @@ class NetAppClientCmodeTestCase(test.TestCase):
'type': 'rw', 'type': 'rw',
'style': 'flex', 'style': 'flex',
'size': fake.SHARE_SIZE, 'size': fake.SHARE_SIZE,
'owning-vserver-name': fake.VSERVER_NAME
} }
self.client.send_iter_request.assert_has_calls([ self.client.send_iter_request.assert_has_calls([
mock.call('volume-get-iter', volume_get_iter_args)]) mock.call('volume-get-iter', volume_get_iter_args)])
@ -5014,3 +5016,67 @@ class NetAppClientCmodeTestCase(test.TestCase):
} }
self.client.send_request.assert_has_calls([ self.client.send_request.assert_has_calls([
mock.call('snapmirror-resync', snapmirror_resync_args)]) mock.call('snapmirror-resync', snapmirror_resync_args)])
@ddt.data('source', 'destination', None)
def test_volume_has_snapmirror_relationships(self, snapmirror_rel_type):
"""Snapmirror relationships can be both ways."""
vol = fake.FAKE_MANAGE_VOLUME
snapmirror = {
'source-vserver': fake.SM_SOURCE_VSERVER,
'source-volume': fake.SM_SOURCE_VOLUME,
'destination-vserver': fake.SM_DEST_VSERVER,
'destination-volume': fake.SM_DEST_VOLUME,
'is-healthy': 'true',
'mirror-state': 'snapmirrored',
'schedule': 'daily',
}
expected_get_snapmirrors_call_count = 2
expected_get_snapmirrors_calls = [
mock.call(vol['owning-vserver-name'], vol['name'], None, None),
mock.call(None, None, vol['owning-vserver-name'], vol['name']),
]
if snapmirror_rel_type is None:
side_effect = ([], [])
elif snapmirror_rel_type == 'source':
snapmirror['source-vserver'] = vol['owning-vserver-name']
snapmirror['source-volume'] = vol['name']
side_effect = ([snapmirror], None)
expected_get_snapmirrors_call_count = 1
expected_get_snapmirrors_calls.pop()
else:
snapmirror['destination-vserver'] = vol['owning-vserver-name']
snapmirror['destination-volume'] = vol['name']
side_effect = (None, [snapmirror])
mock_get_snapmirrors_call = self.mock_object(
self.client, 'get_snapmirrors', mock.Mock(side_effect=side_effect))
mock_exc_log = self.mock_object(client_cmode.LOG, 'exception')
expected_retval = True if snapmirror_rel_type else False
retval = self.client.volume_has_snapmirror_relationships(vol)
self.assertEqual(expected_retval, retval)
self.assertEqual(expected_get_snapmirrors_call_count,
mock_get_snapmirrors_call.call_count)
mock_get_snapmirrors_call.assert_has_calls(
expected_get_snapmirrors_calls)
self.assertFalse(mock_exc_log.called)
def test_volume_has_snapmirror_relationships_api_error(self):
vol = fake.FAKE_MANAGE_VOLUME
expected_get_snapmirrors_calls = [
mock.call(vol['owning-vserver-name'], vol['name'], None, None),
]
mock_get_snapmirrors_call = self.mock_object(
self.client, 'get_snapmirrors', mock.Mock(
side_effect=self._mock_api_error(netapp_api.EINTERNALERROR)))
mock_exc_log = self.mock_object(client_cmode.LOG, 'exception')
retval = self.client.volume_has_snapmirror_relationships(vol)
self.assertFalse(retval)
self.assertEqual(1, mock_get_snapmirrors_call.call_count)
mock_get_snapmirrors_call.assert_has_calls(
expected_get_snapmirrors_calls)
self.assertTrue(mock_exc_log.called)

View File

@ -1451,6 +1451,8 @@ class NetAppFileStorageLibraryTestCase(test.TestCase):
vserver_client.volume_has_luns = mock.Mock(return_value=False) vserver_client.volume_has_luns = mock.Mock(return_value=False)
vserver_client.volume_has_junctioned_volumes = mock.Mock( vserver_client.volume_has_junctioned_volumes = mock.Mock(
return_value=False) return_value=False)
vserver_client.volume_has_snapmirror_relationships = mock.Mock(
return_value=False)
result = self.library._validate_volume_for_manage( result = self.library._validate_volume_for_manage(
fake.FLEXVOL_TO_MANAGE, vserver_client) fake.FLEXVOL_TO_MANAGE, vserver_client)
@ -1474,6 +1476,8 @@ class NetAppFileStorageLibraryTestCase(test.TestCase):
vserver_client.volume_has_luns = mock.Mock(return_value=False) vserver_client.volume_has_luns = mock.Mock(return_value=False)
vserver_client.volume_has_junctioned_volumes = mock.Mock( vserver_client.volume_has_junctioned_volumes = mock.Mock(
return_value=False) return_value=False)
vserver_client.volume_has_snapmirror_relationships = mock.Mock(
return_value=False)
self.assertRaises(exception.ManageInvalidShare, self.assertRaises(exception.ManageInvalidShare,
self.library._validate_volume_for_manage, self.library._validate_volume_for_manage,
@ -1486,6 +1490,8 @@ class NetAppFileStorageLibraryTestCase(test.TestCase):
vserver_client.volume_has_luns = mock.Mock(return_value=True) vserver_client.volume_has_luns = mock.Mock(return_value=True)
vserver_client.volume_has_junctioned_volumes = mock.Mock( vserver_client.volume_has_junctioned_volumes = mock.Mock(
return_value=False) return_value=False)
vserver_client.volume_has_snapmirror_relationships = mock.Mock(
return_value=False)
self.assertRaises(exception.ManageInvalidShare, self.assertRaises(exception.ManageInvalidShare,
self.library._validate_volume_for_manage, self.library._validate_volume_for_manage,
@ -1498,6 +1504,22 @@ class NetAppFileStorageLibraryTestCase(test.TestCase):
vserver_client.volume_has_luns = mock.Mock(return_value=False) vserver_client.volume_has_luns = mock.Mock(return_value=False)
vserver_client.volume_has_junctioned_volumes = mock.Mock( vserver_client.volume_has_junctioned_volumes = mock.Mock(
return_value=True) return_value=True)
vserver_client.volume_has_snapmirror_relationships = mock.Mock(
return_value=False)
self.assertRaises(exception.ManageInvalidShare,
self.library._validate_volume_for_manage,
fake.FLEXVOL_TO_MANAGE,
vserver_client)
def test_validate_volume_for_manage_snapmirror_relationships_present(self):
vserver_client = mock.Mock()
vserver_client.volume_has_luns = mock.Mock(return_value=False)
vserver_client.volume_has_junctioned_volumes = mock.Mock(
return_value=False)
vserver_client.volume_has_snapmirror_relationships = mock.Mock(
return_value=True)
self.assertRaises(exception.ManageInvalidShare, self.assertRaises(exception.ManageInvalidShare,
self.library._validate_volume_for_manage, self.library._validate_volume_for_manage,

View File

@ -728,7 +728,8 @@ class ShareAPITestCase(test.TestCase):
self.context, request_spec=mock.ANY, filter_properties={}) self.context, request_spec=mock.ANY, filter_properties={})
self.assertFalse(self.api.share_rpcapi.create_share_instance.called) self.assertFalse(self.api.share_rpcapi.create_share_instance.called)
def test_manage_new(self): @ddt.data('dr', 'readable', None)
def test_manage_new(self, replication_type):
share_data = { share_data = {
'host': 'fake', 'host': 'fake',
'export_location': 'fake', 'export_location': 'fake',
@ -746,6 +747,7 @@ class ShareAPITestCase(test.TestCase):
'id': 'fake_type_id', 'id': 'fake_type_id',
'extra_specs': { 'extra_specs': {
'snapshot_support': False, 'snapshot_support': False,
'replication_type': replication_type,
}, },
} }
@ -770,6 +772,7 @@ class ShareAPITestCase(test.TestCase):
'status': constants.STATUS_MANAGING, 'status': constants.STATUS_MANAGING,
'scheduled_at': date, 'scheduled_at': date,
'snapshot_support': fake_type['extra_specs']['snapshot_support'], 'snapshot_support': fake_type['extra_specs']['snapshot_support'],
'replication_type': replication_type,
}) })
expected_request_spec = self._get_request_spec_dict( expected_request_spec = self._get_request_spec_dict(
@ -800,6 +803,7 @@ class ShareAPITestCase(test.TestCase):
'id': 'fake_type_id', 'id': 'fake_type_id',
'extra_specs': { 'extra_specs': {
'snapshot_support': False, 'snapshot_support': False,
'replication_type': 'writable',
}, },
} }
@ -969,6 +973,121 @@ class ShareAPITestCase(test.TestCase):
db_api.share_snapshot_create.assert_called_once_with( db_api.share_snapshot_create.assert_called_once_with(
self.context, options) self.context, options)
def test_manage_snapshot_share_not_found(self):
snapshot = fakes.fake_snapshot(share_id='fake_share',
as_primitive=True)
mock_share_get_call = self.mock_object(
db_api, 'share_get', mock.Mock(side_effect=exception.NotFound))
mock_db_snapshot_call = self.mock_object(
db_api, 'share_snapshot_get_all_for_share')
self.assertRaises(exception.ShareNotFound,
self.api.manage_snapshot,
self.context,
snapshot,
{})
self.assertFalse(mock_db_snapshot_call.called)
mock_share_get_call.assert_called_once_with(
self.context, snapshot['share_id'])
def test_manage_snapshot_share_has_replicas(self):
share_ref = fakes.fake_share(
has_replicas=True, status=constants.STATUS_AVAILABLE)
self.mock_object(
db_api, 'share_get', mock.Mock(return_value=share_ref))
snapshot = fakes.fake_snapshot(create_instance=True, as_primitive=True)
mock_db_snapshot_get_all_for_share_call = self.mock_object(
db_api, 'share_snapshot_get_all_for_share')
self.assertRaises(exception.InvalidShare,
self.api.manage_snapshot,
context,
snapshot,
{})
self.assertFalse(mock_db_snapshot_get_all_for_share_call.called)
def test_manage_snapshot_already_managed(self):
share_ref = fakes.fake_share(
has_replicas=False, status=constants.STATUS_AVAILABLE)
snapshot = fakes.fake_snapshot(create_instance=True, as_primitive=True)
self.mock_object(
db_api, 'share_get', mock.Mock(return_value=share_ref))
mock_db_snapshot_call = self.mock_object(
db_api, 'share_snapshot_get_all_for_share', mock.Mock(
return_value=[snapshot]))
mock_db_snapshot_create_call = self.mock_object(
db_api, 'share_snapshot_create')
self.assertRaises(exception.ManageInvalidShareSnapshot,
self.api.manage_snapshot,
self.context,
snapshot,
{})
mock_db_snapshot_call.assert_called_once_with(
self.context, snapshot['share_id'])
self.assertFalse(mock_db_snapshot_create_call.called)
def test_manage_snapshot(self):
share_ref = fakes.fake_share(
has_replicas=False, status=constants.STATUS_AVAILABLE,
host='fake_host')
existing_snapshot = fakes.fake_snapshot(
create_instance=True, share_id=share_ref['id'])
self.mock_object(db_api, 'share_snapshot_get_all_for_share',
mock.Mock(return_value=[existing_snapshot]))
snapshot_data = {
'share_id': share_ref['id'],
'provider_location': 'someproviderlocation',
}
expected_snapshot_data = {
'user_id': self.context.user_id,
'project_id': self.context.project_id,
'status': constants.STATUS_MANAGING,
'share_size': share_ref['size'],
'progress': '0%',
'share_proto': share_ref['share_proto'],
}
expected_snapshot_data.update(**snapshot_data)
snapshot = fakes.fake_snapshot(
create_instance=True, **expected_snapshot_data)
self.mock_object(
db_api, 'share_get', mock.Mock(return_value=share_ref))
mock_db_snapshot_create_call = self.mock_object(
db_api, 'share_snapshot_create', mock.Mock(return_value=snapshot))
mock_rpc_call = self.mock_object(self.share_rpcapi, 'manage_snapshot',
mock.Mock(return_value=snapshot))
new_snap = self.api.manage_snapshot(
self.context, snapshot_data, {})
self.assertEqual(new_snap, snapshot)
mock_db_snapshot_create_call.assert_called_once_with(
self.context, expected_snapshot_data)
mock_rpc_call.assert_called_once_with(
self.context, snapshot, share_ref['host'], {})
def test_unmanage_snapshot(self):
fake_host = 'fake_host'
snapshot_data = {
'status': constants.STATUS_UNMANAGING,
'terminated_at': timeutils.utcnow(),
}
snapshot = fakes.fake_snapshot(
create_instance=True, share_instance_id='id2', **snapshot_data)
mock_db_snap_update_call = self.mock_object(
db_api, 'share_snapshot_update', mock.Mock(return_value=snapshot))
mock_rpc_call = self.mock_object(
self.share_rpcapi, 'unmanage_snapshot')
retval = self.api.unmanage_snapshot(
self.context, snapshot, fake_host)
self.assertIsNone(retval)
mock_db_snap_update_call.assert_called_once_with(
self.context, snapshot['id'], snapshot_data)
mock_rpc_call.assert_called_once_with(
self.context, snapshot, fake_host)
def test_create_snapshot_for_replicated_share(self): def test_create_snapshot_for_replicated_share(self):
share = fakes.fake_share( share = fakes.fake_share(
has_replicas=True, status=constants.STATUS_AVAILABLE) has_replicas=True, status=constants.STATUS_AVAILABLE)

View File

@ -2136,10 +2136,13 @@ class ShareManagerTestCase(test.TestCase):
share['project_id'], {'shares': 1, 'gigabytes': 3}) share['project_id'], {'shares': 1, 'gigabytes': 3})
@ddt.data( @ddt.data(
{'size': 1}, {'size': 1, 'replication_type': None},
{'size': 2, 'name': 'fake'}, {'size': 2, 'name': 'fake', 'replication_type': 'dr'},
{'size': 3, 'export_locations': ['foo', 'bar', 'quuz']}) {'size': 3, 'export_locations': ['foo', 'bar', 'quuz'],
'replication_type': 'writable'},
)
def test_manage_share_valid_share(self, driver_data): def test_manage_share_valid_share(self, driver_data):
replication_type = driver_data.pop('replication_type')
export_locations = driver_data.get('export_locations') export_locations = driver_data.get('export_locations')
self.mock_object(self.share_manager.db, 'share_update', mock.Mock()) self.mock_object(self.share_manager.db, 'share_update', mock.Mock())
self.mock_object(self.share_manager, 'driver', mock.Mock()) self.mock_object(self.share_manager, 'driver', mock.Mock())
@ -2157,7 +2160,7 @@ class ShareManagerTestCase(test.TestCase):
self.mock_object(self.share_manager.driver, self.mock_object(self.share_manager.driver,
"manage_existing", "manage_existing",
mock.Mock(return_value=driver_data)) mock.Mock(return_value=driver_data))
share = db_utils.create_share() share = db_utils.create_share(replication_type=replication_type)
share_id = share['id'] share_id = share['id']
driver_options = {'fake': 'fake'} driver_options = {'fake': 'fake'}
@ -2175,6 +2178,8 @@ class ShareManagerTestCase(test.TestCase):
self.share_manager.db.share_export_locations_update.called) self.share_manager.db.share_export_locations_update.called)
valid_share_data = { valid_share_data = {
'status': constants.STATUS_AVAILABLE, 'launched_at': mock.ANY} 'status': constants.STATUS_AVAILABLE, 'launched_at': mock.ANY}
if replication_type:
valid_share_data['replica_state'] = constants.REPLICA_STATE_ACTIVE
valid_share_data.update(driver_data) valid_share_data.update(driver_data)
self.share_manager.db.share_update.assert_called_once_with( self.share_manager.db.share_update.assert_called_once_with(
utils.IsAMatcher(context.RequestContext), utils.IsAMatcher(context.RequestContext),

View File

@ -0,0 +1,189 @@
# 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.
from tempest import config
from tempest.lib.common.utils import data_utils
from tempest.lib import exceptions as lib_exc
from tempest import test
import testtools
from manila_tempest_tests import clients_share as clients
from manila_tempest_tests.common import constants
from manila_tempest_tests import share_exceptions
from manila_tempest_tests.tests.api import base
CONF = config.CONF
_MIN_SUPPORTED_MICROVERSION = '2.11'
@testtools.skipUnless(CONF.share.run_replication_tests,
'Replication tests are disabled.')
@testtools.skipIf(
CONF.share.multitenancy_enabled,
"Only for driver_handles_share_servers = False driver mode.")
@base.skip_if_microversion_lt(_MIN_SUPPORTED_MICROVERSION)
class ReplicationAdminTest(base.BaseSharesAdminTest):
@classmethod
def resource_setup(cls):
super(ReplicationAdminTest, cls).resource_setup()
# Create share_type
name = data_utils.rand_name(constants.TEMPEST_MANILA_PREFIX)
cls.admin_client = clients.AdminManager().shares_v2_client
cls.replication_type = CONF.share.backend_replication_type
if cls.replication_type not in constants.REPLICATION_TYPE_CHOICES:
raise share_exceptions.ShareReplicationTypeException(
replication_type=cls.replication_type
)
cls.zones = cls.get_availability_zones(client=cls.admin_client)
cls.share_zone = cls.zones[0]
cls.replica_zone = cls.zones[-1]
cls.extra_specs = cls.add_required_extra_specs_to_dict(
{"replication_type": cls.replication_type})
share_type = cls.create_share_type(
name,
cleanup_in_class=True,
extra_specs=cls.extra_specs,
client=cls.admin_client)
cls.share_type = share_type["share_type"]
# Create share with above share_type
cls.share = cls.create_share(size=2,
share_type_id=cls.share_type["id"],
availability_zone=cls.share_zone,)
cls.replica = cls.shares_v2_client.list_share_replicas(
share_id=cls.share['id'])[0]
@test.attr(type=["gate"])
@testtools.skipUnless(CONF.share.run_extend_tests,
'Extend share tests are disabled.')
def test_extend_replicated_share(self):
# Test extend share
new_size = self.share["size"] + 1
self.shares_v2_client.extend_share(self.share["id"], new_size)
self.shares_v2_client.wait_for_share_status(self.share["id"],
"available")
share = self.shares_v2_client.get_share(self.share["id"])
self.assertEqual(new_size, int(share["size"]))
@test.attr(type=["gate"])
@testtools.skipUnless(CONF.share.run_shrink_tests,
'Shrink share tests are disabled.')
def test_shrink_replicated_share(self):
share = self.shares_v2_client.get_share(self.share["id"])
new_size = self.share["size"] - 1
self.shares_v2_client.shrink_share(self.share["id"], new_size)
self.shares_v2_client.wait_for_share_status(share["id"], "available")
shrink_share = self.shares_v2_client.get_share(self.share["id"])
self.assertEqual(new_size, int(shrink_share["size"]))
@test.attr(type=["gate", "positive"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_tests,
'Manage/Unmanage Tests are disabled.')
def test_manage_share_for_replication_type(self):
"""Manage a share with replication share type."""
# Create a share and unmanage it
share = self.create_share(size=2,
share_type_id=self.share_type["id"],
availability_zone=self.share_zone,
cleanup_in_class=True)
share = self.shares_v2_client.get_share(share["id"])
export_locations = self.shares_v2_client.list_share_export_locations(
share["id"])
export_path = export_locations[0]['path']
self.shares_v2_client.unmanage_share(share['id'])
self.shares_v2_client.wait_for_resource_deletion(share_id=share['id'])
# Manage the previously unmanaged share
managed_share = self.shares_v2_client.manage_share(
share['host'], share['share_proto'],
export_path, self.share_type['id'])
self.shares_v2_client.wait_for_share_status(
managed_share['id'], 'available')
# Add managed share to cleanup queue
self.method_resources.insert(
0, {'type': 'share', 'id': managed_share['id'],
'client': self.shares_v2_client})
# Make sure a replica can be added to newly managed share
self.create_share_replica(managed_share['id'], self.replica_zone,
cleanup=True)
@test.attr(type=["gate", "negative"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_tests,
'Manage/Unmanage Tests are disabled.')
def test_unmanage_replicated_share_with_replica(self):
"""Try to unmanage a share having replica."""
# Create a share replica before unmanaging the share
self.create_share_replica(self.share["id"], self.replica_zone,
cleanup=True)
self.assertRaises(
lib_exc.Conflict,
self.shares_v2_client.unmanage_share,
share_id=self.share['id'])
@test.attr(type=["gate", "positive"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_tests,
'Manage/Unmanage Tests are disabled.')
def test_unmanage_replicated_share_with_no_replica(self):
"""Unmanage a replication type share that does not have replica."""
share = self.create_share(size=2,
share_type_id=self.share_type["id"],
availability_zone=self.share_zone,)
self.shares_v2_client.unmanage_share(share['id'])
self.shares_v2_client.wait_for_resource_deletion(share_id=share['id'])
@test.attr(type=["gate", "negative"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_snapshot_tests,
'Manage/Unmanage Snapshot Tests are disabled.')
def test_manage_replicated_share_snapshot(self):
"""Try to manage a snapshot of the replicated."""
# Create a share replica before managing the snapshot
self.create_share_replica(self.share["id"], self.replica_zone,
cleanup=True)
self.assertRaises(
lib_exc.Conflict,
self.shares_v2_client.manage_snapshot,
share_id=self.share['id'],
provider_location="127.0.0.1:/fake_provider_location/"
"manila_share_9dc61f49_fbc8_48d7_9337_2f9593d9")
@test.attr(type=["gate", "negative"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_snapshot_tests,
'Manage/Unmanage Snapshot Tests are disabled.')
def test_unmanage_replicated_share_snapshot(self):
"""Try to unmanage a snapshot of the replicated share with replica."""
# Create a share replica before unmanaging the snapshot
self.create_share_replica(self.share["id"], self.replica_zone,
cleanup=True)
snapshot = self.create_snapshot_wait_for_active(self.share["id"])
self.assertRaises(
lib_exc.Conflict,
self.shares_v2_client.unmanage_snapshot,
snapshot_id=snapshot['id'])
@test.attr(type=["gate", "positive"])
@testtools.skipUnless(CONF.share.run_manage_unmanage_snapshot_tests,
'Manage/Unmanage Snapshot Tests are disabled.')
def test_unmanage_replicated_share_snapshot_with_no_replica(self):
"""Unmanage a snapshot of the replicated share with no replica."""
share = self.create_share(size=2,
share_type_id=self.share_type["id"],
availability_zone=self.share_zone,)
snapshot = self.create_snapshot_wait_for_active(share["id"])
self.shares_v2_client.unmanage_snapshot(snapshot_id=snapshot['id'])
self.shares_v2_client.wait_for_resource_deletion(
snapshot_id=snapshot['id'])

View File

@ -0,0 +1,12 @@
---
features:
- Share can be managed with replication_type extra-spec
in the share_type
issues:
- Managing a share with replication_type can only be
possible if the share does not already have replicas.
fixes:
- Retrying to manage shares in ``manage_error`` status
works as expected.
- Snapshot manage and unmange operations are disabled
for shares with replicas.