Merge "Improve devmode=flase when building the image"

This commit is contained in:
Zuul 2019-08-25 23:35:33 +00:00 committed by Gerrit Code Review
commit 77dab835e6
36 changed files with 204 additions and 816 deletions

View File

@ -404,7 +404,7 @@ function create_mgmt_subnet_v4 {
local name=$3 local name=$3
local ip_range=$4 local ip_range=$4
subnet_id=$(openstack subnet create --project ${project_id} --ip-version 4 --subnet-range ${ip_range} --gateway none --network ${net_id} $name -c id -f value) subnet_id=$(openstack subnet create --project ${project_id} --ip-version 4 --subnet-range ${ip_range} --gateway none --dns-nameserver 8.8.8.8 --network ${net_id} $name -c id -f value)
die_if_not_set $LINENO subnet_id "Failed to create private IPv4 subnet for network: ${net_id}, project: ${project_id}" die_if_not_set $LINENO subnet_id "Failed to create private IPv4 subnet for network: ${net_id}, project: ${project_id}"
echo $subnet_id echo $subnet_id
} }

View File

@ -55,9 +55,9 @@ Operating System and Database
A Trove Guest Instance contains at least a functioning Operating A Trove Guest Instance contains at least a functioning Operating
System and the database software that the instance wishes to provide System and the database software that the instance wishes to provide
(as a Service). For example, if your chosen operating system is Ubuntu (as a Service). For example, if your chosen operating system is Ubuntu
and you wish to deliver MySQL version 5.5, then your guest instance is and you wish to deliver MySQL version 5.7, then your guest instance is
a Nova instance running the Ubuntu operating system and will have a Nova instance running the Ubuntu operating system and will have
MySQL version 5.5 installed on it. MySQL version 5.7 installed on it.
----------------- -----------------
Trove Guest Agent Trove Guest Agent
@ -77,7 +77,7 @@ Guest Agent API is the common API used by Trove to communicate with
any guest database, and the Guest Agent is the implementation of that any guest database, and the Guest Agent is the implementation of that
API for the specific database. API for the specific database.
The Trove Guest Agent runs on the Trove Guest Instance. The Trove Guest Agent runs inside the Trove Guest Instance.
------------------------------------------ ------------------------------------------
Injected Configuration for the Guest Agent Injected Configuration for the Guest Agent
@ -87,11 +87,11 @@ When TaskManager launches the guest VM it injects the specific settings
for the guest into the VM, into the file /etc/trove/conf.d/guest_info.conf. for the guest into the VM, into the file /etc/trove/conf.d/guest_info.conf.
The file is injected one of three ways. The file is injected one of three ways.
If use_nova_server_config_drive=True, it is injected via ConfigDrive. Otherwise If ``use_nova_server_config_drive=True``, it is injected via ConfigDrive.
it is passed to the nova create call as the 'files' parameter and will be Otherwise it is passed to the nova create call as the 'files' parameter and
injected based on the configuration of Nova; the Nova default is to discard the will be injected based on the configuration of Nova; the Nova default is to
files. If the settings in guest_info.conf are not present on the guest discard the files. If the settings in guest_info.conf are not present on the
Guest Agent will fail to start up. guest Guest Agent will fail to start up.
------------------------------ ------------------------------
Persistent Storage, Networking Persistent Storage, Networking
@ -99,9 +99,13 @@ Persistent Storage, Networking
The database stores data on persistent storage on Cinder (if The database stores data on persistent storage on Cinder (if
configured, see trove.conf and the volume_support parameter) or configured, see trove.conf and the volume_support parameter) or
ephemeral storage on the Nova instance. The database is accessible ephemeral storage on the Nova instance. The database service is accessible
over the network and the Guest Instance is configured for network over the tenant network provided when creating the database instance.
access by client applications.
The cloud administrator is able to config a management
networks(``CONF.management_networks``) that is invisible to the cloud tenants,
database instance can talk to the control plane services(e.g. the message
queue) via that network.
Building Guest Images using DIB Building Guest Images using DIB
=============================== ===============================

View File

@ -1,74 +0,0 @@
.. _guest_cloud_init:
.. role:: bash(code)
:language: bash
===========================
Guest Images via Cloud-Init
===========================
.. If section numbers are desired, unindent this
.. sectnum::
.. If a TOC is desired, unindent this
.. contents::
Overview
========
While creating an image is the preferred method for providing a base
for the Guest Instance, there may be cases where creating an image
is impractical. In those cases a Guest instance can be based on
an available Cloud Image and configured at boot via cloud-init.
Currently the most tested Guest image is Ubunutu 14.04 (trusty).
Setting up the Image
====================
* Visit the `Ubuntu Cloud Archive <https://cloud-images.ubuntu.com/releases/trusty/release>`_ and download ``ubuntu-14.04-server-cloudimg-amd64-disk1.img``.
* Upload that image to glance, and note the glance ID for the image.
* Cloud-Init files go into the directory set by the ``cloudinit_location``
configuration parameter, usually ``/etc/trove/cloudinit``. Files in
that directory are of the format ``[datastore].cloudinit``, for
example ``mysql.cloudinit``.
* Create a cloud-init file for your datastore and put it into place.
For this example, it is assumed you are using Ubuntu 16.04, with
the MySQL database and a Trove Agent from the Pike release. You
would put this into ``/etc/trove/cloudinit/mysql.cloudinit``.
.. code-block:: console
#cloud-config
# For Ubuntu-16.04 cloudimage
apt_sources:
- source: "cloud-archive:pike"
packages:
- trove-guestagent
- mysql-server-5.7
write_files:
- path: /etc/sudoers.d/trove
content: |
Defaults:trove !requiretty
trove ALL=(ALL) NOPASSWD:ALL
runcmd:
- stop trove-guestagent
- cat /etc/trove/trove-guestagent.conf /etc/trove/conf.d/guest_info.conf >/etc/trove/trove.conf
- start trove-guestagent
* If you need to debug guests failing to launch simply append
the cloud-init to add a user to allow you to login and
debug the instance.
* When using ``trove-manage datastore_version_update`` to
define your datastore simply use the Glance ID you have for
the Trusty Cloud image.
When trove launches the Guest Instance, the cloud-init will install
the Pike Trove Guest Agent and MySQL database, and then adjust
the configuration files and launch the Guest Agent.

View File

@ -8,6 +8,5 @@
basics basics
building_guest_images building_guest_images
database_module_usage database_module_usage
guest_cloud_init
secure_oslo_messaging secure_oslo_messaging
trovestack trovestack

View File

@ -34,23 +34,21 @@ The trove guest agent image could be created by running the following command:
.. code-block:: console .. code-block:: console
$ CONTROLLER_IP=10.0.17.132 \ $ ./trovestack build-image \
./trovestack build-image \
${datastore_type} \ ${datastore_type} \
${guest_os} \ ${guest_os} \
${guest_os_release} \ ${guest_os_release} \
${dev_mode} ${dev_mode}
* Currently, only ``guest_os=ubuntu`` and ``guest_os_release=xenial`` are fully * Currently, only ``guest_os=ubuntu`` and ``guest_os_release=xenial`` are fully
tested. tested and supported.
* ``dev_mode=true`` is mainly for testing purpose for trove developers. When * ``dev_mode=true`` is mainly for testing purpose for trove developers and it's
``dev_mode=true``, ``CONTROLLER_IP`` could be ignored. You need to build the necessary to build the image on the trove controller host, because the host
image on the trove controller service host, because the host and the guest VM and the guest VM need to ssh into each other without password. In this mode,
need to ssh into each other without password. In this mode, when the trove when the trove guest agent code is changed, the image doesn't need to be
guest agent code is changed, the image doesn't need to be rebuilt which is rebuilt which is convenient for debugging. Trove guest agent will ssh into
convenient for debugging. Trove guest agent will ssh into the host and the host and download trove code during the service initialization.
download trove code when the service is initialized.
* if ``dev_mode=false``, the trove code for guest agent is injected into the * if ``dev_mode=false``, the trove code for guest agent is injected into the
image at the building time. Now ``dev_mode=false`` is still in experimental image at the building time. Now ``dev_mode=false`` is still in experimental
@ -62,7 +60,8 @@ The trove guest agent image could be created by running the following command:
also need to create a Nova keypair and set ``nova_keypair`` option in Trove also need to create a Nova keypair and set ``nova_keypair`` option in Trove
config file in order to ssh into the guest agent. config file in order to ssh into the guest agent.
For example, build a MySQL image for Ubuntu Xenial operating system: For example, in order to build a MySQL image for Ubuntu Xenial operating
system:
.. code-block:: console .. code-block:: console

View File

@ -1,6 +1,4 @@
Element to install an Trove guest agent. Element to install an Trove guest agent.
Note: this requires a system base image modified to include OpenStack Note: this requires a system base image modified to include Trove source code
repositories repositories
the ubuntu-guest element could be removed.

View File

@ -1,28 +0,0 @@
#!/bin/bash
set -e
set -o xtrace
# CONTEXT: HOST prior to IMAGE BUILD as SCRIPT USER
# PURPOSE: creates the SSH key on the host if it doesn't exist. Then this copies the keys over to a staging area where
# they will be duplicated in the guest VM.
# This process allows the host to log into the guest but more importantly the guest phones home to get the trove
# source
source $_LIB/die
HOST_USERNAME=${HOST_USERNAME:-"ubuntu"}
SSH_DIR=${SSH_DIR:-"/home/${HOST_USERNAME}/.ssh"}
[ -n "${TMP_HOOKS_PATH}" ] || die "Temp hook path not set"
# copy files over the "staging" area for the guest image (they'll later be put in the correct location by the guest user
# not these keys should not be overridden otherwise a) you won't be able to ssh in and b) the guest won't be able to
# rsync the files
if [ -f ${SSH_DIR}/id_rsa ]; then
dd if=${SSH_DIR}/authorized_keys of=${TMP_HOOKS_PATH}/ssh-authorized-keys
dd if=${SSH_DIR}/id_rsa of=${TMP_HOOKS_PATH}/ssh-id_rsa
dd if=${SSH_DIR}/id_rsa.pub of=${TMP_HOOKS_PATH}/ssh-id_rsa.pub
else
die "SSH Authorized Keys file must exist along with pub and private key"
fi

View File

@ -0,0 +1,22 @@
#!/bin/bash
# PURPOSE: Add the guest image user that will own the trove agent source if the
# user does not already exist
if [ ${DIB_DEBUG_TRACE:-1} -gt 0 ]; then
set -x
fi
set -e
set -o pipefail
GUEST_USERNAME=${GUEST_USERNAME:-"ubuntu"}
if ! id -u ${GUEST_USERNAME} >/dev/null 2>&1; then
echo "Adding ${GUEST_USERNAME} user"
useradd -G sudo -m ${GUEST_USERNAME} -s /bin/bash
chown ${GUEST_USERNAME}:${GUEST_USERNAME} /home/${GUEST_USERNAME}
passwd ${GUEST_USERNAME} <<_EOF_
${GUEST_USERNAME}
${GUEST_USERNAME}
_EOF_
fi

View File

@ -8,28 +8,35 @@ set -o pipefail
SCRIPTDIR=$(dirname $0) SCRIPTDIR=$(dirname $0)
GUEST_VENV=/opt/guest-agent-venv GUEST_VENV=/opt/guest-agent-venv
GUEST_USERNAME=${GUEST_USERNAME:-"ubuntu"}
# Create a virtual environment to contain the guest agent # Create a virtual environment for guest agent
${DIB_PYTHON} -m virtualenv $GUEST_VENV ${DIB_PYTHON} -m virtualenv ${GUEST_VENV}
$GUEST_VENV/bin/pip install pip --upgrade ${GUEST_VENV}/bin/pip install pip --upgrade
$GUEST_VENV/bin/pip install -U -c /opt/upper-constraints.txt /opt/guest-agent ${GUEST_VENV}/bin/pip install -U -c /opt/upper-constraints.txt /opt/guest-agent
chown -R ${GUEST_USERNAME}:root ${GUEST_VENV}
# Link the trove-guestagent out to /usr/local/bin where the startup scripts look # Link the trove-guestagent out to /usr/local/bin where the startup scripts look for
ln -s $GUEST_VENV/bin/trove-guestagent /usr/local/bin/guest-agent || true ln -s ${GUEST_VENV}/bin/trove-guestagent /usr/local/bin/guest-agent || true
mkdir -p /var/lib/trove /etc/trove/certs /var/log/trove for folder in "/var/lib/trove" "/etc/trove" "/etc/trove/certs" "/etc/trove/conf.d" "/var/log/trove"; do
mkdir -p ${folder}
chown -R ${GUEST_USERNAME}:root ${folder}
done
install -D -g root -o root -m 0644 ${SCRIPTDIR}/guest-agent.logrotate /etc/logrotate.d/guest-agent install -D -g root -o ${GUEST_USERNAME} -m 0644 ${SCRIPTDIR}/guest-agent.logrotate /etc/logrotate.d/guest-agent
case "$DIB_INIT_SYSTEM" in case "$DIB_INIT_SYSTEM" in
upstart)
install -D -g root -o root -m 0644 ${SCRIPTDIR}/guest-agent.conf /etc/init/guest-agent.conf
;;
systemd) systemd)
install -D -g root -o root -m 0644 ${SCRIPTDIR}/guest-agent.service /usr/lib/systemd/system/guest-agent.service mkdir -p /usr/lib/systemd/system
touch /usr/lib/systemd/system/guest-agent.service
sed "s/GUEST_USERNAME/${GUEST_USERNAME}/g" ${SCRIPTDIR}/guest-agent.service > /usr/lib/systemd/system/guest-agent.service
;;
upstart)
install -D -g root -o ${GUEST_USERNAME} -m 0644 ${SCRIPTDIR}/guest-agent.conf /etc/init/guest-agent.conf
;; ;;
sysv) sysv)
install -D -g root -o root -m 0644 ${SCRIPTDIR}/guest-agent.init /etc/init.d/guest-agent.init install -D -g root -o ${GUEST_USERNAME} -m 0644 ${SCRIPTDIR}/guest-agent.init /etc/init.d/guest-agent.init
;; ;;
*) *)
echo "Unsupported init system" echo "Unsupported init system"

View File

@ -4,11 +4,12 @@ After=network.target syslog.service
Wants=syslog.service Wants=syslog.service
[Service] [Service]
User=GUEST_USERNAME
Group=GUEST_USERNAME
ExecStartPre=/bin/bash -c "sudo chown -R GUEST_USERNAME:root /etc/trove/conf.d"
ExecStart=/usr/local/bin/guest-agent --config-dir=/etc/trove/conf.d ExecStart=/usr/local/bin/guest-agent --config-dir=/etc/trove/conf.d
KillMode=mixed KillMode=mixed
Restart=always Restart=always
ExecStartPost=/bin/sh -c "echo $MAINPID > /var/run/guest-agent.pid"
PIDFile=/var/run/guest-agent.pid
[Install] [Install]
WantedBy=multi-user.target WantedBy=multi-user.target

View File

@ -9,15 +9,6 @@ libssl-dev:
python-dev: python-dev:
installtype: source installtype: source
python-sqlalchemy:
python-lxml:
python-eventlet:
python-webob:
python-httplib2:
python-iso8601:
python-pexpect:
python-mysqldb:
python-migrate:
acl: acl:
acpid: acpid:
apparmor: apparmor:
@ -46,13 +37,14 @@ lsof:
net-tools: net-tools:
netbase: netbase:
netcat-openbsd: netcat-openbsd:
network-scripts:
open-vm-tools: open-vm-tools:
arch: i386, amd64
openssh-client: openssh-client:
openssh-server: openssh-server:
pollinate: pollinate:
psmisc: psmisc:
rsyslog: rsyslog:
screen:
socat: socat:
tcpdump: tcpdump:
ubuntu-cloudimage-keyring: ubuntu-cloudimage-keyring:

View File

@ -10,7 +10,7 @@
"cloud-guest-utils": "", "cloud-guest-utils": "",
"apparmor": "", "apparmor": "",
"dmeventd": "", "dmeventd": "",
"isc-dhcp-client": "", "isc-dhcp-client": "dhcp-client",
"uuid-runtime": "", "uuid-runtime": "",
"ubuntu-cloudimage-keyring": "", "ubuntu-cloudimage-keyring": "",
"vim-tiny": "", "vim-tiny": "",

View File

@ -1,39 +0,0 @@
#!/bin/bash
# CONTEXT: GUEST during CONSTRUCTION as ROOT
# PURPOSE: take "staged" ssh keys (see extra-data.d/62-ssh-key) and put them in the GUEST_USERS home directory
set -e
set -o xtrace
SSH_DIR="/home/${GUEST_USERNAME}/.ssh"
TMP_HOOKS_DIR="/tmp/in_target.d"
if ! id -u ${GUEST_USERNAME} >/dev/null 2>&1; then
echo "Adding ${GUEST_USERNAME} user"
useradd -G sudo -m ${GUEST_USERNAME} -s /bin/bash
chown ${GUEST_USERNAME}:${GUEST_USERNAME} /home/${GUEST_USERNAME}
passwd ${GUEST_USERNAME} <<_EOF_
${GUEST_USERNAME}
${GUEST_USERNAME}
_EOF_
fi
if [ -f "${TMP_HOOKS_DIR}/ssh-authorized-keys" ]; then
if [ ! -d ${SSH_DIR} ]; then
# this method worked more reliable in vmware fusion over doing sudo -Hiu ${GUEST_USERNAME}
mkdir ${SSH_DIR}
chown ${GUEST_USERNAME}:${GUEST_USERNAME} ${SSH_DIR}
fi
sudo -Hiu ${GUEST_USERNAME} dd of=${SSH_DIR}/authorized_keys conv=notrunc if=${TMP_HOOKS_DIR}/ssh-authorized-keys
if [ ! -f "${SSH_DIR}/id_rsa" ]; then
sudo -Hiu ${GUEST_USERNAME} dd of=${SSH_DIR}/id_rsa if=${TMP_HOOKS_DIR}/ssh-id_rsa
# perms have to be right on this file for ssh to work
sudo -Hiu ${GUEST_USERNAME} chmod 600 ${SSH_DIR}/id_rsa
sudo -Hiu ${GUEST_USERNAME} dd of=${SSH_DIR}/id_rsa.pub if=${TMP_HOOKS_DIR}/ssh-id_rsa.pub
fi
else
echo "SSH Keys were not staged by host"
exit -1
fi

View File

@ -7,5 +7,3 @@ set -e
set -o xtrace set -o xtrace
apt-get clean apt-get clean

View File

@ -1,11 +1,8 @@
This element clears out /etc/resolv.conf and prevents dhclient from populating This element clears out /etc/resolv.conf and prevents dhclient from populating
it with data from DHCP. This means that DNS resolution will not work from the it with data from DHCP. This means that DNS resolution will not work from the
amphora. This is OK because all outbound connections from the guest will guest. This is OK because all outbound connections from the guest will
be based using raw IP addresses. be based using raw IP addresses.
In addition we remove dns from the nsswitch.conf hosts setting. In addition we remove dns from the nsswitch.conf hosts setting.
This has the real benefit of speeding up host boot and configutation times. This means that the guest never waits for DNS timeouts to occur.
This is especially helpful when running tempest tests in a devstack environment
where DNS resolution from the guest usually doesn't work anyway. This means
that the guest never waits for DNS timeouts to occur.

View File

@ -5,6 +5,7 @@ set -e
#CONTEXT: chroot on host #CONTEXT: chroot on host
#PURPOSE: Allows mysqld to create temporary files when restoring backups #PURPOSE: Allows mysqld to create temporary files when restoring backups
mkdir -p /etc/apparmor.d/local/
cat <<EOF >>/etc/apparmor.d/local/usr.sbin.mysqld cat <<EOF >>/etc/apparmor.d/local/usr.sbin.mysqld
/tmp/ rw, /tmp/ rw,
/tmp/** rwk, /tmp/** rwk,

View File

@ -21,6 +21,8 @@ function build_vm() {
GUEST_CACHEDIR=${GUEST_CACHEDIR:-"$HOME/.cache/image-create"} GUEST_CACHEDIR=${GUEST_CACHEDIR:-"$HOME/.cache/image-create"}
GUEST_WORKING_DIR=${GUEST_WORKING_DIR:-"$HOME/images"} GUEST_WORKING_DIR=${GUEST_WORKING_DIR:-"$HOME/images"}
export GUEST_USERNAME=${guest_username}
# In dev mode, the trove guest agent needs to download trove code from # In dev mode, the trove guest agent needs to download trove code from
# trove-taskmanager host during service initialization. # trove-taskmanager host during service initialization.
if [[ "${dev_mode}" == "true" ]]; then if [[ "${dev_mode}" == "true" ]]; then
@ -34,7 +36,6 @@ function build_vm() {
export HOST_SCP_USERNAME=$(whoami) export HOST_SCP_USERNAME=$(whoami)
export HOST_USERNAME=${HOST_SCP_USERNAME} export HOST_USERNAME=${HOST_SCP_USERNAME}
export SSH_DIR=${SSH_DIR:-"$HOME/.ssh"} export SSH_DIR=${SSH_DIR:-"$HOME/.ssh"}
export GUEST_USERNAME=${guest_username}
manage_ssh_keys manage_ssh_keys
fi fi
@ -68,7 +69,6 @@ function build_vm() {
if [[ "${dev_mode}" == "false" ]]; then if [[ "${dev_mode}" == "false" ]]; then
elementes="$elementes pip-and-virtualenv" elementes="$elementes pip-and-virtualenv"
elementes="$elementes pip-cache" elementes="$elementes pip-cache"
elementes="$elementes no-resolvconf"
elementes="$elementes guest-agent" elementes="$elementes guest-agent"
else else
elementes="$elementes ${guest_os}-guest" elementes="$elementes ${guest_os}-guest"
@ -101,7 +101,7 @@ function build_guest_image() {
dev_mode=${4:-"true"} dev_mode=${4:-"true"}
guest_username=${5:-"ubuntu"} guest_username=${5:-"ubuntu"}
exclaim "Building a ${datastore_type} image of trove guest agent for ${guest_os} ${guest_release}." exclaim "Building a ${datastore_type} image of trove guest agent for ${guest_os} ${guest_release}, dev_mode=${dev_mode}"
VALID_SERVICES='mysql percona mariadb redis cassandra couchbase mongodb postgresql couchdb vertica db2 pxc' VALID_SERVICES='mysql percona mariadb redis cassandra couchbase mongodb postgresql couchdb vertica db2 pxc'
if ! [[ " $VALID_SERVICES " =~ " $datastore_type " ]]; then if ! [[ " $VALID_SERVICES " =~ " $datastore_type " ]]; then
@ -109,7 +109,7 @@ function build_guest_image() {
exit 1 exit 1
fi fi
image_name=${guest_os}_${datastore_type} image_name=${guest_os}-${datastore_type}
image_folder=$HOME/images image_folder=$HOME/images
mkdir -p $image_folder mkdir -p $image_folder
image_path=${image_folder}/${image_name} image_path=${image_folder}/${image_name}

View File

@ -850,19 +850,27 @@ function cmd_build_and_upload_image() {
exit 1 exit 1
fi fi
glance_imageid=$(openstack $CLOUD_ADMIN_ARG image list | grep "$datastore_type" | get_field 1) image_var="${datastore_type^^}_IMAGE_ID"
echo "IMAGEID: $glance_imageid" glance_imageid=`eval echo '$'"$image_var"`
if [[ -z $glance_imageid ]]; then
build_guest_image ${datastore_type} ${guest_os} ${guest_release} ${dev_mode} ${guest_username}
image_folder=$HOME/images if [[ -z $glance_imageid ]]; then
qcow_image=`find $image_folder -name '*.qcow2'` # Find the first image id with the name contains datastore_type.
image_url="file://$qcow_image" glance_imageid=$(openstack $CLOUD_ADMIN_ARG image list | grep "$datastore_type" | awk 'NR==1 {print}' | awk '{print $2}')
glance_imageid=`get_glance_id upload_image $image_url`
[[ -z "$glance_imageid" ]] && echo "Glance upload failed!" && exit 1 if [[ -z $glance_imageid ]]; then
echo "IMAGE ID: $glance_imageid" build_guest_image ${datastore_type} ${guest_os} ${guest_release} ${dev_mode} ${guest_username}
image_folder=$HOME/images
qcow_image=`find $image_folder -name '*.qcow2'`
image_url="file://$qcow_image"
glance_imageid=`get_glance_id upload_image $image_url`
[[ -z "$glance_imageid" ]] && echo "Glance upload failed!" && exit 1
echo "IMAGE ID: $glance_imageid"
fi
fi fi
echo "IMAGEID: $glance_imageid"
exclaim "Updating Datastores" exclaim "Updating Datastores"
cmd_set_datastore "${glance_imageid}" "${datastore_type}" "${restart_trove}" cmd_set_datastore "${glance_imageid}" "${datastore_type}" "${restart_trove}"
} }
@ -1257,7 +1265,12 @@ function cmd_kick_start() {
exclaim "Running kick-start for $DATASTORE_TYPE (restart trove: $RESTART_TROVE)" exclaim "Running kick-start for $DATASTORE_TYPE (restart trove: $RESTART_TROVE)"
dump_env dump_env
cmd_test_init "${DATASTORE_TYPE}" cmd_test_init "${DATASTORE_TYPE}"
cmd_build_and_upload_image "${DATASTORE_TYPE}" "${RESTART_TROVE}"
export GUEST_OS=${GUEST_OS:-"ubuntu"}
export GUEST_OS_RELEASE=${GUEST_OS_RELEASE:-"xenial"}
export GUEST_OS_USERNAME=${GUEST_OS_USERNAME:-"ubuntu"}
export DEV_MOEE=${DEV_MODE:-"true"}
cmd_build_and_upload_image "${DATASTORE_TYPE}" "${RESTART_TROVE}" "${GUEST_OS}" "${GUEST_OS_RELEASE}" "${DEV_MOEE}" "${GUEST_OS_USERNAME}"
} }
function cmd_gate_tests() { function cmd_gate_tests() {

View File

@ -108,3 +108,6 @@ SWIFT_DISK_IMAGE=${SWIFT_DATA_DIR}/drives/images/swift.img
#export TROVE_RESIZE_TIME_OUT=3600 #export TROVE_RESIZE_TIME_OUT=3600
#export TROVE_USAGE_TIMEOUT=1500 #export TROVE_USAGE_TIMEOUT=1500
#export TROVE_STATE_CHANGE_WAIT_TIME=180 #export TROVE_STATE_CHANGE_WAIT_TIME=180
# Image
MYSQL_IMAGE_ID=${MYSQL_IMAGE_ID:-""}

View File

@ -125,7 +125,6 @@ def import_tests():
if not ADD_DOMAINS: if not ADD_DOMAINS:
from tests.api import delete_all from tests.api import delete_all
from tests.api import instances_pagination from tests.api import instances_pagination
from tests.api import instances_quotas
from tests.api import instances_states from tests.api import instances_states
from tests.dns import dns from tests.dns import dns
from tests import initialize from tests import initialize

View File

@ -1,47 +0,0 @@
from proboscis import before_class
from proboscis import test
from proboscis.asserts import assert_raises
from troveclient.compat import exceptions
from trove.tests.config import CONFIG
from trove.tests.util import create_client
@test(groups=['dbaas.api.instances.quotas'])
class InstanceQuotas(object):
created_instances = []
@before_class
def setup(self):
self.client = create_client(is_admin=False)
@test
def test_too_many_instances(self):
self.created_instances = []
if 'trove_max_instances_per_user' in CONFIG.values:
too_many = CONFIG.values['trove_max_instances_per_user']
already_there = len(self.client.instances.list())
flavor = 1
for i in range(too_many - already_there):
response = self.client.instances.create('too_many_%d' % i,
flavor,
{'size': 1})
self.created_instances.append(response)
# This one better fail, because we just reached our quota.
assert_raises(exceptions.OverLimit,
self.client.instances.create,
"too_many", flavor,
{'size': 1})
@test(runs_after=[test_too_many_instances])
def delete_excessive_entries(self):
# Delete all the instances called too_many*.
for id in self.created_instances:
while True:
try:
self.client.instances.delete(id)
except exceptions.UnprocessableEntity:
continue
except exceptions.NotFound:
break

View File

@ -206,7 +206,6 @@ def import_tests():
from trove.tests.api import configurations # noqa from trove.tests.api import configurations # noqa
from trove.tests.api import databases # noqa from trove.tests.api import databases # noqa
from trove.tests.api import datastores # noqa from trove.tests.api import datastores # noqa
from trove.tests.api import flavors # noqa
from trove.tests.api import header # noqa from trove.tests.api import header # noqa
from trove.tests.api import instances as rd_instances # noqa from trove.tests.api import instances as rd_instances # noqa
from trove.tests.api import instances_actions as rd_actions # noqa from trove.tests.api import instances_actions as rd_actions # noqa

View File

@ -447,6 +447,15 @@ class DBaaSInstanceCreate(DBaaSAPINotification):
return ['instance_id'] return ['instance_id']
class DBaaSInstanceReboot(DBaaSAPINotification):
def event_type(self):
return 'instance_reboot'
def required_start_traits(self):
return ['instance_id']
class DBaaSInstanceRestart(DBaaSAPINotification): class DBaaSInstanceRestart(DBaaSAPINotification):
def event_type(self): def event_type(self):

View File

@ -131,7 +131,14 @@ class MgmtInstanceController(InstanceController):
def _action_reboot(self, context, instance, req, body): def _action_reboot(self, context, instance, req, body):
LOG.debug("Rebooting instance %s.", instance.id) LOG.debug("Rebooting instance %s.", instance.id)
instance.reboot()
context.notification = notification.DBaaSInstanceReboot(
context,
request=req
)
with StartNotification(context, instance_id=instance.id):
instance.reboot()
return wsgi.Result(None, 202) return wsgi.Result(None, 202)
def _action_migrate(self, context, instance, req, body): def _action_migrate(self, context, instance, req, body):

View File

@ -36,13 +36,11 @@ from trove.instance import models, views
from trove.module import models as module_models from trove.module import models as module_models
from trove.module import views as module_views from trove.module import views as module_views
CONF = cfg.CONF CONF = cfg.CONF
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
class InstanceController(wsgi.Controller): class InstanceController(wsgi.Controller):
"""Controller for instance functionality.""" """Controller for instance functionality."""
schemas = apischema.instance.copy() schemas = apischema.instance.copy()
@ -86,7 +84,7 @@ class InstanceController(wsgi.Controller):
'restart': self._action_restart, 'restart': self._action_restart,
'resize': self._action_resize, 'resize': self._action_resize,
'promote_to_replica_source': 'promote_to_replica_source':
self._action_promote_to_replica_source, self._action_promote_to_replica_source,
'eject_replica_source': self._action_eject_replica_source, 'eject_replica_source': self._action_eject_replica_source,
'reset_status': self._action_reset_status, 'reset_status': self._action_reset_status,
} }
@ -478,7 +476,7 @@ class InstanceController(wsgi.Controller):
LOG.debug("Default config for instance %(instance_id)s is %(config)s", LOG.debug("Default config for instance %(instance_id)s is %(config)s",
{'instance_id': id, 'config': config}) {'instance_id': id, 'config': config})
return wsgi.Result(views.DefaultConfigurationView( return wsgi.Result(views.DefaultConfigurationView(
config).data(), 200) config).data(), 200)
def guest_log_list(self, req, tenant_id, id): def guest_log_list(self, req, tenant_id, id):
"""Return all information about all logs for an instance.""" """Return all information about all logs for an instance."""

View File

@ -1259,7 +1259,6 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin):
def reboot(self): def reboot(self):
try: try:
# Issue a guest stop db call to shutdown the db if running
LOG.debug("Stopping datastore on instance %s.", self.id) LOG.debug("Stopping datastore on instance %s.", self.id)
try: try:
self.guest.stop_db() self.guest.stop_db()
@ -1268,15 +1267,27 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin):
# Also we check guest state before issuing reboot # Also we check guest state before issuing reboot
LOG.debug(str(e)) LOG.debug(str(e))
self._refresh_datastore_status() # Wait for the mysql stopped.
if not (self.datastore_status_matches( def _datastore_is_offline():
rd_instance.ServiceStatuses.SHUTDOWN) or self._refresh_datastore_status()
return (
self.datastore_status_matches( self.datastore_status_matches(
rd_instance.ServiceStatuses.CRASHED)): rd_instance.ServiceStatuses.SHUTDOWN) or
# We will bail if db did not get stopped or is blocked self.datastore_status_matches(
LOG.error("Cannot reboot instance. DB status is %s.", rd_instance.ServiceStatuses.CRASHED)
)
try:
utils.poll_until(
_datastore_is_offline,
sleep_time=3,
time_out=CONF.reboot_time_out
)
except exception.PollTimeOut:
LOG.error("Cannot reboot instance, DB status is %s",
self.datastore_status.status) self.datastore_status.status)
return return
LOG.debug("The guest service status is %s.", LOG.debug("The guest service status is %s.",
self.datastore_status.status) self.datastore_status.status)
@ -1291,7 +1302,7 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin):
utils.poll_until( utils.poll_until(
update_server_info, update_server_info,
sleep_time=2, sleep_time=3,
time_out=reboot_time_out) time_out=reboot_time_out)
# Set the status to PAUSED. The guest agent will reset the status # Set the status to PAUSED. The guest agent will reset the status
@ -1302,7 +1313,6 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin):
LOG.error("Failed to reboot instance %(id)s: %(e)s", LOG.error("Failed to reboot instance %(id)s: %(e)s",
{'id': self.id, 'e': str(e)}) {'id': self.id, 'e': str(e)})
finally: finally:
LOG.debug("Rebooting FINALLY %s", self.id)
self.reset_task_status() self.reset_task_status()
def restart(self): def restart(self):

View File

@ -53,7 +53,7 @@ backup_count_prior_to_create = 0
backup_count_for_instance_prior_to_create = 0 backup_count_for_instance_prior_to_create = 0
@test(depends_on_groups=[instances_actions.GROUP_STOP_MYSQL], @test(depends_on_groups=[instances_actions.GROUP_RESIZE],
groups=[BACKUP_GROUP, tests.INSTANCES], groups=[BACKUP_GROUP, tests.INSTANCES],
enabled=CONFIG.swift_enabled) enabled=CONFIG.swift_enabled)
class CreateBackups(object): class CreateBackups(object):
@ -380,7 +380,7 @@ class DeleteRestoreInstance(object):
assert_raises(exceptions.NotFound, instance_info.dbaas.instances.get, assert_raises(exceptions.NotFound, instance_info.dbaas.instances.get,
instance_id) instance_id)
@test(runs_after=[VerifyRestore.test_database_restored_incremental]) @test(depends_on=[VerifyRestore.test_database_restored_incremental])
def test_delete_restored_instance_incremental(self): def test_delete_restored_instance_incremental(self):
try: try:
self._delete(incremental_restore_instance_id) self._delete(incremental_restore_instance_id)

View File

@ -1,265 +0,0 @@
# Copyright (c) 2011 OpenStack Foundation
# 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.
import os
from nose.tools import assert_equal
from nose.tools import assert_false
from nose.tools import assert_true
from proboscis.asserts import assert_raises
from proboscis import before_class
from proboscis.decorators import time_out
from proboscis import test
from trove.common.utils import poll_until
from trove import tests
from trove.tests.api.instances import TIMEOUT_INSTANCE_CREATE
from trove.tests.config import CONFIG
from trove.tests.util.check import AttrCheck
from trove.tests.util import create_dbaas_client
from trove.tests.util import create_nova_client
from trove.tests.util import test_config
from trove.tests.util.users import Requirements
from troveclient.compat import exceptions
from troveclient.v1.flavors import Flavor
GROUP = "dbaas.api.flavors"
GROUP_DS = "dbaas.api.datastores"
FAKE_MODE = test_config.values['fake_mode']
servers_flavors = None
dbaas_flavors = None
user = None
def assert_attributes_equal(name, os_flavor, dbaas_flavor):
"""Given an attribute name and two objects,
ensures the attribute is equal.
"""
assert_true(hasattr(os_flavor, name),
"open stack flavor did not have attribute %s" % name)
assert_true(hasattr(dbaas_flavor, name),
"dbaas flavor did not have attribute %s" % name)
expected = getattr(os_flavor, name)
actual = getattr(dbaas_flavor, name)
assert_equal(expected, actual,
'DBaas flavor differs from Open Stack on attribute ' + name)
def assert_flavors_roughly_equivalent(os_flavor, dbaas_flavor):
assert_attributes_equal('name', os_flavor, dbaas_flavor)
assert_attributes_equal('ram', os_flavor, dbaas_flavor)
def assert_link_list_is_equal(flavor):
assert_true(hasattr(flavor, 'links'))
assert_true(flavor.links)
if flavor.id:
flavor_id = str(flavor.id)
else:
flavor_id = flavor.str_id
for link in flavor.links:
href = link['href']
if "self" in link['rel']:
expected_href = os.path.join(test_config.dbaas_url, "flavors",
str(flavor.id))
url = test_config.dbaas_url.replace('http:', 'https:', 1)
msg = ("REL HREF %s doesn't start with %s" %
(href, test_config.dbaas_url))
assert_true(href.startswith(url), msg)
url = os.path.join("flavors", flavor_id)
msg = "REL HREF %s doesn't end in '%s'" % (href, url)
assert_true(href.endswith(url), msg)
elif "bookmark" in link['rel']:
base_url = test_config.version_url.replace('http:', 'https:', 1)
expected_href = os.path.join(base_url, "flavors", flavor_id)
msg = 'bookmark "href" must be %s, not %s' % (expected_href, href)
assert_equal(href, expected_href, msg)
else:
assert_false(True, "Unexpected rel - %s" % link['rel'])
@test(groups=[tests.DBAAS_API, GROUP, GROUP_DS, tests.PRE_INSTANCES],
depends_on_groups=["services.initialize"])
class Flavors(object):
@before_class
def setUp(self):
rd_user = test_config.users.find_user(
Requirements(is_admin=False, services=["trove"]))
self.rd_client = create_dbaas_client(rd_user)
if test_config.nova_client is not None:
nova_user = test_config.users.find_user(
Requirements(services=["nova"]))
self.nova_client = create_nova_client(nova_user)
def get_expected_flavors(self):
# If we have access to the client, great! Let's use that as the flavors
# returned by Trove should be identical.
if test_config.nova_client is not None:
return self.nova_client.flavors.list()
# If we don't have access to the client the flavors need to be spelled
# out in the config file.
flavors = [Flavor(Flavors, flavor_dict, loaded=True)
for flavor_dict in test_config.flavors]
return flavors
@test
def confirm_flavors_lists_nearly_identical(self):
os_flavors = self.get_expected_flavors()
dbaas_flavors = self.rd_client.flavors.list()
print("Open Stack Flavors:")
print(os_flavors)
print("DBaaS Flavors:")
print(dbaas_flavors)
# Length of both flavors list should be identical.
assert_equal(len(os_flavors), len(dbaas_flavors))
for os_flavor in os_flavors:
found_index = None
for index, dbaas_flavor in enumerate(dbaas_flavors):
if os_flavor.name == dbaas_flavor.name:
msg = ("Flavor ID '%s' appears in elements #%s and #%d." %
(dbaas_flavor.id, str(found_index), index))
assert_true(found_index is None, msg)
assert_flavors_roughly_equivalent(os_flavor, dbaas_flavor)
found_index = index
msg = "Some flavors from OS list were missing in DBAAS list."
assert_false(found_index is None, msg)
for flavor in dbaas_flavors:
assert_link_list_is_equal(flavor)
@test
def test_flavor_list_attrs(self):
allowed_attrs = ['id', 'name', 'ram', 'vcpus', 'disk', 'links',
'ephemeral', 'local_storage', 'str_id']
flavors = self.rd_client.flavors.list()
attrcheck = AttrCheck()
for flavor in flavors:
flavor_dict = flavor._info
attrcheck.contains_allowed_attrs(
flavor_dict, allowed_attrs,
msg="Flavors list")
attrcheck.links(flavor_dict['links'])
@test
def test_flavor_get_attrs(self):
allowed_attrs = ['id', 'name', 'ram', 'vcpus', 'disk', 'links',
'ephemeral', 'local_storage', 'str_id']
flavor = self.rd_client.flavors.get(1)
attrcheck = AttrCheck()
flavor_dict = flavor._info
attrcheck.contains_allowed_attrs(
flavor_dict, allowed_attrs,
msg="Flavor Get 1")
attrcheck.links(flavor_dict['links'])
@test
def test_flavor_not_found(self):
assert_raises(exceptions.NotFound,
self.rd_client.flavors.get, "foo")
@test
def test_flavor_list_datastore_version_associated_flavors(self):
datastore = self.rd_client.datastores.get(
test_config.dbaas_datastore)
dbaas_flavors = (self.rd_client.flavors.
list_datastore_version_associated_flavors(
datastore=test_config.dbaas_datastore,
version_id=datastore.default_version))
os_flavors = self.get_expected_flavors()
assert_equal(len(dbaas_flavors), len(os_flavors))
# verify flavor lists are identical
for os_flavor in os_flavors:
found_index = None
for index, dbaas_flavor in enumerate(dbaas_flavors):
if os_flavor.name == dbaas_flavor.name:
msg = ("Flavor ID '%s' appears in elements #%s and #%d." %
(dbaas_flavor.id, str(found_index), index))
assert_true(found_index is None, msg)
assert_flavors_roughly_equivalent(os_flavor, dbaas_flavor)
found_index = index
msg = "Some flavors from OS list were missing in DBAAS list."
assert_false(found_index is None, msg)
for flavor in dbaas_flavors:
assert_link_list_is_equal(flavor)
@test(runs_after=[Flavors],
groups=[tests.DBAAS_API, GROUP, GROUP_DS],
depends_on_groups=["services.initialize"],
enabled=FAKE_MODE)
class DatastoreFlavorAssociation(object):
@before_class
def setUp(self):
rd_user = test_config.users.find_user(
Requirements(is_admin=False, services=["trove"]))
self.rd_client = create_dbaas_client(rd_user)
self.datastore = self.rd_client.datastores.get(
test_config.dbaas_datastore)
self.name1 = "test_instance1"
self.name2 = "test_instance2"
self.volume = {'size': 2}
self.instance_id = None
self.nics = None
shared_network = CONFIG.get('shared_network', None)
if shared_network:
self.nics = [{'net-id': shared_network}]
@test
@time_out(TIMEOUT_INSTANCE_CREATE)
def test_create_instance_with_valid_flavor_association(self):
# all the nova flavors are associated with the default datastore
result = self.rd_client.instances.create(
name=self.name1, flavor_id='1', volume=self.volume,
datastore=self.datastore.id,
nics=self.nics)
self.instance_id = result.id
assert_equal(200, self.rd_client.last_http_code)
def result_is_active():
instance = self.rd_client.instances.get(self.instance_id)
if instance.status == "ACTIVE":
return True
else:
# If its not ACTIVE, anything but BUILD must be
# an error.
assert_equal("BUILD", instance.status)
return False
poll_until(result_is_active)
self.rd_client.instances.delete(self.instance_id)
@test(runs_after=[test_create_instance_with_valid_flavor_association])
def test_create_instance_with_invalid_flavor_association(self):
dbaas_flavors = (self.rd_client.flavors.
list_datastore_version_associated_flavors(
datastore=test_config.dbaas_datastore,
version_id=self.datastore.default_version))
self.flavor_not_associated = None
os_flavors = Flavors().get_expected_flavors()
for os_flavor in os_flavors:
if os_flavor not in dbaas_flavors:
self.flavor_not_associated = os_flavor.id
break
if self.flavor_not_associated is not None:
assert_raises(exceptions.BadRequest,
self.rd_client.instances.create, self.name2,
flavor_not_associated, self.volume,
datastore=self.datastore.id,
nics=self.nics)

View File

@ -16,11 +16,9 @@
import netaddr import netaddr
import os import os
import time import time
from time import sleep
import unittest import unittest
import uuid import uuid
from proboscis import after_class
from proboscis.asserts import assert_equal from proboscis.asserts import assert_equal
from proboscis.asserts import assert_false from proboscis.asserts import assert_false
from proboscis.asserts import assert_is_not_none from proboscis.asserts import assert_is_not_none
@ -35,7 +33,6 @@ from proboscis import test
from troveclient.compat import exceptions from troveclient.compat import exceptions
from trove.common import cfg from trove.common import cfg
from trove.common import exception as rd_exceptions
from trove.common.utils import poll_until from trove.common.utils import poll_until
from trove.datastore import models as datastore_models from trove.datastore import models as datastore_models
from trove import tests from trove import tests
@ -116,15 +113,17 @@ class InstanceTestInfo(object):
'eph.rd-tiny') 'eph.rd-tiny')
else: else:
flavor_name = CONFIG.values.get('instance_flavor_name', 'm1.tiny') flavor_name = CONFIG.values.get('instance_flavor_name', 'm1.tiny')
flavors = self.dbaas.find_flavors_by_name(flavor_name) flavors = self.dbaas.find_flavors_by_name(flavor_name)
assert_equal(len(flavors), 1, assert_equal(len(flavors), 1,
"Number of flavors with name '%s' " "Number of flavors with name '%s' "
"found was '%d'." % (flavor_name, len(flavors))) "found was '%d'." % (flavor_name, len(flavors)))
flavor = flavors[0] flavor = flavors[0]
assert_true(flavor is not None, "Flavor '%s' not found!" % flavor_name)
flavor_href = self.dbaas.find_flavor_self_href(flavor) flavor_href = self.dbaas.find_flavor_self_href(flavor)
assert_true(flavor_href is not None, assert_true(flavor_href is not None,
"Flavor href '%s' not found!" % flavor_name) "Flavor href '%s' not found!" % flavor_name)
return flavor, flavor_href return flavor, flavor_href
def get_address(self, mgmt=False): def get_address(self, mgmt=False):
@ -255,17 +254,10 @@ def test_delete_instance_not_found():
groups=[GROUP, GROUP_QUOTAS], groups=[GROUP, GROUP_QUOTAS],
runs_after_groups=[tests.PRE_INSTANCES]) runs_after_groups=[tests.PRE_INSTANCES])
class CreateInstanceQuotaTest(unittest.TestCase): class CreateInstanceQuotaTest(unittest.TestCase):
def setUp(self):
import copy
self.test_info = copy.deepcopy(instance_info)
self.test_info.dbaas_datastore = CONFIG.dbaas_datastore
def tearDown(self): def tearDown(self):
quota_dict = {'instances': CONFIG.trove_max_instances_per_tenant, quota_dict = {'instances': CONFIG.trove_max_instances_per_tenant,
'volumes': CONFIG.trove_max_volumes_per_tenant} 'volumes': CONFIG.trove_max_volumes_per_tenant}
dbaas_admin.quota.update(self.test_info.user.tenant_id, dbaas_admin.quota.update(instance_info.user.tenant_id,
quota_dict) quota_dict)
def test_instance_size_too_big(self): def test_instance_size_too_big(self):
@ -273,52 +265,48 @@ class CreateInstanceQuotaTest(unittest.TestCase):
VOLUME_SUPPORT): VOLUME_SUPPORT):
too_big = CONFIG.trove_max_accepted_volume_size too_big = CONFIG.trove_max_accepted_volume_size
self.test_info.volume = {'size': too_big + 1}
self.test_info.name = "way_too_large"
assert_raises(exceptions.OverLimit, assert_raises(exceptions.OverLimit,
dbaas.instances.create, dbaas.instances.create,
self.test_info.name, "volume_size_too_large",
self.test_info.dbaas_flavor_href, instance_info.dbaas_flavor_href,
self.test_info.volume, {'size': too_big + 1},
nics=instance_info.nics) nics=instance_info.nics)
def test_update_quota_invalid_resource_should_fail(self): def test_update_quota_invalid_resource_should_fail(self):
quota_dict = {'invalid_resource': 100} quota_dict = {'invalid_resource': 100}
assert_raises(exceptions.NotFound, dbaas_admin.quota.update, assert_raises(exceptions.NotFound, dbaas_admin.quota.update,
self.test_info.user.tenant_id, quota_dict) instance_info.user.tenant_id, quota_dict)
def test_update_quota_volume_should_fail_volume_not_supported(self): def test_update_quota_volume_should_fail_volume_not_supported(self):
if VOLUME_SUPPORT: if VOLUME_SUPPORT:
raise SkipTest("Volume support needs to be disabled") raise SkipTest("Volume support needs to be disabled")
quota_dict = {'volumes': 100} quota_dict = {'volumes': 100}
assert_raises(exceptions.NotFound, dbaas_admin.quota.update, assert_raises(exceptions.NotFound, dbaas_admin.quota.update,
self.test_info.user.tenant_id, quota_dict) instance_info.user.tenant_id, quota_dict)
def test_create_too_many_instances(self): def test_create_too_many_instances(self):
instance_quota = 0 instance_quota = 0
quota_dict = {'instances': instance_quota} quota_dict = {'instances': instance_quota}
new_quotas = dbaas_admin.quota.update(self.test_info.user.tenant_id, new_quotas = dbaas_admin.quota.update(instance_info.user.tenant_id,
quota_dict) quota_dict)
set_quota = dbaas_admin.quota.show(self.test_info.user.tenant_id) set_quota = dbaas_admin.quota.show(instance_info.user.tenant_id)
verify_quota = {q.resource: q.limit for q in set_quota} verify_quota = {q.resource: q.limit for q in set_quota}
assert_equal(new_quotas['instances'], quota_dict['instances']) assert_equal(new_quotas['instances'], quota_dict['instances'])
assert_equal(0, verify_quota['instances']) assert_equal(0, verify_quota['instances'])
self.test_info.volume = None
volume = None
if VOLUME_SUPPORT: if VOLUME_SUPPORT:
assert_equal(CONFIG.trove_max_volumes_per_tenant, assert_equal(CONFIG.trove_max_volumes_per_tenant,
verify_quota['volumes']) verify_quota['volumes'])
self.test_info.volume = {'size': volume = {'size': CONFIG.get('trove_volume_size', 1)}
CONFIG.get('trove_volume_size', 1)}
self.test_info.name = "too_many_instances"
assert_raises(exceptions.OverLimit, assert_raises(exceptions.OverLimit,
dbaas.instances.create, dbaas.instances.create,
self.test_info.name, "too_many_instances",
self.test_info.dbaas_flavor_href, instance_info.dbaas_flavor_href,
self.test_info.volume, volume,
nics=instance_info.nics) nics=instance_info.nics)
assert_equal(413, dbaas.last_http_code) assert_equal(413, dbaas.last_http_code)
@ -328,17 +316,15 @@ class CreateInstanceQuotaTest(unittest.TestCase):
raise SkipTest("Volume support not enabled") raise SkipTest("Volume support not enabled")
volume_quota = 3 volume_quota = 3
quota_dict = {'volumes': volume_quota} quota_dict = {'volumes': volume_quota}
self.test_info.volume = {'size': volume_quota + 1} new_quotas = dbaas_admin.quota.update(instance_info.user.tenant_id,
new_quotas = dbaas_admin.quota.update(self.test_info.user.tenant_id,
quota_dict) quota_dict)
assert_equal(volume_quota, new_quotas['volumes']) assert_equal(volume_quota, new_quotas['volumes'])
self.test_info.name = "too_large_volume"
assert_raises(exceptions.OverLimit, assert_raises(exceptions.OverLimit,
dbaas.instances.create, dbaas.instances.create,
self.test_info.name, "too_large_volume",
self.test_info.dbaas_flavor_href, instance_info.dbaas_flavor_href,
self.test_info.volume, {'size': volume_quota + 1},
nics=instance_info.nics) nics=instance_info.nics)
assert_equal(413, dbaas.last_http_code) assert_equal(413, dbaas.last_http_code)
@ -474,6 +460,7 @@ class CreateInstanceFail(object):
databases = [] databases = []
flavor_name = CONFIG.values.get('instance_flavor_name', 'm1.tiny') flavor_name = CONFIG.values.get('instance_flavor_name', 'm1.tiny')
flavors = dbaas.find_flavors_by_name(flavor_name) flavors = dbaas.find_flavors_by_name(flavor_name)
assert_raises(exceptions.BadRequest, dbaas.instances.create, assert_raises(exceptions.BadRequest, dbaas.instances.create,
instance_name, flavors[0].id, None, databases, instance_name, flavors[0].id, None, databases,
nics=instance_info.nics) nics=instance_info.nics)
@ -1508,86 +1495,3 @@ class CheckInstance(AttrCheck):
slave, allowed_attrs, slave, allowed_attrs,
msg="Replica links not found") msg="Replica links not found")
self.links(slave['links']) self.links(slave['links'])
@test(groups=[GROUP])
class BadInstanceStatusBug(object):
@before_class()
def setUp(self):
self.instances = []
reqs = Requirements(is_admin=True)
self.user = CONFIG.users.find_user(
reqs, black_list=[])
self.client = create_dbaas_client(self.user)
self.mgmt = self.client.management
@test
def test_instance_status_after_double_migrate(self):
"""
This test is to verify that instance status returned is more
informative than 'Status is {}'. There are several ways to
replicate this error. A double migration is just one of them but
since this is a known way to recreate that error we will use it
here to be sure that the error is fixed. The actual code lives
in trove/instance/models.py in _validate_can_perform_action()
"""
# TODO(imsplitbit): test other instances where this issue could be
# replicated. Resizing a resized instance awaiting confirmation
# can be used as another case. This all boils back to the same
# piece of code so I'm not sure if it's relevant or not but could
# be done.
size = None
if VOLUME_SUPPORT:
size = {'size': 5}
result = self.client.instances.create('testbox',
instance_info.dbaas_flavor_href,
size,
nics=instance_info.nics)
id = result.id
self.instances.append(id)
def verify_instance_is_active():
result = self.client.instances.get(id)
print(result.status)
return result.status == 'ACTIVE'
def attempt_migrate():
print('attempting migration')
try:
self.mgmt.migrate(id)
except exceptions.UnprocessableEntity:
return False
return True
# Timing necessary to make the error occur
poll_until(verify_instance_is_active, time_out=120, sleep_time=1)
try:
poll_until(attempt_migrate, time_out=10, sleep_time=1)
except rd_exceptions.PollTimeOut:
fail('Initial migration timed out')
try:
self.mgmt.migrate(id)
except exceptions.UnprocessableEntity as err:
assert('status was {}' not in err.message)
else:
# If we are trying to test what status is returned when an
# instance is in a confirm_resize state and another
# migration is attempted then we also need to
# assert that an exception is raised when running migrate.
# If one is not then we aren't able to test what the
# returned status is in the exception message.
fail('UnprocessableEntity was not thrown')
@after_class(always_run=True)
def tearDown(self):
while len(self.instances) > 0:
for id in self.instances:
try:
self.client.instances.delete(id)
self.instances.remove(id)
except exceptions.UnprocessableEntity:
sleep(1.0)

View File

@ -45,6 +45,7 @@ GROUP_REBOOT = "dbaas.api.instances.actions.reboot"
GROUP_RESTART = "dbaas.api.instances.actions.restart" GROUP_RESTART = "dbaas.api.instances.actions.restart"
GROUP_RESIZE = "dbaas.api.instances.actions.resize" GROUP_RESIZE = "dbaas.api.instances.actions.resize"
GROUP_STOP_MYSQL = "dbaas.api.instances.actions.stop" GROUP_STOP_MYSQL = "dbaas.api.instances.actions.stop"
GROUP_UPDATE_GUEST = "dbaas.api.instances.actions.update_guest"
MYSQL_USERNAME = "test_user" MYSQL_USERNAME = "test_user"
MYSQL_PASSWORD = "abcde" MYSQL_PASSWORD = "abcde"
# stored in test conf # stored in test conf
@ -104,7 +105,6 @@ def get_resize_timeout():
TIME_OUT_TIME = get_resize_timeout() TIME_OUT_TIME = get_resize_timeout()
USER_WAS_DELETED = False
class ActionTestBase(object): class ActionTestBase(object):
@ -223,23 +223,25 @@ class RebootTestBase(ActionTestBase):
def call_reboot(self): def call_reboot(self):
raise NotImplementedError() raise NotImplementedError()
def wait_for_broken_connection(self):
"""Wait until our connection breaks."""
if not USE_IP:
return
if not hasattr(self, "connection"):
return
poll_until(self.connection.is_connected,
lambda connected: not connected,
time_out=TIME_OUT_TIME)
def wait_for_successful_restart(self): def wait_for_successful_restart(self):
"""Wait until status becomes running.""" """Wait until status becomes running.
def is_finished_rebooting():
Reboot is an async operation, make sure the instance is rebooting
before active.
"""
def _is_rebooting():
instance = self.instance instance = self.instance
if instance.status == "REBOOT": if instance.status == "REBOOT":
return True
return False
poll_until(_is_rebooting, time_out=TIME_OUT_TIME)
def is_finished_rebooting():
instance = self.instance
asserts.assert_not_equal(instance.status, "ERROR")
if instance.status != "ACTIVE":
return False return False
asserts.assert_equal("ACTIVE", instance.status)
return True return True
poll_until(is_finished_rebooting, time_out=TIME_OUT_TIME) poll_until(is_finished_rebooting, time_out=TIME_OUT_TIME)
@ -253,45 +255,10 @@ class RebootTestBase(ActionTestBase):
def successful_restart(self): def successful_restart(self):
"""Restart MySQL via the REST API successfully.""" """Restart MySQL via the REST API successfully."""
self.fix_mysql()
self.call_reboot() self.call_reboot()
self.wait_for_broken_connection()
self.wait_for_successful_restart() self.wait_for_successful_restart()
self.assert_mysql_proc_is_different() self.assert_mysql_proc_is_different()
def mess_up_mysql(self):
"""Ruin MySQL's ability to restart."""
server = create_server_connection(self.instance_id,
self.instance_mgmt_address)
cmd_template = "sudo cp /dev/null /var/lib/mysql/data/ib_logfile%d"
instance_info.dbaas_admin.management.stop(self.instance_id)
for index in range(2):
cmd = cmd_template % index
try:
server.execute(cmd)
except Exception as e:
asserts.fail("Failed to execute command %s, error: %s" %
(cmd, str(e)))
def fix_mysql(self):
"""Fix MySQL's ability to restart."""
if not FAKE_MODE:
server = create_server_connection(self.instance_id,
self.instance_mgmt_address)
cmd_template = "sudo rm /var/lib/mysql/data/ib_logfile%d"
# We want to stop mysql so that upstart does not keep trying to
# respawn it and block the guest agent from accessing the logs.
instance_info.dbaas_admin.management.stop(self.instance_id)
for index in range(2):
cmd = cmd_template % index
try:
server.execute(cmd)
except Exception as e:
asserts.fail("Failed to execute command %s, error: %s" %
(cmd, str(e)))
def wait_for_failure_status(self): def wait_for_failure_status(self):
"""Wait until status becomes running.""" """Wait until status becomes running."""
def is_finished_rebooting(): def is_finished_rebooting():
@ -306,19 +273,6 @@ class RebootTestBase(ActionTestBase):
poll_until(is_finished_rebooting, time_out=TIME_OUT_TIME) poll_until(is_finished_rebooting, time_out=TIME_OUT_TIME)
def unsuccessful_restart(self):
"""Restart MySQL via the REST when it should fail, assert it does."""
assert not FAKE_MODE
self.mess_up_mysql()
self.call_reboot()
self.wait_for_broken_connection()
self.wait_for_failure_status()
def restart_normally(self):
"""Fix iblogs and reboot normally."""
self.fix_mysql()
self.test_successful_restart()
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_RESTART], @test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_RESTART],
depends_on_groups=[GROUP_START], depends_on=[create_user]) depends_on_groups=[GROUP_START], depends_on=[create_user])
@ -338,22 +292,14 @@ class RestartTests(RebootTestBase):
"""Make sure MySQL is accessible before restarting.""" """Make sure MySQL is accessible before restarting."""
self.ensure_mysql_is_running() self.ensure_mysql_is_running()
@test(depends_on=[test_ensure_mysql_is_running], enabled=not FAKE_MODE) @test(depends_on=[test_ensure_mysql_is_running])
def test_unsuccessful_restart(self):
"""Restart MySQL via the REST when it should fail, assert it does."""
if FAKE_MODE:
raise SkipTest("Cannot run this in fake mode.")
self.unsuccessful_restart()
@test(depends_on=[test_set_up],
runs_after=[test_ensure_mysql_is_running, test_unsuccessful_restart])
def test_successful_restart(self): def test_successful_restart(self):
"""Restart MySQL via the REST API successfully.""" """Restart MySQL via the REST API successfully."""
self.successful_restart() self.successful_restart()
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_STOP_MYSQL], @test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_STOP_MYSQL],
depends_on_groups=[GROUP_START], depends_on=[create_user]) depends_on_groups=[GROUP_RESTART], depends_on=[create_user])
class StopTests(RebootTestBase): class StopTests(RebootTestBase):
"""Tests which involve stopping MySQL.""" """Tests which involve stopping MySQL."""
@ -373,11 +319,10 @@ class StopTests(RebootTestBase):
def test_stop_mysql(self): def test_stop_mysql(self):
"""Stops MySQL.""" """Stops MySQL."""
instance_info.dbaas_admin.management.stop(self.instance_id) instance_info.dbaas_admin.management.stop(self.instance_id)
self.wait_for_broken_connection()
self.wait_for_failure_status() self.wait_for_failure_status()
@test(depends_on=[test_stop_mysql]) @test(depends_on=[test_stop_mysql])
def test_instance_get_shows_volume_info_while_mysql_is_down(self): def test_volume_info_while_mysql_is_down(self):
""" """
Confirms the get call behaves appropriately while an instance is Confirms the get call behaves appropriately while an instance is
down. down.
@ -392,15 +337,14 @@ class StopTests(RebootTestBase):
check.true(isinstance(instance.volume.get('size', None), int)) check.true(isinstance(instance.volume.get('size', None), int))
check.true(isinstance(instance.volume.get('used', None), float)) check.true(isinstance(instance.volume.get('used', None), float))
@test(depends_on=[test_set_up], @test(depends_on=[test_volume_info_while_mysql_is_down])
runs_after=[test_instance_get_shows_volume_info_while_mysql_is_down])
def test_successful_restart_when_in_shutdown_state(self): def test_successful_restart_when_in_shutdown_state(self):
"""Restart MySQL via the REST API successfully when MySQL is down.""" """Restart MySQL via the REST API successfully when MySQL is down."""
self.successful_restart() self.successful_restart()
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_REBOOT], @test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_REBOOT],
depends_on_groups=[GROUP_START], depends_on=[RestartTests, create_user]) depends_on_groups=[GROUP_STOP_MYSQL])
class RebootTests(RebootTestBase): class RebootTests(RebootTestBase):
"""Tests restarting instance.""" """Tests restarting instance."""
@ -418,14 +362,7 @@ class RebootTests(RebootTestBase):
"""Make sure MySQL is accessible before restarting.""" """Make sure MySQL is accessible before restarting."""
self.ensure_mysql_is_running() self.ensure_mysql_is_running()
@test(depends_on=[test_ensure_mysql_is_running]) @after_class(depends_on=[test_ensure_mysql_is_running])
def test_unsuccessful_restart(self):
"""Restart MySQL via the REST when it should fail, assert it does."""
if FAKE_MODE:
raise SkipTest("Cannot run this in fake mode.")
self.unsuccessful_restart()
@after_class(depends_on=[test_set_up])
def test_successful_restart(self): def test_successful_restart(self):
"""Restart MySQL via the REST API successfully.""" """Restart MySQL via the REST API successfully."""
if FAKE_MODE: if FAKE_MODE:
@ -434,8 +371,7 @@ class RebootTests(RebootTestBase):
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_RESIZE], @test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_RESIZE],
depends_on_groups=[GROUP_START], depends_on=[create_user], depends_on_groups=[GROUP_REBOOT])
runs_after=[RebootTests])
class ResizeInstanceTest(ActionTestBase): class ResizeInstanceTest(ActionTestBase):
""" """
@ -466,7 +402,6 @@ class ResizeInstanceTest(ActionTestBase):
self.connection.connect() self.connection.connect()
asserts.assert_true(self.connection.is_connected(), asserts.assert_true(self.connection.is_connected(),
"Should be able to connect before resize.") "Should be able to connect before resize.")
self.user_was_deleted = False
@test @test
def test_instance_resize_same_size_should_fail(self): def test_instance_resize_same_size_should_fail(self):
@ -484,8 +419,6 @@ class ResizeInstanceTest(ActionTestBase):
poll_until(is_active, time_out=TIME_OUT_TIME) poll_until(is_active, time_out=TIME_OUT_TIME)
asserts.assert_equal(self.instance.status, 'ACTIVE') asserts.assert_equal(self.instance.status, 'ACTIVE')
self.get_flavor_href(
flavor_id=self.expected_old_flavor_id)
asserts.assert_raises(HTTPNotImplemented, asserts.assert_raises(HTTPNotImplemented,
self.dbaas.instances.resize_instance, self.dbaas.instances.resize_instance,
self.instance_id, flavors[0].id) self.instance_id, flavors[0].id)
@ -517,11 +450,6 @@ class ResizeInstanceTest(ActionTestBase):
flavor = flavors[0] flavor = flavors[0]
self.old_dbaas_flavor = instance_info.dbaas_flavor self.old_dbaas_flavor = instance_info.dbaas_flavor
instance_info.dbaas_flavor = flavor instance_info.dbaas_flavor = flavor
asserts.assert_true(flavor is not None,
"Flavor '%s' not found!" % flavor_name)
flavor_href = self.dbaas.find_flavor_self_href(flavor)
asserts.assert_true(flavor_href is not None,
"Flavor href '%s' not found!" % flavor_name)
self.expected_new_flavor_id = flavor.id self.expected_new_flavor_id = flavor.id
@test(depends_on=[test_instance_resize_same_size_should_fail]) @test(depends_on=[test_instance_resize_same_size_should_fail])
@ -579,45 +507,6 @@ class ResizeInstanceTest(ActionTestBase):
expected = self.get_flavor_href(flavor_id=self.expected_new_flavor_id) expected = self.get_flavor_href(flavor_id=self.expected_new_flavor_id)
asserts.assert_equal(actual, expected) asserts.assert_equal(actual, expected)
@test(depends_on=[test_instance_has_new_flavor_after_resize])
@time_out(TIME_OUT_TIME)
def test_resize_down(self):
expected_dbaas_flavor = self.expected_dbaas_flavor
def is_active():
return self.instance.status == 'ACTIVE'
poll_until(is_active, time_out=TIME_OUT_TIME)
asserts.assert_equal(self.instance.status, 'ACTIVE')
old_flavor_href = self.get_flavor_href(
flavor_id=self.expected_old_flavor_id)
self.dbaas.instances.resize_instance(self.instance_id, old_flavor_href)
asserts.assert_equal(202, self.dbaas.last_http_code)
self.old_dbaas_flavor = instance_info.dbaas_flavor
instance_info.dbaas_flavor = expected_dbaas_flavor
self.wait_for_resize()
asserts.assert_equal(str(self.instance.flavor['id']),
str(self.expected_old_flavor_id))
@test(depends_on=[test_resize_down],
groups=["dbaas.usage"])
def test_resize_instance_down_usage_event_sent(self):
expected = self._build_expected_msg()
expected['old_instance_size'] = self.old_dbaas_flavor.ram
instance_info.consumer.check_message(instance_info.id,
'trove.instance.modify_flavor',
**expected)
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP,
GROUP + ".resize.instance"],
depends_on_groups=[GROUP_START], depends_on=[create_user],
runs_after=[RebootTests, ResizeInstanceTest])
def resize_should_not_delete_users():
if USER_WAS_DELETED:
asserts.fail("Somehow, the resize made the test user disappear.")
@test(depends_on=[ResizeInstanceTest], @test(depends_on=[ResizeInstanceTest],
groups=[GROUP, tests.INSTANCES, INSTANCE_GROUP, GROUP_RESIZE], groups=[GROUP, tests.INSTANCES, INSTANCE_GROUP, GROUP_RESIZE],
@ -708,9 +597,8 @@ class ResizeInstanceVolume(ActionTestBase):
UPDATE_GUEST_CONF = CONFIG.values.get("guest-update-test", None) UPDATE_GUEST_CONF = CONFIG.values.get("guest-update-test", None)
@test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP + ".update_guest"], @test(groups=[tests.INSTANCES, INSTANCE_GROUP, GROUP, GROUP_UPDATE_GUEST],
depends_on=[create_user], depends_on_groups=[GROUP_RESIZE])
depends_on_groups=[GROUP_START])
class UpdateGuest(object): class UpdateGuest(object):
def get_version(self): def get_version(self):

View File

@ -52,6 +52,7 @@ class TestBase(object):
'm1.tiny') 'm1.tiny')
flavor2_name = test_config.values.get( flavor2_name = test_config.values.get(
'instance_bigger_flavor_name', 'm1.small') 'instance_bigger_flavor_name', 'm1.small')
flavors = self.client.find_flavors_by_name(flavor_name) flavors = self.client.find_flavors_by_name(flavor_name)
self.flavor_id = flavors[0].id self.flavor_id = flavors[0].id
self.name = "TEST_" + str(uuid.uuid4()) self.name = "TEST_" + str(uuid.uuid4())

View File

@ -18,7 +18,6 @@ from trove.tests.api import backups
from trove.tests.api import configurations from trove.tests.api import configurations
from trove.tests.api import databases from trove.tests.api import databases
from trove.tests.api import datastores from trove.tests.api import datastores
from trove.tests.api import flavors
from trove.tests.api import instances from trove.tests.api import instances
from trove.tests.api import instances_actions from trove.tests.api import instances_actions
from trove.tests.api.mgmt import accounts from trove.tests.api.mgmt import accounts
@ -86,7 +85,6 @@ def register(group_names, *test_groups, **kwargs):
depends_on_groups=build_group(*test_groups)) depends_on_groups=build_group(*test_groups))
black_box_groups = [ black_box_groups = [
flavors.GROUP,
users.GROUP, users.GROUP,
user_access.GROUP, user_access.GROUP,
databases.GROUP, databases.GROUP,
@ -114,7 +112,6 @@ proboscis.register(groups=["blackbox", "mysql"],
simple_black_box_groups = [ simple_black_box_groups = [
GROUP_SERVICES_INITIALIZE, GROUP_SERVICES_INITIALIZE,
flavors.GROUP,
versions.GROUP, versions.GROUP,
instances.GROUP_START_SIMPLE, instances.GROUP_START_SIMPLE,
admin_required.GROUP, admin_required.GROUP,
@ -141,7 +138,6 @@ proboscis.register(groups=["blackbox_mgmt"],
# Base groups for all other groups # Base groups for all other groups
base_groups = [ base_groups = [
GROUP_SERVICES_INITIALIZE, GROUP_SERVICES_INITIALIZE,
flavors.GROUP,
versions.GROUP, versions.GROUP,
GROUP_SETUP GROUP_SETUP
] ]

View File

@ -199,8 +199,8 @@ class InstanceCreateRunner(TestRunner):
self.assert_equal(instance_info.name, instance._info['name'], self.assert_equal(instance_info.name, instance._info['name'],
"Unexpected instance name") "Unexpected instance name")
self.assert_equal(flavor.id, self.assert_equal(str(flavor.id),
int(instance._info['flavor']['id']), str(instance._info['flavor']['id']),
"Unexpected instance flavor") "Unexpected instance flavor")
self.assert_equal(instance_info.dbaas_datastore, self.assert_equal(instance_info.dbaas_datastore,
instance._info['datastore']['type'], instance._info['datastore']['type'],

View File

@ -802,10 +802,8 @@ class TestRunner(object):
self.assert_equal( self.assert_equal(
1, len(flavors), 1, len(flavors),
"Unexpected number of flavors with name '%s' found." % flavor_name) "Unexpected number of flavors with name '%s' found." % flavor_name)
flavor = flavors[0]
self.assert_is_not_none(flavor, "Flavor '%s' not found." % flavor_name)
return flavor return flavors[0]
def get_instance_flavor(self, fault_num=None): def get_instance_flavor(self, fault_num=None):
name_format = 'instance%s%s_flavor_name' name_format = 'instance%s%s_flavor_name'

View File

@ -52,7 +52,7 @@ class TestDatastoreVersion(trove_testtools.TestCase):
def test_version_create(self, mock_glance_client): def test_version_create(self, mock_glance_client):
body = {"version": { body = {"version": {
"datastore_name": "test_ds", "datastore_name": "test_ds",
"name": "test_vr", "name": "test_version",
"datastore_manager": "mysql", "datastore_manager": "mysql",
"image": "image-id", "image": "image-id",
"packages": "test-pkg", "packages": "test-pkg",

View File

@ -309,6 +309,7 @@ class FreshInstanceTasksTest(BaseFreshInstanceTasksTest):
new_callable=PropertyMock, new_callable=PropertyMock,
return_value='fake-hostname') return_value='fake-hostname')
def test_servers_create_block_device_mapping_v2(self, mock_hostname): def test_servers_create_block_device_mapping_v2(self, mock_hostname):
self.freshinstancetasks._prepare_userdata = Mock(return_value=None)
mock_nova_client = self.freshinstancetasks.nova_client = Mock() mock_nova_client = self.freshinstancetasks.nova_client = Mock()
mock_servers_create = mock_nova_client.servers.create mock_servers_create = mock_nova_client.servers.create
self.freshinstancetasks._create_server('fake-flavor', 'fake-image', self.freshinstancetasks._create_server('fake-flavor', 'fake-image',
@ -867,26 +868,23 @@ class BuiltInstanceTasksTest(trove_testtools.TestCase):
@patch.object(utils, 'poll_until') @patch.object(utils, 'poll_until')
def test_reboot(self, mock_poll): def test_reboot(self, mock_poll):
self.instance_task.datastore_status_matches = Mock(return_value=True)
self.instance_task._refresh_datastore_status = Mock()
self.instance_task.server.reboot = Mock() self.instance_task.server.reboot = Mock()
self.instance_task.set_datastore_status_to_paused = Mock() self.instance_task.set_datastore_status_to_paused = Mock()
self.instance_task.reboot() self.instance_task.reboot()
self.instance_task._guest.stop_db.assert_any_call() self.instance_task._guest.stop_db.assert_any_call()
self.instance_task._refresh_datastore_status.assert_any_call()
self.instance_task.server.reboot.assert_any_call() self.instance_task.server.reboot.assert_any_call()
self.instance_task.set_datastore_status_to_paused.assert_any_call() self.instance_task.set_datastore_status_to_paused.assert_any_call()
@patch.object(utils, 'poll_until') @patch.object(utils, 'poll_until')
@patch('trove.taskmanager.models.LOG') @patch('trove.taskmanager.models.LOG')
def test_reboot_datastore_not_ready(self, mock_logging, mock_poll): def test_reboot_datastore_not_ready(self, mock_logging, mock_poll):
self.instance_task.datastore_status_matches = Mock(return_value=False) mock_poll.side_effect = PollTimeOut
self.instance_task._refresh_datastore_status = Mock()
self.instance_task.server.reboot = Mock() self.instance_task.server.reboot = Mock()
self.instance_task.set_datastore_status_to_paused = Mock() self.instance_task.set_datastore_status_to_paused = Mock()
self.instance_task.reboot() self.instance_task.reboot()
self.instance_task._guest.stop_db.assert_any_call() self.instance_task._guest.stop_db.assert_any_call()
self.instance_task._refresh_datastore_status.assert_any_call()
assert not self.instance_task.server.reboot.called assert not self.instance_task.server.reboot.called
assert not self.instance_task.set_datastore_status_to_paused.called assert not self.instance_task.set_datastore_status_to_paused.called