Merge "Improve the function tests"
This commit is contained in:
commit
2d301d0a21
@ -142,6 +142,9 @@
|
|||||||
devstack_localrc:
|
devstack_localrc:
|
||||||
TROVE_RESIZE_TIME_OUT: 1800
|
TROVE_RESIZE_TIME_OUT: 1800
|
||||||
trove_resize_time_out: 1800
|
trove_resize_time_out: 1800
|
||||||
|
trove_test_datastore: 'mysql'
|
||||||
|
trove_test_group: 'mysql'
|
||||||
|
trove_test_datastore_version: '5.7'
|
||||||
|
|
||||||
- job:
|
- job:
|
||||||
name: trove-grenade
|
name: trove-grenade
|
||||||
@ -196,6 +199,7 @@
|
|||||||
vars:
|
vars:
|
||||||
trove_test_datastore: mariadb
|
trove_test_datastore: mariadb
|
||||||
trove_test_group: mariadb-supported-single
|
trove_test_group: mariadb-supported-single
|
||||||
|
trove_test_datastore_version: 10.4
|
||||||
devstack_localrc:
|
devstack_localrc:
|
||||||
TROVE_ENABLE_IMAGE_BUILD: false
|
TROVE_ENABLE_IMAGE_BUILD: false
|
||||||
|
|
||||||
@ -205,6 +209,7 @@
|
|||||||
vars:
|
vars:
|
||||||
trove_test_datastore: mariadb
|
trove_test_datastore: mariadb
|
||||||
trove_test_group: mariadb-supported-multi
|
trove_test_group: mariadb-supported-multi
|
||||||
|
trove_test_datastore_version: 10.4
|
||||||
devstack_localrc:
|
devstack_localrc:
|
||||||
TROVE_ENABLE_IMAGE_BUILD: false
|
TROVE_ENABLE_IMAGE_BUILD: false
|
||||||
|
|
||||||
@ -214,6 +219,7 @@
|
|||||||
vars:
|
vars:
|
||||||
trove_test_datastore: mysql
|
trove_test_datastore: mysql
|
||||||
trove_test_group: mysql-supported-single
|
trove_test_group: mysql-supported-single
|
||||||
|
trove_test_datastore_version: 5.7
|
||||||
|
|
||||||
- job:
|
- job:
|
||||||
name: trove-scenario-mysql-multi
|
name: trove-scenario-mysql-multi
|
||||||
@ -221,6 +227,7 @@
|
|||||||
vars:
|
vars:
|
||||||
trove_test_datastore: mysql
|
trove_test_datastore: mysql
|
||||||
trove_test_group: mysql-supported-multi
|
trove_test_group: mysql-supported-multi
|
||||||
|
trove_test_datastore_version: 5.7
|
||||||
|
|
||||||
- job:
|
- job:
|
||||||
name: trove-scenario-percona-multi
|
name: trove-scenario-percona-multi
|
||||||
|
@ -390,9 +390,11 @@ function setup_mgmt_network() {
|
|||||||
die_if_not_set $LINENO network_id "Failed to create network: $NET_NAME, project: ${PROJECT_ID}"
|
die_if_not_set $LINENO network_id "Failed to create network: $NET_NAME, project: ${PROJECT_ID}"
|
||||||
|
|
||||||
if [[ "$IP_VERSION" =~ 4.* ]]; then
|
if [[ "$IP_VERSION" =~ 4.* ]]; then
|
||||||
NEW_SUBNET_ID=$(create_mgmt_subnet_v4 ${PROJECT_ID} ${network_id} ${SUBNET_NAME} ${SUBNET_RANGE})
|
net_subnet_id=$(create_mgmt_subnet_v4 ${PROJECT_ID} ${network_id} ${SUBNET_NAME} ${SUBNET_RANGE})
|
||||||
openstack router add subnet $ROUTER_ID $NEW_SUBNET_ID
|
# 'openstack router add' has a bug that cound't show the error message
|
||||||
|
# openstack router add subnet ${ROUTER_ID} ${net_subnet_id} --debug
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Trove doesn't support IPv6 for now.
|
# Trove doesn't support IPv6 for now.
|
||||||
# if [[ "$IP_VERSION" =~ .*6 ]]; then
|
# if [[ "$IP_VERSION" =~ .*6 ]]; then
|
||||||
# NEW_IPV6_SUBNET_ID=$(create_subnet_v6 ${PROJECT_ID} ${network_id} ${IPV6_SUBNET_NAME})
|
# NEW_IPV6_SUBNET_ID=$(create_subnet_v6 ${PROJECT_ID} ${network_id} ${IPV6_SUBNET_NAME})
|
||||||
@ -454,32 +456,25 @@ function create_guest_image {
|
|||||||
${TROVE_IMAGE_OS_RELEASE} \
|
${TROVE_IMAGE_OS_RELEASE} \
|
||||||
true
|
true
|
||||||
|
|
||||||
image_name="trove-${TROVE_IMAGE_OS}-${TROVE_IMAGE_OS_RELEASE}-${TROVE_DATASTORE_TYPE}"
|
image_name="trove-datastore-${TROVE_IMAGE_OS}-${TROVE_IMAGE_OS_RELEASE}-${TROVE_DATASTORE_TYPE}"
|
||||||
image_file=$HOME/images/${image_name}.qcow2
|
image_file=$HOME/images/${image_name}.qcow2
|
||||||
if [ ! -f ${image_file} ]; then
|
if [ ! -f ${image_file} ]; then
|
||||||
echo "Image file was not found at ${image_file}. Probably it was not created."
|
echo "Image file was not found at ${image_file}"
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
ACTIVE=1
|
|
||||||
INACTIVE=0
|
|
||||||
|
|
||||||
echo "Add the image to glance"
|
echo "Add the image to glance"
|
||||||
glance_image_id=$(openstack --os-region-name RegionOne --os-password ${SERVICE_PASSWORD} \
|
glance_image_id=$(openstack --os-region-name RegionOne --os-password ${SERVICE_PASSWORD} \
|
||||||
--os-project-name service --os-username trove \
|
--os-project-name service --os-username trove \
|
||||||
image create ${TROVE_IMAGE_OS}-${TROVE_IMAGE_OS_RELEASE}-${TROVE_DATASTORE_TYPE} \
|
image create ${image_name} \
|
||||||
--disk-format qcow2 --container-format bare --property hw_rng_model='virtio' --file ${image_file} \
|
--disk-format qcow2 --container-format bare --property hw_rng_model='virtio' --file ${image_file} \
|
||||||
-c id -f value)
|
-c id -f value)
|
||||||
|
|
||||||
echo "Register the image in datastore"
|
echo "Register the image in datastore"
|
||||||
$TROVE_MANAGE datastore_update $TROVE_DATASTORE_TYPE ""
|
$TROVE_MANAGE datastore_update $TROVE_DATASTORE_TYPE ""
|
||||||
$TROVE_MANAGE datastore_version_update $TROVE_DATASTORE_TYPE $TROVE_DATASTORE_VERSION $TROVE_DATASTORE_TYPE $glance_image_id "" $ACTIVE
|
$TROVE_MANAGE datastore_version_update $TROVE_DATASTORE_TYPE $TROVE_DATASTORE_VERSION $TROVE_DATASTORE_TYPE $glance_image_id "" 1
|
||||||
$TROVE_MANAGE datastore_update $TROVE_DATASTORE_TYPE $TROVE_DATASTORE_VERSION
|
$TROVE_MANAGE datastore_update $TROVE_DATASTORE_TYPE $TROVE_DATASTORE_VERSION
|
||||||
|
|
||||||
# just for tests
|
|
||||||
$TROVE_MANAGE datastore_version_update "$TROVE_DATASTORE_TYPE" "inactive_version" "manager1" $glance_image_id "" $INACTIVE
|
|
||||||
$TROVE_MANAGE datastore_update Test_Datastore_1 ""
|
|
||||||
|
|
||||||
echo "Add parameter validation rules if available"
|
echo "Add parameter validation rules if available"
|
||||||
if [ -f $DEST/trove/trove/templates/$TROVE_DATASTORE_TYPE/validation-rules.json ]; then
|
if [ -f $DEST/trove/trove/templates/$TROVE_DATASTORE_TYPE/validation-rules.json ]; then
|
||||||
$TROVE_MANAGE db_load_datastore_config_parameters "$TROVE_DATASTORE_TYPE" "$TROVE_DATASTORE_VERSION" \
|
$TROVE_MANAGE db_load_datastore_config_parameters "$TROVE_DATASTORE_TYPE" "$TROVE_DATASTORE_VERSION" \
|
||||||
@ -546,6 +541,8 @@ function config_trove_network {
|
|||||||
openstack network list
|
openstack network list
|
||||||
echo "Neutron subnet list:"
|
echo "Neutron subnet list:"
|
||||||
openstack subnet list
|
openstack subnet list
|
||||||
|
echo "Neutron router:"
|
||||||
|
openstack router show ${ROUTER_ID} -f yaml
|
||||||
echo "ip route:"
|
echo "ip route:"
|
||||||
sudo ip route
|
sudo ip route
|
||||||
|
|
||||||
|
@ -56,6 +56,8 @@ if is_service_enabled neutron; then
|
|||||||
TROVE_MGMT_NETWORK_NAME=${TROVE_MGMT_NETWORK_NAME:-"trove-mgmt"}
|
TROVE_MGMT_NETWORK_NAME=${TROVE_MGMT_NETWORK_NAME:-"trove-mgmt"}
|
||||||
TROVE_MGMT_SUBNET_NAME=${TROVE_MGMT_SUBNET_NAME:-${TROVE_MGMT_NETWORK_NAME}-subnet}
|
TROVE_MGMT_SUBNET_NAME=${TROVE_MGMT_SUBNET_NAME:-${TROVE_MGMT_NETWORK_NAME}-subnet}
|
||||||
TROVE_MGMT_SUBNET_RANGE=${TROVE_MGMT_SUBNET_RANGE:-"192.168.254.0/24"}
|
TROVE_MGMT_SUBNET_RANGE=${TROVE_MGMT_SUBNET_RANGE:-"192.168.254.0/24"}
|
||||||
|
TROVE_MGMT_SUBNET_START=${TROVE_MGMT_SUBNET_START:-"192.168.254.2"}
|
||||||
|
TROVE_MGMT_SUBNET_END=${TROVE_MGMT_SUBNET_END:-"192.168.254.200"}
|
||||||
else
|
else
|
||||||
TROVE_HOST_GATEWAY=${NETWORK_GATEWAY:-10.0.0.1}
|
TROVE_HOST_GATEWAY=${NETWORK_GATEWAY:-10.0.0.1}
|
||||||
fi
|
fi
|
||||||
|
@ -1,360 +0,0 @@
|
|||||||
.. _create_trove_instance:
|
|
||||||
|
|
||||||
==============================
|
|
||||||
How to create a trove instance
|
|
||||||
==============================
|
|
||||||
|
|
||||||
While creating a trove instance, I often have problems with cinder
|
|
||||||
volumes and nova servers, this is due to my lack of knowledge in the area.
|
|
||||||
This post is to help describe my journey on creating a trove instance.
|
|
||||||
|
|
||||||
----------------
|
|
||||||
Installing trove
|
|
||||||
----------------
|
|
||||||
|
|
||||||
I use the integration tools provided by trove to install the required services.
|
|
||||||
This is already covered in the install guide.
|
|
||||||
|
|
||||||
Install trove
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
/trove/integration/scripts$ ./trovestack install
|
|
||||||
|
|
||||||
Once that completes, I use the command kick-start that gets a datastore
|
|
||||||
ready for us to use and target for our trove instance. This shows the
|
|
||||||
mysql datastore.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
/trove/integration/scripts$ ./trovestack kick-start mysql
|
|
||||||
|
|
||||||
Note: This command doesn't give you a completion message.
|
|
||||||
|
|
||||||
You can view the available datastores by running the following command
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ trove datastore-list
|
|
||||||
|
|
||||||
+--------------------------------------+------------------+
|
|
||||||
| ID | Name |
|
|
||||||
+--------------------------------------+------------------+
|
|
||||||
| 137c27ee-d491-4a54-90ab-06307e9f6bf6 | mysql |
|
|
||||||
| aea3d4c5-9c2e-48ae-b100-527b18d4eb02 | Test_Datastore_1 |
|
|
||||||
| b8583e8c-8177-480e-889e-a73c5290b558 | test_ds |
|
|
||||||
+--------------------------------------+------------------+
|
|
||||||
|
|
||||||
Once that is done, view the image that was built for the datastore you have
|
|
||||||
kick-started and identify the resources required for it.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ openstack image list
|
|
||||||
|
|
||||||
+--------------------------------------+--------------------------+--------+
|
|
||||||
| ID | Name | Status |
|
|
||||||
+--------------------------------------+--------------------------+--------+
|
|
||||||
| 37d4b996-14c2-4981-820e-3ac87bb4c5a2 | cirros-0.3.5-x86_64-disk | active |
|
|
||||||
| 2d7d930a-d606-4934-8602-851207546fee | ubuntu_mysql | active |
|
|
||||||
+--------------------------------------+--------------------------+--------+
|
|
||||||
|
|
||||||
Grab the ID from the list and run the following command to view the size of
|
|
||||||
the image.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ openstack image show ubuntu_mysql
|
|
||||||
|
|
||||||
+------------------+------------------------------------------------------+
|
|
||||||
| Field | Value |
|
|
||||||
+------------------+------------------------------------------------------+
|
|
||||||
| checksum | 9facdf0670ccb58ea27bf665e4fdcdf5 |
|
|
||||||
| container_format | bare |
|
|
||||||
| created_at | 2017-05-26T14:35:39Z |
|
|
||||||
| disk_format | qcow2 |
|
|
||||||
| file | /v2/images/2d7d930a-d606-4934-8602-851207546fee/file |
|
|
||||||
| id | 2d7d930a-d606-4934-8602-851207546fee |
|
|
||||||
| min_disk | 0 |
|
|
||||||
| min_ram | 0 |
|
|
||||||
| name | ubuntu_mysql |
|
|
||||||
| owner | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| protected | False |
|
|
||||||
| schema | /v2/schemas/image |
|
|
||||||
| size | 633423872 |
|
|
||||||
| status | active |
|
|
||||||
| tags | |
|
|
||||||
| updated_at | 2017-05-26T14:35:42Z |
|
|
||||||
| virtual_size | None |
|
|
||||||
| visibility | public |
|
|
||||||
+------------------+------------------------------------------------------+
|
|
||||||
|
|
||||||
Take the value that says size, this is 633423872 in bytes. Cinder volumes are
|
|
||||||
in gigabytes so 633423872 becomes:
|
|
||||||
|
|
||||||
633423872 / 1024
|
|
||||||
618578 # KB
|
|
||||||
618578 / 1024
|
|
||||||
604 # MB
|
|
||||||
604 / 1024
|
|
||||||
0 # < 1 GB so we will round up.
|
|
||||||
|
|
||||||
Then test that you can create the cinder volume:
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ cinder create --name my-v 1
|
|
||||||
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
| Property | Value |
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
| attachments | [] |
|
|
||||||
| availability_zone | nova |
|
|
||||||
| bootable | false |
|
|
||||||
| consistencygroup_id | None |
|
|
||||||
| created_at | 2017-05-26T16:37:55.000000 |
|
|
||||||
| description | None |
|
|
||||||
| encrypted | False |
|
|
||||||
| id | 7a2da60f-cc1b-4798-ba7a-1f0215c74615 |
|
|
||||||
| metadata | {} |
|
|
||||||
| migration_status | None |
|
|
||||||
| multiattach | False |
|
|
||||||
| name | my-v |
|
|
||||||
| os-vol-host-attr:host | None |
|
|
||||||
| os-vol-mig-status-attr:migstat | None |
|
|
||||||
| os-vol-mig-status-attr:name_id | None |
|
|
||||||
| os-vol-tenant-attr:tenant_id | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| replication_status | None |
|
|
||||||
| size | 1 |
|
|
||||||
| snapshot_id | None |
|
|
||||||
| source_volid | None |
|
|
||||||
| status | creating |
|
|
||||||
| updated_at | None |
|
|
||||||
| user_id | cf1e59dc2e4d4aeca51aa050faac15c2 |
|
|
||||||
| volume_type | lvmdriver-1 |
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
|
|
||||||
Next, verify the cinder volume status has moved from creating to available.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ cinder show my-v
|
|
||||||
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
| Property | Value |
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
| attachments | [] |
|
|
||||||
| availability_zone | nova |
|
|
||||||
| bootable | false |
|
|
||||||
| consistencygroup_id | None |
|
|
||||||
| created_at | 2017-05-26T16:37:55.000000 |
|
|
||||||
| description | None |
|
|
||||||
| encrypted | False |
|
|
||||||
| id | 7a2da60f-cc1b-4798-ba7a-1f0215c74615 |
|
|
||||||
| metadata | {} |
|
|
||||||
| migration_status | None |
|
|
||||||
| multiattach | False |
|
|
||||||
| name | my-v |
|
|
||||||
| os-vol-host-attr:host | ubuntu@lvmdriver-1#lvmdriver-1 |
|
|
||||||
| os-vol-mig-status-attr:migstat | None |
|
|
||||||
| os-vol-mig-status-attr:name_id | None |
|
|
||||||
| os-vol-tenant-attr:tenant_id | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| replication_status | None |
|
|
||||||
| size | 1 |
|
|
||||||
| snapshot_id | None |
|
|
||||||
| source_volid | None |
|
|
||||||
| status | available |
|
|
||||||
| updated_at | 2017-05-26T16:37:56.000000 |
|
|
||||||
| user_id | cf1e59dc2e4d4aeca51aa050faac15c2 |
|
|
||||||
| volume_type | lvmdriver-1 |
|
|
||||||
+--------------------------------+--------------------------------------+
|
|
||||||
|
|
||||||
Ok, now we know that works so lets delete it.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ cinder delete my-v
|
|
||||||
|
|
||||||
Next is to choose a server flavor that fits the requirements of your datastore
|
|
||||||
and do not exceed your computer hardware limitations.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ trove flavor-list
|
|
||||||
|
|
||||||
+------+--------------------------+--------+-------+------+-----------+
|
|
||||||
| ID | Name | RAM | vCPUs | Disk | Ephemeral |
|
|
||||||
+------+--------------------------+--------+-------+------+-----------+
|
|
||||||
| 1 | m1.tiny | 512 | 1 | 1 | 0 |
|
|
||||||
| 10 | test.tiny-3 | 512 | 1 | 3 | 0 |
|
|
||||||
| 10e | test.eph.tiny-3 | 512 | 1 | 3 | 1 |
|
|
||||||
| 10er | test.eph.tiny-3.resize | 528 | 2 | 3 | 1 |
|
|
||||||
| 10r | test.tiny-3.resize | 528 | 2 | 3 | 0 |
|
|
||||||
| 15 | test.small-3 | 768 | 1 | 3 | 0 |
|
|
||||||
| 15e | test.eph.small-3 | 768 | 1 | 3 | 1 |
|
|
||||||
| 15er | test.eph.small-3.resize | 784 | 2 | 3 | 1 |
|
|
||||||
| 15r | test.small-3.resize | 784 | 2 | 3 | 0 |
|
|
||||||
| 16 | test.small-4 | 768 | 1 | 4 | 0 |
|
|
||||||
| 16e | test.eph.small-4 | 768 | 1 | 4 | 1 |
|
|
||||||
| 16er | test.eph.small-4.resize | 784 | 2 | 4 | 1 |
|
|
||||||
| 16r | test.small-4.resize | 784 | 2 | 4 | 0 |
|
|
||||||
| 17 | test.small-5 | 768 | 1 | 5 | 0 |
|
|
||||||
| 17e | test.eph.small-5 | 768 | 1 | 5 | 1 |
|
|
||||||
| 17er | test.eph.small-5.resize | 784 | 2 | 5 | 1 |
|
|
||||||
| 17r | test.small-5.resize | 784 | 2 | 5 | 0 |
|
|
||||||
| 2 | m1.small | 2048 | 1 | 20 | 0 |
|
|
||||||
| 20 | test.medium-4 | 1024 | 1 | 4 | 0 |
|
|
||||||
| 20e | test.eph.medium-4 | 1024 | 1 | 4 | 1 |
|
|
||||||
| 20er | test.eph.medium-4.resize | 1040 | 2 | 4 | 1 |
|
|
||||||
| 20r | test.medium-4.resize | 1040 | 2 | 4 | 0 |
|
|
||||||
| 21 | test.medium-5 | 1024 | 1 | 5 | 0 |
|
|
||||||
| 21e | test.eph.medium-5 | 1024 | 1 | 5 | 1 |
|
|
||||||
| 21er | test.eph.medium-5.resize | 1040 | 2 | 5 | 1 |
|
|
||||||
| 21r | test.medium-5.resize | 1040 | 2 | 5 | 0 |
|
|
||||||
| 25 | test.large-5 | 2048 | 1 | 5 | 0 |
|
|
||||||
| 25e | test.eph.large-5 | 2048 | 1 | 5 | 1 |
|
|
||||||
| 25er | test.eph.large-5.resize | 2064 | 2 | 5 | 1 |
|
|
||||||
| 25r | test.large-5.resize | 2064 | 2 | 5 | 0 |
|
|
||||||
| 26 | test.large-10 | 2048 | 1 | 10 | 0 |
|
|
||||||
| 26e | test.eph.large-10 | 2048 | 1 | 10 | 1 |
|
|
||||||
| 26er | test.eph.large-10.resize | 2064 | 2 | 10 | 1 |
|
|
||||||
| 26r | test.large-10.resize | 2064 | 2 | 10 | 0 |
|
|
||||||
| 27 | test.large-15 | 2048 | 1 | 15 | 0 |
|
|
||||||
| 27e | test.eph.large-15 | 2048 | 1 | 15 | 1 |
|
|
||||||
| 27er | test.eph.large-15.resize | 2064 | 2 | 15 | 1 |
|
|
||||||
| 27r | test.large-15.resize | 2064 | 2 | 15 | 0 |
|
|
||||||
| 3 | m1.medium | 4096 | 2 | 40 | 0 |
|
|
||||||
| 30 | test.fault_1-1 | 512 | 1 | 1 | 0 |
|
|
||||||
| 30e | test.eph.fault_1-1 | 512 | 1 | 1 | 1 |
|
|
||||||
| 31 | test.fault_2-5 | 131072 | 1 | 5 | 0 |
|
|
||||||
| 31e | test.eph.fault_2-5 | 131072 | 1 | 5 | 1 |
|
|
||||||
| 4 | m1.large | 8192 | 4 | 80 | 0 |
|
|
||||||
| 42 | m1.nano | 64 | 1 | 0 | 0 |
|
|
||||||
| 451 | m1.heat | 512 | 1 | 0 | 0 |
|
|
||||||
| 5 | m1.xlarge | 16384 | 8 | 160 | 0 |
|
|
||||||
| 84 | m1.micro | 128 | 1 | 0 | 0 |
|
|
||||||
| c1 | cirros256 | 256 | 1 | 0 | 0 |
|
|
||||||
| d1 | ds512M | 512 | 1 | 5 | 0 |
|
|
||||||
| d2 | ds1G | 1024 | 1 | 10 | 0 |
|
|
||||||
| d3 | ds2G | 2048 | 2 | 10 | 0 |
|
|
||||||
| d4 | ds4G | 4096 | 4 | 20 | 0 |
|
|
||||||
+------+--------------------------+--------+-------+------+-----------+
|
|
||||||
|
|
||||||
|
|
||||||
The flavor sizes are in megabytes, check your computer disk space and pick a
|
|
||||||
flavor less than your limitations.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ df -h
|
|
||||||
|
|
||||||
Filesystem Size Used Avail Use% Mounted on
|
|
||||||
udev 7.9G 0 7.9G 0% /dev
|
|
||||||
tmpfs 1.6G 162M 1.5G 11% /run
|
|
||||||
/dev/mapper/ubuntu--vg-root 33G 11G 21G 34% /
|
|
||||||
tmpfs 7.9G 4.0K 7.9G 1% /dev/shm
|
|
||||||
tmpfs 5.0M 0 5.0M 0% /run/lock
|
|
||||||
tmpfs 7.9G 0 7.9G 0% /sys/fs/cgroup
|
|
||||||
/dev/vda1 472M 102M 346M 23% /boot
|
|
||||||
tmpfs 1.6G 0 1.6G 0% /run/user/1000
|
|
||||||
/dev/loop0 6.0G 650M 5.4G 11% /opt/stack/data/swift/drives/sdb1
|
|
||||||
|
|
||||||
I have a lot of partitions I don't understand but ubuntu--vg-root is the one
|
|
||||||
setup by LVM during the install and it is the largest one so I'm going to use 21G
|
|
||||||
as my upper limit. Now I only need 1G, this information is still good to know when
|
|
||||||
you are dealing with multiple instances, larger images, or limited disk space.
|
|
||||||
|
|
||||||
Flavors also use RAM so it's important to check your free memory.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ free -h
|
|
||||||
|
|
||||||
total used free shared buff/cache available
|
|
||||||
Mem: 15G 5.1G 5.0G 150M 5.5G 10G
|
|
||||||
Swap: 15G 4.1M 15G
|
|
||||||
|
|
||||||
I have given my VM 16GB RAM and it shows I have 5GB free. So In order to be safe,
|
|
||||||
I will choose test-small-3 (ID 15), this is RAM 768 and disk size 3GB. The disk size must be
|
|
||||||
greater than 604MB from the ubuntu_mysql image requirements, but we rounded to 1GB to
|
|
||||||
be safe.
|
|
||||||
|
|
||||||
After all of this we are ready to create our trove instance.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ trove create my-inst 15 --size 1
|
|
||||||
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| Property | Value |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| created | 2017-05-26T16:53:06 |
|
|
||||||
| datastore | mysql |
|
|
||||||
| datastore_version | 5.6 |
|
|
||||||
| encrypted_rpc_messaging | True |
|
|
||||||
| flavor | 15 |
|
|
||||||
| id | 39f8ac9e-2935-40fb-8b09-8a963fb235bd |
|
|
||||||
| name | my-inst |
|
|
||||||
| region | RegionOne |
|
|
||||||
| server_id | None |
|
|
||||||
| status | BUILD |
|
|
||||||
| tenant_id | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| updated | 2017-05-26T16:53:06 |
|
|
||||||
| volume | 1 |
|
|
||||||
| volume_id | None |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
|
|
||||||
Now we view the details to see if it is successful.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ trove show my-inst
|
|
||||||
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| Property | Value |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| created | 2017-05-26T16:53:07 |
|
|
||||||
| datastore | mysql |
|
|
||||||
| datastore_version | 5.6 |
|
|
||||||
| encrypted_rpc_messaging | True |
|
|
||||||
| flavor | 15 |
|
|
||||||
| id | 39f8ac9e-2935-40fb-8b09-8a963fb235bd |
|
|
||||||
| name | my-inst |
|
|
||||||
| region | RegionOne |
|
|
||||||
| server_id | 62399b7e-dec1-4606-9297-3b3711a62d68 |
|
|
||||||
| status | BUILD |
|
|
||||||
| tenant_id | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| updated | 2017-05-26T16:53:13 |
|
|
||||||
| volume | 1 |
|
|
||||||
| volume_id | da3b3951-7f7a-4c71-86b9-f0059da814f8 |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
|
|
||||||
Notice, status still says BUILD but we now have a server_id and volume_id.
|
|
||||||
|
|
||||||
After waiting a few moments, check it again.
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
$ trove show my-inst
|
|
||||||
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| Property | Value |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
| created | 2017-05-26T16:53:07 |
|
|
||||||
| datastore | mysql |
|
|
||||||
| datastore_version | 5.6 |
|
|
||||||
| encrypted_rpc_messaging | True |
|
|
||||||
| flavor | 15 |
|
|
||||||
| id | 39f8ac9e-2935-40fb-8b09-8a963fb235bd |
|
|
||||||
| name | my-inst |
|
|
||||||
| region | RegionOne |
|
|
||||||
| server_id | 62399b7e-dec1-4606-9297-3b3711a62d68 |
|
|
||||||
| status | ACTIVE |
|
|
||||||
| tenant_id | e765230cd96f47f294f910551ec3c1f4 |
|
|
||||||
| updated | 2017-05-26T16:53:13 |
|
|
||||||
| volume | 1 |
|
|
||||||
| volume_id | da3b3951-7f7a-4c71-86b9-f0059da814f8 |
|
|
||||||
| volume_used | 0.1 |
|
|
||||||
+-------------------------+--------------------------------------+
|
|
||||||
|
|
||||||
The status is now set to ACTIVE and you are done!
|
|
@ -11,4 +11,3 @@ functionality, the following resources are provided.
|
|||||||
|
|
||||||
design
|
design
|
||||||
testing
|
testing
|
||||||
how_to_create_a_trove_instance.rst
|
|
||||||
|
@ -33,21 +33,8 @@ For an in-depth look at the project's design and structure, see the
|
|||||||
- `Trove`_
|
- `Trove`_
|
||||||
- `Trove Client`_
|
- `Trove Client`_
|
||||||
|
|
||||||
* `Trove Wiki`_ on OpenStack
|
|
||||||
* `Trove API Documentation`_ on docs.openstack.org
|
* `Trove API Documentation`_ on docs.openstack.org
|
||||||
* `Trove Blueprints`_ on storyboard.openstack.org
|
* `Trove storyboard`_ on storyboard.openstack.org
|
||||||
* `Trove Bugs`_ on storyboard.openstack.org
|
|
||||||
|
|
||||||
|
|
||||||
Guest Images
|
|
||||||
============
|
|
||||||
|
|
||||||
In order to use Trove, you need to have Guest Images for each
|
|
||||||
datastore and version. These images are loaded into Glance and
|
|
||||||
registered with Trove.
|
|
||||||
|
|
||||||
For those wishing to develop guest images, please refer to the
|
|
||||||
:ref:`build_guest_images` page.
|
|
||||||
|
|
||||||
|
|
||||||
Search Trove Documentation
|
Search Trove Documentation
|
||||||
@ -56,9 +43,7 @@ Search Trove Documentation
|
|||||||
* :ref:`search`
|
* :ref:`search`
|
||||||
|
|
||||||
|
|
||||||
.. _Trove Wiki: https://wiki.openstack.org/wiki/Trove
|
|
||||||
.. _Trove: https://opendev.org/openstack/trove
|
.. _Trove: https://opendev.org/openstack/trove
|
||||||
.. _Trove Client: https://opendev.org/openstack/python-troveclient
|
.. _Trove Client: https://opendev.org/openstack/python-troveclient
|
||||||
.. _Trove API Documentation: https://docs.openstack.org/api-ref/database/
|
.. _Trove API Documentation: https://docs.openstack.org/api-ref/database/
|
||||||
.. _Trove Blueprints: https://storyboard.openstack.org/#!/project/openstack/trove
|
.. _Trove storyboard: https://storyboard.openstack.org/#!/project/openstack/trove
|
||||||
.. _Trove Bugs: https://storyboard.openstack.org/#!/project/openstack/trove
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
{
|
{
|
||||||
"report_directory":"rdli-test-report",
|
"report_directory":"rdli-test-report",
|
||||||
"start_services": false,
|
|
||||||
|
|
||||||
"test_mgmt":false,
|
"test_mgmt":false,
|
||||||
"use_local_ovz":false,
|
"use_local_ovz":false,
|
||||||
@ -16,7 +15,6 @@
|
|||||||
"nova_conf":"/home/vagrant/nova.conf",
|
"nova_conf":"/home/vagrant/nova.conf",
|
||||||
"keystone_code_root":"/opt/stack/keystone",
|
"keystone_code_root":"/opt/stack/keystone",
|
||||||
"keystone_conf":"/etc/keystone/keystone.conf",
|
"keystone_conf":"/etc/keystone/keystone.conf",
|
||||||
"keystone_use_combined":true,
|
|
||||||
"trove_code_root":"/opt/stack/trove",
|
"trove_code_root":"/opt/stack/trove",
|
||||||
"trove_conf":"/tmp/trove.conf",
|
"trove_conf":"/tmp/trove.conf",
|
||||||
"trove_version":"v1.0",
|
"trove_version":"v1.0",
|
||||||
|
@ -37,23 +37,11 @@
|
|||||||
],
|
],
|
||||||
"flavors": null,
|
"flavors": null,
|
||||||
"white_box":false,
|
"white_box":false,
|
||||||
"start_services": %startservices%,
|
|
||||||
"test_mgmt":false,
|
"test_mgmt":false,
|
||||||
"use_local_ovz":false,
|
"use_local_ovz":false,
|
||||||
"use_venv":false,
|
"use_venv":false,
|
||||||
"glance_code_root":"%glance_path%",
|
|
||||||
"glance_api_conf":"/vagrant/conf/glance-api.conf",
|
|
||||||
"glance_reg_conf":"/vagrant/conf/glance-reg.conf",
|
|
||||||
"glance_images_directory": "/glance_images",
|
|
||||||
"glance_image": "debian-squeeze-x86_64-openvz.tar.gz",
|
|
||||||
"report_directory":"%report_directory%",
|
"report_directory":"%report_directory%",
|
||||||
"usr_bin_dir":"%bin_path%",
|
"usr_bin_dir":"%bin_path%",
|
||||||
"nova_code_root":"%nova_path%",
|
|
||||||
"nova_conf":"/home/vagrant/nova.conf",
|
|
||||||
"keystone_code_root":"%keystone_path%",
|
|
||||||
"keystone_conf":"/etc/keystone/keystone.conf",
|
|
||||||
"keystone_use_combined":true,
|
|
||||||
"trove_code_root":"%trove_path%",
|
|
||||||
"trove_conf":"/tmp/trove.conf",
|
"trove_conf":"/tmp/trove.conf",
|
||||||
"trove_version":"v1.0",
|
"trove_version":"v1.0",
|
||||||
"trove_api_updated":"2012-08-01T00:00:00Z",
|
"trove_api_updated":"2012-08-01T00:00:00Z",
|
||||||
|
@ -1,3 +0,0 @@
|
|||||||
Sets up a MySQL server install in the image.
|
|
||||||
|
|
||||||
TODO: auto-tune settings based on host resources or metadata service.
|
|
@ -6,5 +6,3 @@
|
|||||||
dd if=/tmp/in_target.d/trove-guest.service of=/etc/systemd/system/trove-guest.service
|
dd if=/tmp/in_target.d/trove-guest.service of=/etc/systemd/system/trove-guest.service
|
||||||
|
|
||||||
systemctl enable trove-guest.service
|
systemctl enable trove-guest.service
|
||||||
|
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
GUEST_UNIT_DROPINS="/etc/systemd/system/trove-guest.service.d"
|
GUEST_UNIT_DROPINS="/etc/systemd/system/trove-guest.service.d"
|
||||||
|
|
||||||
mkdir -v -p ${GUEST_UNIT_DROPINS}
|
mkdir -v -p ${GUEST_UNIT_DROPINS}
|
||||||
echo -e '[Service]\nEnvironment=REQUESTS_CA_BUNDLE=/etc/ssl/certs' > ${GUEST_UNIT_DROPINS}/30-use-system-certificates.conf
|
cat <<EOF > ${GUEST_UNIT_DROPINS}/30-use-system-certificates.conf
|
||||||
|
[Service]
|
||||||
|
Environment=REQUESTS_CA_BUNDLE=/etc/ssl/certs
|
||||||
|
EOF
|
||||||
|
@ -14,17 +14,17 @@ apt-key adv --recv-keys --keyserver hkp://keyserver.ubuntu.com:80 0xF1656F24C74C
|
|||||||
curl -sS https://downloads.mariadb.com/MariaDB/mariadb_repo_setup |
|
curl -sS https://downloads.mariadb.com/MariaDB/mariadb_repo_setup |
|
||||||
bash -s -- --mariadb-server-version="mariadb-10.4" --skip-key-import --skip-maxscale
|
bash -s -- --mariadb-server-version="mariadb-10.4" --skip-key-import --skip-maxscale
|
||||||
|
|
||||||
|
apt-get install -y -qq apt-transport-https ca-certificates gnupg2
|
||||||
|
|
||||||
# NOTE(lxkong): Refer to https://www.percona.com/doc/percona-xtrabackup/2.4/installation/apt_repo.html
|
# NOTE(lxkong): Refer to https://www.percona.com/doc/percona-xtrabackup/2.4/installation/apt_repo.html
|
||||||
wget https://repo.percona.com/apt/percona-release_latest.$(lsb_release -sc)_all.deb
|
wget https://repo.percona.com/apt/percona-release_latest.$(lsb_release -sc)_all.deb
|
||||||
dpkg -i percona-release_latest.$(lsb_release -sc)_all.deb
|
dpkg -i percona-release_latest.$(lsb_release -sc)_all.deb
|
||||||
|
|
||||||
apt-get install -y -qq apt-transport-https ca-certificates
|
|
||||||
apt-get update -qq
|
|
||||||
|
|
||||||
# Disable password prompt
|
# Disable password prompt
|
||||||
debconf-set-selections <<< "mariadb-server mysql-server/root_password password ''"
|
debconf-set-selections <<< "mariadb-server mysql-server/root_password password ''"
|
||||||
debconf-set-selections <<< "mariadb-server mysql-server/root_password_again password ''"
|
debconf-set-selections <<< "mariadb-server mysql-server/root_password_again password ''"
|
||||||
|
|
||||||
|
apt-get update -qq
|
||||||
apt-get install -y -qq --allow-unauthenticated mariadb-server mariadb-client galera-4 libmariadb3 mariadb-backup mariadb-common
|
apt-get install -y -qq --allow-unauthenticated mariadb-server mariadb-client galera-4 libmariadb3 mariadb-backup mariadb-common
|
||||||
|
|
||||||
cat <<EOF >/etc/mysql/conf.d/no_perf_schema.cnf
|
cat <<EOF >/etc/mysql/conf.d/no_perf_schema.cnf
|
||||||
@ -34,5 +34,6 @@ EOF
|
|||||||
|
|
||||||
chown mysql:mysql /etc/mysql/my.cnf
|
chown mysql:mysql /etc/mysql/my.cnf
|
||||||
rm -f /etc/init.d/mysql
|
rm -f /etc/init.d/mysql
|
||||||
|
|
||||||
systemctl daemon-reload
|
systemctl daemon-reload
|
||||||
systemctl enable mariadb
|
systemctl enable mariadb
|
@ -8,44 +8,15 @@ set -o xtrace
|
|||||||
|
|
||||||
export DEBIAN_FRONTEND=noninteractive
|
export DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
apt-get --allow-unauthenticated -y install mysql-client mysql-server
|
apt-get --allow-unauthenticated -y install mysql-client mysql-server gnupg2
|
||||||
|
|
||||||
# Xenial provides mysql 5.7 which requires percona-xtrabackup-24
|
|
||||||
PXB_VERSION_OVERRIDE=24
|
|
||||||
#PKGS=$(apt-cache search percona-xtrabackup-${PXB_VERSION_OVERRIDE})
|
|
||||||
#if [[ "$PKGS" == *"percona-xtrabackup-$PXB_VERSION_OVERRIDE"* ]]; then
|
|
||||||
# apt-get --allow-unauthenticated -y install percona-xtrabackup-${PXB_VERSION_OVERRIDE}
|
|
||||||
#else
|
|
||||||
# # Architecture is not supported by percona website. Compile and install it
|
|
||||||
# PXB_VERSION=${PXB_VERSION_OVERRIDE:0:1}.${PXB_VERSION_OVERRIDE:1:1}
|
|
||||||
#
|
|
||||||
# apt-get --allow-unauthenticated -y install build-essential flex bison automake autoconf \
|
|
||||||
# libtool cmake libaio-dev mysql-client libncurses-dev zlib1g-dev \
|
|
||||||
# libgcrypt11-dev libev-dev libcurl4-gnutls-dev vim-common
|
|
||||||
#
|
|
||||||
# pushd /tmp
|
|
||||||
#
|
|
||||||
# git clone https://github.com/percona/percona-xtrabackup.git
|
|
||||||
# cd percona-xtrabackup
|
|
||||||
# git checkout $PXB_VERSION
|
|
||||||
#
|
|
||||||
# mkdir /tmp/boost
|
|
||||||
# cmake -DDOWNLOAD_BOOST=1 -DWITH_BOOST=/tmp/boost -DBUILD_CONFIG=xtrabackup_release -DWITH_MAN_PAGES=OFF && make -j4
|
|
||||||
# make install
|
|
||||||
# ln -s /usr/local/xtrabackup/bin/* /usr/bin/
|
|
||||||
#
|
|
||||||
# dpkg -P build-essential automake autoconf libtool cmake
|
|
||||||
# apt-get -y clean
|
|
||||||
#
|
|
||||||
# popd
|
|
||||||
#
|
|
||||||
# rm -rf /tmp/boost /tmp/percona-xtrabackup
|
|
||||||
#fi
|
|
||||||
|
|
||||||
# NOTE(lxkong): Refer to https://www.percona.com/doc/percona-xtrabackup/2.4/installation/apt_repo.html
|
# NOTE(lxkong): Refer to https://www.percona.com/doc/percona-xtrabackup/2.4/installation/apt_repo.html
|
||||||
wget https://repo.percona.com/apt/percona-release_latest.$(lsb_release -sc)_all.deb
|
wget https://repo.percona.com/apt/percona-release_latest.$(lsb_release -sc)_all.deb
|
||||||
dpkg -i percona-release_latest.$(lsb_release -sc)_all.deb
|
dpkg -i percona-release_latest.$(lsb_release -sc)_all.deb
|
||||||
apt-get update
|
apt-get update
|
||||||
|
|
||||||
|
# Xenial provides mysql 5.7 which requires percona-xtrabackup-24
|
||||||
|
PXB_VERSION_OVERRIDE=24
|
||||||
apt-get --allow-unauthenticated -y install percona-xtrabackup-${PXB_VERSION_OVERRIDE}
|
apt-get --allow-unauthenticated -y install percona-xtrabackup-${PXB_VERSION_OVERRIDE}
|
||||||
|
|
||||||
cat >/etc/mysql/conf.d/no_perf_schema.cnf <<_EOF_
|
cat >/etc/mysql/conf.d/no_perf_schema.cnf <<_EOF_
|
||||||
@ -58,7 +29,6 @@ mv /etc/mysql/my.cnf.fallback /etc/mysql/my.cnf
|
|||||||
chown mysql:mysql /etc/mysql/my.cnf
|
chown mysql:mysql /etc/mysql/my.cnf
|
||||||
cat >/etc/mysql/my.cnf <<_EOF_
|
cat >/etc/mysql/my.cnf <<_EOF_
|
||||||
[mysql]
|
[mysql]
|
||||||
|
|
||||||
!includedir /etc/mysql/conf.d/
|
!includedir /etc/mysql/conf.d/
|
||||||
_EOF_
|
_EOF_
|
||||||
|
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
[Unit]
|
[Unit]
|
||||||
Description=Trove Guest
|
Description=Trove Guest
|
||||||
After=syslog.target
|
After=syslog.target network.target
|
||||||
After=network.target
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
Type=simple
|
Type=simple
|
||||||
@ -13,25 +15,23 @@ Group=GUEST_USERNAME
|
|||||||
# CONTROLLER=192.168.32.151
|
# CONTROLLER=192.168.32.151
|
||||||
EnvironmentFile=/etc/trove/controller.conf
|
EnvironmentFile=/etc/trove/controller.conf
|
||||||
|
|
||||||
ExecStartPre=/bin/bash -c "sudo mkdir -p GUEST_LOGDIR ; sudo chown GUEST_USERNAME:root GUEST_LOGDIR"
|
ExecStartPre=/bin/bash -c "sudo mkdir -p GUEST_LOGDIR"
|
||||||
|
|
||||||
# If ~/trove-installed does not exist, copy the trove source from
|
# If ~/trove-installed does not exist, copy the trove source from
|
||||||
# the user's development environment, then touch the sentinel file
|
# the user's development environment, then touch the sentinel file
|
||||||
ExecStartPre=/bin/bash -c "test -e /home/GUEST_USERNAME/trove-installed || sudo -u GUEST_USERNAME rsync -e 'ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no' -avz --exclude='.*' HOST_SCP_USERNAME@$CONTROLLER:PATH_TROVE/ /home/GUEST_USERNAME/trove && touch /home/GUEST_USERNAME/trove-installed"
|
ExecStartPre=/bin/bash -c "test -e /home/GUEST_USERNAME/trove-installed || sudo rsync -e 'ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no -i /home/GUEST_USERNAME/.ssh/id_rsa' -avz --exclude='.*' HOST_SCP_USERNAME@$CONTROLLER:PATH_TROVE/ /home/GUEST_USERNAME/trove && touch /home/GUEST_USERNAME/trove-installed"
|
||||||
|
|
||||||
# If /etc/trove does not exist, create it and then copy the trove-guestagent.conf
|
# If /etc/trove does not exist, create it and then copy the trove-guestagent.conf
|
||||||
# from /etc/trove on the user's development environment,
|
# from /etc/trove on the user's development environment,
|
||||||
ExecStartPre=/bin/bash -c "test -d /etc/trove/conf.d || sudo mkdir -p /etc/trove/conf.d && sudo -u GUEST_USERNAME rsync -e 'ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no' -avz --exclude='.*' HOST_SCP_USERNAME@$CONTROLLER:/etc/trove/trove-guestagent.conf ~GUEST_USERNAME/ && sudo mv ~GUEST_USERNAME/trove-guestagent.conf /etc/trove/conf.d/trove-guestagent.conf"
|
ExecStartPre=/bin/bash -c "test -d /etc/trove/conf.d || sudo mkdir -p /etc/trove/conf.d && sudo rsync -e 'ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no -i /home/GUEST_USERNAME/.ssh/id_rsa' -avz --exclude='.*' HOST_SCP_USERNAME@$CONTROLLER:/etc/trove/trove-guestagent.conf ~GUEST_USERNAME/ && sudo mv ~GUEST_USERNAME/trove-guestagent.conf /etc/trove/conf.d/trove-guestagent.conf"
|
||||||
|
|
||||||
ExecStartPre=/bin/bash -c "sudo chown -R GUEST_USERNAME:root /etc/trove"
|
ExecStartPre=/bin/bash -c "sudo chown -R GUEST_USERNAME:root /etc/trove /home/GUEST_USERNAME/trove GUEST_LOGDIR"
|
||||||
|
|
||||||
ExecStart=/home/GUEST_USERNAME/trove/contrib/trove-guestagent --config-dir=/etc/trove/conf.d
|
# Start trove-guest.service
|
||||||
|
ExecStart=/bin/bash -c "/home/GUEST_USERNAME/trove/contrib/trove-guestagent --config-dir=/etc/trove/conf.d"
|
||||||
|
|
||||||
# Give a reasonable amount of time for the server to start up/shut down
|
|
||||||
TimeoutSec=300
|
TimeoutSec=300
|
||||||
|
Restart=on-failure
|
||||||
|
|
||||||
# PgSql doesn't play nice with PrivateTmp
|
# PostgreSQL doesn't play nice with PrivateTmp
|
||||||
PrivateTmp=false
|
PrivateTmp=false
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
|
@ -66,7 +66,9 @@ function build_vm() {
|
|||||||
elementes="$elementes pip-cache"
|
elementes="$elementes pip-cache"
|
||||||
elementes="$elementes guest-agent"
|
elementes="$elementes guest-agent"
|
||||||
else
|
else
|
||||||
|
# Install guest agent dependencies, user, etc.
|
||||||
elementes="$elementes ${guest_os}-guest"
|
elementes="$elementes ${guest_os}-guest"
|
||||||
|
# Install guest agent service
|
||||||
elementes="$elementes ${guest_os}-${guest_release}-guest"
|
elementes="$elementes ${guest_os}-${guest_release}-guest"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
@ -514,10 +514,6 @@ function get_field() {
|
|||||||
done
|
done
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_glance_id () {
|
|
||||||
echo `$@ | grep ' id ' | get_field 2`
|
|
||||||
}
|
|
||||||
|
|
||||||
function set_bin_path() {
|
function set_bin_path() {
|
||||||
if is_fedora; then
|
if is_fedora; then
|
||||||
sed -i "s|%bin_path%|/usr/bin|g" $TEST_CONF
|
sed -i "s|%bin_path%|/usr/bin|g" $TEST_CONF
|
||||||
@ -526,35 +522,16 @@ function set_bin_path() {
|
|||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
function set_mysql_pkg() {
|
|
||||||
if is_fedora; then
|
|
||||||
MYSQL_PKG="mysql-community-server"
|
|
||||||
MYSQL_VER="5.6"
|
|
||||||
else
|
|
||||||
if [[ "$RELEASE" == "xenial" || "$RELEASE" == "bionic" ]]; then
|
|
||||||
MYSQL_PKG="mysql-server-5.7"
|
|
||||||
MYSQL_VER="5.7"
|
|
||||||
else
|
|
||||||
MYSQL_PKG="mysql-server-5.6"
|
|
||||||
MYSQL_VER="5.6"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function cmd_set_datastore() {
|
function cmd_set_datastore() {
|
||||||
local IMAGEID=$1
|
local IMAGEID=$1
|
||||||
local DATASTORE_TYPE=$2
|
local DATASTORE_TYPE=$2
|
||||||
local RESTART_TROVE=${3:-$(get_bool RESTART_TROVE "true")}
|
|
||||||
|
|
||||||
# rd_manage datastore_update <datastore_name> <default_version>
|
# rd_manage datastore_update <datastore_name> <default_version>
|
||||||
rd_manage datastore_update "$DATASTORE_TYPE" ""
|
rd_manage datastore_update "$DATASTORE_TYPE" ""
|
||||||
PACKAGES=${PACKAGES:-""}
|
PACKAGES=${PACKAGES:-""}
|
||||||
|
|
||||||
if [ "$DATASTORE_TYPE" == "mysql" ]; then
|
if [ "$DATASTORE_TYPE" == "mysql" ]; then
|
||||||
set_mysql_pkg
|
VERSION="5.7"
|
||||||
PACKAGES=${PACKAGES:-$MYSQL_PKG}
|
|
||||||
VERSION=$MYSQL_VER
|
|
||||||
elif [ "$DATASTORE_TYPE" == "percona" ]; then
|
elif [ "$DATASTORE_TYPE" == "percona" ]; then
|
||||||
PACKAGES=${PACKAGES:-"percona-server-server-5.6"}
|
PACKAGES=${PACKAGES:-"percona-server-server-5.6"}
|
||||||
VERSION="5.6"
|
VERSION="5.6"
|
||||||
@ -562,7 +539,6 @@ function cmd_set_datastore() {
|
|||||||
PACKAGES=${PACKAGES:-"percona-xtradb-cluster-server-5.6"}
|
PACKAGES=${PACKAGES:-"percona-xtradb-cluster-server-5.6"}
|
||||||
VERSION="5.6"
|
VERSION="5.6"
|
||||||
elif [ "$DATASTORE_TYPE" == "mariadb" ]; then
|
elif [ "$DATASTORE_TYPE" == "mariadb" ]; then
|
||||||
PACKAGES=${PACKAGES:-"mariadb-server"}
|
|
||||||
VERSION="10.4"
|
VERSION="10.4"
|
||||||
elif [ "$DATASTORE_TYPE" == "mongodb" ]; then
|
elif [ "$DATASTORE_TYPE" == "mongodb" ]; then
|
||||||
PACKAGES=${PACKAGES:-"mongodb-org"}
|
PACKAGES=${PACKAGES:-"mongodb-org"}
|
||||||
@ -577,7 +553,6 @@ function cmd_set_datastore() {
|
|||||||
PACKAGES=${PACKAGES:-"couchbase-server"}
|
PACKAGES=${PACKAGES:-"couchbase-server"}
|
||||||
VERSION="2.2.0"
|
VERSION="2.2.0"
|
||||||
elif [ "$DATASTORE_TYPE" == "postgresql" ]; then
|
elif [ "$DATASTORE_TYPE" == "postgresql" ]; then
|
||||||
PACKAGES=${PACKAGES:-"postgresql-9.6"}
|
|
||||||
VERSION="9.6"
|
VERSION="9.6"
|
||||||
elif [ "$DATASTORE_TYPE" == "couchdb" ]; then
|
elif [ "$DATASTORE_TYPE" == "couchdb" ]; then
|
||||||
PACKAGES=${PACKAGES:-"couchdb"}
|
PACKAGES=${PACKAGES:-"couchdb"}
|
||||||
@ -593,28 +568,14 @@ function cmd_set_datastore() {
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
sed -i "s/%datastore_type%/$DATASTORE_TYPE/g" $TEST_CONF
|
# trove-manage datastore_version_update <datastore_name> <version_name> <datastore_manager> <image_id> <packages> <active>
|
||||||
sed -i "s/%datastore_version%/$VERSION/g" $TEST_CONF
|
|
||||||
|
|
||||||
#rd_manage datastore_version_update <datastore_name> <version_name> <datastore_manager> <image_id> <packages> <active>
|
|
||||||
rd_manage datastore_version_update "$DATASTORE_TYPE" "$VERSION" "$DATASTORE_TYPE" $IMAGEID "$PACKAGES" 1
|
rd_manage datastore_version_update "$DATASTORE_TYPE" "$VERSION" "$DATASTORE_TYPE" $IMAGEID "$PACKAGES" 1
|
||||||
rd_manage datastore_version_update "$DATASTORE_TYPE" "inactive_version" "manager1" $IMAGEID "" 0
|
|
||||||
rd_manage datastore_update "$DATASTORE_TYPE" "$VERSION"
|
rd_manage datastore_update "$DATASTORE_TYPE" "$VERSION"
|
||||||
rd_manage datastore_update Test_Datastore_1 ""
|
|
||||||
|
|
||||||
if [ -f "$PATH_TROVE"/trove/templates/$DATASTORE_TYPE/validation-rules.json ]; then
|
if [ -f "$PATH_TROVE"/trove/templates/$DATASTORE_TYPE/validation-rules.json ]; then
|
||||||
# add the configuration parameters to the database for the kick-start datastore
|
# add the configuration parameters to the database for the kick-start datastore
|
||||||
rd_manage db_load_datastore_config_parameters "$DATASTORE_TYPE" "$VERSION" "$PATH_TROVE"/trove/templates/$DATASTORE_TYPE/validation-rules.json
|
rd_manage db_load_datastore_config_parameters "$DATASTORE_TYPE" "$VERSION" "$PATH_TROVE"/trove/templates/$DATASTORE_TYPE/validation-rules.json
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ "${RESTART_TROVE}" == true ]]; then
|
|
||||||
cmd_stop
|
|
||||||
fi
|
|
||||||
iniset $TROVE_CONF DEFAULT default_datastore "$DATASTORE_TYPE"
|
|
||||||
sleep 1.5
|
|
||||||
if [[ "${RESTART_TROVE}" == true ]]; then
|
|
||||||
cmd_start
|
|
||||||
fi
|
|
||||||
}
|
}
|
||||||
|
|
||||||
###############################################################################
|
###############################################################################
|
||||||
@ -681,7 +642,8 @@ function install_test_packages() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function mod_confs() {
|
function mod_confs() {
|
||||||
DATASTORE_TYPE=$1
|
local DATASTORE_TYPE=$1
|
||||||
|
local DATASTORE_VERSION=$2
|
||||||
exclaim "Running mod_confs ..."
|
exclaim "Running mod_confs ..."
|
||||||
|
|
||||||
sudo install -b --mode 0664 $TROVESTACK_SCRIPTS/conf/test_begin.conf $TEST_CONF
|
sudo install -b --mode 0664 $TROVESTACK_SCRIPTS/conf/test_begin.conf $TEST_CONF
|
||||||
@ -701,20 +663,9 @@ function mod_confs() {
|
|||||||
cat $DATASTORE_CONF | sudo tee -a $TEST_CONF > /dev/null
|
cat $DATASTORE_CONF | sudo tee -a $TEST_CONF > /dev/null
|
||||||
cat $TROVESTACK_SCRIPTS/conf/test_end.conf | sudo tee -a $TEST_CONF > /dev/null
|
cat $TROVESTACK_SCRIPTS/conf/test_end.conf | sudo tee -a $TEST_CONF > /dev/null
|
||||||
|
|
||||||
#When running in the gate, don't start services
|
|
||||||
if [ "${DEVSTACK_GATE_TROVE}" == "1" ]; then
|
|
||||||
sed -i "s,%startservices%,false,g" ${TEST_CONF}
|
|
||||||
else
|
|
||||||
sed -i "s,%startservices%,true,g" ${TEST_CONF}
|
|
||||||
fi
|
|
||||||
#Add the paths to the test conf
|
#Add the paths to the test conf
|
||||||
sed -i "s,%report_directory%,$TROVE_REPORT_DIR,g" $TEST_CONF
|
sed -i "s,%report_directory%,$TROVE_REPORT_DIR,g" $TEST_CONF
|
||||||
sed -i "s,%keystone_path%,$PATH_KEYSTONE,g" $TEST_CONF
|
|
||||||
sed -i "s,%nova_path%,$PATH_NOVA,g" $TEST_CONF
|
|
||||||
sed -i "s,%glance_path%,$PATH_GLANCE,g" $TEST_CONF
|
|
||||||
sed -i "s,%trove_path%,$PATH_TROVE,g" $TEST_CONF
|
|
||||||
sed -i "s,%service_host%,$SERVICE_HOST,g" $TEST_CONF
|
sed -i "s,%service_host%,$SERVICE_HOST,g" $TEST_CONF
|
||||||
sed -i "s,%swifth_path%,$PATH_SWIFT,g" $TEST_CONF
|
|
||||||
|
|
||||||
# Add the region name into test.conf
|
# Add the region name into test.conf
|
||||||
sed -i "s/%region_name%/${REGION_NAME}/g" $TEST_CONF
|
sed -i "s/%region_name%/${REGION_NAME}/g" $TEST_CONF
|
||||||
@ -759,8 +710,10 @@ function mod_confs() {
|
|||||||
iniset $TROVE_CONF $DATASTORE_TYPE num_config_servers_per_cluster 1
|
iniset $TROVE_CONF $DATASTORE_TYPE num_config_servers_per_cluster 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
set_bin_path
|
sed -i "s/%datastore_type%/$DATASTORE_TYPE/g" $TEST_CONF
|
||||||
|
sed -i "s/%datastore_version%/${DATASTORE_VERSION}/g" $TEST_CONF
|
||||||
|
|
||||||
|
set_bin_path
|
||||||
}
|
}
|
||||||
|
|
||||||
function setup_cluster_configs() {
|
function setup_cluster_configs() {
|
||||||
@ -794,6 +747,7 @@ function add_test_flavors() {
|
|||||||
|
|
||||||
function cmd_test_init() {
|
function cmd_test_init() {
|
||||||
local DATASTORE_TYPE=$1
|
local DATASTORE_TYPE=$1
|
||||||
|
local DATASTORE_VERSION=$2
|
||||||
|
|
||||||
if [ -z "${DATASTORE_TYPE}" ]; then
|
if [ -z "${DATASTORE_TYPE}" ]; then
|
||||||
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
||||||
@ -806,7 +760,7 @@ function cmd_test_init() {
|
|||||||
install_test_packages "${DATASTORE_TYPE}"
|
install_test_packages "${DATASTORE_TYPE}"
|
||||||
|
|
||||||
exclaim "Modifying test.conf and guest.conf with appropriate values."
|
exclaim "Modifying test.conf and guest.conf with appropriate values."
|
||||||
mod_confs "${DATASTORE_TYPE}"
|
mod_confs "${DATASTORE_TYPE}" "${DATASTORE_VERSION}"
|
||||||
|
|
||||||
exclaim "Creating Test Flavors."
|
exclaim "Creating Test Flavors."
|
||||||
add_test_flavors
|
add_test_flavors
|
||||||
@ -823,7 +777,7 @@ function cmd_build_image() {
|
|||||||
local output=$6
|
local output=$6
|
||||||
|
|
||||||
if [[ -z "$output" ]]; then
|
if [[ -z "$output" ]]; then
|
||||||
image_name="trove-${IMAGE_GUEST_OS}-${IMAGE_GUEST_RELEASE}-${IMAGE_DATASTORE_TYPE}"
|
image_name="trove-datastore-${IMAGE_GUEST_OS}-${IMAGE_GUEST_RELEASE}-${IMAGE_DATASTORE_TYPE}"
|
||||||
image_folder=$HOME/images
|
image_folder=$HOME/images
|
||||||
output="${image_folder}/${image_name}"
|
output="${image_folder}/${image_name}"
|
||||||
fi
|
fi
|
||||||
@ -840,13 +794,17 @@ function cmd_build_image() {
|
|||||||
build_guest_image $IMAGE_DATASTORE_TYPE $IMAGE_GUEST_OS $IMAGE_GUEST_RELEASE $DEV_MODE ${guest_username} $output
|
build_guest_image $IMAGE_DATASTORE_TYPE $IMAGE_GUEST_OS $IMAGE_GUEST_RELEASE $DEV_MODE ${guest_username} $output
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Build guest image and upload to Glance, register the datastore and configuration parameters.
|
||||||
|
# We could skip the image build and upload by:
|
||||||
|
# 1. MYSQL_IMAGE_ID is passed, or
|
||||||
|
# 2. There is an image in Glance contains the datastore name
|
||||||
function cmd_build_and_upload_image() {
|
function cmd_build_and_upload_image() {
|
||||||
local datastore_type=$1
|
local datastore_type=$1
|
||||||
local restart_trove=${2:-$(get_bool RESTART_TROVE "true")}
|
local guest_os=${2:-"ubuntu"}
|
||||||
local guest_os=${3:-"ubuntu"}
|
local guest_release=${3:-"xenial"}
|
||||||
local guest_release=${4:-"xenial"}
|
local dev_mode=${4:-"true"}
|
||||||
local dev_mode=${5:-"true"}
|
local guest_username=${5:-"ubuntu"}
|
||||||
local guest_username=${6:-"ubuntu"}
|
local output_dir=${6:-"$HOME/images"}
|
||||||
|
|
||||||
if [ -z "${datastore_type}" ]; then
|
if [ -z "${datastore_type}" ]; then
|
||||||
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
||||||
@ -861,21 +819,20 @@ function cmd_build_and_upload_image() {
|
|||||||
glance_imageid=$(openstack $CLOUD_ADMIN_ARG image list | grep "$datastore_type" | awk 'NR==1 {print}' | awk '{print $2}')
|
glance_imageid=$(openstack $CLOUD_ADMIN_ARG image list | grep "$datastore_type" | awk 'NR==1 {print}' | awk '{print $2}')
|
||||||
|
|
||||||
if [[ -z $glance_imageid ]]; then
|
if [[ -z $glance_imageid ]]; then
|
||||||
cmd_build_image ${datastore_type} ${guest_os} ${guest_release} ${dev_mode} ${guest_username}
|
mkdir -p ${output_dir}
|
||||||
|
name=trove-datastore-${guest_os}-${guest_release}-${datastore_type}
|
||||||
|
output=${output_dir}/$name.qcow2
|
||||||
|
cmd_build_image ${datastore_type} ${guest_os} ${guest_release} ${dev_mode} ${guest_username} $output
|
||||||
|
|
||||||
image_folder=$HOME/images
|
glance_imageid=$(openstack ${CLOUD_ADMIN_ARG} image create $name --public --disk-format qcow2 --container-format bare --file $output --property hw_rng_model='virtio' -c id -f value)
|
||||||
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
|
[[ -z "$glance_imageid" ]] && echo "Glance upload failed!" && exit 1
|
||||||
echo "IMAGE ID: $glance_imageid"
|
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo "IMAGEID: $glance_imageid"
|
exclaim "Using Glance image ID: $glance_imageid"
|
||||||
|
|
||||||
exclaim "Updating Datastores"
|
exclaim "Updating Datastores"
|
||||||
cmd_set_datastore "${glance_imageid}" "${datastore_type}" "${restart_trove}"
|
cmd_set_datastore "${glance_imageid}" "${datastore_type}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1037,7 +994,7 @@ function cmd_int_tests() {
|
|||||||
fi
|
fi
|
||||||
cd $TROVESTACK_SCRIPTS
|
cd $TROVESTACK_SCRIPTS
|
||||||
if [ $# -lt 1 ]; then
|
if [ $# -lt 1 ]; then
|
||||||
args="--group=blackbox"
|
args="--group=mysql"
|
||||||
else
|
else
|
||||||
args="$@"
|
args="$@"
|
||||||
fi
|
fi
|
||||||
@ -1055,23 +1012,6 @@ function cmd_int_tests() {
|
|||||||
python $args
|
python $args
|
||||||
}
|
}
|
||||||
|
|
||||||
function cmd_int_tests_simple() {
|
|
||||||
exclaim "Running Trove Simple Integration Tests..."
|
|
||||||
cd $TROVESTACK_SCRIPTS
|
|
||||||
if [ $# -lt 1 ]; then
|
|
||||||
args="--group=simple_blackbox"
|
|
||||||
else
|
|
||||||
args="$@"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# -- verbose makes it prettier.
|
|
||||||
# -- logging-clear-handlers keeps the novaclient and other things from
|
|
||||||
# spewing logs to stdout.
|
|
||||||
args="$INT_TEST_OPTIONS -B $TROVESTACK_TESTS/integration/int_tests.py --verbose --logging-clear-handlers $args"
|
|
||||||
echo "python $args"
|
|
||||||
python $args
|
|
||||||
}
|
|
||||||
|
|
||||||
function cmd_int_tests_white_box() {
|
function cmd_int_tests_white_box() {
|
||||||
export PYTHONPATH=$PYTHONPATH:$PATH_TROVE
|
export PYTHONPATH=$PYTHONPATH:$PATH_TROVE
|
||||||
export PYTHONPATH=$PYTHONPATH:$PATH_NOVA
|
export PYTHONPATH=$PYTHONPATH:$PATH_NOVA
|
||||||
@ -1259,49 +1199,45 @@ function cmd_clean() {
|
|||||||
|
|
||||||
function cmd_kick_start() {
|
function cmd_kick_start() {
|
||||||
local DATASTORE_TYPE=$1
|
local DATASTORE_TYPE=$1
|
||||||
local RESTART_TROVE=${2:-$(get_bool RESTART_TROVE "true")}
|
local DATASTORE_VERSION=$2
|
||||||
|
|
||||||
if [ -z "${DATASTORE_TYPE}" ]; then
|
if [ -z "${DATASTORE_TYPE}" ]; then
|
||||||
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
exclaim "${COLOR_RED}Datastore argument was not specified.${COLOR_NONE}"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
exclaim "Running kick-start for $DATASTORE_TYPE (restart trove: $RESTART_TROVE)"
|
exclaim "Running kick-start for $DATASTORE_TYPE"
|
||||||
dump_env
|
dump_env
|
||||||
cmd_test_init "${DATASTORE_TYPE}"
|
cmd_test_init "${DATASTORE_TYPE}" "${DATASTORE_VERSION}"
|
||||||
|
|
||||||
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}"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Start functional test. The guest image should be created and registered in
|
||||||
|
# appropriate datastore before the test, the configuration parameters should
|
||||||
|
# also be loaded as well. DevStack has done all of that.
|
||||||
function cmd_gate_tests() {
|
function cmd_gate_tests() {
|
||||||
local DATASTORE_TYPE=${1:-'mysql'}
|
local DATASTORE_TYPE=${1:-'mysql'}
|
||||||
local TEST_GROUP=${2:-${DATASTORE_TYPE}}
|
local TEST_GROUP=${2:-${DATASTORE_TYPE}}
|
||||||
local HOST_SCP_USERNAME=${3:-$(whoami)}
|
local DATASTORE_VERSION=${3:-'5.7'}
|
||||||
local GUEST_USERNAME=${4:-'ubuntu'}
|
local HOST_SCP_USERNAME=${4:-$(whoami)}
|
||||||
# We're not using devstack-gate in Zuul v3 job
|
local GUEST_USERNAME=${5:-'ubuntu'}
|
||||||
if [[ $GATE_JOB_VER == "v2" ]]; then
|
|
||||||
local ESCAPED_PATH_TROVE=${5:-'\/opt\/stack\/new\/trove'}
|
|
||||||
fi
|
|
||||||
|
|
||||||
exclaim "Running cmd_gate_tests ..."
|
exclaim "Running cmd_gate_tests ..."
|
||||||
|
|
||||||
export REPORT_DIRECTORY=${REPORT_DIRECTORY:=$HOME/gate-tests-report/}
|
export REPORT_DIRECTORY=${REPORT_DIRECTORY:=$HOME/gate-tests-report/}
|
||||||
export TROVE_REPORT_DIR=$HOME/gate-tests-report/
|
export TROVE_REPORT_DIR=$HOME/gate-tests-report/
|
||||||
TROVESTACK_DUMP_ENV=true
|
export TROVESTACK_DUMP_ENV=true
|
||||||
|
|
||||||
export SSH_DIR=${SSH_DIR:-"$HOME/.ssh"}
|
export SSH_DIR=${SSH_DIR:-"$HOME/.ssh"}
|
||||||
# The user used to connect the db instance.
|
# The user is used to connect with the db instance during testing.
|
||||||
export TROVE_TEST_SSH_USER=${TROVE_TEST_SSH_USER:-"ubuntu"}
|
export TROVE_TEST_SSH_USER=${TROVE_TEST_SSH_USER:-"ubuntu"}
|
||||||
# This var is used to ssh into the db instance during the test.
|
# This var is used to ssh into the db instance during testing.
|
||||||
export TROVE_TEST_SSH_KEY_FILE=${SSH_DIR}/id_rsa
|
export TROVE_TEST_SSH_KEY_FILE=${SSH_DIR}/id_rsa
|
||||||
|
|
||||||
cd $TROVESTACK_SCRIPTS
|
cd $TROVESTACK_SCRIPTS
|
||||||
local RESTART_TROVE=false
|
|
||||||
cmd_kick_start "${DATASTORE_TYPE}" "${RESTART_TROVE}"
|
# Build and upload guest image, register datastore version.
|
||||||
|
cmd_build_and_upload_image ${DATASTORE_TYPE}
|
||||||
|
|
||||||
|
cmd_kick_start "${DATASTORE_TYPE}" "${DATASTORE_VERSION}"
|
||||||
|
|
||||||
cmd_int_tests --group=$TEST_GROUP
|
cmd_int_tests --group=$TEST_GROUP
|
||||||
}
|
}
|
||||||
@ -1441,42 +1377,13 @@ function run_command() {
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
case "$1" in
|
case "$1" in
|
||||||
"install" ) cmd_install;;
|
|
||||||
"test-init" ) shift; cmd_test_init $@;;
|
|
||||||
"build-image" ) shift; cmd_build_image $@;;
|
"build-image" ) shift; cmd_build_image $@;;
|
||||||
"initialize" ) cmd_initialize;;
|
"upload-image" ) shift; cmd_build_and_upload_image $@;;
|
||||||
"unit-tests" ) cmd_unit_tests;;
|
|
||||||
"start-deps" ) cmd_start_deps;;
|
|
||||||
"stop-deps" ) cmd_stop_deps;;
|
|
||||||
"start" ) cmd_start;;
|
|
||||||
"int-tests" ) shift; cmd_int_tests $@;;
|
"int-tests" ) shift; cmd_int_tests $@;;
|
||||||
"int-tests-wb" ) shift; cmd_int_tests_white_box $@;;
|
|
||||||
"simple-tests") shift; cmd_int_tests_simple $@;;
|
|
||||||
"stop" ) cmd_stop;;
|
|
||||||
"restart" ) cmd_stop; cmd_start;;
|
|
||||||
"wipe-logs" ) cmd_wipe_logs;;
|
|
||||||
"rd-sql" ) shift; cmd_rd_sql $@;;
|
|
||||||
"fake-sql" ) shift; cmd_fake_sql $@;;
|
|
||||||
"run-ci" ) shift; cmd_run_ci $@;;
|
|
||||||
"vagrant-ssh" ) shift; cmd_vagrant_ssh $@;;
|
|
||||||
"debug" ) shift; echo "Enabling debugging."; \
|
"debug" ) shift; echo "Enabling debugging."; \
|
||||||
set -o xtrace; TROVESTACK_DUMP_ENV=true; run_command $@;;
|
set -o xtrace; TROVESTACK_DUMP_ENV=true; run_command $@;;
|
||||||
"clear" ) shift; cmd_clear $@;;
|
|
||||||
"clean" ) shift; cmd_clean $@;;
|
|
||||||
"run" ) shift; cmd_run $@;;
|
|
||||||
"kick-start" ) shift; cmd_kick_start $@;;
|
|
||||||
"dsvm-gate-tests" ) shift; export GATE_JOB_VER=v2; \
|
|
||||||
cmd_gate_tests $@;;
|
|
||||||
"gate-tests" ) shift; cmd_gate_tests $@;;
|
"gate-tests" ) shift; cmd_gate_tests $@;;
|
||||||
"run-fake" ) shift; cmd_run_fake $@;;
|
|
||||||
"start-fake" ) shift; cmd_start_fake $@;;
|
|
||||||
"update-projects" ) cmd_clone_projects force_update \
|
|
||||||
$TROVESTACK_SCRIPTS/projects-list \
|
|
||||||
$TROVESTACK_SCRIPTS/image-projects-list;;
|
|
||||||
"reset-task" ) shift; cmd_reset_task $@;;
|
|
||||||
"wipe-queues" ) shift; cmd_wipe_queues $@;;
|
"wipe-queues" ) shift; cmd_wipe_queues $@;;
|
||||||
"repl" ) shift; cmd_repl $@;;
|
|
||||||
"help" ) print_usage;;
|
|
||||||
* )
|
* )
|
||||||
echo "'$1' not a valid command"
|
echo "'$1' not a valid command"
|
||||||
exit 1
|
exit 1
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
{
|
{
|
||||||
"report_directory":"rdli-test-report",
|
"report_directory":"rdli-test-report",
|
||||||
"start_services": false,
|
|
||||||
|
|
||||||
|
|
||||||
"white_box":false,
|
"white_box":false,
|
||||||
"test_mgmt":false,
|
"test_mgmt":false,
|
||||||
@ -18,7 +16,6 @@
|
|||||||
"nova_conf":"/home/vagrant/nova.conf",
|
"nova_conf":"/home/vagrant/nova.conf",
|
||||||
"keystone_code_root":"/opt/stack/keystone",
|
"keystone_code_root":"/opt/stack/keystone",
|
||||||
"keystone_conf":"/etc/keystone/keystone.conf",
|
"keystone_conf":"/etc/keystone/keystone.conf",
|
||||||
"keystone_use_combined":true,
|
|
||||||
"trove_code_root":"/opt/stack/trove",
|
"trove_code_root":"/opt/stack/trove",
|
||||||
"trove_conf":"/tmp/trove.conf",
|
"trove_conf":"/tmp/trove.conf",
|
||||||
"trove_version":"v1.0",
|
"trove_version":"v1.0",
|
||||||
@ -29,9 +26,8 @@
|
|||||||
"trove_max_accepted_volume_size": 1000,
|
"trove_max_accepted_volume_size": 1000,
|
||||||
"trove_max_instances_per_user": 55,
|
"trove_max_instances_per_user": 55,
|
||||||
"trove_max_volumes_per_user": 100,
|
"trove_max_volumes_per_user": 100,
|
||||||
"use_nova_volume": false,
|
|
||||||
"use_reaper":false,
|
"use_reaper":false,
|
||||||
"root_removed_from_instance_api": true,
|
"root_removed_from_instance_api": true,
|
||||||
"root_timestamp_disabled": false,
|
"root_timestamp_disabled": false,
|
||||||
"openvz_disabled": false,
|
"openvz_disabled": false,
|
||||||
"management_api_disabled": true,
|
"management_api_disabled": true,
|
||||||
@ -43,6 +39,6 @@
|
|||||||
"users_page_size": 20,
|
"users_page_size": 20,
|
||||||
"rabbit_runs_locally":false,
|
"rabbit_runs_locally":false,
|
||||||
|
|
||||||
"dns_instance_entry_factory":"trove.dns.rsdns.driver.RsDnsInstanceEntryFactory",
|
"dns_instance_entry_factory":"trove.dns.rsdns.driver.RsDnsInstanceEntryFactory",
|
||||||
"sentinel": null
|
"sentinel": null
|
||||||
}
|
}
|
||||||
|
@ -112,44 +112,33 @@ def _clean_up():
|
|||||||
|
|
||||||
|
|
||||||
def import_tests():
|
def import_tests():
|
||||||
|
|
||||||
# TODO(tim.simpson): Import these again once white box test functionality
|
|
||||||
# is restored.
|
|
||||||
# from tests.dns import check_domain
|
|
||||||
# from tests.dns import concurrency
|
|
||||||
# from tests.dns import conversion
|
|
||||||
|
|
||||||
# The DNS stuff is problematic. Not loading the other tests allow us to
|
# The DNS stuff is problematic. Not loading the other tests allow us to
|
||||||
# run its functional tests only.
|
# run its functional tests only.
|
||||||
ADD_DOMAINS = os.environ.get("ADD_DOMAINS", "False") == 'True'
|
ADD_DOMAINS = os.environ.get("ADD_DOMAINS", "False") == 'True'
|
||||||
if not ADD_DOMAINS:
|
if not ADD_DOMAINS:
|
||||||
from tests.api import delete_all
|
# F401 unused imports needed for tox tests
|
||||||
from tests.api import instances_pagination
|
from trove.tests.api import backups # noqa
|
||||||
from tests.api import instances_states
|
from trove.tests.api import configurations # noqa
|
||||||
from tests.dns import dns
|
from trove.tests.api import databases # noqa
|
||||||
from tests import initialize
|
from trove.tests.api import datastores # noqa
|
||||||
from tests.smoke import instance
|
from trove.tests.api import instances as rd_instances # noqa
|
||||||
from tests.volumes import driver
|
from trove.tests.api import instances_actions as acts # noqa
|
||||||
|
from trove.tests.api import instances_delete # noqa
|
||||||
|
from trove.tests.api import instances_resize # noqa
|
||||||
|
from trove.tests.api import limits # noqa
|
||||||
|
from trove.tests.api.mgmt import datastore_versions # noqa
|
||||||
|
from trove.tests.api.mgmt import instances_actions as mgmt_acts # noqa
|
||||||
|
from trove.tests.api import replication # noqa
|
||||||
|
from trove.tests.api import root # noqa
|
||||||
|
from trove.tests.api import user_access # noqa
|
||||||
|
from trove.tests.api import users # noqa
|
||||||
|
from trove.tests.api import versions # noqa
|
||||||
|
from trove.tests.db import migrations # noqa
|
||||||
|
|
||||||
# Groups that exist as core int-tests are registered from the
|
# Groups that exist as core int-tests are registered from the
|
||||||
# trove.tests.int_tests module
|
# trove.tests.int_tests module
|
||||||
from trove.tests import int_tests
|
from trove.tests import int_tests
|
||||||
|
|
||||||
# Groups defined in trove/integration, or any other externally
|
|
||||||
# defined groups can be registered here
|
|
||||||
heavy_black_box_groups = [
|
|
||||||
"dbaas.api.instances.pagination",
|
|
||||||
"dbaas.api.instances.delete",
|
|
||||||
"dbaas.api.instances.status",
|
|
||||||
"dbaas.api.instances.down",
|
|
||||||
"dbaas.api.mgmt.hosts.update",
|
|
||||||
"fake.dbaas.api.mgmt.instances",
|
|
||||||
"fake.dbaas.api.mgmt.accounts.broken",
|
|
||||||
"fake.dbaas.api.mgmt.allaccounts"
|
|
||||||
]
|
|
||||||
proboscis.register(groups=["heavy_blackbox"],
|
|
||||||
depends_on_groups=heavy_black_box_groups)
|
|
||||||
|
|
||||||
|
|
||||||
def run_main(test_importer):
|
def run_main(test_importer):
|
||||||
|
|
||||||
|
@ -1,57 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
# Specify the path to the Trove repo as argument one.
|
|
||||||
# This script will create a .pid file and report in the current directory.
|
|
||||||
|
|
||||||
set -e
|
|
||||||
if [ $# -lt 1 ]; then
|
|
||||||
echo "Please give the path to the Trove repo as argument one."
|
|
||||||
exit 5
|
|
||||||
else
|
|
||||||
TROVE_PATH=$1
|
|
||||||
fi
|
|
||||||
if [ $# -lt 2 ]; then
|
|
||||||
echo "Please give the path to the Trove Client as argument two."
|
|
||||||
exit 5
|
|
||||||
else
|
|
||||||
TROVECLIENT_PATH=$2
|
|
||||||
fi
|
|
||||||
shift;
|
|
||||||
shift;
|
|
||||||
|
|
||||||
|
|
||||||
PID_FILE="`pwd`.pid"
|
|
||||||
|
|
||||||
function start_server() {
|
|
||||||
pushd $TROVE_PATH
|
|
||||||
bin/start_server.sh --pid_file=$PID_FILE
|
|
||||||
popd
|
|
||||||
}
|
|
||||||
|
|
||||||
function stop_server() {
|
|
||||||
if [ -f $PID_FILE ];
|
|
||||||
then
|
|
||||||
pushd $TROVE_PATH
|
|
||||||
bin/stop_server.sh $PID_FILE
|
|
||||||
popd
|
|
||||||
else
|
|
||||||
echo "The pid file did not exist, so not stopping server."
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
function on_error() {
|
|
||||||
echo "Something went wrong!"
|
|
||||||
stop_server
|
|
||||||
}
|
|
||||||
|
|
||||||
trap on_error EXIT # Proceed to trap - END in event of failure.
|
|
||||||
|
|
||||||
TROVE_CLIENT_PATH=$TROVECLIENT_PATH tox -e py26
|
|
||||||
start_server
|
|
||||||
.tox/py26/bin/pip install -U $TROVECLIENT_PATH
|
|
||||||
PYTHONPATH=$PYTHONPATH:$TROVECLIENT_PATH .tox/py26/bin/python int_tests.py \
|
|
||||||
--conf=localhost.test.conf -- $@
|
|
||||||
stop_server
|
|
||||||
|
|
||||||
|
|
||||||
trap - EXIT
|
|
||||||
echo "Ran tests successfully. :)"
|
|
||||||
exit 0
|
|
@ -1,30 +0,0 @@
|
|||||||
# 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 setuptools import setup
|
|
||||||
|
|
||||||
|
|
||||||
def read(fname):
|
|
||||||
return open(os.path.join(os.path.dirname(__file__), fname)).read()
|
|
||||||
|
|
||||||
|
|
||||||
setup(
|
|
||||||
name="Trove Integration Tests",
|
|
||||||
version="0.0.9.9",
|
|
||||||
author='OpenStack',
|
|
||||||
description="Runs integration tests on Ridley.",
|
|
||||||
license='Apache',
|
|
||||||
py_modules=[],
|
|
||||||
packages=['tests'],
|
|
||||||
scripts=[]
|
|
||||||
)
|
|
@ -1 +0,0 @@
|
|||||||
Integration tests.
|
|
@ -1,32 +0,0 @@
|
|||||||
# Copyright 2011 OpenStack LLC.
|
|
||||||
# All Rights Reserved.
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
# not use this file except in compliance with the License. You may obtain
|
|
||||||
# a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
|
|
||||||
from proboscis import test
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
from trove.tests.util import create_dbaas_client
|
|
||||||
from trove.tests.util.users import Requirements
|
|
||||||
|
|
||||||
GROUP = "dbaas.api.instances.delete"
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[GROUP])
|
|
||||||
def delete_all():
|
|
||||||
"""Delete every single one."""
|
|
||||||
user = CONFIG.users.find_user(Requirements(is_admin=False))
|
|
||||||
dbaas = create_dbaas_client(user)
|
|
||||||
instances = dbaas.instances.list()
|
|
||||||
for instance in instances:
|
|
||||||
instance.delete()
|
|
@ -1,219 +0,0 @@
|
|||||||
|
|
||||||
from proboscis import after_class
|
|
||||||
from proboscis import before_class
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis.asserts import assert_equal
|
|
||||||
from proboscis.asserts import assert_is_not
|
|
||||||
from proboscis.asserts import assert_is_none
|
|
||||||
from proboscis.asserts import assert_true
|
|
||||||
|
|
||||||
|
|
||||||
from troveclient.compat import exceptions
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
from trove.tests.util import create_dbaas_client
|
|
||||||
from trove.tests.util.users import Requirements
|
|
||||||
|
|
||||||
|
|
||||||
class TestBase(object):
|
|
||||||
|
|
||||||
def set_up(self):
|
|
||||||
"""Create a ton of instances."""
|
|
||||||
reqs = Requirements(is_admin=False)
|
|
||||||
self.user = CONFIG.users.find_user(reqs)
|
|
||||||
self.dbaas = create_dbaas_client(self.user)
|
|
||||||
|
|
||||||
def delete_instances(self):
|
|
||||||
chunk = 0
|
|
||||||
while True:
|
|
||||||
chunk += 1
|
|
||||||
attempts = 0
|
|
||||||
instances = self.dbaas.instances.list()
|
|
||||||
if len(instances) == 0:
|
|
||||||
break
|
|
||||||
# Sit around and try to delete this chunk.
|
|
||||||
while True:
|
|
||||||
instance_results = []
|
|
||||||
attempts += 1
|
|
||||||
deleted_count = 0
|
|
||||||
for instance in instances:
|
|
||||||
try:
|
|
||||||
instance.delete()
|
|
||||||
result = "[w]"
|
|
||||||
except exceptions.UnprocessableEntity:
|
|
||||||
result = "[W]"
|
|
||||||
except exceptions.NotFound:
|
|
||||||
result = "[O]"
|
|
||||||
deleted_count += 1
|
|
||||||
except Exception:
|
|
||||||
result = "[X]"
|
|
||||||
instance_results.append(result)
|
|
||||||
print("Chunk %d, attempt %d : %s"
|
|
||||||
% (chunk, attempts, ",".join(instance_results)))
|
|
||||||
if deleted_count == len(instances):
|
|
||||||
break
|
|
||||||
|
|
||||||
def create_instances(self):
|
|
||||||
self.ids = []
|
|
||||||
for index in range(self.max):
|
|
||||||
name = "multi-%03d" % index
|
|
||||||
result = self.dbaas.instances.create(name, 1,
|
|
||||||
{'size': 1}, [], [])
|
|
||||||
self.ids.append(result.id)
|
|
||||||
# Sort the list of IDs in order, so we can confirm the lists pagination
|
|
||||||
# returns is also sorted correctly.
|
|
||||||
self.ids.sort()
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def assert_instances_sorted_by_ids(instances):
|
|
||||||
# Assert that the strings are always increasing.
|
|
||||||
last_id = ""
|
|
||||||
for instance in instances:
|
|
||||||
assert_true(last_id < instance.id)
|
|
||||||
|
|
||||||
def print_list(self, instances):
|
|
||||||
print("Length = %d" % len(instances))
|
|
||||||
print(",".join([instance.id for instance in instances]))
|
|
||||||
|
|
||||||
def test_pagination(self, requested_limit, requested_marker,
|
|
||||||
expected_length, expected_marker, expected_last_item):
|
|
||||||
instances = self.dbaas.instances.list(limit=requested_limit,
|
|
||||||
marker=requested_marker)
|
|
||||||
marker = instances.next
|
|
||||||
|
|
||||||
self.print_list(instances)
|
|
||||||
|
|
||||||
# Better get as many as we asked for.
|
|
||||||
assert_equal(len(instances), expected_length)
|
|
||||||
# The last one should be roughly this one in the list.
|
|
||||||
assert_equal(instances[-1].id, expected_last_item)
|
|
||||||
# Because limit < count, the marker must be something.
|
|
||||||
if expected_marker:
|
|
||||||
assert_is_not(marker, None)
|
|
||||||
assert_equal(marker, expected_marker)
|
|
||||||
else:
|
|
||||||
assert_is_none(marker)
|
|
||||||
self.assert_instances_sorted_by_ids(instances)
|
|
||||||
|
|
||||||
|
|
||||||
@test(runs_after_groups=["dbaas.guest.shutdown"],
|
|
||||||
groups=['dbaas.api.instances.pagination'])
|
|
||||||
class SimpleCreateAndDestroy(TestBase):
|
|
||||||
"""
|
|
||||||
It turns out a big part of guaranteeing pagination works is to make sure
|
|
||||||
we can create a big batch of instances and delete them without problems.
|
|
||||||
Even in fake mode though its worth it to check this is the case.
|
|
||||||
"""
|
|
||||||
|
|
||||||
max = 5
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def set_up(self):
|
|
||||||
"""Create a ton of instances."""
|
|
||||||
super(SimpleCreateAndDestroy, self).set_up()
|
|
||||||
self.delete_instances()
|
|
||||||
|
|
||||||
@test
|
|
||||||
def spin_up(self):
|
|
||||||
self.create_instances()
|
|
||||||
|
|
||||||
@after_class(always_run=True)
|
|
||||||
def tear_down(self):
|
|
||||||
self.delete_instances()
|
|
||||||
|
|
||||||
|
|
||||||
@test(runs_after_groups=["dbaas.guest.shutdown"],
|
|
||||||
groups=['dbaas.api.instances.pagination'])
|
|
||||||
class InstancePagination50(TestBase):
|
|
||||||
|
|
||||||
max = 50
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def set_up(self):
|
|
||||||
"""Create a ton of instances."""
|
|
||||||
super(InstancePagination50, self).set_up()
|
|
||||||
self.delete_instances()
|
|
||||||
self.create_instances()
|
|
||||||
|
|
||||||
@after_class(always_run=True)
|
|
||||||
def tear_down(self):
|
|
||||||
"""Tear down all instances."""
|
|
||||||
self.delete_instances()
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_short(self):
|
|
||||||
self.test_pagination(requested_limit=10, requested_marker=None,
|
|
||||||
expected_length=10, expected_marker=self.ids[9],
|
|
||||||
expected_last_item=self.ids[9])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_default(self):
|
|
||||||
self.test_pagination(requested_limit=None, requested_marker=None,
|
|
||||||
expected_length=20, expected_marker=self.ids[19],
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_full(self):
|
|
||||||
self.test_pagination(requested_limit=50, requested_marker=None,
|
|
||||||
expected_length=20, expected_marker=self.ids[19],
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
|
|
||||||
@test(runs_after_groups=["dbaas.guest.shutdown"],
|
|
||||||
groups=['dbaas.api.instances.pagination'])
|
|
||||||
class InstancePagination20(TestBase):
|
|
||||||
|
|
||||||
max = 20
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def set_up(self):
|
|
||||||
"""Create a ton of instances."""
|
|
||||||
super(InstancePagination20, self).set_up()
|
|
||||||
self.delete_instances()
|
|
||||||
self.create_instances()
|
|
||||||
|
|
||||||
@after_class(always_run=True)
|
|
||||||
def tear_down(self):
|
|
||||||
"""Tear down all instances."""
|
|
||||||
self.delete_instances()
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_short(self):
|
|
||||||
self.test_pagination(requested_limit=10, requested_marker=None,
|
|
||||||
expected_length=10, expected_marker=self.ids[9],
|
|
||||||
expected_last_item=self.ids[9])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_default(self):
|
|
||||||
self.test_pagination(requested_limit=None, requested_marker=None,
|
|
||||||
expected_length=20, expected_marker=None,
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_full(self):
|
|
||||||
self.test_pagination(requested_limit=20, requested_marker=None,
|
|
||||||
expected_length=20, expected_marker=None,
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_overkill(self):
|
|
||||||
self.test_pagination(requested_limit=30, requested_marker=None,
|
|
||||||
expected_length=20, expected_marker=None,
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_last_half(self):
|
|
||||||
self.test_pagination(requested_limit=10, requested_marker=self.ids[9],
|
|
||||||
expected_length=10, expected_marker=None,
|
|
||||||
expected_last_item=self.ids[19])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_third_quarter(self):
|
|
||||||
self.test_pagination(requested_limit=5, requested_marker=self.ids[9],
|
|
||||||
expected_length=5, expected_marker=self.ids[14],
|
|
||||||
expected_last_item=self.ids[14])
|
|
||||||
|
|
||||||
@test
|
|
||||||
def pagination_fourth_quarter(self):
|
|
||||||
self.test_pagination(requested_limit=20, requested_marker=self.ids[14],
|
|
||||||
expected_length=5, expected_marker=None,
|
|
||||||
expected_last_item=self.ids[19])
|
|
@ -1,76 +0,0 @@
|
|||||||
# Copyright 2012 OpenStack LLC.
|
|
||||||
# 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.
|
|
||||||
|
|
||||||
|
|
||||||
GROUP = "dbaas.api.instances.status"
|
|
||||||
|
|
||||||
from proboscis import before_class
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis.asserts import assert_equal
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
from trove.tests.util import create_dbaas_client
|
|
||||||
from trove.tests.util.users import Requirements
|
|
||||||
from trove.common.utils import poll_until
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[GROUP])
|
|
||||||
class InstanceStatusTests(object):
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def set_up(self):
|
|
||||||
reqs = Requirements(is_admin=False)
|
|
||||||
self.user = CONFIG.users.find_user(reqs)
|
|
||||||
self.dbaas = create_dbaas_client(self.user)
|
|
||||||
|
|
||||||
@test
|
|
||||||
def test_create_failure_on_volume_prov_failure(self):
|
|
||||||
# Fake nova will fail a volume of size 9.
|
|
||||||
response = self.dbaas.instances.create('volume_fail', 1,
|
|
||||||
{'size': 9}, [])
|
|
||||||
poll_until(lambda: self.dbaas.instances.get(response.id),
|
|
||||||
lambda instance: instance.status == 'ERROR',
|
|
||||||
time_out=10)
|
|
||||||
instance = self.dbaas.instances.get(response.id)
|
|
||||||
print("Status: %s" % instance.status)
|
|
||||||
assert_equal(instance.status, "ERROR",
|
|
||||||
"Instance did not drop to error after volume prov failure.")
|
|
||||||
|
|
||||||
@test
|
|
||||||
def test_create_failure_on_server_failure(self):
|
|
||||||
# Fake nova will fail a server ending with 'SERVER_ERROR'."
|
|
||||||
response = self.dbaas.instances.create('test_SERVER_ERROR', 1,
|
|
||||||
{'size': 1}, [])
|
|
||||||
poll_until(lambda: self.dbaas.instances.get(response.id),
|
|
||||||
lambda instance: instance.status == 'ERROR',
|
|
||||||
time_out=10)
|
|
||||||
instance = self.dbaas.instances.get(response.id)
|
|
||||||
print("Status: %s" % instance.status)
|
|
||||||
assert_equal(instance.status, "ERROR",
|
|
||||||
"Instance did not drop to error after server prov failure.")
|
|
||||||
|
|
||||||
###TODO(ed-): We don't at present have a way to test DNS in FAKE_MODE.
|
|
||||||
@test(enabled=False)
|
|
||||||
def test_create_failure_on_dns_failure(self):
|
|
||||||
#TODO(ed-): Throw DNS-specific monkeywrench into works
|
|
||||||
response = self.dbaas.instances.create('test_DNS_ERROR', 1,
|
|
||||||
{'size': 1}, [])
|
|
||||||
poll_until(lambda: self.dbaas.instances.get(response.id),
|
|
||||||
lambda instance: instance.status == 'ERROR',
|
|
||||||
time_out=10)
|
|
||||||
instance = self.dbaas.instances.get(response.id)
|
|
||||||
print("Status: %s" % instance.status)
|
|
||||||
assert_equal(instance.status, "ERROR",
|
|
||||||
"Instance did not drop to error after DNS prov failure.")
|
|
@ -1,171 +0,0 @@
|
|||||||
# Copyright 2011 OpenStack LLC.
|
|
||||||
# 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.
|
|
||||||
"""Checks that the domain specified in the flag file exists and is valid.
|
|
||||||
|
|
||||||
If you define the environment variable ADD_DOMAINS=True when running the tests,
|
|
||||||
they will create the domain if its not found (see below for details).
|
|
||||||
|
|
||||||
"""
|
|
||||||
import time
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis import before_class
|
|
||||||
from proboscis.asserts import assert_equal
|
|
||||||
from proboscis.asserts import assert_not_equal
|
|
||||||
from proboscis.decorators import time_out
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
|
|
||||||
WHITE_BOX = CONFIG.white_box
|
|
||||||
RUN_DNS = CONFIG.values.get("trove_dns_support", False)
|
|
||||||
|
|
||||||
if WHITE_BOX:
|
|
||||||
from nova import utils
|
|
||||||
from nova import flags
|
|
||||||
import rsdns
|
|
||||||
from trove.dns.rsdns.driver import create_client_with_flag_values
|
|
||||||
from trove.dns.driver import DnsEntry
|
|
||||||
from trove.dns.rsdns.driver import RsDnsInstanceEntryFactory
|
|
||||||
from trove.dns.rsdns.driver import RsDnsDriver
|
|
||||||
from trove.dns.rsdns.driver import RsDnsZone
|
|
||||||
from trove.utils import poll_until
|
|
||||||
FLAGS = flags.FLAGS
|
|
||||||
TEST_CONTENT = "126.1.1.1"
|
|
||||||
TEST_NAME = "hiwassup.%s" % FLAGS.dns_domain_name
|
|
||||||
DNS_DOMAIN_ID = None
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["rsdns.domains", "rsdns.show_entries"],
|
|
||||||
enabled=WHITE_BOX and RUN_DNS)
|
|
||||||
class ClientTests(object):
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def increase_logging(self):
|
|
||||||
import httplib2
|
|
||||||
httplib2.debuglevel = 1
|
|
||||||
|
|
||||||
@test
|
|
||||||
def can_auth(self):
|
|
||||||
self.client = create_client_with_flag_values()
|
|
||||||
self.client.authenticate()
|
|
||||||
|
|
||||||
@test(depends_on=[can_auth])
|
|
||||||
def list_domains(self):
|
|
||||||
domains = self.client.domains.list()
|
|
||||||
print(domains)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["rsdns.domains"], depends_on=[ClientTests],
|
|
||||||
enabled=WHITE_BOX and RUN_DNS)
|
|
||||||
class RsDnsDriverTests(object):
|
|
||||||
"""Tests the RS DNS Driver."""
|
|
||||||
|
|
||||||
def create_domain_if_needed(self):
|
|
||||||
"""Adds the domain specified in the flags."""
|
|
||||||
print("Creating domain %s" % self.driver.default_dns_zone.name)
|
|
||||||
future = self.driver.dns_client.domains.create(
|
|
||||||
self.driver.default_dns_zone.name)
|
|
||||||
while not future.ready:
|
|
||||||
time.sleep(2)
|
|
||||||
print("Got something: %s" % future.resource)
|
|
||||||
with open('/home/vagrant/dns_resource.txt', 'w') as f:
|
|
||||||
f.write('%r\n' % future.result[0].id)
|
|
||||||
global DNS_DOMAIN_ID
|
|
||||||
DNS_DOMAIN_ID = future.result[0].id
|
|
||||||
print("The domain should have been created with id=%s" % DNS_DOMAIN_ID)
|
|
||||||
|
|
||||||
@test
|
|
||||||
@time_out(2 * 60)
|
|
||||||
def ensure_domain_specified_in_flags_exists(self):
|
|
||||||
"""Make sure the domain in the FLAGS exists."""
|
|
||||||
self.driver = RsDnsDriver(raise_if_zone_missing=False)
|
|
||||||
assert_not_equal(None, self.driver.default_dns_zone)
|
|
||||||
|
|
||||||
def zone_found():
|
|
||||||
zones = self.driver.get_dns_zones()
|
|
||||||
print("Retrieving zones.")
|
|
||||||
for zone in zones:
|
|
||||||
print("zone %s" % zone)
|
|
||||||
if zone.name == self.driver.default_dns_zone.name:
|
|
||||||
self.driver.default_dns_zone.id = zone.id
|
|
||||||
global DNS_DOMAIN_ID
|
|
||||||
DNS_DOMAIN_ID = zone.id
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
if zone_found():
|
|
||||||
return
|
|
||||||
self.create_domain_if_needed()
|
|
||||||
for i in range(5):
|
|
||||||
if zone_found():
|
|
||||||
return
|
|
||||||
self.fail("""Could not find default dns zone.
|
|
||||||
This happens when they clear the staging DNS service of data.
|
|
||||||
To fix it, manually run the tests as follows:
|
|
||||||
$ ADD_DOMAINS=True python int_tests.py
|
|
||||||
and if all goes well the tests will create a new domain
|
|
||||||
record.""")
|
|
||||||
|
|
||||||
@test(depends_on=[ensure_domain_specified_in_flags_exists],
|
|
||||||
enabled=WHITE_BOX and FLAGS.dns_domain_name != "dbaas.rackspace.com")
|
|
||||||
def delete_all_entries(self):
|
|
||||||
"""Deletes all entries under the default domain."""
|
|
||||||
list = self.driver.get_entries()
|
|
||||||
for entry in list:
|
|
||||||
if entry.type == "A":
|
|
||||||
self.driver.delete_entry(name=entry.name, type=entry.type,
|
|
||||||
dns_zone=entry.dns_zone)
|
|
||||||
# It takes awhile for them to be deleted.
|
|
||||||
poll_until(lambda: self.driver.get_entries_by_name(TEST_NAME),
|
|
||||||
lambda list: len(list) == 0,
|
|
||||||
sleep_time=4, time_out=60)
|
|
||||||
|
|
||||||
@test(depends_on=[delete_all_entries])
|
|
||||||
def create_test_entry(self):
|
|
||||||
fullname = TEST_NAME
|
|
||||||
entry = DnsEntry(name=fullname, content=TEST_CONTENT, type="A",
|
|
||||||
ttl=3600)
|
|
||||||
self.driver.create_entry(entry)
|
|
||||||
list = None
|
|
||||||
for i in range(500):
|
|
||||||
list = self.driver.get_entries_by_name(name=fullname)
|
|
||||||
if len(list) > 0:
|
|
||||||
break
|
|
||||||
time.sleep(1)
|
|
||||||
print("This is the list: %r" % list)
|
|
||||||
assert_equal(1, len(list))
|
|
||||||
list2 = self.driver.get_entries_by_content(content=TEST_CONTENT)
|
|
||||||
assert_equal(1, len(list2))
|
|
||||||
|
|
||||||
@test(depends_on=[delete_all_entries])
|
|
||||||
def create_test_rsdns_entry(self):
|
|
||||||
"""Create an entry using the RsDnsInstanceEntryFactory."""
|
|
||||||
instance = {'uuid': '000136c0-effa-4711-a747-a5b9fbfcb3bd', 'id': '10'}
|
|
||||||
ip = "10.100.2.7"
|
|
||||||
factory = RsDnsInstanceEntryFactory(dns_domain_id=DNS_DOMAIN_ID)
|
|
||||||
entry = factory.create_entry(instance)
|
|
||||||
entry.content = ip
|
|
||||||
self.driver.create_entry(entry)
|
|
||||||
entries = self.driver.get_entries_by_name(name=entry.name)
|
|
||||||
assert_equal(1, len(entries))
|
|
||||||
assert_equal(ip, entries[0].content)
|
|
||||||
assert_equal(FLAGS.dns_ttl, entries[0].ttl)
|
|
||||||
|
|
||||||
@test(depends_on=[create_test_entry])
|
|
||||||
def delete_test_entry(self):
|
|
||||||
fullname = TEST_NAME
|
|
||||||
self.driver.delete_entry(fullname, "A")
|
|
||||||
# It takes awhile for them to be deleted.
|
|
||||||
poll_until(lambda: self.driver.get_entries_by_name(TEST_NAME),
|
|
||||||
lambda list: len(list) == 0,
|
|
||||||
sleep_time=2, time_out=60)
|
|
@ -1,111 +0,0 @@
|
|||||||
# Copyright 2011 OpenStack LLC.
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
This test recreates an issue we had with eventlet. In the logs, we'd see that
|
|
||||||
the JSON response was malformed; instead of JSON, it contained the following
|
|
||||||
string:
|
|
||||||
Second simultaneous read on fileno 5 detected. Unless you really know what
|
|
||||||
you're doing, make sure that only one greenthread can read any particular
|
|
||||||
socket. Consider using a pools.Pool. If you do know what you're doing and want
|
|
||||||
to disable this error, call
|
|
||||||
eventlet.debug.hub_multiple_reader_prevention(False)
|
|
||||||
|
|
||||||
It is perhaps the most helpful error message ever created.
|
|
||||||
|
|
||||||
The root issue was that a subclass of httplib2.Http was created at program
|
|
||||||
started and used in all threads.
|
|
||||||
|
|
||||||
Using the old (broken) RsDNS client code this test recreates the greatest error
|
|
||||||
message ever.
|
|
||||||
"""
|
|
||||||
|
|
||||||
try:
|
|
||||||
import eventlet
|
|
||||||
CAN_USE_EVENTLET = True
|
|
||||||
except ImportError:
|
|
||||||
CAN_USE_EVENTLET = False
|
|
||||||
import uuid
|
|
||||||
|
|
||||||
from proboscis import before_class
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis.asserts import assert_true
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
|
|
||||||
WHITE_BOX = CONFIG.white_box
|
|
||||||
RUN_DNS = CONFIG.values.get("trove_dns_support", False)
|
|
||||||
|
|
||||||
|
|
||||||
if CONFIG.white_box:
|
|
||||||
from trove.dns.rsdns.driver import RsDnsInstanceEntryFactory
|
|
||||||
from nova import flags
|
|
||||||
from nova import utils
|
|
||||||
FLAGS = flags.FLAGS
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["rsdns.eventlet"], enabled=CAN_USE_EVENTLET)
|
|
||||||
class RsdnsEventletTests(object):
|
|
||||||
"""Makes sure the RSDNS client can be used from multiple green threads."""
|
|
||||||
|
|
||||||
def assert_record_created(self, index):
|
|
||||||
msg = "Record %d wasn't created!" % index
|
|
||||||
assert_true(index in self.new_records, msg)
|
|
||||||
|
|
||||||
@before_class(enabled=WHITE_BOX and RUN_DNS)
|
|
||||||
def create_driver(self):
|
|
||||||
"""Creates the DNS Driver used in subsequent tests."""
|
|
||||||
self.driver = utils.import_object(FLAGS.dns_driver)
|
|
||||||
self.entry_factory = RsDnsInstanceEntryFactory()
|
|
||||||
self.test_uuid = uuid.uuid4().hex
|
|
||||||
self.new_records = {}
|
|
||||||
|
|
||||||
def make_record(self, index):
|
|
||||||
"""Creates a record with the form 'eventlet-%s-%d'."""
|
|
||||||
uuid = "eventlet-%s-%d" % (self.test_uuid, index)
|
|
||||||
instance = {'uuid': uuid}
|
|
||||||
entry = self.entry_factory.create_entry(instance)
|
|
||||||
entry.name = uuid + "." + self.entry_factory.default_dns_zone.name
|
|
||||||
entry.content = "123.123.123.123"
|
|
||||||
self.driver.create_entry(entry)
|
|
||||||
self.new_records[index] = True
|
|
||||||
|
|
||||||
@test(enabled=WHITE_BOX and RUN_DNS)
|
|
||||||
def use_dns_from_a_single_thread(self):
|
|
||||||
"""Add DNS records one at a time."""
|
|
||||||
self.new_records = {}
|
|
||||||
for index in range(-1, -5, -1):
|
|
||||||
self.make_record(index)
|
|
||||||
self.assert_record_created(index)
|
|
||||||
|
|
||||||
@test(enabled=WHITE_BOX and RUN_DNS)
|
|
||||||
def use_dns_from_multiple_greenthreads(self):
|
|
||||||
"""Add multiple DNS records at once."""
|
|
||||||
self.new_records = {}
|
|
||||||
|
|
||||||
def make_record(index):
|
|
||||||
def __cb():
|
|
||||||
self.make_record(index)
|
|
||||||
self.assert_record_created(index)
|
|
||||||
return index
|
|
||||||
return __cb
|
|
||||||
|
|
||||||
pile = eventlet.GreenPile()
|
|
||||||
indices = range(1, 4)
|
|
||||||
for index in indices:
|
|
||||||
pile.spawn(make_record(index))
|
|
||||||
|
|
||||||
list(pile) # Wait for them to finish
|
|
||||||
for index in indices:
|
|
||||||
self.assert_record_created(index)
|
|
@ -1,103 +0,0 @@
|
|||||||
# Copyright 2011 OpenStack LLC.
|
|
||||||
# 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.
|
|
||||||
"""Tests classes which convert RS style-entries to Nova DNS entries."""
|
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import unittest
|
|
||||||
from proboscis import test
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
|
|
||||||
|
|
||||||
if CONFIG.white_box:
|
|
||||||
from nova import flags
|
|
||||||
from rsdns.client.records import Record
|
|
||||||
from trove.dns.rsdns.driver import EntryToRecordConverter
|
|
||||||
from trove.dns.rsdns.driver import RsDnsInstanceEntryFactory
|
|
||||||
from trove.dns.rsdns.driver import RsDnsZone
|
|
||||||
FLAGS = flags.FLAGS
|
|
||||||
driver = None
|
|
||||||
DEFAULT_ZONE = RsDnsZone(1, "dbaas.rackspace.org")
|
|
||||||
TEST_CONTENT = "126.1.1.1"
|
|
||||||
TEST_NAME = "hiwassup.dbaas.rackspace.org"
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["unit", "rsdns.conversion"],
|
|
||||||
enabled=CONFIG.white_box)
|
|
||||||
class ConvertingNovaEntryNamesToRecordNames(unittest.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.converter = EntryToRecordConverter(DEFAULT_ZONE)
|
|
||||||
self.fake_zone = RsDnsZone(id=5, name="blah.org")
|
|
||||||
|
|
||||||
def test_normal_name(self):
|
|
||||||
long_name = self.converter.name_to_long_name("hi", self.fake_zone)
|
|
||||||
self.assertEqual("hi.blah.org", long_name)
|
|
||||||
|
|
||||||
def test_short_name(self):
|
|
||||||
long_name = self.converter.name_to_long_name("", self.fake_zone)
|
|
||||||
self.assertEqual("", long_name)
|
|
||||||
|
|
||||||
def test_long_name(self):
|
|
||||||
long_name = self.converter.name_to_long_name("blah.org.",
|
|
||||||
self.fake_zone)
|
|
||||||
self.assertEqual("blah.org..blah.org", long_name)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["unit", "rsdns.conversion"],
|
|
||||||
enabled=CONFIG.white_box)
|
|
||||||
class ConvertingRecordsToEntries(unittest.TestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.converter = EntryToRecordConverter(DEFAULT_ZONE)
|
|
||||||
self.fake_zone = RsDnsZone(id=5, name="blah.org")
|
|
||||||
|
|
||||||
def test_normal_name(self):
|
|
||||||
record = Record(None, {"id": 5, "name": "hi.blah.org",
|
|
||||||
"data": "stacker.com blah@blah 13452378",
|
|
||||||
"ttl": 5,
|
|
||||||
"type": "SOA"})
|
|
||||||
entry = self.converter.record_to_entry(record=record,
|
|
||||||
dns_zone=self.fake_zone)
|
|
||||||
self.assertEqual("stacker.com blah@blah 13452378", entry.content)
|
|
||||||
self.assertEqual("hi.blah.org", entry.name)
|
|
||||||
self.assertEqual("5", str(entry.ttl))
|
|
||||||
self.assertEqual("SOA", entry.type)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["rsdns.conversion"],
|
|
||||||
enabled=CONFIG.white_box)
|
|
||||||
class WhenCreatingAnEntryForAnInstance(unittest.TestCase):
|
|
||||||
# This isn't a unit test because RsDnsInstanceEntryFactory connects to the
|
|
||||||
# service.
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.creator = RsDnsInstanceEntryFactory()
|
|
||||||
|
|
||||||
def test_should_concatanate_strings(self):
|
|
||||||
instance = {'id': '56',
|
|
||||||
'uuid': '000136c0-effa-4711-a747-a5b9fbfcb3bd'}
|
|
||||||
entry = self.creator.create_entry(instance)
|
|
||||||
expected_name = "%s.%s" % (hashlib.sha1(instance['uuid']).hexdigest(),
|
|
||||||
FLAGS.dns_domain_name)
|
|
||||||
self.assertEqual(expected_name, entry.name,
|
|
||||||
msg="Entry name should match - %s" % entry.name)
|
|
||||||
self.assertIsNone(entry.content)
|
|
||||||
self.assertEqual("A", entry.type)
|
|
||||||
self.assertEqual(FLAGS.dns_ttl, entry.ttl)
|
|
||||||
self.assertIsNone(entry.priority)
|
|
||||||
self.assertEqual(FLAGS.dns_domain_name, entry.dns_zone.name)
|
|
||||||
if not entry.dns_zone.id:
|
|
||||||
self.fail(msg="DNS Zone Id should not be empty")
|
|
@ -1,104 +0,0 @@
|
|||||||
|
|
||||||
import unittest
|
|
||||||
|
|
||||||
from proboscis import test
|
|
||||||
|
|
||||||
from trove.tests.api.instances import instance_info
|
|
||||||
from trove.tests.api.instances import GROUP_START as INSTANCE_START
|
|
||||||
from trove.tests.api.instances import GROUP_TEST
|
|
||||||
from trove.tests.api.instances import GROUP_STOP as INSTANCE_STOP
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
from trove.common.utils import import_object
|
|
||||||
from trove.common.utils import poll_until
|
|
||||||
|
|
||||||
WHITE_BOX = CONFIG.white_box
|
|
||||||
|
|
||||||
if WHITE_BOX:
|
|
||||||
# TODO(tim.simpson): Restore this once white box functionality can be
|
|
||||||
# added back to this test module.
|
|
||||||
pass
|
|
||||||
# import rsdns
|
|
||||||
# from nova import flags
|
|
||||||
# from nova import utils
|
|
||||||
|
|
||||||
# from trove import exception
|
|
||||||
# from trove.utils import poll_until
|
|
||||||
|
|
||||||
# FLAGS = flags.FLAGS
|
|
||||||
|
|
||||||
dns_driver = None
|
|
||||||
|
|
||||||
GROUP = "dbaas.guest.dns"
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[GROUP, GROUP_TEST])
|
|
||||||
class Setup(unittest.TestCase):
|
|
||||||
"""Creates the DNS Driver and entry factory used in subsequent tests."""
|
|
||||||
|
|
||||||
def test_create_rs_dns_driver(self):
|
|
||||||
global dns_driver
|
|
||||||
dns_driver = import_object(FLAGS.dns_driver)
|
|
||||||
|
|
||||||
|
|
||||||
def expected_dns_entry():
|
|
||||||
"""Returns expected DNS entry for this instance.
|
|
||||||
|
|
||||||
:rtype: Instance of :class:`DnsEntry`.
|
|
||||||
|
|
||||||
"""
|
|
||||||
return create_dns_entry(instance_info.local_id, instance_info.id)
|
|
||||||
|
|
||||||
|
|
||||||
@test(depends_on_classes=[Setup],
|
|
||||||
depends_on_groups=[INSTANCE_START],
|
|
||||||
groups=[GROUP, GROUP_TEST])
|
|
||||||
class WhenInstanceIsCreated(unittest.TestCase):
|
|
||||||
"""Make sure the DNS name was provisioned.
|
|
||||||
|
|
||||||
This class actually calls the DNS driver to confirm the entry that should
|
|
||||||
exist for the given instance does exist.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
def test_dns_entry_should_exist(self):
|
|
||||||
entry = expected_dns_entry()
|
|
||||||
if entry:
|
|
||||||
def get_entries():
|
|
||||||
return dns_driver.get_entries_by_name(entry.name)
|
|
||||||
try:
|
|
||||||
poll_until(get_entries, lambda entries: len(entries) > 0,
|
|
||||||
sleep_time=2, time_out=60)
|
|
||||||
except exception.PollTimeOut:
|
|
||||||
self.fail("Did not find name " + entry.name + \
|
|
||||||
" in the entries, which were as follows:"
|
|
||||||
+ str(dns_driver.get_entries()))
|
|
||||||
|
|
||||||
|
|
||||||
@test(depends_on_classes=[Setup, WhenInstanceIsCreated],
|
|
||||||
depends_on_groups=[INSTANCE_STOP],
|
|
||||||
groups=[GROUP])
|
|
||||||
class AfterInstanceIsDestroyed(unittest.TestCase):
|
|
||||||
"""Make sure the DNS name is removed along with an instance.
|
|
||||||
|
|
||||||
Because the compute manager calls the DNS manager with RPC cast, it can
|
|
||||||
take awhile. So we wait for 30 seconds for it to disappear.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
def test_dns_entry_exist_should_be_removed_shortly_thereafter(self):
|
|
||||||
entry = expected_dns_entry()
|
|
||||||
|
|
||||||
if not entry:
|
|
||||||
return
|
|
||||||
|
|
||||||
def get_entries():
|
|
||||||
return dns_driver.get_entries_by_name(entry.name)
|
|
||||||
|
|
||||||
try:
|
|
||||||
poll_until(get_entries, lambda entries: len(entries) == 0,
|
|
||||||
sleep_time=2, time_out=60)
|
|
||||||
except exception.PollTimeOut:
|
|
||||||
# Manually delete the rogue item
|
|
||||||
dns_driver.delete_entry(entry.name, entry.type, entry.dns_zone)
|
|
||||||
self.fail("The DNS entry was never deleted when the instance "
|
|
||||||
"was destroyed.")
|
|
@ -21,27 +21,14 @@ from tests.util.services import Service
|
|||||||
from trove.tests.config import CONFIG
|
from trove.tests.config import CONFIG
|
||||||
|
|
||||||
|
|
||||||
FAKE = CONFIG.fake_mode
|
|
||||||
START_SERVICES = (not FAKE) and CONFIG.values.get('start_services', False)
|
|
||||||
START_NOVA_NETWORK = (START_SERVICES and
|
|
||||||
not CONFIG.values.get('neutron_enabled',
|
|
||||||
False))
|
|
||||||
KEYSTONE_ALL = CONFIG.values.get('keystone_use_combined', True)
|
|
||||||
USE_NOVA_VOLUME = CONFIG.values.get('use_nova_volume', False)
|
|
||||||
|
|
||||||
dbaas_image = None
|
|
||||||
instance_name = None
|
|
||||||
success_statuses = ["build", "active"]
|
|
||||||
|
|
||||||
|
|
||||||
def dbaas_url():
|
def dbaas_url():
|
||||||
return str(CONFIG.values.get("dbaas_url"))
|
return str(CONFIG.values.get("dbaas_url"))
|
||||||
|
|
||||||
|
|
||||||
def nova_url():
|
def nova_url():
|
||||||
return str(CONFIG.values.get("nova_client")['url'])
|
return str(CONFIG.values.get("nova_client")['url'])
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class Daemon(object):
|
class Daemon(object):
|
||||||
"""Starts a daemon."""
|
"""Starts a daemon."""
|
||||||
|
|
||||||
@ -74,95 +61,3 @@ class Daemon(object):
|
|||||||
self.service = Service(cmds)
|
self.service = Service(cmds)
|
||||||
if not self.service.is_service_alive():
|
if not self.service.is_service_alive():
|
||||||
self.service.start()
|
self.service.start()
|
||||||
|
|
||||||
@test(groups=["services.initialize"],
|
|
||||||
enabled=START_SERVICES and (not KEYSTONE_ALL))
|
|
||||||
def start_keystone_all():
|
|
||||||
"""Starts the Keystone API."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/keystone-all",
|
|
||||||
extra_cmds=['--config-file'],
|
|
||||||
conf_file_name="keystone_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize", "services.initialize.glance"],
|
|
||||||
enabled=START_SERVICES)
|
|
||||||
def start_glance_registry():
|
|
||||||
"""Starts the Glance Registry."""
|
|
||||||
Daemon(alternate_path="/usr/bin/glance-registry",
|
|
||||||
conf_file_name="glance_reg_conf",
|
|
||||||
service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/glance-registry").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize", "services.initialize.glance"],
|
|
||||||
depends_on=[start_glance_registry], enabled=START_SERVICES)
|
|
||||||
def start_glance_api():
|
|
||||||
"""Starts the Glance API."""
|
|
||||||
Daemon(alternate_path="/usr/bin/glance-api",
|
|
||||||
conf_file_name="glance_reg_conf",
|
|
||||||
service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/glance-api").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"], depends_on_classes=[start_glance_api],
|
|
||||||
enabled=START_NOVA_NETWORK)
|
|
||||||
def start_nova_network():
|
|
||||||
"""Starts the Nova Network Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/nova-network",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="nova_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"], enabled=START_SERVICES)
|
|
||||||
def start_scheduler():
|
|
||||||
"""Starts the Scheduler Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/nova-scheduler",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="nova_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"],
|
|
||||||
depends_on_classes=[start_glance_api],
|
|
||||||
enabled=START_SERVICES)
|
|
||||||
def start_compute():
|
|
||||||
"""Starts the Nova Compute Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/nova-compute",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="nova_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"], depends_on_classes=[start_scheduler],
|
|
||||||
enabled=START_SERVICES and USE_NOVA_VOLUME)
|
|
||||||
def start_volume():
|
|
||||||
"""Starts the Nova Compute Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/nova-volume",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="nova_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"],
|
|
||||||
depends_on_classes=[start_glance_api, start_nova_network, start_compute,
|
|
||||||
start_volume],
|
|
||||||
enabled=START_SERVICES)
|
|
||||||
def start_nova_api():
|
|
||||||
"""Starts the Nova Compute Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/nova-api",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="nova_conf").run()
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=["services.initialize"],
|
|
||||||
depends_on_classes=[start_nova_api],
|
|
||||||
enabled=START_SERVICES)
|
|
||||||
def start_trove_api():
|
|
||||||
"""Starts the Trove Service."""
|
|
||||||
Daemon(service_path_root="usr_bin_dir",
|
|
||||||
service_path="%s/trove-api",
|
|
||||||
extra_cmds=['--config-file='],
|
|
||||||
conf_file_name="trove_conf").run()
|
|
||||||
|
@ -1,103 +0,0 @@
|
|||||||
|
|
||||||
from proboscis.asserts import assert_equal
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis import before_class
|
|
||||||
|
|
||||||
from trove.common.utils import poll_until
|
|
||||||
from trove.tests.util import create_client
|
|
||||||
|
|
||||||
|
|
||||||
class InstanceGenerator(object):
|
|
||||||
|
|
||||||
def __init__(self, client, status=None, name=None, flavor=None,
|
|
||||||
account_id=None, created_at=None, databases=None, users=None,
|
|
||||||
volume_size=None):
|
|
||||||
self.client = client
|
|
||||||
self.status = status
|
|
||||||
self.name = name
|
|
||||||
self.flavor = flavor
|
|
||||||
self.account_id = account_id
|
|
||||||
self.databases = databases
|
|
||||||
self.users = users
|
|
||||||
self.volume_size = volume_size
|
|
||||||
self.id = None
|
|
||||||
|
|
||||||
def create_instance(self):
|
|
||||||
#make the call to create the instance
|
|
||||||
instance = self.client.instances.create(self.name, self.flavor,
|
|
||||||
self.volume_size, self.databases, self.users)
|
|
||||||
self.client.assert_http_code(200)
|
|
||||||
|
|
||||||
#verify we are in a build state
|
|
||||||
assert_equal(instance.status, "BUILD")
|
|
||||||
#pull out the ID
|
|
||||||
self.id = instance.id
|
|
||||||
|
|
||||||
return instance
|
|
||||||
|
|
||||||
def wait_for_build_to_finish(self):
|
|
||||||
poll_until(lambda: self.client.instance.get(self.id),
|
|
||||||
lambda instance: instance.status != "BUILD",
|
|
||||||
time_out=600)
|
|
||||||
|
|
||||||
def get_active_instance(self):
|
|
||||||
instance = self.client.instance.get(self.id)
|
|
||||||
self.client.assert_http_code(200)
|
|
||||||
|
|
||||||
#check the container name
|
|
||||||
assert_equal(instance.name, self.name)
|
|
||||||
|
|
||||||
#pull out volume info and verify
|
|
||||||
assert_equal(str(instance.volume_size), str(self.volume_size))
|
|
||||||
|
|
||||||
#pull out the flavor and verify
|
|
||||||
assert_equal(str(instance.flavor), str(self.flavor))
|
|
||||||
|
|
||||||
return instance
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=['smoke', 'positive'])
|
|
||||||
class CreateInstance(object):
|
|
||||||
|
|
||||||
@before_class
|
|
||||||
def set_up(self):
|
|
||||||
client = create_client(is_admin=False)
|
|
||||||
name = 'test_createInstance_container'
|
|
||||||
flavor = 1
|
|
||||||
volume_size = 1
|
|
||||||
db_name = 'test_db'
|
|
||||||
databases = [
|
|
||||||
{
|
|
||||||
"name": db_name
|
|
||||||
}
|
|
||||||
]
|
|
||||||
users = [
|
|
||||||
{
|
|
||||||
"name": "lite",
|
|
||||||
"password": "litepass",
|
|
||||||
"databases": [{"name": db_name}]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
#create the Instance
|
|
||||||
instance = InstanceGenerator(client, name=self.name,
|
|
||||||
flavor=flavor,
|
|
||||||
volume_size=self.volume_size,
|
|
||||||
databases=databases, users=users)
|
|
||||||
instance.create_instance()
|
|
||||||
|
|
||||||
#wait for the instance
|
|
||||||
instance.wait_for_build_to_finish()
|
|
||||||
|
|
||||||
#get the active instance
|
|
||||||
inst = instance.get_active_instance()
|
|
||||||
|
|
||||||
#list out the databases for our instance and verify the db name
|
|
||||||
dbs = client.databases.list(inst.id)
|
|
||||||
client.assert_http_code(200)
|
|
||||||
|
|
||||||
assert_equal(len(dbs), 1)
|
|
||||||
assert_equal(dbs[0].name, instance.db_name)
|
|
||||||
|
|
||||||
client.instance.delete(inst.id)
|
|
||||||
client.assert_http_code(202)
|
|
@ -1,25 +0,0 @@
|
|||||||
# Copyright (c) 2011 OpenStack, LLC.
|
|
||||||
# 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.
|
|
||||||
|
|
||||||
"""
|
|
||||||
:mod:`volumes` -- Tests for volumes.
|
|
||||||
===================================
|
|
||||||
"""
|
|
||||||
|
|
||||||
""""Tests for Volumes."""
|
|
||||||
|
|
||||||
# Is a set of tests written directly against the VolumeManager and VolumeClient
|
|
||||||
# classes which doesn't require standing up Nova daemons or anything.
|
|
||||||
VOLUMES_DRIVER = "trove.volumes.driver"
|
|
@ -1,547 +0,0 @@
|
|||||||
# Copyright (c) 2012 OpenStack, LLC.
|
|
||||||
# All Rights Reserved.
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
# not use this file except in compliance with the License. You may obtain
|
|
||||||
# a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
|
|
||||||
from numbers import Number
|
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import shutil
|
|
||||||
import six
|
|
||||||
import socket
|
|
||||||
import time
|
|
||||||
import unittest
|
|
||||||
|
|
||||||
import pexpect
|
|
||||||
|
|
||||||
from proboscis import test
|
|
||||||
from proboscis.asserts import assert_raises
|
|
||||||
from proboscis.decorators import expect_exception
|
|
||||||
from proboscis.decorators import time_out
|
|
||||||
|
|
||||||
from trove.tests.config import CONFIG
|
|
||||||
from trove.common.utils import poll_until
|
|
||||||
from trove.tests.util import process
|
|
||||||
from trove.common.utils import import_class
|
|
||||||
from tests import initialize
|
|
||||||
|
|
||||||
|
|
||||||
WHITE_BOX = CONFIG.white_box
|
|
||||||
VOLUMES_DRIVER = "trove.volumes.driver"
|
|
||||||
|
|
||||||
if WHITE_BOX:
|
|
||||||
# TODO(tim.simpson): Restore this once white box functionality can be
|
|
||||||
# added back to this test module.
|
|
||||||
pass
|
|
||||||
# from nova import context
|
|
||||||
# from nova import exception
|
|
||||||
# from nova import flags
|
|
||||||
# from nova import utils
|
|
||||||
# from trove import exception as trove_exception
|
|
||||||
# from trove.utils import poll_until
|
|
||||||
# from trove import volume
|
|
||||||
# from trove.tests.volume import driver as test_driver
|
|
||||||
|
|
||||||
# FLAGS = flags.FLAGS
|
|
||||||
|
|
||||||
|
|
||||||
UUID_PATTERN = re.compile('^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-'
|
|
||||||
'[0-9a-f]{4}-[0-9a-f]{12}$')
|
|
||||||
|
|
||||||
HUGE_VOLUME = 5000
|
|
||||||
|
|
||||||
|
|
||||||
def is_uuid(text):
|
|
||||||
return UUID_PATTERN.search(text) is not None
|
|
||||||
|
|
||||||
|
|
||||||
class StoryDetails(object):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.api = volume.API()
|
|
||||||
self.client = volume.Client()
|
|
||||||
self.context = context.get_admin_context()
|
|
||||||
self.device_path = None
|
|
||||||
self.volume_desc = None
|
|
||||||
self.volume_id = None
|
|
||||||
self.volume_name = None
|
|
||||||
self.volume = None
|
|
||||||
self.host = socket.gethostname()
|
|
||||||
self.original_uuid = None
|
|
||||||
self.original_device_info = None
|
|
||||||
self.resize_volume_size = 2
|
|
||||||
|
|
||||||
def get_volume(self):
|
|
||||||
return self.api.get(self.context, self.volume_id)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def mount_point(self):
|
|
||||||
return "%s/%s" % (LOCAL_MOUNT_PATH, self.volume_id)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def test_mount_file_path(self):
|
|
||||||
return "%s/test.txt" % self.mount_point
|
|
||||||
|
|
||||||
|
|
||||||
story = None
|
|
||||||
storyFail = None
|
|
||||||
|
|
||||||
LOCAL_MOUNT_PATH = "/testsmnt"
|
|
||||||
|
|
||||||
|
|
||||||
class VolumeTest(unittest.TestCase):
|
|
||||||
"""This test tells the story of a volume, from cradle to grave."""
|
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
unittest.TestCase.__init__(self, *args, **kwargs)
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
global story, storyFail
|
|
||||||
self.story = story
|
|
||||||
self.storyFail = storyFail
|
|
||||||
|
|
||||||
def assert_volume_as_expected(self, volume):
|
|
||||||
self.assertIsInstance(volume["id"], Number)
|
|
||||||
self.assertEqual(self.story.volume_name, volume["display_name"])
|
|
||||||
self.assertEqual(self.story.volume_desc, volume["display_description"])
|
|
||||||
self.assertEqual(1, volume["size"])
|
|
||||||
self.assertEqual(self.story.context.user_id, volume["user_id"])
|
|
||||||
self.assertEqual(self.story.context.project_id, volume["project_id"])
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[initialize.start_volume])
|
|
||||||
class SetUp(VolumeTest):
|
|
||||||
|
|
||||||
def test_05_create_story(self):
|
|
||||||
"""Creating 'story' vars used by the rest of these tests."""
|
|
||||||
global story, storyFail
|
|
||||||
story = StoryDetails()
|
|
||||||
storyFail = StoryDetails()
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_10_wait_for_topics(self):
|
|
||||||
"""Wait until the volume topic is up before proceeding."""
|
|
||||||
topics = ["volume"]
|
|
||||||
from tests.util.topics import hosts_up
|
|
||||||
while not all(hosts_up(topic) for topic in topics):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def test_20_refresh_local_folders(self):
|
|
||||||
"""Delete the local folders used as mount locations if they exist."""
|
|
||||||
if os.path.exists(LOCAL_MOUNT_PATH):
|
|
||||||
#TODO(rnirmal): Also need to remove any existing mounts.
|
|
||||||
shutil.rmtree(LOCAL_MOUNT_PATH)
|
|
||||||
os.mkdir(LOCAL_MOUNT_PATH)
|
|
||||||
# Give some time for the services to startup
|
|
||||||
time.sleep(10)
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_30_mgmt_volume_check(self):
|
|
||||||
"""Get the volume information from the mgmt API"""
|
|
||||||
story_context = self.story.context
|
|
||||||
device_info = self.story.api.get_storage_device_info(story_context)
|
|
||||||
print("device_info : %r" % device_info)
|
|
||||||
self.assertNotEqual(device_info, None,
|
|
||||||
"the storage device information should exist")
|
|
||||||
self.story.original_device_info = device_info
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_31_mgmt_volume_info(self):
|
|
||||||
"""Check the available space against the mgmt API info."""
|
|
||||||
story_context = self.story.context
|
|
||||||
device_info = self.story.api.get_storage_device_info(story_context)
|
|
||||||
print("device_info : %r" % device_info)
|
|
||||||
info = {'spaceTotal': device_info['raw_total'],
|
|
||||||
'spaceAvail': device_info['raw_avail']}
|
|
||||||
self._assert_available_space(info)
|
|
||||||
|
|
||||||
def _assert_available_space(self, device_info, fail=False):
|
|
||||||
"""
|
|
||||||
Give the SAN device_info(fake or not) and get the asserts for free
|
|
||||||
"""
|
|
||||||
print("DEVICE_INFO on SAN : %r" % device_info)
|
|
||||||
# Calculate the GBs; Divide by 2 for the FLAGS.san_network_raid_factor
|
|
||||||
gbs = 1.0 / 1024 / 1024 / 1024 / 2
|
|
||||||
total = int(device_info['spaceTotal']) * gbs
|
|
||||||
free = int(device_info['spaceAvail']) * gbs
|
|
||||||
used = total - free
|
|
||||||
usable = total * (FLAGS.san_max_provision_percent * 0.01)
|
|
||||||
real_free = float(int(usable - used))
|
|
||||||
|
|
||||||
print("total : %r" % total)
|
|
||||||
print("free : %r" % free)
|
|
||||||
print("used : %r" % used)
|
|
||||||
print("usable : %r" % usable)
|
|
||||||
print("real_free : %r" % real_free)
|
|
||||||
|
|
||||||
check_space = self.story.api.check_for_available_space
|
|
||||||
self.assertFalse(check_space(self.story.context, HUGE_VOLUME))
|
|
||||||
self.assertFalse(check_space(self.story.context, real_free + 1))
|
|
||||||
|
|
||||||
if fail:
|
|
||||||
self.assertFalse(check_space(self.story.context, real_free))
|
|
||||||
self.assertFalse(check_space(self.story.context, real_free - 1))
|
|
||||||
self.assertFalse(check_space(self.story.context, 1))
|
|
||||||
else:
|
|
||||||
self.assertTrue(check_space(self.story.context, real_free))
|
|
||||||
self.assertTrue(check_space(self.story.context, real_free - 1))
|
|
||||||
self.assertTrue(check_space(self.story.context, 1))
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[SetUp])
|
|
||||||
class AddVolumeFailure(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_add(self):
|
|
||||||
"""
|
|
||||||
Make call to FAIL a prov. volume and assert the return value is a
|
|
||||||
FAILURE.
|
|
||||||
"""
|
|
||||||
self.assertIsNone(self.storyFail.volume_id)
|
|
||||||
name = "TestVolume"
|
|
||||||
desc = "A volume that was created for testing."
|
|
||||||
self.storyFail.volume_name = name
|
|
||||||
self.storyFail.volume_desc = desc
|
|
||||||
volume = self.storyFail.api.create(self.storyFail.context,
|
|
||||||
size=HUGE_VOLUME,
|
|
||||||
snapshot_id=None, name=name,
|
|
||||||
description=desc)
|
|
||||||
self.assertEqual(HUGE_VOLUME, volume["size"])
|
|
||||||
self.assertEqual("creating", volume["status"])
|
|
||||||
self.assertEqual("detached", volume["attach_status"])
|
|
||||||
self.storyFail.volume = volume
|
|
||||||
self.storyFail.volume_id = volume["id"]
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[AddVolumeFailure])
|
|
||||||
class AfterVolumeFailureIsAdded(VolumeTest):
|
|
||||||
"""Check that the volume can be retrieved via the API, and setup.
|
|
||||||
|
|
||||||
All we want to see returned is a list-like with an initial string.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
@time_out(120)
|
|
||||||
def test_api_get(self):
|
|
||||||
"""Wait until the volume is a FAILURE."""
|
|
||||||
volume = poll_until(lambda: self.storyFail.get_volume(),
|
|
||||||
lambda volume: volume["status"] != "creating")
|
|
||||||
self.assertEqual("error", volume["status"])
|
|
||||||
self.assertEqual("detached", volume["attach_status"])
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_mgmt_volume_check(self):
|
|
||||||
"""Get the volume information from the mgmt API"""
|
|
||||||
info = self.story.api.get_storage_device_info(self.story.context)
|
|
||||||
print("device_info : %r" % info)
|
|
||||||
self.assertNotEqual(info, None,
|
|
||||||
"the storage device information should exist")
|
|
||||||
self.assertEqual(self.story.original_device_info['raw_total'],
|
|
||||||
info['raw_total'])
|
|
||||||
self.assertEqual(self.story.original_device_info['raw_avail'],
|
|
||||||
info['raw_avail'])
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[SetUp])
|
|
||||||
class AddVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_add(self):
|
|
||||||
"""Make call to prov. a volume and assert the return value is OK."""
|
|
||||||
self.assertIsNone(self.story.volume_id)
|
|
||||||
name = "TestVolume"
|
|
||||||
desc = "A volume that was created for testing."
|
|
||||||
self.story.volume_name = name
|
|
||||||
self.story.volume_desc = desc
|
|
||||||
volume = self.story.api.create(self.story.context, size=1,
|
|
||||||
snapshot_id=None, name=name,
|
|
||||||
description=desc)
|
|
||||||
self.assert_volume_as_expected(volume)
|
|
||||||
self.assertEqual("creating", volume["status"])
|
|
||||||
self.assertEqual("detached", volume["attach_status"])
|
|
||||||
self.story.volume = volume
|
|
||||||
self.story.volume_id = volume["id"]
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[AddVolume])
|
|
||||||
class AfterVolumeIsAdded(VolumeTest):
|
|
||||||
"""Check that the volume can be retrieved via the API, and setup.
|
|
||||||
|
|
||||||
All we want to see returned is a list-like with an initial string.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
@time_out(120)
|
|
||||||
def test_api_get(self):
|
|
||||||
"""Wait until the volume is finished provisioning."""
|
|
||||||
volume = poll_until(lambda: self.story.get_volume(),
|
|
||||||
lambda volume: volume["status"] != "creating")
|
|
||||||
self.assertEqual("available", volume["status"])
|
|
||||||
self.assert_volume_as_expected(volume)
|
|
||||||
self.assertEqual("detached", volume["attach_status"])
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_mgmt_volume_check(self):
|
|
||||||
"""Get the volume information from the mgmt API"""
|
|
||||||
print("self.story.original_device_info : %r" %
|
|
||||||
self.story.original_device_info)
|
|
||||||
info = self.story.api.get_storage_device_info(self.story.context)
|
|
||||||
print("device_info : %r" % info)
|
|
||||||
self.assertNotEqual(info, None,
|
|
||||||
"the storage device information should exist")
|
|
||||||
self.assertEqual(self.story.original_device_info['raw_total'],
|
|
||||||
info['raw_total'])
|
|
||||||
volume_size = int(self.story.volume['size']) * (1024 ** 3) * 2
|
|
||||||
print("volume_size: %r" % volume_size)
|
|
||||||
print("self.story.volume['size']: %r" % self.story.volume['size'])
|
|
||||||
avail = int(self.story.original_device_info['raw_avail']) - volume_size
|
|
||||||
print("avail space: %r" % avail)
|
|
||||||
self.assertEqual(int(info['raw_avail']), avail)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[AfterVolumeIsAdded])
|
|
||||||
class SetupVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_assign_volume(self):
|
|
||||||
"""Tell the volume it belongs to this host node."""
|
|
||||||
#TODO(tim.simpson) If this is important, could we add a test to
|
|
||||||
# make sure some kind of exception is thrown if it
|
|
||||||
# isn't added to certain drivers?
|
|
||||||
self.assertNotEqual(None, self.story.volume_id)
|
|
||||||
self.story.api.assign_to_compute(self.story.context,
|
|
||||||
self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_setup_volume(self):
|
|
||||||
"""Set up the volume on this host. AKA discovery."""
|
|
||||||
self.assertNotEqual(None, self.story.volume_id)
|
|
||||||
device = self.story.client._setup_volume(self.story.context,
|
|
||||||
self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
if not isinstance(device, six.string_types):
|
|
||||||
self.fail("Expected device to be a string, but instead it was " +
|
|
||||||
str(type(device)) + ".")
|
|
||||||
self.story.device_path = device
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[SetupVolume])
|
|
||||||
class FormatVolume(VolumeTest):
|
|
||||||
|
|
||||||
@expect_exception(IOError)
|
|
||||||
@time_out(60)
|
|
||||||
def test_10_should_raise_IOError_if_format_fails(self):
|
|
||||||
"""
|
|
||||||
|
|
||||||
Tests that if the driver's _format method fails, its
|
|
||||||
public format method will perform an assertion properly, discover
|
|
||||||
it failed, and raise an exception.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
volume_driver_cls = import_class(FLAGS.volume_driver)
|
|
||||||
|
|
||||||
class BadFormatter(volume_driver_cls):
|
|
||||||
|
|
||||||
def _format(self, device_path):
|
|
||||||
pass
|
|
||||||
|
|
||||||
bad_client = volume.Client(volume_driver=BadFormatter())
|
|
||||||
bad_client._format(self.story.device_path)
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_20_format(self):
|
|
||||||
self.assertNotEqual(None, self.story.device_path)
|
|
||||||
self.story.client._format(self.story.device_path)
|
|
||||||
|
|
||||||
def test_30_check_options(self):
|
|
||||||
cmd = ("sudo dumpe2fs -h %s 2> /dev/null | "
|
|
||||||
"awk -F ':' '{ if($1 == \"Reserved block count\") "
|
|
||||||
"{ rescnt=$2 } } { if($1 == \"Block count\") "
|
|
||||||
"{ blkcnt=$2 } } END { print (rescnt/blkcnt)*100 }'")
|
|
||||||
cmd = cmd % self.story.device_path
|
|
||||||
out, err = process(cmd)
|
|
||||||
self.assertEqual(float(5), round(float(out)), msg=out)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[FormatVolume])
|
|
||||||
class MountVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_mount(self):
|
|
||||||
self.story.client._mount(self.story.device_path,
|
|
||||||
self.story.mount_point)
|
|
||||||
with open(self.story.test_mount_file_path, 'w') as file:
|
|
||||||
file.write("Yep, it's mounted alright.")
|
|
||||||
self.assertTrue(os.path.exists(self.story.test_mount_file_path))
|
|
||||||
|
|
||||||
def test_mount_options(self):
|
|
||||||
cmd = "mount -l | awk '/%s.*noatime/ { print $1 }'"
|
|
||||||
cmd %= LOCAL_MOUNT_PATH.replace('/', '')
|
|
||||||
out, err = process(cmd)
|
|
||||||
self.assertEqual(os.path.realpath(self.story.device_path), out.strip(),
|
|
||||||
msg=out)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[MountVolume])
|
|
||||||
class ResizeVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(300)
|
|
||||||
def test_resize(self):
|
|
||||||
self.story.api.resize(self.story.context, self.story.volume_id,
|
|
||||||
self.story.resize_volume_size)
|
|
||||||
|
|
||||||
volume = poll_until(lambda: self.story.get_volume(),
|
|
||||||
lambda volume: volume["status"] == "resized")
|
|
||||||
self.assertEqual("resized", volume["status"])
|
|
||||||
self.assertEqual("attached", volume["attach_status"])
|
|
||||||
self.assertEqual(self.story.resize_volume_size, volume['size'])
|
|
||||||
|
|
||||||
@time_out(300)
|
|
||||||
def test_resizefs_rescan(self):
|
|
||||||
self.story.client.resize_fs(self.story.context,
|
|
||||||
self.story.volume_id)
|
|
||||||
expected = "trove.tests.volume.driver.ISCSITestDriver"
|
|
||||||
if FLAGS.volume_driver is expected:
|
|
||||||
size = self.story.resize_volume_size * \
|
|
||||||
test_driver.TESTS_VOLUME_SIZE_MULTIPLIER * 1024 * 1024
|
|
||||||
else:
|
|
||||||
size = self.story.resize_volume_size * 1024 * 1024
|
|
||||||
out, err = process('sudo blockdev --getsize64 %s' %
|
|
||||||
os.path.realpath(self.story.device_path))
|
|
||||||
if int(out) < (size * 0.8):
|
|
||||||
self.fail("Size %s is not more or less %s" % (out, size))
|
|
||||||
|
|
||||||
# Reset the volume status to available
|
|
||||||
self.story.api.update(self.story.context, self.story.volume_id,
|
|
||||||
{'status': 'available'})
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[MountVolume])
|
|
||||||
class UnmountVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_unmount(self):
|
|
||||||
self.story.client._unmount(self.story.mount_point)
|
|
||||||
child = pexpect.spawn("sudo mount %s" % self.story.mount_point)
|
|
||||||
child.expect("mount: can't find %s in" % self.story.mount_point)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[UnmountVolume])
|
|
||||||
class GrabUuid(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_uuid_must_match_pattern(self):
|
|
||||||
"""UUID must be hex chars in the form 8-4-4-4-12."""
|
|
||||||
client = self.story.client # volume.Client()
|
|
||||||
device_path = self.story.device_path # '/dev/sda5'
|
|
||||||
uuid = client.get_uuid(device_path)
|
|
||||||
self.story.original_uuid = uuid
|
|
||||||
self.assertTrue(is_uuid(uuid), "uuid must match regex")
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_get_invalid_uuid(self):
|
|
||||||
"""DevicePathInvalidForUuid is raised if device_path is wrong."""
|
|
||||||
client = self.story.client
|
|
||||||
device_path = "gdfjghsfjkhggrsyiyerreygghdsghsdfjhf"
|
|
||||||
self.assertRaises(trove_exception.DevicePathInvalidForUuid,
|
|
||||||
client.get_uuid, device_path)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[GrabUuid])
|
|
||||||
class RemoveVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_remove(self):
|
|
||||||
self.story.client.remove_volume(self.story.context,
|
|
||||||
self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
self.assertRaises(Exception,
|
|
||||||
self.story.client._format, self.story.device_path)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[GrabUuid])
|
|
||||||
class Initialize(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(300)
|
|
||||||
def test_10_initialize_will_format(self):
|
|
||||||
"""initialize will setup, format, and store the UUID of a volume"""
|
|
||||||
self.assertTrue(self.story.get_volume()['uuid'] is None)
|
|
||||||
self.story.client.initialize(self.story.context, self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
volume = self.story.get_volume()
|
|
||||||
self.assertTrue(is_uuid(volume['uuid']), "uuid must match regex")
|
|
||||||
self.assertNotEqual(self.story.original_uuid, volume['uuid'],
|
|
||||||
"Validate our assumption that the volume UUID "
|
|
||||||
"will change when the volume is formatted.")
|
|
||||||
self.story.client.remove_volume(self.story.context,
|
|
||||||
self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_20_initialize_the_second_time_will_not_format(self):
|
|
||||||
"""If initialize is called but a UUID exists, it should not format."""
|
|
||||||
old_uuid = self.story.get_volume()['uuid']
|
|
||||||
self.assertTrue(old_uuid is not None)
|
|
||||||
|
|
||||||
class VolumeClientNoFmt(volume.Client):
|
|
||||||
|
|
||||||
def _format(self, device_path):
|
|
||||||
raise RuntimeError("_format should not be called!")
|
|
||||||
|
|
||||||
no_fmt_client = VolumeClientNoFmt()
|
|
||||||
no_fmt_client.initialize(self.story.context, self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
self.assertEqual(old_uuid, self.story.get_volume()['uuid'],
|
|
||||||
"UUID should be the same as no formatting occurred.")
|
|
||||||
self.story.client.remove_volume(self.story.context,
|
|
||||||
self.story.volume_id,
|
|
||||||
self.story.host)
|
|
||||||
|
|
||||||
def test_30_check_device_exists(self):
|
|
||||||
assert_raises(exception.InvalidDevicePath, self.story.client._format,
|
|
||||||
self.story.device_path)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[Initialize])
|
|
||||||
class DeleteVolume(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_delete(self):
|
|
||||||
self.story.api.delete(self.story.context, self.story.volume_id)
|
|
||||||
|
|
||||||
|
|
||||||
@test(groups=[VOLUMES_DRIVER], depends_on_classes=[DeleteVolume])
|
|
||||||
class ConfirmMissing(VolumeTest):
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_discover_should_fail(self):
|
|
||||||
try:
|
|
||||||
self.story.client.driver.discover_volume(self.story.context,
|
|
||||||
self.story.volume)
|
|
||||||
self.fail("Expecting an error but did not get one.")
|
|
||||||
except exception.Error:
|
|
||||||
pass
|
|
||||||
except trove_exception.ISCSITargetNotDiscoverable:
|
|
||||||
pass
|
|
||||||
|
|
||||||
@time_out(60)
|
|
||||||
def test_get_missing_volume(self):
|
|
||||||
try:
|
|
||||||
volume = poll_until(lambda: self.story.api.get(self.story.context,
|
|
||||||
self.story.volume_id),
|
|
||||||
lambda volume: volume["status"] != "deleted")
|
|
||||||
self.assertEqual(volume["deleted"], False)
|
|
||||||
except exception.VolumeNotFound:
|
|
||||||
pass
|
|
@ -1,28 +0,0 @@
|
|||||||
# Examples:
|
|
||||||
# Run tests against Trove running locally in fake mode:
|
|
||||||
# TROVE_CLIENT_PATH=../some_path tox -e local -- --group=blackbox
|
|
||||||
[tox]
|
|
||||||
envlist = py26
|
|
||||||
|
|
||||||
[testenv]
|
|
||||||
deps =
|
|
||||||
coverage
|
|
||||||
nose
|
|
||||||
pexpect
|
|
||||||
proboscis
|
|
||||||
sqlalchemy
|
|
||||||
{env:TROVE_PATH}
|
|
||||||
{env:TROVE_CLIENT_PATH}
|
|
||||||
|
|
||||||
[testenv:py26]
|
|
||||||
|
|
||||||
[testenv:local]
|
|
||||||
deps =
|
|
||||||
nose
|
|
||||||
pexpect
|
|
||||||
proboscis
|
|
||||||
sqlalchemy
|
|
||||||
{env:TROVE_PATH}
|
|
||||||
{env:TROVE_CLIENT_PATH}
|
|
||||||
commands =
|
|
||||||
{envpython} int_tests.py --conf=localhost.test.conf {posargs:DEFAULTS}
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user