python-manilaclient/manilaclient/tests/functional/osc/base.py
2023-08-31 14:30:51 +00:00

475 lines
17 KiB
Python

# 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 json
import time
from tempest.lib.cli import base
from tempest.lib.common.utils import data_utils
from manilaclient import config
CONF = config.CONF
class OSCClientTestBase(base.ClientTestBase):
"""Base class for OSC manila functional tests"""
@classmethod
def get_admin_client(cls):
admin_client = base.CLIClient(
username=CONF.admin_username,
password=CONF.admin_password,
tenant_name=CONF.admin_tenant_name,
uri=CONF.admin_auth_url,
cli_dir=CONF.manila_exec_dir,
insecure=CONF.insecure,
project_domain_name=CONF.admin_project_domain_name or None,
project_domain_id=CONF.admin_project_domain_id or None,
user_domain_name=CONF.admin_user_domain_name or None,
user_domain_id=CONF.admin_user_domain_id or None
)
return admin_client
@classmethod
def get_user_client(cls):
user_client = base.CLIClient(
username=CONF.username,
password=CONF.password,
tenant_name=CONF.tenant_name,
uri=CONF.auth_url,
cli_dir=CONF.manila_exec_dir,
insecure=CONF.insecure,
project_domain_name=CONF.project_domain_name or None,
project_domain_id=CONF.project_domain_id or None,
user_domain_name=CONF.user_domain_name or None,
user_domain_id=CONF.user_domain_id or None
)
return user_client
@property
def admin_client(self):
if not hasattr(self, '_admin_client'):
self._admin_client = self.get_admin_client()
return self._admin_client
@property
def user_client(self):
if not hasattr(self, '_user_client'):
self._user_client = self.get_user_client()
return self._user_client
def _get_clients(self):
return self.admin_client
def _get_property_from_output(self, output):
"""Creates a dictionary from the given output"""
obj = {}
items = self.parser.listing(output)
for item in items:
obj[item['Field']] = str(item['Value'])
return obj
def _wait_for_object_status(self, object_name, object_id, status,
timeout=CONF.build_timeout,
interval=CONF.build_interval):
"""Waits for a object to reach a given status."""
start_time = time.time()
while time.time() - start_time < timeout:
if status == self.openstack(
'%(obj)s show -c status -f value %(id)s'
% {'obj': object_name,
'id': object_id}).rstrip():
break
time.sleep(interval)
else:
self.fail("%s %s did not reach status %s after %d seconds."
% (object_name, object_id, status, timeout))
def check_object_deleted(self, object_name, object_id,
timeout=CONF.build_timeout,
interval=CONF.build_interval):
"""Check that object deleted successfully"""
cmd = '%s list -c ID -f value' % object_name
start_time = time.time()
while time.time() - start_time < timeout:
if object_id not in self.openstack(cmd):
break
time.sleep(interval)
else:
self.fail("%s %s not deleted after %d seconds."
% (object_name, object_id, timeout))
def openstack(self, action, flags='', params='', fail_ok=False,
merge_stderr=False, client=None):
"""Executes openstack command for given action"""
if '--os-share-api-version' not in flags:
flags = (
flags + '--os-share-api-version %s'
% CONF.max_api_microversion)
if client is None:
client = self.admin_client
return client.openstack(action, flags=flags, params=params,
fail_ok=fail_ok,
merge_stderr=merge_stderr)
def listing_result(self, object_name, command, client=None):
"""Returns output for the given command as list of dictionaries"""
output = self.openstack(object_name, params=command, client=client)
result = self.parser.listing(output)
return result
def dict_result(self, object_name, command, client=None):
"""Returns output for the given command as dictionary"""
output = self.openstack(object_name, params=command, client=client)
result_dict = self._get_property_from_output(output)
return result_dict
def create_share(self, share_protocol=None, size=None, name=None,
snapshot_id=None, properties=None, share_network=None,
description=None, public=False, share_type=None,
availability_zone=None, share_group=None,
add_cleanup=True, client=None,
wait_for_status='available'):
name = name or data_utils.rand_name('autotest_share_name')
# share_type = dhss_false until we have implemented
# share network commands for osc
share_type = share_type or 'dhss_false'
cmd = ('create '
'%(protocol)s %(size)s %(name)s %(desc)s %(public)s %(stype)s'
% {'protocol': share_protocol or 'NFS',
'size': size or '1',
'name': '--name %s' % name,
'desc': '--description %s' % description,
'public': '--public %s' % public,
'stype': '--share-type %s' % share_type})
if snapshot_id:
cmd = cmd + ' --snapshot-id %s' % snapshot_id
if properties:
for key, value in properties.items():
cmd = (cmd + ' --property %(key)s=%(value)s'
% {'key': key, 'value': value})
if share_network:
cmd = cmd + ' --share-network %s' % share_network
if availability_zone:
cmd = cmd + ' --availability-zone %s' % availability_zone
if share_group:
cmd = cmd + ' --share-group %s' % share_group
share_object = self.dict_result('share', cmd, client=client)
self._wait_for_object_status(
'share', share_object['id'], wait_for_status)
if add_cleanup:
self.addCleanup(
self.openstack, 'share delete %s --wait' % share_object['id']
)
return share_object
def list_pools(self, backend=None, host=None, pool=None, detail=False):
cmd = 'pool list '
if backend:
cmd += f'--backend {backend} '
if pool:
cmd += f'--pool {pool} '
if host:
cmd += f'--host {host} '
if detail:
cmd += '--detail'
pools = self.listing_result('share', cmd)
return pools
def create_share_type(self, name=None, dhss=False, description=None,
snapshot_support=None,
create_share_from_snapshot_support=None,
revert_to_snapshot_support=False,
mount_snapshot_support=False, extra_specs={},
public=True, add_cleanup=True, client=None,
formatter=None):
name = name or data_utils.rand_name('autotest_share_type_name')
cmd = (f'create {name} {dhss} --public {public}')
if description:
cmd += f' --description {description}'
if snapshot_support:
cmd += f' --snapshot-support {snapshot_support}'
if create_share_from_snapshot_support:
cmd += (' --create-share-from-snapshot-support '
f'{create_share_from_snapshot_support}')
if revert_to_snapshot_support:
cmd += (' --revert-to-snapshot-support '
f' {revert_to_snapshot_support}')
if mount_snapshot_support:
cmd += f' --mount-snapshot-support {mount_snapshot_support}'
if extra_specs:
specs = ' --extra-specs'
for key, value in extra_specs.items():
specs += f' {key}={value}'
cmd += specs
if formatter == 'json':
cmd = f'share type {cmd} -f {formatter} '
share_type = json.loads(self.openstack(cmd, client=client))
else:
share_type = self.dict_result('share type', cmd, client=client)
if add_cleanup:
self.addCleanup(
self.openstack, f'share type delete {share_type["id"]}'
)
return share_type
def list_services(self, host=None, status=None, state=None, zone=None):
cmd = 'service list '
if host:
cmd += f'--host {host} '
if status:
cmd += f'--status {status} '
if state:
cmd += f'--state {state} '
if zone:
cmd += f'--zone {zone} '
services = self.listing_result('share', cmd)
return services
def create_share_access_rule(self, share, access_type,
access_to, properties=None,
access_level=None, wait=False,
lock_visibility=False,
lock_deletion=False,
lock_reason=None,
add_cleanup=False):
cmd = f'access create {share} {access_type} {access_to} '
if access_level:
cmd += f'--access-level {access_level} '
if properties:
cmd += f'--properties {properties} '
if wait:
cmd += '--wait '
if lock_visibility:
cmd += '--lock-visibility '
if lock_deletion:
cmd += '--lock-deletion '
if lock_reason:
cmd += f'--lock-reason {lock_reason}'
access_rule = self.dict_result('share', cmd)
return access_rule
def get_share_export_locations(self, share):
cmd = (f'export location list {share}')
export_locations = json.loads(self.openstack(f'share {cmd} -f json'))
return export_locations
def create_snapshot(self, share, name=None,
description=None, wait=True, force=None,
add_cleanup=True, client=None):
name = name or data_utils.rand_name('autotest_snapshot_name')
cmd = (f'snapshot create {share} --name {name} ')
if description:
cmd += f' --description {description}'
if wait:
cmd += ' --wait'
if force:
cmd += ' --force'
snapshot_object = self.dict_result('share', cmd, client=client)
if add_cleanup:
self.addCleanup(
self.openstack,
f'share snapshot delete {snapshot_object["id"]} --wait')
return snapshot_object
def create_share_transfer(self, share, name=None, client=None):
name = name or data_utils.rand_name('autotest_share_transfer_name')
cmd = (f'transfer create {share} --name {name} ')
transfer_object = self.dict_result('share', cmd, client=client)
return transfer_object
def create_share_network(self, neutron_net_id=None,
neutron_subnet_id=None, name=None,
description=None, availability_zone=None,
add_cleanup=True):
name = name or data_utils.rand_name('autotest_share_network_name')
cmd = (f'network create --name {name} --description {description}')
if neutron_net_id:
cmd = cmd + f' --neutron-net-id {neutron_net_id}'
if neutron_subnet_id:
cmd = cmd + f' --neutron-subnet-id {neutron_subnet_id}'
if availability_zone:
cmd = cmd + f' --availability-zone {availability_zone}'
share_network_obj = self.dict_result('share', cmd)
self._wait_for_object_status(
'share network', share_network_obj['id'], 'active')
if add_cleanup:
self.addCleanup(
self.openstack,
f'share network delete {share_network_obj["id"]}'
)
return share_network_obj
def create_share_replica(self, share, availability_zone=None,
share_network=None, wait=None,
add_cleanup=True):
cmd = (f'replica create {share}')
if availability_zone:
cmd = cmd + f' --availability-zone {availability_zone}'
if wait:
cmd = cmd + ' --wait'
if share_network:
cmd = cmd + ' --share-network %s' % share_network
replica_object = self.dict_result('share', cmd)
self._wait_for_object_status(
'share replica', replica_object['id'], 'available')
if add_cleanup:
self.addCleanup(
self.openstack,
f'share replica delete {replica_object["id"]} --wait'
)
return replica_object
def get_share_replica_export_locations(self, replica):
cmd = (f'replica export location list {replica}')
export_locations = self.listing_result('share', cmd)
return export_locations
def create_share_group_type(self, name=None, share_types=None,
group_specs=None, public=True,
add_cleanup=True):
name = name or data_utils.rand_name('autotest_share_group_types_name')
share_types = share_types or 'None'
cmd = (f'group type create '
f'{name} '
f'{share_types} ')
if group_specs:
cmd = cmd + f' --group-specs {group_specs} '
if not public:
cmd = cmd + f' --public {public} '
share_object = self.dict_result('share', cmd)
if add_cleanup:
self.addCleanup(
self.openstack,
'share group type delete %s' % share_object['id'])
return share_object
def share_group_type_access_create(self, group_type, project):
cmd = (f'group type access create '
f'{group_type} '
f'{project} ')
self.dict_result('share', cmd)
def share_group_type_access_delete(self, group_type, access_id):
cmd = (f'group type access delete '
f'{group_type} '
f'{access_id} ')
self.dict_result('share', cmd)
def check_create_network_subnet(self, share_network, neutron_net_id=None,
neutron_subnet_id=None,
availability_zone=None,
restart_check=None):
cmd = f'network subnet create {share_network} --check-only'
if neutron_net_id:
cmd += f' --neutron-net-id {neutron_net_id}'
if neutron_subnet_id:
cmd += f' --neutron-subnet-id {neutron_subnet_id}'
if availability_zone:
cmd += f' --availability-zone {availability_zone}'
if restart_check:
cmd += f' --restart-check'
check_result = self.dict_result('share', cmd)
return check_result
def create_resource_lock(self, resource_id, resource_type='share',
resource_action='delete', lock_reason=None,
add_cleanup=True, client=None):
cmd = f'lock create {resource_id} {resource_type}'
cmd += f' --resource-action {resource_action}'
if lock_reason:
cmd += f' --reason "{lock_reason}"'
lock = self.dict_result('share', cmd, client=client)
if add_cleanup:
self.addCleanup(self.openstack,
'share lock delete %s' % lock['id'],
client=client)
return lock
def create_backup(self, share_id, name=None, description=None,
backup_options=None, add_cleanup=True):
name = name or data_utils.rand_name('autotest_backup_name')
cmd = (f'backup create {share_id} ')
if name:
cmd += f' --name {name}'
if description:
cmd += f' --description {description}'
if backup_options:
options = ' --backup-options'
for key, value in backup_options.items():
options += f' {key}={value}'
cmd += options
backup_object = self.dict_result('share', cmd)
self._wait_for_object_status(
'share backup', backup_object['id'], 'available')
if add_cleanup:
self.addCleanup(
self.openstack,
f'share backup delete {backup_object["id"]} --wait')
return backup_object