512 lines
17 KiB
Bash
Executable File
512 lines
17 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
set -o xtrace
|
|
set -o errexit
|
|
set -o pipefail
|
|
|
|
# Enable unbuffered output for Ansible in Jenkins.
|
|
export PYTHONUNBUFFERED=1
|
|
|
|
function test_smoke {
|
|
openstack --debug endpoint list
|
|
openstack --debug compute service list
|
|
openstack --debug network agent list
|
|
openstack --debug orchestration service list
|
|
if [[ $SCENARIO == "cephadm" ]] || [[ $SCENARIO == "zun" ]]; then
|
|
openstack --debug volume service list
|
|
fi
|
|
}
|
|
|
|
function create_a_volume {
|
|
local volume_name=$1
|
|
|
|
local attempt
|
|
|
|
openstack volume create --size 1 $volume_name
|
|
attempt=1
|
|
while [[ $(openstack volume show $volume_name -f value -c status) != "available" ]]; do
|
|
echo "Volume $volume_name not available yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 10 ]]; then
|
|
echo "Volume $volume_name failed to become available"
|
|
openstack volume show $volume_name
|
|
return 1
|
|
fi
|
|
sleep 10
|
|
done
|
|
}
|
|
|
|
function create_a_volume_from_image {
|
|
local volume_name=$1
|
|
local image_name=$2
|
|
|
|
local attempt
|
|
|
|
openstack volume create --image $image_name --size 1 $volume_name
|
|
attempt=1
|
|
while [[ $(openstack volume show $volume_name -f value -c status) != "available" ]]; do
|
|
echo "Volume $volume_name not available yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 11 ]]; then
|
|
echo "Volume $volume_name failed to become available"
|
|
openstack volume show $volume_name
|
|
return 1
|
|
fi
|
|
sleep 30
|
|
done
|
|
}
|
|
|
|
function create_an_image_from_volume {
|
|
local image_name=$1
|
|
local volume_name=$2
|
|
|
|
local attempt
|
|
|
|
# NOTE(yoctozepto): Adding explicit microversion of Victoria as a sane default to work
|
|
# around the bug: https://storyboard.openstack.org/#!/story/2009287
|
|
openstack --os-volume-api-version 3.62 image create --volume $volume_name $image_name
|
|
attempt=1
|
|
while [[ $(openstack image show $image_name -f value -c status) != "active" ]]; do
|
|
echo "Image $image_name not active yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 11 ]]; then
|
|
echo "Image $image_name failed to become active"
|
|
openstack image show $image_name
|
|
return 1
|
|
fi
|
|
sleep 30
|
|
done
|
|
}
|
|
|
|
function create_an_image_from_instance {
|
|
local image_name=$1
|
|
local instance_name=$2
|
|
|
|
local attempt
|
|
|
|
openstack server image create $instance_name --name $image_name
|
|
attempt=1
|
|
while [[ $(openstack image show $image_name -f value -c status) != "active" ]]; do
|
|
echo "Image $image_name not active yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 11 ]]; then
|
|
echo "Image $image_name failed to become active"
|
|
openstack image show $image_name
|
|
return 1
|
|
fi
|
|
sleep 30
|
|
done
|
|
}
|
|
|
|
function attach_and_detach_a_volume {
|
|
local volume_name=$1
|
|
local instance_name=$2
|
|
|
|
local attempt
|
|
|
|
openstack server add volume $instance_name $volume_name --device /dev/vdb
|
|
attempt=1
|
|
while [[ $(openstack volume show $volume_name -f value -c status) != "in-use" ]]; do
|
|
echo "Volume $volume_name not attached yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 10 ]]; then
|
|
echo "Volume failed to attach"
|
|
openstack volume show $volume_name
|
|
return 1
|
|
fi
|
|
sleep 10
|
|
done
|
|
|
|
openstack server remove volume $instance_name $volume_name
|
|
attempt=1
|
|
while [[ $(openstack volume show $volume_name -f value -c status) != "available" ]]; do
|
|
echo "Volume $volume_name not detached yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 10 ]]; then
|
|
echo "Volume failed to detach"
|
|
openstack volume show $volume_name
|
|
return 1
|
|
fi
|
|
sleep 10
|
|
done
|
|
}
|
|
|
|
function delete_a_volume {
|
|
local volume_name=$1
|
|
|
|
local attempt
|
|
local result
|
|
|
|
openstack volume delete $volume_name
|
|
|
|
attempt=1
|
|
# NOTE(yoctozepto): This is executed outside of the `while` clause
|
|
# *on purpose*. You see, bash is evil (TM) and will silence any error
|
|
# happening in any "condition" clause (such as `if` or `while`) even with
|
|
# `errexit` being set.
|
|
result=$(openstack volume list --name $volume_name -f value -c ID)
|
|
while [[ -n "$result" ]]; do
|
|
echo "Volume $volume_name not deleted yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 10 ]]; then
|
|
echo "Volume failed to delete"
|
|
openstack volume show $volume_name
|
|
return 1
|
|
fi
|
|
sleep 10
|
|
result=$(openstack volume list --name $volume_name -f value -c ID)
|
|
done
|
|
}
|
|
|
|
function create_instance {
|
|
local name=$1
|
|
local server_create_extra
|
|
|
|
if [[ $IP_VERSION -eq 6 ]]; then
|
|
# NOTE(yoctozepto): CirrOS has no IPv6 metadata support, hence need to use configdrive
|
|
server_create_extra="${server_create_extra} --config-drive True"
|
|
fi
|
|
|
|
openstack server create --wait --image cirros --flavor m1.tiny --key-name mykey --network demo-net ${server_create_extra} ${name}
|
|
# If the status is not ACTIVE, print info and exit 1
|
|
if [[ $(openstack server show ${name} -f value -c status) != "ACTIVE" ]]; then
|
|
echo "FAILED: Instance is not active"
|
|
openstack --debug server show ${name}
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
function resize_instance {
|
|
local name=$1
|
|
|
|
# TODO(priteau): Remove once previous_release includes m2.tiny in
|
|
# init-runonce
|
|
if ! openstack flavor list -f value | grep m2.tiny; then
|
|
openstack flavor create --id 6 --ram 512 --disk 1 --vcpus 2 m2.tiny
|
|
fi
|
|
|
|
openstack server resize --flavor m2.tiny --wait ${name}
|
|
# If the status is not VERIFY_RESIZE, print info and exit 1
|
|
if [[ $(openstack server show ${name} -f value -c status) != "VERIFY_RESIZE" ]]; then
|
|
echo "FAILED: Instance is not resized"
|
|
openstack --debug server show ${name}
|
|
return 1
|
|
fi
|
|
|
|
openstack server resize confirm ${name}
|
|
|
|
# Confirming the resize operation is not instantaneous. Wait for change to
|
|
# be reflected in server status.
|
|
attempt=1
|
|
while [[ $(openstack server show ${name} -f value -c status) != "ACTIVE" ]]; do
|
|
echo "Instance is not active yet"
|
|
attempt=$((attempt+1))
|
|
if [[ $attempt -eq 5 ]]; then
|
|
echo "FAILED: Instance failed to become active after resize confirm"
|
|
openstack --debug server show ${name}
|
|
return 1
|
|
fi
|
|
sleep 2
|
|
done
|
|
}
|
|
|
|
function delete_instance {
|
|
local name=$1
|
|
openstack server delete --wait ${name}
|
|
}
|
|
|
|
function create_fip {
|
|
openstack floating ip create public1 -f value -c floating_ip_address
|
|
}
|
|
|
|
function delete_fip {
|
|
local fip_addr=$1
|
|
openstack floating ip delete ${fip_addr}
|
|
}
|
|
|
|
function attach_fip {
|
|
local instance_name=$1
|
|
local fip_addr=$2
|
|
openstack server add floating ip ${instance_name} ${fip_addr}
|
|
}
|
|
|
|
function detach_fip {
|
|
local instance_name=$1
|
|
local fip_addr=$2
|
|
openstack server remove floating ip ${instance_name} ${fip_addr}
|
|
}
|
|
|
|
function set_cirros_image_q35_machine_type {
|
|
openstack image set --property hw_machine_type=q35 cirros
|
|
}
|
|
|
|
function unset_cirros_image_q35_machine_type {
|
|
openstack image unset --property hw_machine_type cirros
|
|
}
|
|
|
|
function test_neutron_modules {
|
|
# Exit the function if scenario is "ovn" or if there's an upgrade
|
|
# as it only concerns ml2/ovs
|
|
if [[ $SCENARIO == "ovn" ]] || [[ $HAS_UPGRADE == "yes" ]]; then
|
|
return
|
|
fi
|
|
|
|
local modules
|
|
modules=( $(sed -n '/neutron_modules_extra:/,/^[^ ]/p' /etc/kolla/globals.yml | grep -oP '^ - name: \K[^ ]+' | tr -d "'") )
|
|
for module in "${modules[@]}"; do
|
|
if ! grep -q "^${module} " /proc/modules; then
|
|
echo "Error: Module $module is not loaded."
|
|
exit 1
|
|
else
|
|
echo "Module $module is loaded."
|
|
fi
|
|
done
|
|
}
|
|
|
|
function test_ssh {
|
|
local instance_name=$1
|
|
local fip_addr=$2
|
|
local attempts
|
|
attempts=12
|
|
for i in $(seq 1 ${attempts}); do
|
|
if ping -c1 -W1 ${fip_addr} && ssh -v -o BatchMode=yes -o StrictHostKeyChecking=no cirros@${fip_addr} hostname; then
|
|
break
|
|
elif [[ $i -eq ${attempts} ]]; then
|
|
echo "Failed to access server via SSH after ${attempts} attempts"
|
|
echo "Console log:"
|
|
openstack console log show ${instance_name} || true
|
|
openstack --debug server show ${instance_name}
|
|
return 1
|
|
else
|
|
echo "Cannot access server - retrying"
|
|
fi
|
|
sleep 10
|
|
done
|
|
}
|
|
|
|
function test_instance_boot {
|
|
local fip_addr
|
|
local machine_type="${1}"
|
|
local fip_file="/tmp/kolla_ci_pre_upgrade_fip_addr${machine_type:+_$machine_type}"
|
|
local upgrade_instance_name="kolla_upgrade_test${machine_type:+_$machine_type}"
|
|
local volume_name="durable_volume${machine_type:+_$machine_type}"
|
|
|
|
echo "TESTING: Server creation"
|
|
create_instance kolla_boot_test
|
|
echo "SUCCESS: Server creation"
|
|
|
|
if [[ $SCENARIO == "cephadm" ]] || [[ $SCENARIO == "zun" ]]; then
|
|
echo "TESTING: Cinder volume creation and attachment"
|
|
|
|
create_a_volume test_volume
|
|
openstack volume show test_volume
|
|
attach_and_detach_a_volume test_volume kolla_boot_test
|
|
delete_a_volume test_volume
|
|
|
|
# test a qcow2 image (non-cloneable)
|
|
create_a_volume_from_image test_volume_from_image cirros
|
|
openstack volume show test_volume_from_image
|
|
attach_and_detach_a_volume test_volume_from_image kolla_boot_test
|
|
delete_a_volume test_volume_from_image
|
|
|
|
# test a raw image (cloneable)
|
|
openstack image create --disk-format raw --container-format bare --public \
|
|
--file /etc/passwd raw-image
|
|
create_a_volume_from_image test_volume_from_image raw-image
|
|
openstack volume show test_volume_from_image
|
|
attach_and_detach_a_volume test_volume_from_image kolla_boot_test
|
|
delete_a_volume test_volume_from_image
|
|
openstack image delete raw-image
|
|
|
|
echo "SUCCESS: Cinder volume creation and attachment"
|
|
|
|
if [[ $HAS_UPGRADE == 'yes' ]]; then
|
|
echo "TESTING: Cinder volume upgrade stability (PHASE: $PHASE)"
|
|
|
|
if [[ $PHASE == 'deploy' ]]; then
|
|
create_a_volume $volume_name
|
|
openstack volume show $volume_name
|
|
elif [[ $PHASE == 'upgrade' ]]; then
|
|
openstack volume show $volume_name
|
|
attach_and_detach_a_volume $volume_name kolla_boot_test
|
|
delete_a_volume $volume_name
|
|
fi
|
|
|
|
echo "SUCCESS: Cinder volume upgrade stability (PHASE: $PHASE)"
|
|
fi
|
|
|
|
echo "TESTING: Glance image from Cinder volume and back to volume"
|
|
|
|
create_a_volume test_volume_to_image
|
|
openstack volume show test_volume_to_image
|
|
create_an_image_from_volume test_image_from_volume test_volume_to_image
|
|
|
|
create_a_volume_from_image test_volume_from_image_from_volume test_image_from_volume
|
|
openstack volume show test_volume_from_image_from_volume
|
|
attach_and_detach_a_volume test_volume_from_image_from_volume kolla_boot_test
|
|
|
|
delete_a_volume test_volume_from_image_from_volume
|
|
openstack image delete test_image_from_volume
|
|
delete_a_volume test_volume_to_image
|
|
|
|
echo "SUCCESS: Glance image from Cinder volume and back to volume"
|
|
fi
|
|
|
|
echo "TESTING: Instance image upload"
|
|
create_an_image_from_instance image_from_instance kolla_boot_test
|
|
openstack image delete image_from_instance
|
|
echo "SUCCESS: Instance image upload"
|
|
|
|
if [[ $IP_VERSION -eq 4 ]]; then
|
|
echo "TESTING: Floating ip allocation"
|
|
fip_addr=$(create_fip)
|
|
attach_fip kolla_boot_test ${fip_addr}
|
|
echo "SUCCESS: Floating ip allocation"
|
|
else
|
|
# NOTE(yoctozepto): Neutron has no IPv6 NAT support, hence no floating ip addresses
|
|
local instance_addresses
|
|
fip_addr=$(openstack server show kolla_boot_test -f yaml -c addresses|tail -1|cut -d- -f2)
|
|
fi
|
|
|
|
echo "TESTING: PING&SSH to instance"
|
|
test_ssh kolla_boot_test ${fip_addr}
|
|
echo "SUCCESS: PING&SSH to instance"
|
|
|
|
if [[ $IP_VERSION -eq 4 ]]; then
|
|
echo "TESTING: Floating ip deallocation"
|
|
detach_fip kolla_boot_test ${fip_addr}
|
|
delete_fip ${fip_addr}
|
|
echo "SUCCESS: Floating ip deallocation"
|
|
fi
|
|
|
|
echo "TESTING: Server resize"
|
|
resize_instance kolla_boot_test
|
|
echo "SUCCESS: Server resize"
|
|
|
|
echo "TESTING: Server deletion"
|
|
delete_instance kolla_boot_test
|
|
echo "SUCCESS: Server deletion"
|
|
|
|
if [[ $HAS_UPGRADE == 'yes' ]]; then
|
|
echo "TESTING: Instance (Nova and Neutron) upgrade stability (PHASE: $PHASE)"
|
|
|
|
if [[ $PHASE == 'deploy' ]]; then
|
|
create_instance $upgrade_instance_name
|
|
fip_addr=$(create_fip)
|
|
attach_fip $upgrade_instance_name ${fip_addr}
|
|
test_ssh $upgrade_instance_name ${fip_addr} # tested to see if the instance has not just failed booting already
|
|
echo ${fip_addr} > $fip_file
|
|
elif [[ $PHASE == 'upgrade' ]]; then
|
|
fip_addr=$(cat $fip_file)
|
|
test_ssh $upgrade_instance_name ${fip_addr}
|
|
detach_fip $upgrade_instance_name ${fip_addr}
|
|
delete_fip ${fip_addr}
|
|
delete_instance $upgrade_instance_name
|
|
fi
|
|
|
|
echo "SUCCESS: Instance (Nova and Neutron) upgrade stability (PHASE: $PHASE)"
|
|
fi
|
|
}
|
|
|
|
function test_internal_dns_integration {
|
|
|
|
# As per test globals - neutron integration is turned off
|
|
if openstack extension list --network -f value -c Alias | grep -q dns-integration; then
|
|
DNS_NAME="my-port"
|
|
PORT_NAME="${DNS_NAME}"
|
|
DNS_DOMAIN=$(awk -F ':' '/neutron_dns_domain:/ { print $2 }' /etc/kolla/globals.yml \
|
|
| sed -e 's/"//g' -e "s/'//g" -e "s/\ *//g")
|
|
|
|
openstack network create dns-test-network
|
|
openstack subnet create --network dns-test-network --subnet-range 192.168.88.0/24 dns-test-subnet
|
|
openstack port create --network dns-test-network --dns-name ${DNS_NAME} ${PORT_NAME}
|
|
|
|
DNS_ASSIGNMENT=$(openstack port show ${DNS_NAME} -f json -c dns_assignment)
|
|
FQDN=$(echo ${DNS_ASSIGNMENT} | python -c 'import json,sys;obj=json.load(sys.stdin);print(obj["dns_assignment"][0]["fqdn"]);')
|
|
HOSTNAME=$(echo ${DNS_ASSIGNMENT} | python -c 'import json,sys;obj=json.load(sys.stdin);print(obj["dns_assignment"][0]["hostname"]);')
|
|
|
|
if [ "${DNS_NAME}.${DNS_DOMAIN}" == "${FQDN}" ]; then
|
|
echo "[i] Test neutron internal DNS integration FQDN check port - PASS"
|
|
else
|
|
echo "[e] Test neutron internal DNS integration FQDN check port - FAIL"
|
|
exit 1
|
|
fi
|
|
|
|
if [ "${DNS_NAME}" == "${HOSTNAME}" ]; then
|
|
echo "[i] Test neutron internal DNS integration HOSTNAME check port - PASS"
|
|
else
|
|
echo "[e] Test neutron internal DNS integration HOSTNAME check port - FAIL"
|
|
exit 1
|
|
fi
|
|
|
|
openstack port delete ${PORT_NAME}
|
|
|
|
SERVER_NAME="my_vm"
|
|
SERVER_NAME_SANITIZED=$(echo ${SERVER_NAME} | sed -e 's/_/-/g')
|
|
|
|
openstack server create --image cirros --flavor m1.tiny --network dns-test-network ${SERVER_NAME}
|
|
|
|
SERVER_ID=$(openstack server show ${SERVER_NAME} -f value -c id)
|
|
PORT_ID=$(openstack port list --device-id ${SERVER_ID} -f value -c ID)
|
|
|
|
DNS_ASSIGNMENT=$(openstack port show ${PORT_ID} -f json -c dns_assignment)
|
|
FQDN=$(echo ${DNS_ASSIGNMENT} | python -c 'import json,sys;obj=json.load(sys.stdin);print(obj["dns_assignment"][0]["fqdn"]);')
|
|
HOSTNAME=$(echo ${DNS_ASSIGNMENT} | python -c 'import json,sys;obj=json.load(sys.stdin);print(obj["dns_assignment"][0]["hostname"]);')
|
|
|
|
if [ "${SERVER_NAME_SANITIZED}.${DNS_DOMAIN}" == "${FQDN}" ]; then
|
|
echo "[i] Test neutron internal DNS integration FQDN check instance create - PASS"
|
|
else
|
|
echo "[e] Test neutron internal DNS integration FQDN check instance create - FAIL"
|
|
exit 1
|
|
fi
|
|
|
|
if [ "${SERVER_NAME_SANITIZED}" == "${HOSTNAME}" ]; then
|
|
echo "[i] Test neutron internal DNS integration HOSTNAME check instance create - PASS"
|
|
else
|
|
echo "[e] Test neutron internal DNS integration HOSTNAME check instance create - FAIL"
|
|
exit 1
|
|
fi
|
|
|
|
openstack server delete --wait ${SERVER_NAME}
|
|
openstack subnet delete dns-test-subnet
|
|
openstack network delete dns-test-network
|
|
|
|
else
|
|
echo "[i] DNS Integration is not enabled."
|
|
fi
|
|
}
|
|
|
|
function test_openstack_logged {
|
|
. /etc/kolla/admin-openrc.sh
|
|
. ~/openstackclient-venv/bin/activate
|
|
test_smoke
|
|
test_neutron_modules
|
|
test_instance_boot
|
|
test_internal_dns_integration
|
|
|
|
# Check for x86_64 architecture to run q35 tests
|
|
if [[ $(uname -m) == "x86_64" ]]; then
|
|
set_cirros_image_q35_machine_type
|
|
test_instance_boot q35
|
|
unset_cirros_image_q35_machine_type
|
|
fi
|
|
}
|
|
|
|
function test_openstack {
|
|
echo "Testing OpenStack"
|
|
log_file=/tmp/logs/ansible/test-core-openstack
|
|
if [[ -f $log_file ]]; then
|
|
log_file=${log_file}-upgrade
|
|
fi
|
|
test_openstack_logged > $log_file 2>&1
|
|
result=$?
|
|
if [[ $result != 0 ]]; then
|
|
echo "Testing OpenStack failed. See ansible/test-core-openstack for details"
|
|
else
|
|
echo "Successfully tested OpenStack. See ansible/test-core-openstack for details"
|
|
fi
|
|
return $result
|
|
}
|
|
|
|
test_openstack
|