e6100d9dd5
Switch to using bundletester for execution of functional tests, leveraging tox to build out test virtualenvs. Rename amulet tests inline with gate-*, dev-* and dfs-* naming standards. Update README to refer to functional testing section of the charm guide. Change-Id: I6d9582386b77cae8a00726760dc7cead85af7527
797 lines
33 KiB
Python
797 lines
33 KiB
Python
#!/usr/bin/env python
|
|
#
|
|
# Copyright 2016 Canonical Ltd
|
|
#
|
|
# 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 amulet
|
|
import os
|
|
import yaml
|
|
import time
|
|
import json
|
|
import subprocess
|
|
|
|
from charmhelpers.contrib.openstack.amulet.deployment import (
|
|
OpenStackAmuletDeployment
|
|
)
|
|
|
|
from charmhelpers.contrib.openstack.amulet.utils import (
|
|
OpenStackAmuletUtils,
|
|
DEBUG,
|
|
# ERROR
|
|
)
|
|
|
|
# Use DEBUG to turn on debug logging
|
|
u = OpenStackAmuletUtils(DEBUG)
|
|
|
|
|
|
class CinderBasicDeployment(OpenStackAmuletDeployment):
|
|
"""Amulet tests on a basic lvm-backed cinder deployment. Verify
|
|
relations, service status, users and endpoint service catalog.
|
|
Create, clone, delete volumes. Create volume from glance image.
|
|
Create volume snapshot. Create volume from snapshot."""
|
|
|
|
def __init__(self, series=None, openstack=None, source=None, git=False,
|
|
stable=False):
|
|
"""Deploy the entire test environment."""
|
|
super(CinderBasicDeployment, self).__init__(series, openstack, source,
|
|
stable)
|
|
self.git = git
|
|
self._add_services()
|
|
self._add_relations()
|
|
self._configure_services()
|
|
self._deploy()
|
|
|
|
u.log.info('Waiting on extended status checks...')
|
|
exclude_services = ['mysql']
|
|
self._auto_wait_for_status(exclude_services=exclude_services)
|
|
|
|
self._initialize_tests()
|
|
|
|
def _add_services(self):
|
|
"""Add services
|
|
|
|
Add the services that we're testing, where cinder is local,
|
|
and the rest of the service are from lp branches that are
|
|
compatible with the local charm (e.g. stable or next).
|
|
"""
|
|
this_service = {'name': 'cinder'}
|
|
other_services = [{'name': 'mysql'}, {'name': 'rabbitmq-server'},
|
|
{'name': 'keystone'}, {'name': 'glance'}]
|
|
super(CinderBasicDeployment, self)._add_services(this_service,
|
|
other_services)
|
|
|
|
def _add_relations(self):
|
|
"""Add relations for the services."""
|
|
relations = {
|
|
'keystone:shared-db': 'mysql:shared-db',
|
|
'cinder:shared-db': 'mysql:shared-db',
|
|
'cinder:identity-service': 'keystone:identity-service',
|
|
'cinder:amqp': 'rabbitmq-server:amqp',
|
|
'cinder:image-service': 'glance:image-service',
|
|
'glance:identity-service': 'keystone:identity-service',
|
|
'glance:shared-db': 'mysql:shared-db',
|
|
'glance:amqp': 'rabbitmq-server:amqp'
|
|
}
|
|
super(CinderBasicDeployment, self)._add_relations(relations)
|
|
|
|
def _configure_services(self):
|
|
"""Configure all of the services."""
|
|
cinder_config = {'block-device': 'vdb',
|
|
'glance-api-version': '2',
|
|
'overwrite': 'true',
|
|
'ephemeral-unmount': '/mnt'}
|
|
if self.git:
|
|
amulet_http_proxy = os.environ.get('AMULET_HTTP_PROXY')
|
|
|
|
reqs_repo = 'git://github.com/openstack/requirements'
|
|
cinder_repo = 'git://github.com/openstack/cinder'
|
|
if self._get_openstack_release() == self.trusty_icehouse:
|
|
reqs_repo = 'git://github.com/coreycb/requirements'
|
|
cinder_repo = 'git://github.com/coreycb/cinder'
|
|
|
|
branch = 'stable/' + self._get_openstack_release_string()
|
|
|
|
openstack_origin_git = {
|
|
'repositories': [
|
|
{'name': 'requirements',
|
|
'repository': reqs_repo,
|
|
'branch': branch},
|
|
{'name': 'cinder',
|
|
'repository': cinder_repo,
|
|
'branch': branch},
|
|
],
|
|
# Most tests use /mnt/openstack-git but cinder's using /dev/vdb
|
|
# to store block devices so leave /mnt alone.
|
|
'directory': '/tmp/openstack-git',
|
|
'http_proxy': amulet_http_proxy,
|
|
'https_proxy': amulet_http_proxy,
|
|
}
|
|
cinder_config['openstack-origin-git'] = \
|
|
yaml.dump(openstack_origin_git)
|
|
|
|
keystone_config = {'admin-password': 'openstack',
|
|
'admin-token': 'ubuntutesting'}
|
|
mysql_config = {'dataset-size': '50%'}
|
|
configs = {'cinder': cinder_config,
|
|
'keystone': keystone_config,
|
|
'mysql': mysql_config}
|
|
super(CinderBasicDeployment, self)._configure_services(configs)
|
|
|
|
def _initialize_tests(self):
|
|
"""Perform final initialization before tests get run."""
|
|
# Access the sentries for inspecting service units
|
|
self.cinder_sentry = self.d.sentry['cinder'][0]
|
|
self.glance_sentry = self.d.sentry['glance'][0]
|
|
self.mysql_sentry = self.d.sentry['mysql'][0]
|
|
self.keystone_sentry = self.d.sentry['keystone'][0]
|
|
self.rabbitmq_sentry = self.d.sentry['rabbitmq-server'][0]
|
|
u.log.debug('openstack release val: {}'.format(
|
|
self._get_openstack_release()))
|
|
u.log.debug('openstack release str: {}'.format(
|
|
self._get_openstack_release_string()))
|
|
|
|
# Authenticate admin with keystone
|
|
self.keystone = u.authenticate_keystone_admin(self.keystone_sentry,
|
|
user='admin',
|
|
password='openstack',
|
|
tenant='admin')
|
|
|
|
# Authenticate admin with cinder endpoint
|
|
self.cinder = u.authenticate_cinder_admin(self.keystone_sentry,
|
|
username='admin',
|
|
password='openstack',
|
|
tenant='admin')
|
|
|
|
# Authenticate admin with glance endpoint
|
|
self.glance = u.authenticate_glance_admin(self.keystone)
|
|
|
|
def _run_action(self, unit_id, action, *args):
|
|
command = ["juju", "action", "do", "--format=json", unit_id, action]
|
|
command.extend(args)
|
|
print("Running command: %s\n" % " ".join(command))
|
|
output = subprocess.check_output(command)
|
|
output_json = output.decode(encoding="UTF-8")
|
|
data = json.loads(output_json)
|
|
action_id = data[u'Action queued with id']
|
|
return action_id
|
|
|
|
def _wait_on_action(self, action_id):
|
|
command = ["juju", "action", "fetch", "--format=json", action_id]
|
|
while True:
|
|
try:
|
|
output = subprocess.check_output(command)
|
|
except Exception as e:
|
|
print(e)
|
|
return False
|
|
output_json = output.decode(encoding="UTF-8")
|
|
data = json.loads(output_json)
|
|
if data[u"status"] == "completed":
|
|
return True
|
|
elif data[u"status"] == "failed":
|
|
return False
|
|
time.sleep(2)
|
|
|
|
def _extend_cinder_volume(self, vol_id, new_size=2):
|
|
"""Extend an existing cinder volume size.
|
|
|
|
:param vol_id: existing cinder volume to extend
|
|
:param new_size: new size in gigabytes
|
|
:returns: None if successful; Failure message otherwise
|
|
"""
|
|
# Extend existing volume size
|
|
try:
|
|
self.cinder.volumes.extend(vol_id, new_size)
|
|
vol_size_org = self.cinder.volumes.get(vol_id).size
|
|
except Exception as e:
|
|
msg = 'Failed to extend volume: {}'.format(e)
|
|
amulet.raise_status(amulet.FAIL, msg=msg)
|
|
|
|
# Confirm that the volume reaches available status.
|
|
ret = u.resource_reaches_status(self.cinder.volumes, vol_id,
|
|
expected_stat="available",
|
|
msg="Volume status wait")
|
|
if not ret:
|
|
msg = ('Cinder volume failed to reach expected state '
|
|
'while extending.')
|
|
return ret
|
|
|
|
# Validate volume size and status
|
|
u.log.debug('Validating volume attributes...')
|
|
vol_size_ext = self.cinder.volumes.get(vol_id).size
|
|
vol_stat = self.cinder.volumes.get(vol_id).status
|
|
msg_attr = ('Volume attributes - orig size:{} extended size:{} '
|
|
'stat:{}'.format(vol_size_org, vol_size_ext, vol_stat))
|
|
|
|
if vol_size_ext > vol_size_org and vol_stat == 'available':
|
|
u.log.debug(msg_attr)
|
|
else:
|
|
msg = ('Volume validation failed, {}'.format(msg_attr))
|
|
return ret
|
|
|
|
return None
|
|
|
|
def _snapshot_cinder_volume(self, name='demo-snapshot', vol_id=None):
|
|
"""Create a snapshot of an existing cinder volume.
|
|
|
|
:param name: display name to assign to snapshot
|
|
:param vol_id: existing cinder volume to snapshot
|
|
:returns: None if successful; Failure message otherwise
|
|
"""
|
|
u.log.debug('Creating snapshot of volume ({})...'.format(vol_id))
|
|
# Create snapshot of an existing cinder volume
|
|
try:
|
|
snap_new = self.cinder.volume_snapshots.create(
|
|
volume_id=vol_id, display_name=name)
|
|
snap_id = snap_new.id
|
|
except Exception as e:
|
|
msg = 'Failed to snapshot the volume: {}'.format(e)
|
|
amulet.raise_status(amulet.FAIL, msg=msg)
|
|
|
|
# Confirm that the volume reaches available status.
|
|
ret = u.resource_reaches_status(self.cinder.volume_snapshots,
|
|
snap_id,
|
|
expected_stat="available",
|
|
msg="Volume status wait")
|
|
if not ret:
|
|
msg = ('Cinder volume failed to reach expected state '
|
|
'while snapshotting.')
|
|
return ret
|
|
|
|
# Validate snapshot
|
|
u.log.debug('Validating snapshot attributes...')
|
|
snap_name = self.cinder.volume_snapshots.get(snap_id).display_name
|
|
snap_stat = self.cinder.volume_snapshots.get(snap_id).status
|
|
snap_vol_id = self.cinder.volume_snapshots.get(snap_id).volume_id
|
|
msg_attr = ('Snapshot attributes - name:{} status:{} '
|
|
'vol_id:{}'.format(snap_name, snap_stat, snap_vol_id))
|
|
|
|
if snap_name == name and snap_stat == 'available' \
|
|
and snap_vol_id == vol_id:
|
|
u.log.debug(msg_attr)
|
|
else:
|
|
msg = ('Snapshot validation failed, {}'.format(msg_attr))
|
|
amulet.raise_status(amulet.FAIL, msg=msg)
|
|
|
|
return snap_new
|
|
|
|
def _check_cinder_lvm(self):
|
|
"""Inspect lvm on cinder unit, do basic validation against
|
|
cinder volumes and snapshots that exist."""
|
|
u.log.debug('Checking cinder volumes against lvm volumes...')
|
|
# Inspect
|
|
cmd = 'sudo lvs | grep cinder-volumes | awk \'{ print $1 }\''
|
|
output, code = self.cinder_sentry.run(cmd)
|
|
u.log.debug('{} `{}` returned '
|
|
'{}'.format(self.cinder_sentry.info['unit_name'],
|
|
cmd, code))
|
|
if code != 0:
|
|
return "command `{}` returned {}".format(cmd, str(code))
|
|
|
|
vol_list = self.cinder.volumes.list()
|
|
lv_id_list = output.split('\n')
|
|
lv_count = len(lv_id_list)
|
|
vol_count = len(vol_list)
|
|
snap_count = len(self.cinder.volume_snapshots.list())
|
|
|
|
# Expect cinder vol + snap count to match lvm log vol count
|
|
u.log.debug('vols:{} snaps:{} lvs:{}'.format(vol_count,
|
|
snap_count,
|
|
lv_count))
|
|
if (vol_count + snap_count) != len(lv_id_list):
|
|
msg = ('lvm volume count ({}) != cinder volume + snap count '
|
|
'({})'.format(len(vol_list), len(lv_id_list)))
|
|
return msg
|
|
|
|
# Expect all cinder vol IDs to exist in the LVM volume list
|
|
for vol_this in vol_list:
|
|
try:
|
|
vol_id = vol_this.id
|
|
vol_name = vol_this.display_name
|
|
lv_id = 'volume-{}'.format(vol_id)
|
|
_index = lv_id_list.index(lv_id)
|
|
u.log.info('Volume ({}) correlates to lv '
|
|
'{} ({})'.format(vol_name,
|
|
_index,
|
|
lv_id))
|
|
except:
|
|
u.log.error('lvs output: {}'.format(output))
|
|
msg = ('Volume ID {} not found in '
|
|
'LVM volume list.'.format(vol_this.id))
|
|
return msg
|
|
|
|
return None
|
|
|
|
# def test_100_services(self):
|
|
# """Verify that the expected services are running on the
|
|
# corresponding service units."""
|
|
# services = {
|
|
# self.cinder_sentry: ['cinder-api',
|
|
# 'cinder-scheduler',
|
|
# 'cinder-volume'],
|
|
# self.glance_sentry: ['glance-registry',
|
|
# 'glance-api'],
|
|
# self.mysql_sentry: ['mysql'],
|
|
# self.keystone_sentry: ['keystone'],
|
|
# self.rabbitmq_sentry: ['rabbitmq-server']
|
|
# }
|
|
# ret = u.validate_services_by_name(services)
|
|
# if ret:
|
|
# amulet.raise_status(amulet.FAIL, msg=ret)
|
|
#
|
|
# def test_110_users(self):
|
|
# """Verify expected users."""
|
|
# u.log.debug('Checking keystone users...')
|
|
# user0 = {'name': 'cinder_cinderv2',
|
|
# 'enabled': True,
|
|
# 'tenantId': u.not_null,
|
|
# 'id': u.not_null,
|
|
# 'email': 'juju@localhost'}
|
|
# user1 = {'name': 'admin',
|
|
# 'enabled': True,
|
|
# 'tenantId': u.not_null,
|
|
# 'id': u.not_null,
|
|
# 'email': 'juju@localhost'}
|
|
# user2 = {'name': 'glance',
|
|
# 'enabled': True,
|
|
# 'tenantId': u.not_null,
|
|
# 'id': u.not_null,
|
|
# 'email': 'juju@localhost'}
|
|
# expected = [user0, user1, user2]
|
|
# actual = self.keystone.users.list()
|
|
#
|
|
# ret = u.validate_user_data(expected, actual)
|
|
# if ret:
|
|
# amulet.raise_status(amulet.FAIL, msg=ret)
|
|
#
|
|
# def test_112_service_catalog(self):
|
|
# """Verify that the service catalog endpoint data"""
|
|
# u.log.debug('Checking keystone service catalog...')
|
|
# endpoint_vol = {'adminURL': u.valid_url,
|
|
# 'region': 'RegionOne',
|
|
# 'publicURL': u.valid_url,
|
|
# 'internalURL': u.valid_url}
|
|
# endpoint_id = {'adminURL': u.valid_url,
|
|
# 'region': 'RegionOne',
|
|
# 'publicURL': u.valid_url,
|
|
# 'internalURL': u.valid_url}
|
|
# if self._get_openstack_release() >= self.trusty_icehouse:
|
|
# endpoint_vol['id'] = u.not_null
|
|
# endpoint_id['id'] = u.not_null
|
|
#
|
|
# expected = {'image': [endpoint_id],
|
|
# 'identity': [endpoint_id],
|
|
# 'volume': [endpoint_id]}
|
|
# actual = self.keystone.service_catalog.get_endpoints()
|
|
#
|
|
# ret = u.validate_svc_catalog_endpoint_data(expected, actual)
|
|
# if ret:
|
|
# amulet.raise_status(amulet.FAIL, msg=ret)
|
|
#
|
|
# def test_114_cinder_endpoint(self):
|
|
# """Verify the cinder endpoint data."""
|
|
# u.log.debug('Checking cinder endpoint...')
|
|
# endpoints = self.keystone.endpoints.list()
|
|
# admin_port = internal_port = public_port = '8776'
|
|
# expected = {'id': u.not_null,
|
|
# 'region': 'RegionOne',
|
|
# 'adminurl': u.valid_url,
|
|
# 'internalurl': u.valid_url,
|
|
# 'publicurl': u.valid_url,
|
|
# 'service_id': u.not_null}
|
|
#
|
|
# ret = u.validate_endpoint_data(endpoints, admin_port, internal_port,
|
|
# public_port, expected)
|
|
# if ret:
|
|
# amulet.raise_status(amulet.FAIL,
|
|
# msg='cinder endpoint: {}'.format(ret))
|
|
#
|
|
# def test_202_cinder_glance_image_service_relation(self):
|
|
# """Verify the cinder:glance image-service relation data"""
|
|
# u.log.debug('Checking cinder:glance image-service relation data...')
|
|
# unit = self.cinder_sentry
|
|
# relation = ['image-service', 'glance:image-service']
|
|
# expected = {'private-address': u.valid_ip}
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('cinder image-service', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_203_glance_cinder_image_service_relation(self):
|
|
# """Verify the glance:cinder image-service relation data"""
|
|
# u.log.debug('Checking glance:cinder image-service relation data...')
|
|
# unit = self.glance_sentry
|
|
# relation = ['image-service', 'cinder:image-service']
|
|
# expected = {
|
|
# 'private-address': u.valid_ip,
|
|
# 'glance-api-server': u.valid_url
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('glance image-service', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_204_mysql_cinder_db_relation(self):
|
|
# """Verify the mysql:glance shared-db relation data"""
|
|
# u.log.debug('Checking mysql:cinder db relation data...')
|
|
# unit = self.mysql_sentry
|
|
# relation = ['shared-db', 'cinder:shared-db']
|
|
# expected = {
|
|
# 'private-address': u.valid_ip,
|
|
# 'db_host': u.valid_ip
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('mysql shared-db', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_205_cinder_mysql_db_relation(self):
|
|
# """Verify the cinder:mysql shared-db relation data"""
|
|
# u.log.debug('Checking cinder:mysql db relation data...')
|
|
# unit = self.cinder_sentry
|
|
# relation = ['shared-db', 'mysql:shared-db']
|
|
# expected = {
|
|
# 'private-address': u.valid_ip,
|
|
# 'hostname': u.valid_ip,
|
|
# 'username': 'cinder',
|
|
# 'database': 'cinder'
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('cinder shared-db', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_206_keystone_cinder_id_relation(self):
|
|
# """Verify the keystone:cinder identity-service relation data"""
|
|
# u.log.debug('Checking keystone:cinder id relation data...')
|
|
# unit = self.keystone_sentry
|
|
# relation = ['identity-service',
|
|
# 'cinder:identity-service']
|
|
# expected = {
|
|
# 'service_protocol': 'http',
|
|
# 'service_tenant': 'services',
|
|
# 'admin_token': 'ubuntutesting',
|
|
# 'service_password': u.not_null,
|
|
# 'service_port': '5000',
|
|
# 'auth_port': '35357',
|
|
# 'auth_protocol': 'http',
|
|
# 'private-address': u.valid_ip,
|
|
# 'auth_host': u.valid_ip,
|
|
# 'service_username': 'cinder_cinderv2',
|
|
# 'service_tenant_id': u.not_null,
|
|
# 'service_host': u.valid_ip
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('identity-service cinder', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_207_cinder_keystone_id_relation(self):
|
|
# """Verify the cinder:keystone identity-service relation data"""
|
|
# u.log.debug('Checking cinder:keystone id relation data...')
|
|
# unit = self.cinder_sentry
|
|
# relation = ['identity-service',
|
|
# 'keystone:identity-service']
|
|
# expected = {
|
|
# 'cinder_service': 'cinder',
|
|
# 'cinder_region': 'RegionOne',
|
|
# 'cinder_public_url': u.valid_url,
|
|
# 'cinder_internal_url': u.valid_url,
|
|
# 'cinder_admin_url': u.valid_url,
|
|
# 'private-address': u.valid_ip
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('cinder identity-service', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_208_rabbitmq_cinder_amqp_relation(self):
|
|
# """Verify the rabbitmq-server:cinder amqp relation data"""
|
|
# u.log.debug('Checking rmq:cinder amqp relation data...')
|
|
# unit = self.rabbitmq_sentry
|
|
# relation = ['amqp', 'cinder:amqp']
|
|
# expected = {
|
|
# 'private-address': u.valid_ip,
|
|
# 'password': u.not_null,
|
|
# 'hostname': u.valid_ip
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('amqp cinder', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
# def test_209_cinder_rabbitmq_amqp_relation(self):
|
|
# """Verify the cinder:rabbitmq-server amqp relation data"""
|
|
# u.log.debug('Checking cinder:rmq amqp relation data...')
|
|
# unit = self.cinder_sentry
|
|
# relation = ['amqp', 'rabbitmq-server:amqp']
|
|
# expected = {
|
|
# 'private-address': u.valid_ip,
|
|
# 'vhost': 'openstack',
|
|
# 'username': u.not_null
|
|
# }
|
|
# ret = u.validate_relation_data(unit, relation, expected)
|
|
# if ret:
|
|
# msg = u.relation_error('cinder amqp', ret)
|
|
# amulet.raise_status(amulet.FAIL, msg=msg)
|
|
#
|
|
def test_300_cinder_config(self):
|
|
"""Verify the data in the cinder.conf file."""
|
|
u.log.debug('Checking cinder config file data...')
|
|
unit = self.cinder_sentry
|
|
conf = '/etc/cinder/cinder.conf'
|
|
unit_mq = self.rabbitmq_sentry
|
|
unit_ks = self.keystone_sentry
|
|
rel_mq_ci = unit_mq.relation('amqp', 'cinder:amqp')
|
|
rel_ks_ci = unit_ks.relation('identity-service',
|
|
'cinder:identity-service')
|
|
auth_uri = ('http://%s:%s/' %
|
|
(rel_ks_ci['auth_host'], rel_ks_ci['service_port']))
|
|
auth_url = ('http://%s:%s/' %
|
|
(rel_ks_ci['auth_host'], rel_ks_ci['auth_port']))
|
|
|
|
expected = {
|
|
'DEFAULT': {
|
|
'use_syslog': 'False',
|
|
'debug': 'False',
|
|
'verbose': 'False',
|
|
'iscsi_helper': 'tgtadm',
|
|
'volume_group': 'cinder-volumes',
|
|
'auth_strategy': 'keystone',
|
|
'volumes_dir': '/var/lib/cinder/volumes'
|
|
},
|
|
'keystone_authtoken': {
|
|
'admin_user': rel_ks_ci['service_username'],
|
|
'admin_password': rel_ks_ci['service_password'],
|
|
'admin_tenant_name': rel_ks_ci['service_tenant'],
|
|
'auth_uri': auth_uri,
|
|
'signing_dir': '/var/cache/cinder'
|
|
}
|
|
}
|
|
|
|
expected_rmq = {
|
|
'rabbit_userid': 'cinder',
|
|
'rabbit_virtual_host': 'openstack',
|
|
'rabbit_password': rel_mq_ci['password'],
|
|
'rabbit_host': rel_mq_ci['hostname'],
|
|
}
|
|
if self._get_openstack_release() >= self.trusty_mitaka:
|
|
expected['keystone_authtoken'] = {
|
|
'auth_uri': auth_uri.rstrip('/'),
|
|
'auth_url': auth_url.rstrip('/'),
|
|
'auth_type': 'password',
|
|
'project_domain_name': 'default',
|
|
'user_domain_name': 'default',
|
|
'project_name': 'services',
|
|
'username': rel_ks_ci['service_username'],
|
|
'password': rel_ks_ci['service_password'],
|
|
'signing_dir': '/var/cache/cinder'
|
|
}
|
|
elif self._get_openstack_release() >= self.trusty_liberty:
|
|
expected['keystone_authtoken'] = {
|
|
'auth_uri': auth_uri.rstrip('/'),
|
|
'auth_url': auth_url.rstrip('/'),
|
|
'auth_plugin': 'password',
|
|
'project_domain_id': 'default',
|
|
'user_domain_id': 'default',
|
|
'project_name': 'services',
|
|
'username': rel_ks_ci['service_username'],
|
|
'password': rel_ks_ci['service_password'],
|
|
'signing_dir': '/var/cache/cinder'
|
|
}
|
|
|
|
if self._get_openstack_release() == self.trusty_kilo:
|
|
expected['keystone_authtoken']['auth_uri'] = auth_uri
|
|
expected['keystone_authtoken']['identity_uri'] = \
|
|
auth_url.strip('/')
|
|
|
|
if self._get_openstack_release() >= self.trusty_kilo:
|
|
# Kilo or later
|
|
expected['oslo_messaging_rabbit'] = expected_rmq
|
|
else:
|
|
# Juno or earlier
|
|
expected['DEFAULT'].update(expected_rmq)
|
|
expected['keystone_authtoken']['auth_host'] = \
|
|
rel_ks_ci['auth_host']
|
|
|
|
for section, pairs in expected.iteritems():
|
|
ret = u.validate_config_data(unit, conf, section, pairs)
|
|
if ret:
|
|
message = "cinder config error: {}".format(ret)
|
|
amulet.raise_status(amulet.FAIL, msg=message)
|
|
|
|
def test_301_cinder_logging_config(self):
|
|
"""Verify the data in the cinder logging conf file."""
|
|
u.log.debug('Checking cinder logging config file data...')
|
|
unit = self.cinder_sentry
|
|
conf = '/etc/cinder/logging.conf'
|
|
|
|
expected = {
|
|
'loggers': {
|
|
'keys': 'root, cinder'
|
|
},
|
|
'logger_cinder': {
|
|
'level': 'INFO',
|
|
'handlers': 'stderr',
|
|
'qualname': 'cinder'
|
|
},
|
|
'logger_root': {
|
|
'level': 'WARNING',
|
|
'handlers': 'null'
|
|
}
|
|
}
|
|
|
|
for section, pairs in expected.iteritems():
|
|
ret = u.validate_config_data(unit, conf, section, pairs)
|
|
if ret:
|
|
message = "cinder logging config error: {}".format(ret)
|
|
amulet.raise_status(amulet.FAIL, msg=message)
|
|
|
|
def test_303_cinder_rootwrap_config(self):
|
|
"""Inspect select config pairs in rootwrap.conf."""
|
|
u.log.debug('Checking cinder rootwrap config file data...')
|
|
unit = self.cinder_sentry
|
|
conf = '/etc/cinder/rootwrap.conf'
|
|
section = 'DEFAULT'
|
|
expected = {
|
|
'filters_path': '/etc/cinder/rootwrap.d,'
|
|
'/usr/share/cinder/rootwrap',
|
|
'use_syslog': 'False',
|
|
}
|
|
|
|
ret = u.validate_config_data(unit, conf, section, expected)
|
|
if ret:
|
|
msg = "cinder rootwrap config error: {}".format(ret)
|
|
amulet.raise_status(amulet.FAIL, msg=msg)
|
|
|
|
def test_400_cinder_api_connection(self):
|
|
"""Simple api call to check service is up and responding"""
|
|
u.log.debug('Checking basic cinder api functionality...')
|
|
check = list(self.cinder.volumes.list())
|
|
u.log.debug('Cinder api check (volumes.list): {}'.format(check))
|
|
assert(check == [])
|
|
|
|
def test_401_create_delete_volume(self):
|
|
"""Create a cinder volume and delete it."""
|
|
u.log.debug('Creating, checking and deleting cinder volume...')
|
|
vol_new = u.create_cinder_volume(self.cinder)
|
|
vol_id = vol_new.id
|
|
u.delete_resource(self.cinder.volumes, vol_id, msg="cinder volume")
|
|
|
|
def test_402_create_delete_volume_from_image(self):
|
|
"""Create a cinder volume from a glance image, and delete it."""
|
|
u.log.debug('Creating, checking and deleting cinder volume'
|
|
'from glance image...')
|
|
img_new = u.create_cirros_image(self.glance, "cirros-image-1")
|
|
img_id = img_new.id
|
|
vol_new = u.create_cinder_volume(self.cinder,
|
|
vol_name="demo-vol-cirros",
|
|
img_id=img_id)
|
|
vol_id = vol_new.id
|
|
u.delete_resource(self.glance.images, img_id, msg="glance image")
|
|
u.delete_resource(self.cinder.volumes, vol_id, msg="cinder volume")
|
|
|
|
def test_403_volume_snap_clone_extend_inspect(self):
|
|
"""Create a cinder volume, clone it, extend its size, create a
|
|
snapshot of the volume, create a volume from a snapshot, check
|
|
status of each, inspect underlying lvm, then delete the resources."""
|
|
u.log.debug('Creating, snapshotting, cloning, extending a '
|
|
'cinder volume...')
|
|
vols = []
|
|
|
|
# Create a 1GB volume
|
|
vol_new = u.create_cinder_volume(self.cinder, vol_size=1)
|
|
vols.append(vol_new)
|
|
vol_id = vol_new.id
|
|
|
|
# Snapshot the volume
|
|
snap = self._snapshot_cinder_volume(vol_id=vol_id)
|
|
snap_id = snap.id
|
|
|
|
# Create a volume from the snapshot
|
|
vol_from_snap = u.create_cinder_volume(self.cinder,
|
|
vol_name="demo-vol-from-snap",
|
|
snap_id=snap_id)
|
|
vols.append(vol_from_snap)
|
|
|
|
# Clone an existing volume
|
|
vol_clone = u.create_cinder_volume(self.cinder,
|
|
vol_name="demo-vol-clone",
|
|
src_vol_id=vol_id)
|
|
vols.append(vol_clone)
|
|
vol_clone_id = vol_clone.id
|
|
|
|
# Extend the cloned volume and confirm new size
|
|
ret = self._extend_cinder_volume(vol_clone_id, new_size=2)
|
|
if ret:
|
|
amulet.raise_status(amulet.FAIL, msg=ret)
|
|
|
|
# Inspect logical volumes (lvm) on cinder unit
|
|
ret = self._check_cinder_lvm()
|
|
if ret:
|
|
amulet.raise_status(amulet.FAIL, msg=ret)
|
|
|
|
# Cleanup
|
|
u.log.debug('Deleting snapshot {}...'.format(snap_id))
|
|
u.delete_resource(self.cinder.volume_snapshots,
|
|
snap_id, msg="cinder volume")
|
|
|
|
for vol in vols:
|
|
u.log.debug('Deleting volume {}...'.format(vol.id))
|
|
u.delete_resource(self.cinder.volumes, vol.id, msg="cinder volume")
|
|
|
|
def test_900_restart_on_config_change(self):
|
|
"""Verify that the specified services are restarted when the
|
|
config is changed."""
|
|
|
|
sentry = self.cinder_sentry
|
|
juju_service = 'cinder'
|
|
|
|
# Expected default and alternate values
|
|
set_default = {'debug': 'False'}
|
|
set_alternate = {'debug': 'True'}
|
|
|
|
# Config file affected by juju set config change
|
|
conf_file = '/etc/cinder/cinder.conf'
|
|
|
|
# Services which are expected to restart upon config change
|
|
services = {
|
|
'cinder-api': conf_file,
|
|
'cinder-scheduler': conf_file,
|
|
'cinder-volume': conf_file
|
|
}
|
|
|
|
# Make config change, check for service restarts
|
|
u.log.debug('Making config change on {}...'.format(juju_service))
|
|
mtime = u.get_sentry_time(sentry)
|
|
self.d.configure(juju_service, set_alternate)
|
|
|
|
sleep_time = 40
|
|
for s, conf_file in services.iteritems():
|
|
u.log.debug("Checking that service restarted: {}".format(s))
|
|
if not u.validate_service_config_changed(sentry, mtime, s,
|
|
conf_file,
|
|
retry_count=4,
|
|
retry_sleep_time=20,
|
|
sleep_time=sleep_time):
|
|
self.d.configure(juju_service, set_default)
|
|
msg = "service {} didn't restart after config change".format(s)
|
|
amulet.raise_status(amulet.FAIL, msg=msg)
|
|
sleep_time = 0
|
|
|
|
self.d.configure(juju_service, set_default)
|
|
|
|
def test_910_pause_and_resume(self):
|
|
"""The services can be paused and resumed. """
|
|
u.log.debug('Checking pause and resume actions...')
|
|
unit = self.d.sentry['cinder'][0]
|
|
unit_name = unit.info['unit_name']
|
|
|
|
u.log.debug('Checking for active status on {}'.format(unit_name))
|
|
assert u.status_get(unit)[0] == "active"
|
|
|
|
u.log.debug('Running pause action on {}'.format(unit_name))
|
|
action_id = self._run_action(unit_name, "pause")
|
|
u.log.debug('Waiting on action {}'.format(action_id))
|
|
assert self._wait_on_action(action_id), "Pause action failed."
|
|
u.log.debug('Checking for maintenance status on {}'.format(unit_name))
|
|
assert u.status_get(unit)[0] == "maintenance"
|
|
|
|
u.log.debug('Running resume action on {}'.format(unit_name))
|
|
action_id = self._run_action(unit_name, "resume")
|
|
u.log.debug('Waiting on action {}'.format(action_id))
|
|
assert self._wait_on_action(action_id), "Resume action failed."
|
|
u.log.debug('Checking for active status on {}'.format(unit_name))
|
|
assert u.status_get(unit)[0] == "active"
|
|
u.log.debug('OK')
|