ba45f40e12
The end user can soft delete or restore share by manila client. and can list shares in recycle bin. update micversion to 2.69 Depends-On: Ic838eec5fea890be6513514053329b1d2d86b3ba Partially-Implements: blueprint manila-share-support-recycle-bin Change-Id: Ic4748b59e2d99388c832969b7bef537b4115e4b1
307 lines
12 KiB
Python
307 lines
12 KiB
Python
# Copyright 2015 Mirantis Inc.
|
|
# All Rights Reserved.
|
|
#
|
|
# 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.
|
|
|
|
import ddt
|
|
import testtools
|
|
|
|
from tempest.lib.common.utils import data_utils
|
|
from tempest.lib import exceptions
|
|
|
|
from manilaclient.common import constants
|
|
from manilaclient import config
|
|
from manilaclient.tests.functional import base
|
|
from manilaclient.tests.functional import utils
|
|
|
|
CONF = config.CONF
|
|
|
|
|
|
@ddt.ddt
|
|
class SharesReadWriteBase(base.BaseTestCase):
|
|
protocol = None
|
|
|
|
def setUp(self):
|
|
super(SharesReadWriteBase, self).setUp()
|
|
if self.protocol not in CONF.enable_protocols:
|
|
message = "%s tests are disabled" % self.protocol
|
|
raise self.skipException(message)
|
|
self.name = data_utils.rand_name('autotest_share_name')
|
|
self.description = data_utils.rand_name('autotest_share_description')
|
|
|
|
# NOTE(vponomaryov): following share is used only in one test
|
|
# until tests for snapshots appear.
|
|
self.share = self.create_share(
|
|
share_protocol=self.protocol,
|
|
size=1,
|
|
name=self.name,
|
|
description=self.description,
|
|
client=self.get_user_client())
|
|
|
|
def test_create_delete_share(self):
|
|
name = data_utils.rand_name('autotest_share_name')
|
|
|
|
create = self.create_share(
|
|
self.protocol, name=name, client=self.user_client)
|
|
|
|
self.assertEqual("creating", create['status'])
|
|
self.assertEqual(name, create['name'])
|
|
self.assertEqual('1', create['size'])
|
|
self.assertEqual(self.protocol.upper(), create['share_proto'])
|
|
|
|
self.user_client.delete_share(create['id'])
|
|
|
|
self.user_client.wait_for_share_deletion(create['id'])
|
|
|
|
def test_create_update_share(self):
|
|
name = data_utils.rand_name('autotest_share_name')
|
|
new_name = 'new_' + name
|
|
description = data_utils.rand_name('autotest_share_description')
|
|
new_description = 'new_' + description
|
|
|
|
create = self.create_share(
|
|
self.protocol, name=name, description=description,
|
|
client=self.user_client)
|
|
|
|
self.assertEqual(name, create['name'])
|
|
self.assertEqual(description, create['description'])
|
|
self.assertEqual('False', create['is_public'])
|
|
|
|
self.user_client.update_share(
|
|
create['id'], name=new_name, description=new_description)
|
|
get = self.user_client.get_share(create['id'])
|
|
|
|
self.assertEqual(new_name, get['name'])
|
|
self.assertEqual(new_description, get['description'])
|
|
self.assertEqual('False', get['is_public'])
|
|
|
|
# only admins operating at system scope can set a share to be public
|
|
self.admin_client.update_share(create['id'], is_public=True)
|
|
get = self.user_client.get_share(create['id'])
|
|
|
|
self.assertEqual(new_name, get['name'])
|
|
self.assertEqual(new_description, get['description'])
|
|
self.assertEqual('True', get['is_public'])
|
|
|
|
def test_get_share(self):
|
|
get = self.user_client.get_share(self.share['id'])
|
|
|
|
self.assertEqual(self.name, get['name'])
|
|
self.assertEqual(self.description, get['description'])
|
|
self.assertEqual('1', get['size'])
|
|
self.assertEqual(self.protocol.upper(), get['share_proto'])
|
|
|
|
def test_create_delete_with_wait(self):
|
|
name = data_utils.rand_name('share-with-wait-%s')
|
|
description = data_utils.rand_name('we-wait-until-share-is-ready')
|
|
|
|
share_1, share_2 = (
|
|
self.create_share(self.protocol, name=(name % num),
|
|
description=description,
|
|
use_wait_option=True,
|
|
client=self.user_client)
|
|
for num in range(0, 2)
|
|
)
|
|
|
|
self.assertEqual("available", share_1['status'])
|
|
self.assertEqual("available", share_2['status'])
|
|
|
|
self.delete_share([share_1['id'], share_2['id']],
|
|
wait=True,
|
|
client=self.user_client)
|
|
|
|
for share in (share_1, share_2):
|
|
self.assertRaises(exceptions.NotFound,
|
|
self.user_client.get_share,
|
|
share['id'])
|
|
|
|
def test_create_soft_delete_and_restore_share(self):
|
|
self.skip_if_microversion_not_supported('2.69')
|
|
microversion = '2.69'
|
|
description = data_utils.rand_name('we-wait-until-share-is-ready')
|
|
|
|
share = self.create_share(self.protocol,
|
|
name='share_name',
|
|
description=description,
|
|
use_wait_option=True,
|
|
client=self.user_client)
|
|
|
|
self.assertEqual("available", share['status'])
|
|
|
|
# soft delete the share to recycle bin
|
|
self.soft_delete_share([share['id']], client=self.user_client,
|
|
microversion=microversion)
|
|
self.user_client.wait_for_share_soft_deletion(share['id'])
|
|
|
|
# get shares list in recycle bin
|
|
result = self.user_client.list_shares(is_soft_deleted=True,
|
|
microversion=microversion)
|
|
share_ids = [sh['ID'] for sh in result]
|
|
# check share is in recycle bin
|
|
self.assertIn(share['id'], share_ids)
|
|
|
|
# restore the share from recycle bin
|
|
self.restore_share([share['id']], client=self.user_client,
|
|
microversion=microversion)
|
|
self.user_client.wait_for_share_restore(share['id'])
|
|
|
|
result1 = self.user_client.list_shares(is_soft_deleted=True,
|
|
microversion=microversion)
|
|
share_ids1 = [sh['ID'] for sh in result1]
|
|
# check share not in recycle bin
|
|
self.assertNotIn(share['id'], share_ids1)
|
|
|
|
|
|
@ddt.ddt
|
|
class SharesTestMigration(base.BaseTestCase):
|
|
|
|
def setUp(self):
|
|
super(SharesTestMigration, self).setUp()
|
|
|
|
self.old_type = self.create_share_type(
|
|
data_utils.rand_name('test_share_type'),
|
|
driver_handles_share_servers=True)
|
|
self.new_type = self.create_share_type(
|
|
data_utils.rand_name('test_share_type'),
|
|
driver_handles_share_servers=True)
|
|
self.error_type = self.create_share_type(
|
|
data_utils.rand_name('test_share_type'),
|
|
driver_handles_share_servers=True,
|
|
extra_specs={'cause_error': 'no_valid_host'})
|
|
|
|
self.old_share_net = self.get_user_client().get_share_network(
|
|
self.get_user_client().share_network)
|
|
share_net_info = (
|
|
utils.get_default_subnet(self.get_user_client(),
|
|
self.old_share_net['id'])
|
|
if utils.share_network_subnets_are_supported()
|
|
else self.old_share_net)
|
|
self.new_share_net = self.create_share_network(
|
|
neutron_net_id=share_net_info['neutron_net_id'],
|
|
neutron_subnet_id=share_net_info['neutron_subnet_id'])
|
|
|
|
@utils.skip_if_microversion_not_supported('2.22')
|
|
@ddt.data('migration_error', 'migration_success', 'None')
|
|
def test_reset_task_state(self, state):
|
|
|
|
share = self.create_share(
|
|
share_protocol='nfs',
|
|
size=1,
|
|
name=data_utils.rand_name('autotest_share_name'),
|
|
client=self.get_user_client(),
|
|
share_type=self.old_type['ID'],
|
|
share_network=self.old_share_net['id'],
|
|
wait_for_creation=True)
|
|
share = self.user_client.get_share(share['id'])
|
|
|
|
self.admin_client.reset_task_state(share['id'], state)
|
|
share = self.user_client.get_share(share['id'])
|
|
self.assertEqual(state, share['task_state'])
|
|
|
|
@utils.skip_if_microversion_not_supported('2.29')
|
|
@testtools.skipUnless(
|
|
CONF.run_migration_tests, 'Share migration tests are disabled.')
|
|
@ddt.data('cancel', 'success', 'error')
|
|
def test_full_migration(self, test_type):
|
|
# We are testing with DHSS=True only because it allows us to specify
|
|
# new_share_network.
|
|
|
|
share = self.create_share(
|
|
share_protocol='nfs',
|
|
size=1,
|
|
name=data_utils.rand_name('autotest_share_name'),
|
|
client=self.get_user_client(),
|
|
share_type=self.old_type['ID'],
|
|
share_network=self.old_share_net['id'],
|
|
wait_for_creation=True)
|
|
share = self.admin_client.get_share(share['id'])
|
|
|
|
pools = self.admin_client.pool_list(detail=True)
|
|
|
|
dest_pool = utils.choose_matching_backend(
|
|
share, pools, self.new_type)
|
|
|
|
self.assertIsNotNone(dest_pool)
|
|
|
|
source_pool = share['host']
|
|
|
|
new_type = self.new_type
|
|
if test_type == 'error':
|
|
statuses = constants.TASK_STATE_MIGRATION_ERROR
|
|
new_type = self.error_type
|
|
else:
|
|
statuses = (constants.TASK_STATE_MIGRATION_DRIVER_PHASE1_DONE,
|
|
constants.TASK_STATE_DATA_COPYING_COMPLETED)
|
|
|
|
self.admin_client.migration_start(
|
|
share['id'], dest_pool, writable=True, nondisruptive=False,
|
|
preserve_metadata=True, preserve_snapshots=True,
|
|
force_host_assisted_migration=False,
|
|
new_share_network=self.new_share_net['id'],
|
|
new_share_type=new_type['ID'])
|
|
|
|
share = self.admin_client.wait_for_migration_task_state(
|
|
share['id'], dest_pool, statuses)
|
|
|
|
progress = self.admin_client.migration_get_progress(share['id'])
|
|
self.assertEqual('100', progress['total_progress'])
|
|
|
|
self.assertEqual(source_pool, share['host'])
|
|
self.assertEqual(self.old_type['ID'], share['share_type'])
|
|
self.assertEqual(self.old_share_net['id'], share['share_network_id'])
|
|
|
|
if test_type == 'error':
|
|
self.assertEqual(statuses, progress['task_state'])
|
|
else:
|
|
if test_type == 'success':
|
|
self.admin_client.migration_complete(share['id'])
|
|
statuses = constants.TASK_STATE_MIGRATION_SUCCESS
|
|
elif test_type == 'cancel':
|
|
self.admin_client.migration_cancel(share['id'])
|
|
statuses = constants.TASK_STATE_MIGRATION_CANCELLED
|
|
|
|
share = self.admin_client.wait_for_migration_task_state(
|
|
share['id'], dest_pool, statuses)
|
|
progress = self.admin_client.migration_get_progress(share['id'])
|
|
self.assertEqual(statuses, progress['task_state'])
|
|
if test_type == 'success':
|
|
self.assertEqual(dest_pool, share['host'])
|
|
self.assertEqual(new_type['ID'], share['share_type'])
|
|
self.assertEqual(self.new_share_net['id'],
|
|
share['share_network_id'])
|
|
else:
|
|
self.assertEqual(source_pool, share['host'])
|
|
self.assertEqual(self.old_type['ID'], share['share_type'])
|
|
self.assertEqual(self.old_share_net['id'],
|
|
share['share_network_id'])
|
|
|
|
|
|
class NFSSharesReadWriteTest(SharesReadWriteBase):
|
|
protocol = 'nfs'
|
|
|
|
|
|
class CIFSSharesReadWriteTest(SharesReadWriteBase):
|
|
protocol = 'cifs'
|
|
|
|
|
|
class GlusterFSSharesReadWriteTest(SharesReadWriteBase):
|
|
protocol = 'glusterfs'
|
|
|
|
|
|
class HDFSSharesReadWriteTest(SharesReadWriteBase):
|
|
protocol = 'hdfs'
|
|
|
|
|
|
class MAPRFSSharesReadWriteTest(SharesReadWriteBase):
|
|
protocol = 'maprfs'
|