Migration to utilize tempest plugin
This is a migration step to utilize tempest plugin system instead of directly calling functional api tests. This is the approach used by a number of other projects as well as an approved process by openstack-qa. The difference in execution is that we will need to execute tempest's tox instead of our own: tox -eall-plugin magnum.tests.functional.api.v1 -- --concurrency=1 Implements: blueprint magnum-tempest Change-Id: Ic3eadae7fb5d88b776f9ded9589ef25279a2e1be
This commit is contained in:
parent
6f2cfe9bc2
commit
63de8e5540
@ -61,15 +61,23 @@ Use follow command lines to check what functional testing is supported::
|
|||||||
cd /opt/stack/magnum
|
cd /opt/stack/magnum
|
||||||
cat tox.ini | grep functional- | awk -F: '{print $2}' | sed s/]//
|
cat tox.ini | grep functional- | awk -F: '{print $2}' | sed s/]//
|
||||||
|
|
||||||
To do some specify functional testing, for example, test api functional
|
To do some specify functional testing, for example, test kubernetes functional
|
||||||
cases::
|
cases::
|
||||||
|
|
||||||
tox -e functional-api -- --concurrency=1
|
tox -e functional-k8s -- --concurrency=1
|
||||||
|
|
||||||
Test specified case of api functional cases::
|
Test specified case of kubernetes functional cases::
|
||||||
|
|
||||||
|
tox -e functional-k8s -- --concurrency=1 <test path>
|
||||||
|
|
||||||
tox -e functional-api -- --concurrency=1 <test path>
|
|
||||||
|
|
||||||
The following is an example for test path::
|
The following is an example for test path::
|
||||||
|
|
||||||
magnum.tests.functional.api.v1.test_baymodel.BayModelTest.test_create_baymodel
|
magnum.tests.functional.k8s.v1.test_k8s_python_client.TestBayModelResource.test_baymodel_create_and_delete
|
||||||
|
|
||||||
|
We also have api tests being added. To run those, go to
|
||||||
|
tempest directory (tempest is installed with devstack by default), install
|
||||||
|
magnum, and execute tempest tox::
|
||||||
|
|
||||||
|
cd /opt/stack/tempest
|
||||||
|
tox -eall-plugin magnum.tests.functional.api.v1 -- --concurrency=1
|
||||||
|
@ -22,46 +22,17 @@ function function_exists {
|
|||||||
declare -f -F $1 > /dev/null
|
declare -f -F $1 > /dev/null
|
||||||
}
|
}
|
||||||
|
|
||||||
if ! function_exists echo_summary; then
|
# Set up all necessary test data
|
||||||
function echo_summary {
|
function create_test_data {
|
||||||
echo $@
|
# First we test Magnum's command line to see if we can stand up
|
||||||
}
|
# a baymodel, bay and a pod
|
||||||
fi
|
|
||||||
|
|
||||||
# Save trace setting
|
export NIC_ID=$(neutron net-show public | awk '/ id /{print $4}')
|
||||||
XTRACE=$(set +o | grep xtrace)
|
export IMAGE_ID=$(glance --os-image-api-version 1 image-show fedora-21-atomic-5 | awk '/ id /{print $4}')
|
||||||
set -o xtrace
|
|
||||||
|
|
||||||
echo_summary "magnum's post_test_hook.sh was called..."
|
# pass the appropriate variables via a config file
|
||||||
(set -o posix; set)
|
CREDS_FILE=$MAGNUM_DIR/functional_creds.conf
|
||||||
|
cat <<EOF > $CREDS_FILE
|
||||||
constraints="-c $REQUIREMENTS_DIR/upper-constraints.txt"
|
|
||||||
sudo pip install $constraints -U -r requirements.txt -r test-requirements.txt
|
|
||||||
|
|
||||||
export MAGNUM_DIR="$BASE/new/magnum"
|
|
||||||
sudo chown -R jenkins:stack $MAGNUM_DIR
|
|
||||||
|
|
||||||
# Get admin credentials
|
|
||||||
pushd ../devstack
|
|
||||||
source openrc admin admin
|
|
||||||
# NOTE(hongbin): This is a temporary work around. These variables are for
|
|
||||||
# keystone v3, but magnum is using v2 API. Therefore, unset them to make the
|
|
||||||
# keystoneclient work.
|
|
||||||
# Bug: #1473600
|
|
||||||
unset OS_PROJECT_DOMAIN_ID
|
|
||||||
unset OS_USER_DOMAIN_ID
|
|
||||||
unset OS_AUTH_TYPE
|
|
||||||
popd
|
|
||||||
|
|
||||||
# First we test Magnum's command line to see if we can stand up
|
|
||||||
# a baymodel, bay and a pod
|
|
||||||
export NIC_ID=$(neutron net-show public | awk '/ id /{print $4}')
|
|
||||||
export IMAGE_ID=$(glance --os-image-api-version 1 image-show fedora-21-atomic-5 | awk '/ id /{print $4}')
|
|
||||||
|
|
||||||
|
|
||||||
# pass the appropriate variables via a config file
|
|
||||||
CREDS_FILE=$MAGNUM_DIR/functional_creds.conf
|
|
||||||
cat <<EOF > $CREDS_FILE
|
|
||||||
# Credentials for functional testing
|
# Credentials for functional testing
|
||||||
|
|
||||||
[auth]
|
[auth]
|
||||||
@ -84,25 +55,113 @@ flavor_id = m1.magnum
|
|||||||
copy_logs = true
|
copy_logs = true
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
# Note(eliqiao): Let's keep this only for debugging on gate.
|
# Note(eliqiao): Let's keep this only for debugging on gate.
|
||||||
echo_summary $CREDS_FILE
|
echo_summary $CREDS_FILE
|
||||||
cat $CREDS_FILE
|
cat $CREDS_FILE
|
||||||
|
|
||||||
# Create a keypair for use in the functional tests.
|
# Create a keypair for use in the functional tests.
|
||||||
echo_summary "Generate a key-pair"
|
echo_summary "Generate a key-pair"
|
||||||
ssh-keygen -t rsa -N "" -f ~/.ssh/id_rsa
|
ssh-keygen -t rsa -N "" -f ~/.ssh/id_rsa
|
||||||
nova keypair-add --pub-key ~/.ssh/id_rsa.pub default
|
nova keypair-add --pub-key ~/.ssh/id_rsa.pub default
|
||||||
|
|
||||||
# Create magnum specific flavor for use in functional tests.
|
}
|
||||||
echo_summary "Create a flavor"
|
|
||||||
nova flavor-create m1.magnum 100 1024 8 1
|
function add_flavor {
|
||||||
|
# because of policy.json change in nova, flavor-create is now an admin-only feature
|
||||||
|
# moving this out to only be used by admins
|
||||||
|
|
||||||
|
# Get admin credentials
|
||||||
|
pushd ../devstack
|
||||||
|
source openrc admin admin
|
||||||
|
# NOTE(hongbin): This is a temporary work around. These variables are for
|
||||||
|
# keystone v3, but magnum is using v2 API. Therefore, unset them to make the
|
||||||
|
# keystoneclient work.
|
||||||
|
# Bug: #1473600
|
||||||
|
unset OS_PROJECT_DOMAIN_ID
|
||||||
|
unset OS_USER_DOMAIN_ID
|
||||||
|
unset OS_AUTH_TYPE
|
||||||
|
popd
|
||||||
|
|
||||||
|
# Create magnum specific flavor for use in functional tests.
|
||||||
|
echo_summary "Create a flavor"
|
||||||
|
nova flavor-create m1.magnum 100 1024 8 1
|
||||||
|
}
|
||||||
|
|
||||||
|
if ! function_exists echo_summary; then
|
||||||
|
function echo_summary {
|
||||||
|
echo $@
|
||||||
|
}
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Save trace setting
|
||||||
|
XTRACE=$(set +o | grep xtrace)
|
||||||
|
set -o xtrace
|
||||||
|
|
||||||
|
echo_summary "magnum's post_test_hook.sh was called..."
|
||||||
|
(set -o posix; set)
|
||||||
|
|
||||||
|
constraints="-c $REQUIREMENTS_DIR/upper-constraints.txt"
|
||||||
|
sudo pip install $constraints -U -r requirements.txt -r test-requirements.txt
|
||||||
|
|
||||||
|
export MAGNUM_DIR="$BASE/new/magnum"
|
||||||
|
sudo chown -R jenkins:stack $MAGNUM_DIR
|
||||||
|
|
||||||
# Run functional tests
|
# Run functional tests
|
||||||
# Currently we support functional-api, functional-k8s, will support swarm,
|
# Currently we support functional-api, functional-k8s, will support swarm,
|
||||||
# mesos later.
|
# mesos later.
|
||||||
|
|
||||||
echo "Running magnum functional test suite for $1"
|
echo "Running magnum functional test suite for $1"
|
||||||
sudo -E -H -u jenkins tox -e functional-$1 -- --concurrency=1
|
|
||||||
|
# For api, we will run tempest tests
|
||||||
|
if [[ "api" == $1 ]]; then
|
||||||
|
# Import devstack functions 'iniset', 'iniget' and 'trueorfalse'
|
||||||
|
source $BASE/new/devstack/functions
|
||||||
|
echo "TEMPEST_SERVICES+=,magnum" >> $localrc_path
|
||||||
|
pushd $BASE/new/tempest
|
||||||
|
sudo chown -R jenkins:stack $BASE/new/tempest
|
||||||
|
|
||||||
|
add_flavor
|
||||||
|
|
||||||
|
# Set demo credentials
|
||||||
|
source $BASE/new/devstack/accrc/demo/demo
|
||||||
|
unset OS_AUTH_TYPE
|
||||||
|
|
||||||
|
create_test_data
|
||||||
|
|
||||||
|
# Set up tempest config with magnum goodness
|
||||||
|
iniset $BASE/new/tempest/etc/tempest.conf magnum image_id $IMAGE_ID
|
||||||
|
iniset $BASE/new/tempest/etc/tempest.conf magnum nic_id $NIC_ID
|
||||||
|
iniset $BASE/new/tempest/etc/tempest.conf magnum keypair_id default
|
||||||
|
iniset $BASE/new/tempest/etc/tempest.conf magnum flavor_id m1.magnum
|
||||||
|
|
||||||
|
# show tempest config with magnum
|
||||||
|
cat etc/tempest.conf
|
||||||
|
|
||||||
|
# Set up concurrency and test regex
|
||||||
|
export MAGNUM_TEMPEST_CONCURRENCY=${MAGNUM_TEMPEST_CONCURRENCY:-1}
|
||||||
|
export MAGNUM_TESTS=${MAGNUM_TESTS:-'magnum.tests.functional.api.v1'}
|
||||||
|
|
||||||
|
echo "Running tempest magnum test suites"
|
||||||
|
sudo -H -u jenkins tox -eall-plugin -- $MAGNUM_TESTS --concurrency=$MAGNUM_TEMPEST_CONCURRENCY
|
||||||
|
else
|
||||||
|
# Get admin credentials
|
||||||
|
pushd ../devstack
|
||||||
|
source openrc admin admin
|
||||||
|
# NOTE(hongbin): This is a temporary work around. These variables are for
|
||||||
|
# keystone v3, but magnum is using v2 API. Therefore, unset them to make the
|
||||||
|
# keystoneclient work.
|
||||||
|
# Bug: #1473600
|
||||||
|
unset OS_PROJECT_DOMAIN_ID
|
||||||
|
unset OS_USER_DOMAIN_ID
|
||||||
|
unset OS_AUTH_TYPE
|
||||||
|
popd
|
||||||
|
|
||||||
|
add_flavor
|
||||||
|
|
||||||
|
create_test_data
|
||||||
|
|
||||||
|
sudo -E -H -u jenkins tox -e functional-$1 -- --concurrency=1
|
||||||
|
fi
|
||||||
EXIT_CODE=$?
|
EXIT_CODE=$?
|
||||||
|
|
||||||
# Delete the keypair used in the functional test.
|
# Delete the keypair used in the functional test.
|
||||||
|
@ -10,7 +10,10 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
import ConfigParser
|
from tempest import config
|
||||||
|
|
||||||
|
|
||||||
|
CONF = config.CONF
|
||||||
|
|
||||||
|
|
||||||
class Config(object):
|
class Config(object):
|
||||||
@ -19,77 +22,65 @@ class Config(object):
|
|||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_admin_creds(cls, config):
|
def set_admin_creds(cls, config):
|
||||||
cls.admin_user = config.get('admin', 'user')
|
cls.admin_user = CONF.auth.admin_username
|
||||||
cls.admin_passwd = config.get('admin', 'pass')
|
cls.admin_passwd = CONF.auth.admin_password
|
||||||
cls.admin_tenant = config.get('admin', 'tenant')
|
cls.admin_tenant = CONF.auth.admin_tenant_name
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_user_creds(cls, config):
|
def set_user_creds(cls, config):
|
||||||
# normal user creds
|
# normal user creds
|
||||||
cls.user = config.get('auth', 'username')
|
cls.user = CONF.identity.username
|
||||||
cls.passwd = config.get('auth', 'password')
|
cls.passwd = CONF.identity.password
|
||||||
cls.tenant = config.get('auth', 'tenant_name')
|
cls.tenant = CONF.identity.tenant_name
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_auth_version(cls, config):
|
def set_auth_version(cls, config):
|
||||||
# auth version for client authentication
|
# auth version for client authentication
|
||||||
if config.has_option('auth', 'auth_version'):
|
cls.auth_version = CONF.identity.auth_version
|
||||||
cls.auth_version = config.get('auth', 'auth_version')
|
|
||||||
else:
|
|
||||||
cls.auth_version = 'v3'
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_auth_url(cls, config):
|
def set_auth_url(cls, config):
|
||||||
# auth_url for client authentication
|
# auth_url for client authentication
|
||||||
if cls.auth_version == 'v3':
|
if cls.auth_version == 'v3':
|
||||||
if not config.has_option('auth', 'auth_v3_url'):
|
cls.auth_v3_url = CONF.identity.uri_v3
|
||||||
raise Exception('config missing auth_v3_url key')
|
|
||||||
cls.auth_v3_url = config.get('auth', 'auth_v3_url')
|
|
||||||
else:
|
else:
|
||||||
if not config.has_option('auth', 'auth_url'):
|
if 'uri' not in CONF.identity:
|
||||||
raise Exception('config missing auth_url key')
|
raise Exception('config missing auth_url key')
|
||||||
cls.auth_url = config.get('auth', 'auth_url')
|
cls.auth_url = CONF.identity.uri
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_region(cls, config):
|
def set_region(cls, config):
|
||||||
if config.has_option('auth', 'region'):
|
if 'region' in CONF.identity:
|
||||||
cls.region = config.get('auth', 'region')
|
cls.region = CONF.identity.region
|
||||||
else:
|
else:
|
||||||
cls.region = 'RegionOne'
|
cls.region = 'RegionOne'
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_image_id(cls, config):
|
def set_image_id(cls, config):
|
||||||
cls.image_id = config.get('magnum', 'image_id')
|
if 'image_id' not in CONF.magnum:
|
||||||
if not config.has_option('magnum', 'image_id'):
|
|
||||||
raise Exception('config missing image_id key')
|
raise Exception('config missing image_id key')
|
||||||
|
cls.image_id = CONF.magnum.image_id
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_nic_id(cls, config):
|
def set_nic_id(cls, config):
|
||||||
cls.nic_id = config.get('magnum', 'nic_id')
|
if 'nic_id' not in CONF.magnum:
|
||||||
if not config.has_option('magnum', 'nic_id'):
|
|
||||||
raise Exception('config missing nic_id key')
|
raise Exception('config missing nic_id key')
|
||||||
|
cls.nic_id = CONF.magnum.nic_id
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def set_keypair_id(cls, config):
|
def set_keypair_id(cls, config):
|
||||||
cls.keypair_id = config.get('magnum', 'keypair_id')
|
if 'keypair_id' not in CONF.magnum:
|
||||||
if not config.has_option('magnum', 'keypair_id'):
|
|
||||||
raise Exception('config missing keypair_id key')
|
raise Exception('config missing keypair_id key')
|
||||||
|
cls.keypair_id = CONF.magnum.keypair_id
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def setUp(cls):
|
def setUp(cls):
|
||||||
config = ConfigParser.RawConfigParser()
|
|
||||||
if config.read('functional_creds.conf'):
|
|
||||||
cls.set_admin_creds(config)
|
cls.set_admin_creds(config)
|
||||||
cls.set_user_creds(config)
|
cls.set_user_creds(config)
|
||||||
cls.set_auth_version(config)
|
cls.set_auth_version(config)
|
||||||
cls.set_auth_url(config)
|
cls.set_auth_url(config)
|
||||||
|
|
||||||
# optional magnum bypass url
|
|
||||||
cls.magnum_url = config.get('auth', 'magnum_url')
|
|
||||||
|
|
||||||
cls.set_region(config)
|
cls.set_region(config)
|
||||||
cls.set_image_id(config)
|
cls.set_image_id(config)
|
||||||
cls.set_nic_id(config)
|
cls.set_nic_id(config)
|
||||||
cls.set_keypair_id(config)
|
cls.set_keypair_id(config)
|
||||||
else:
|
|
||||||
raise Exception('missing functional_creds.conf file')
|
|
||||||
|
@ -61,7 +61,7 @@ def generate_random_port():
|
|||||||
|
|
||||||
|
|
||||||
def generate_random_docker_volume_size():
|
def generate_random_docker_volume_size():
|
||||||
return random.randrange(1, 100)
|
return random.randrange(1, 3)
|
||||||
|
|
||||||
|
|
||||||
def generate_fake_ssh_pubkey():
|
def generate_fake_ssh_pubkey():
|
||||||
|
0
magnum/tests/functional/tempest_tests/__init__.py
Normal file
0
magnum/tests/functional/tempest_tests/__init__.py
Normal file
46
magnum/tests/functional/tempest_tests/config.py
Normal file
46
magnum/tests/functional/tempest_tests/config.py
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
# 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 print_function
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
|
|
||||||
|
from tempest import config # noqa
|
||||||
|
|
||||||
|
service_available_group = cfg.OptGroup(name="service_available",
|
||||||
|
title="Available OpenStack Services")
|
||||||
|
|
||||||
|
ServiceAvailableGroup = [
|
||||||
|
cfg.BoolOpt("magnum",
|
||||||
|
default=True,
|
||||||
|
help="Whether or not magnum is expected to be available"),
|
||||||
|
]
|
||||||
|
|
||||||
|
magnum_group = cfg.OptGroup(name="magnum", title="Magnum Options")
|
||||||
|
|
||||||
|
MagnumGroup = [
|
||||||
|
cfg.StrOpt("image_id",
|
||||||
|
default="fedora-21-atomic-5",
|
||||||
|
help="Image id to be used for baymodel."),
|
||||||
|
|
||||||
|
cfg.StrOpt("nic_id",
|
||||||
|
default="public",
|
||||||
|
help="NIC id."),
|
||||||
|
|
||||||
|
cfg.StrOpt("keypair_id",
|
||||||
|
default="default",
|
||||||
|
help="Keypair id to use to log into nova instances."),
|
||||||
|
|
||||||
|
cfg.StrOpt("flavor_id",
|
||||||
|
default="m1.magnum",
|
||||||
|
help="Flavor id to use for baymodels."),
|
||||||
|
]
|
39
magnum/tests/functional/tempest_tests/plugin.py
Normal file
39
magnum/tests/functional/tempest_tests/plugin.py
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
# 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 os
|
||||||
|
|
||||||
|
from tempest import config
|
||||||
|
from tempest.test_discover import plugins
|
||||||
|
|
||||||
|
import magnum
|
||||||
|
from magnum.tests.functional.tempest_tests import config as magnum_config
|
||||||
|
|
||||||
|
|
||||||
|
class MagnumTempestPlugin(plugins.TempestPlugin):
|
||||||
|
def load_tests(self):
|
||||||
|
base_path = os.path.split(os.path.dirname(
|
||||||
|
os.path.abspath(magnum.__file__)))[0]
|
||||||
|
test_dir = "magnum/tests/functional/api/v1"
|
||||||
|
full_test_dir = os.path.join(base_path, test_dir)
|
||||||
|
return full_test_dir, base_path
|
||||||
|
|
||||||
|
def register_opts(self, conf):
|
||||||
|
config.register_opt_group(
|
||||||
|
conf, magnum_config.service_available_group,
|
||||||
|
magnum_config.ServiceAvailableGroup)
|
||||||
|
config.register_opt_group(conf, magnum_config.magnum_group,
|
||||||
|
magnum_config.MagnumGroup)
|
||||||
|
|
||||||
|
def get_opt_lists(self):
|
||||||
|
return [(magnum_config.magnum_group.name, magnum_config.MagnumGroup)]
|
@ -65,5 +65,7 @@ magnum.cert_manager.backend =
|
|||||||
barbican = magnum.common.cert_manager.barbican_cert_manager
|
barbican = magnum.common.cert_manager.barbican_cert_manager
|
||||||
local = magnum.common.cert_manager.local_cert_manager
|
local = magnum.common.cert_manager.local_cert_manager
|
||||||
|
|
||||||
|
tempest.test_plugins =
|
||||||
|
magnum_tests = magnum.tests.functional.tempest_tests.plugin:MagnumTempestPlugin
|
||||||
[wheel]
|
[wheel]
|
||||||
universal = 1
|
universal = 1
|
||||||
|
Loading…
x
Reference in New Issue
Block a user