[Manila] pre-upgrade resource creation and post-procedure validation
There's a need to create a Manila share resource before the upgrade process and then validate the share after the process is complete. To achieve this objective, it is essential to incorporate support for the Manila client, a task that is accomplished through this commit. Change-Id: I0d520e40a1a491ee864e707d63413d10035b99ca
This commit is contained in:
parent
1449078813
commit
5d04e37ab0
@ -25,6 +25,7 @@ python-designateclient==4.4.0
|
||||
python-glanceclient==3.2.2
|
||||
python-heatclient==2.3.0
|
||||
python-ironicclient==4.6.1
|
||||
python-manilaclient==4.5.1
|
||||
python-neutronclient==7.2.1
|
||||
python-novaclient==17.2.1
|
||||
python-octaviaclient==2.2.0
|
||||
|
@ -0,0 +1,4 @@
|
||||
---
|
||||
features:
|
||||
- |
|
||||
Added Manila support and testing. Now Manila tests can be run on tobiko as well.
|
@ -22,6 +22,7 @@ python-designateclient>=4.4.0 # Apache-2.0
|
||||
python-glanceclient>=3.2.2 # Apache-2.0
|
||||
python-heatclient>=2.3.0 # Apache-2.0
|
||||
python-ironicclient>=4.6.1 # Apache-2.0
|
||||
python-manilaclient>=4.5.1 # Apache-2.0
|
||||
python-neutronclient>=7.2.1 # Apache-2.0
|
||||
python-novaclient>=17.2.1 # Apache-2.0
|
||||
python-octaviaclient>=2.2.0 # Apache-2.0
|
||||
|
@ -38,3 +38,11 @@
|
||||
grep "^tobiko\." | \
|
||||
xargs -r {{ openstack_cmd }} image delete
|
||||
ignore_errors: yes
|
||||
|
||||
- name: "cleanup Manila shares created by Tobiko tests"
|
||||
shell: |
|
||||
source {{ stackrc_file }}
|
||||
openstack share list -f value -c 'Name' | \
|
||||
grep "^tobiko" | \
|
||||
xargs -r openstack share delete --force
|
||||
ignore_errors: yes
|
||||
|
@ -0,0 +1,8 @@
|
||||
---
|
||||
|
||||
test_workflow_steps:
|
||||
- tox_description: 'check manila resources'
|
||||
tox_envlist: manila
|
||||
tox_step_name: check_manila_resources
|
||||
tox_environment:
|
||||
TOBIKO_PREVENT_CREATE: yes
|
@ -0,0 +1,8 @@
|
||||
---
|
||||
|
||||
test_workflow_steps:
|
||||
- tox_description: 'create manila resources'
|
||||
tox_envlist: manila
|
||||
tox_step_name: create_manila_resources
|
||||
tox_environment:
|
||||
TOBIKO_PREVENT_CREATE: no
|
@ -30,6 +30,7 @@ LOG = log.getLogger(__name__)
|
||||
|
||||
CONFIG_MODULES = ['tobiko.common._case',
|
||||
'tobiko.openstack.glance.config',
|
||||
'tobiko.openstack.manila.config',
|
||||
'tobiko.openstack.keystone.config',
|
||||
'tobiko.openstack.neutron.config',
|
||||
'tobiko.openstack.nova.config',
|
||||
@ -418,3 +419,8 @@ def is_prevent_create() -> bool:
|
||||
def skip_if_prevent_create(reason='TOBIKO_PREVENT_CREATE is True'):
|
||||
return tobiko.skip_if(reason=reason,
|
||||
predicate=is_prevent_create)
|
||||
|
||||
|
||||
def skip_unless_prevent_create(reason='TOBIKO_PREVENT_CREATE is False'):
|
||||
return tobiko.skip_unless(reason=reason,
|
||||
predicate=is_prevent_create)
|
||||
|
44
tobiko/openstack/manila/__init__.py
Normal file
44
tobiko/openstack/manila/__init__.py
Normal file
@ -0,0 +1,44 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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 __future__ import absolute_import
|
||||
|
||||
from tobiko.openstack.manila import _client
|
||||
from tobiko.openstack.manila import _constants
|
||||
from tobiko.openstack.manila import _exceptions
|
||||
from tobiko.openstack.manila import _waiters
|
||||
|
||||
manila_client = _client.manila_client
|
||||
get_manila_client = _client.get_manila_client
|
||||
ManilaClientFixture = _client.ManilaClientFixture
|
||||
create_share = _client.create_share
|
||||
get_share = _client.get_share
|
||||
get_shares_by_name = _client.get_shares_by_name
|
||||
delete_share = _client.delete_share
|
||||
extend_share = _client.extend_share
|
||||
list_shares = _client.list_shares
|
||||
|
||||
# Waiters
|
||||
wait_for_share_status = _waiters.wait_for_share_status
|
||||
wait_for_resource_deletion = _waiters.wait_for_resource_deletion
|
||||
|
||||
# Exceptions
|
||||
ShareNotFound = _exceptions.ShareNotFound
|
||||
ShareReleaseFailed = _exceptions.ShareReleaseFailed
|
||||
|
||||
# Constants
|
||||
RESOURCE_STATUS = _constants.RESOURCE_STATUS
|
||||
STATUS_AVAILABLE = _constants.STATUS_AVAILABLE
|
||||
STATUS_ERROR = _constants.STATUS_ERROR
|
||||
STATUS_ERROR_DELETING = _constants.STATUS_ERROR_DELETING
|
||||
SHARE_NAME = _constants.SHARE_NAME
|
104
tobiko/openstack/manila/_client.py
Normal file
104
tobiko/openstack/manila/_client.py
Normal file
@ -0,0 +1,104 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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 __future__ import absolute_import
|
||||
|
||||
from manilaclient.v2 import client as manilaclient
|
||||
from manilaclient import exceptions
|
||||
from oslo_log import log
|
||||
|
||||
import tobiko
|
||||
from tobiko import config
|
||||
from tobiko.openstack import keystone
|
||||
from tobiko.openstack import _client
|
||||
from tobiko.openstack.manila import _exceptions
|
||||
|
||||
LOG = log.getLogger(__name__)
|
||||
CONF = config.CONF
|
||||
|
||||
|
||||
class ManilaClientFixture(_client.OpenstackClientFixture):
|
||||
|
||||
def init_client(self, session):
|
||||
return manilaclient.Client(session=session)
|
||||
|
||||
|
||||
class ManilaClientManager(_client.OpenstackClientManager):
|
||||
|
||||
def create_client(self, session):
|
||||
return ManilaClientFixture(session=session)
|
||||
|
||||
|
||||
CLIENTS = ManilaClientManager()
|
||||
|
||||
|
||||
@keystone.skip_if_missing_service(name='manila')
|
||||
def manila_client(obj=None):
|
||||
obj = obj or default_manila_client()
|
||||
if tobiko.is_fixture(obj):
|
||||
obj = tobiko.setup_fixture(obj).client
|
||||
return tobiko.check_valid_type(obj, manilaclient.Client)
|
||||
|
||||
|
||||
def default_manila_client():
|
||||
return get_manila_client()
|
||||
|
||||
|
||||
def get_manila_client(session=None, shared=True, init_client=None,
|
||||
manager=None):
|
||||
manager = manager or CLIENTS
|
||||
fixture = manager.get_client(session=session, shared=shared,
|
||||
init_client=init_client)
|
||||
return manila_client(fixture)
|
||||
|
||||
|
||||
def create_share(share_protocol=None, size=None, client=None, **kwargs):
|
||||
share_protocol = share_protocol or CONF.tobiko.manila.share_protocol
|
||||
share_size = size or CONF.tobiko.manila.size
|
||||
return manila_client(client).shares.create(
|
||||
share_proto=share_protocol, size=share_size, return_raw=True,
|
||||
**kwargs)
|
||||
|
||||
|
||||
def list_shares(client=None, **kwargs):
|
||||
return manila_client(client).shares.list(return_raw=True, **kwargs)
|
||||
|
||||
|
||||
def delete_share(share_id, client=None, **kwargs):
|
||||
try:
|
||||
manila_client(client).shares.delete(share_id, **kwargs)
|
||||
except exceptions.NotFound:
|
||||
LOG.debug(f'Share {share_id} was not found')
|
||||
return False
|
||||
else:
|
||||
LOG.debug(f'Share {share_id} was deleted successfully')
|
||||
return True
|
||||
|
||||
|
||||
def extend_share(share_id, new_size, client=None):
|
||||
return manila_client(client).shares.extend(share_id, new_size)
|
||||
|
||||
|
||||
def get_share(share_id, client=None):
|
||||
try:
|
||||
return manila_client(client).shares.get(share_id, return_raw=True)
|
||||
except exceptions.NotFound as ex:
|
||||
raise _exceptions.ShareNotFound(id=share_id) from ex
|
||||
|
||||
|
||||
def get_shares_by_name(share_name, client=None):
|
||||
share_list = list_shares(client=client)
|
||||
shares = [
|
||||
s for s in share_list if s['name'] == share_name
|
||||
]
|
||||
return shares
|
24
tobiko/openstack/manila/_constants.py
Normal file
24
tobiko/openstack/manila/_constants.py
Normal file
@ -0,0 +1,24 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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.
|
||||
|
||||
# Manila attributes
|
||||
RESOURCE_STATUS = 'status'
|
||||
|
||||
# Shares
|
||||
STATUS_AVAILABLE = 'available'
|
||||
STATUS_ERROR = 'error'
|
||||
STATUS_ERROR_DELETING = 'error_deleting'
|
||||
|
||||
# Manila resource names
|
||||
SHARE_NAME = "tobiko-manila-share"
|
24
tobiko/openstack/manila/_exceptions.py
Normal file
24
tobiko/openstack/manila/_exceptions.py
Normal file
@ -0,0 +1,24 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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 __future__ import absolute_import
|
||||
|
||||
import tobiko
|
||||
|
||||
|
||||
class ShareNotFound(tobiko.ObjectNotFound):
|
||||
message = "No such manila share {id!r}"
|
||||
|
||||
|
||||
class ShareReleaseFailed(tobiko.ObjectNotFound):
|
||||
message = "Share has 'error_deleting' status and can not be deleted"
|
93
tobiko/openstack/manila/_waiters.py
Normal file
93
tobiko/openstack/manila/_waiters.py
Normal file
@ -0,0 +1,93 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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 __future__ import absolute_import
|
||||
|
||||
import typing
|
||||
import time
|
||||
|
||||
from oslo_log import log
|
||||
from manilaclient import exceptions
|
||||
|
||||
import tobiko
|
||||
from tobiko.openstack.manila import _client
|
||||
from tobiko.openstack.manila import _constants
|
||||
from tobiko.openstack.manila import _exceptions
|
||||
|
||||
LOG = log.getLogger(__name__)
|
||||
|
||||
|
||||
def wait_for_status(object_id: str,
|
||||
status_key: str = _constants.RESOURCE_STATUS,
|
||||
status: str = _constants.STATUS_AVAILABLE,
|
||||
get_client: typing.Callable = None,
|
||||
interval: tobiko.Seconds = None,
|
||||
timeout: tobiko.Seconds = None,
|
||||
**kwargs):
|
||||
"""Waits for an object to reach a specific status.
|
||||
|
||||
:param status_key: The key of the status field in the response.
|
||||
Ex. status
|
||||
:param status: The status to wait for. Ex. "ACTIVE"
|
||||
:param get_client: The tobiko client get method.
|
||||
Ex. _client.get_zone
|
||||
:param object_id: The id of the object to query.
|
||||
:param interval: How often to check the status, in seconds.
|
||||
:param timeout: The maximum time, in seconds, to check the status.
|
||||
:raises TimeoutException: The object did not achieve the status or ERROR in
|
||||
the check_timeout period.
|
||||
:raises UnexpectedStatusException: The request returned an unexpected
|
||||
response code.
|
||||
"""
|
||||
|
||||
get_client = get_client or _client.get_share
|
||||
|
||||
for attempt in tobiko.retry(timeout=timeout,
|
||||
interval=interval,
|
||||
default_timeout=300.,
|
||||
default_interval=5.):
|
||||
response = get_client(object_id, **kwargs)
|
||||
if response[status_key] == status:
|
||||
return response
|
||||
|
||||
attempt.check_limits()
|
||||
|
||||
LOG.debug(f"Waiting for {get_client.__name__} {status_key} to get "
|
||||
f"from '{response[status_key]}' to '{status}'...")
|
||||
|
||||
|
||||
def wait_for_share_status(share_id):
|
||||
wait_for_status(object_id=share_id)
|
||||
|
||||
|
||||
def _is_share_deleted(share_id):
|
||||
try:
|
||||
res = _client.get_share(share_id)
|
||||
except _exceptions.ShareNotFound:
|
||||
return True
|
||||
if res.get(_constants.RESOURCE_STATUS) in [
|
||||
_constants.STATUS_ERROR, _constants.STATUS_ERROR_DELETING]:
|
||||
# Share has "error_deleting" status and can not be deleted.
|
||||
raise _exceptions.ShareReleaseFailed(id=share_id)
|
||||
return False
|
||||
|
||||
|
||||
def wait_for_resource_deletion(share_id, build_interval=1, build_timeout=60):
|
||||
"""Waits for a resource to be deleted."""
|
||||
start_time = int(time.time())
|
||||
while True:
|
||||
if _is_share_deleted(share_id):
|
||||
return
|
||||
if int(time.time()) - start_time >= build_timeout:
|
||||
raise exceptions.TimeoutException
|
||||
time.sleep(build_interval)
|
36
tobiko/openstack/manila/config.py
Normal file
36
tobiko/openstack/manila/config.py
Normal file
@ -0,0 +1,36 @@
|
||||
# Copyright 2023 Red Hat
|
||||
#
|
||||
# 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 __future__ import absolute_import
|
||||
|
||||
import itertools
|
||||
|
||||
from oslo_config import cfg
|
||||
|
||||
GROUP_NAME = 'manila'
|
||||
OPTIONS = [
|
||||
cfg.StrOpt('share_protocol',
|
||||
default='nfs',
|
||||
help="Share protocol"),
|
||||
cfg.IntOpt('size',
|
||||
default=1,
|
||||
help="Default size in GB for shares created by share tests."),
|
||||
]
|
||||
|
||||
|
||||
def register_tobiko_options(conf):
|
||||
conf.register_opts(group=cfg.OptGroup(GROUP_NAME), opts=OPTIONS)
|
||||
|
||||
|
||||
def list_options():
|
||||
return [(GROUP_NAME, itertools.chain(OPTIONS))]
|
0
tobiko/tests/scenario/manila/__init__.py
Normal file
0
tobiko/tests/scenario/manila/__init__.py
Normal file
58
tobiko/tests/scenario/manila/test_manila.py
Normal file
58
tobiko/tests/scenario/manila/test_manila.py
Normal file
@ -0,0 +1,58 @@
|
||||
# Copyright (c) 2023 Red Hat, 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.
|
||||
from __future__ import absolute_import
|
||||
|
||||
import testtools
|
||||
from oslo_log import log
|
||||
|
||||
from tobiko import config
|
||||
from tobiko.openstack import keystone
|
||||
from tobiko.openstack import manila
|
||||
|
||||
LOG = log.getLogger(__name__)
|
||||
CONF = config.CONF
|
||||
|
||||
|
||||
@keystone.skip_if_missing_service(name='manila')
|
||||
class ManilaApiTestCase(testtools.TestCase):
|
||||
"""Manila scenario tests.
|
||||
|
||||
Create a manila share.
|
||||
Check it reaches status 'available'.
|
||||
After upgrade/disruptions/etc, check the share is still valid and it can be
|
||||
extended.
|
||||
"""
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
if config.get_bool_env('TOBIKO_PREVENT_CREATE'):
|
||||
LOG.debug('skipping creation of manila resources')
|
||||
cls.share = manila.get_shares_by_name(manila.SHARE_NAME)[0]
|
||||
else:
|
||||
cls.share = manila.create_share(name=manila.SHARE_NAME)
|
||||
|
||||
manila.wait_for_share_status(cls.share['id'])
|
||||
|
||||
@config.skip_if_prevent_create()
|
||||
def test_1_create_share(self):
|
||||
self.assertEqual(manila.SHARE_NAME, self.share['name'])
|
||||
|
||||
@config.skip_unless_prevent_create()
|
||||
def test_2_extend_share(self):
|
||||
share_id = self.share['id']
|
||||
manila.extend_share(share_id, new_size=CONF.tobiko.manila.size + 1)
|
||||
manila.wait_for_share_status(share_id)
|
||||
share_size = manila.get_share(share_id)['size']
|
||||
self.assertEqual(CONF.tobiko.manila.size + 1, share_size)
|
9
tox.ini
9
tox.ini
@ -172,6 +172,15 @@ setenv =
|
||||
OS_TEST_PATH = {toxinidir}/tobiko/tests/functional
|
||||
|
||||
|
||||
[testenv:manila]
|
||||
basepython = {[integration]basepython}
|
||||
envdir = {[integration]envdir}
|
||||
passenv = {[integration]passenv}
|
||||
setenv =
|
||||
{[integration]setenv}
|
||||
OS_TEST_PATH = {toxinidir}/tobiko/tests/scenario/manila
|
||||
|
||||
|
||||
[testenv:scenario]
|
||||
|
||||
basepython = {[integration]basepython}
|
||||
|
@ -359,7 +359,7 @@ python-keystoneclient===4.4.0
|
||||
python-ldap===3.4.0
|
||||
python-linstor===1.13.0
|
||||
python-magnumclient===3.6.0
|
||||
python-manilaclient===3.3.1
|
||||
python-manilaclient===4.5.1
|
||||
python-masakariclient===7.2.0
|
||||
python-memcached===1.59
|
||||
python-mistralclient===4.4.0
|
||||
|
Loading…
Reference in New Issue
Block a user