blob: 24930837eaaca0e3582b0e77bf8031ee8f0e3b80 [file] [log] [blame]
# Copyright 2017 Mirantis, Inc.
#
# 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 pytest
from devops.helpers import helpers
from tcp_tests import logger
from tcp_tests.utils import get_jenkins_job_stages
from tcp_tests.utils import run_jenkins_job
LOG = logger.logger
class TestFailoverCeph(object):
"""Test class for testing MCP Ceph failover"""
TEMPEST_JOB_NAME = 'cvp-tempest'
TEMPEST_JOB_PARAMETERS = {
'TEMPEST_ENDPOINT_TYPE': 'internalURL',
'TEMPEST_TEST_PATTERN': 'set=smoke'
}
SANITY_JOB_NAME = 'cvp-sanity'
SANITY_JOB_PARAMETERS = {
'EXTRA_PARAMS': {
'envs': [
"tests_set=-k "
"'not test_ceph_health and not test_prometheus_alert_count'"
]
}
}
JENKINS_START_TIMEOUT = 60
JENKINS_BUILD_TIMEOUT = 60 * 15
def get_ceph_health(self, ssh, node_names):
"""Get Ceph health status on specified nodes
:param ssh: UnderlaySSHManager, tcp-qa SSH manager instance
:param node_names: list, full hostnames of Ceph OSD nodes
:return: dict, Ceph health status from each OSD node (output of
'ceph -s' command executed on each node)
"""
return {
node_name: ssh.check_call(
"ceph -s",
node_name=node_name,
raise_on_err=False)['stdout_str']
for node_name in node_names
}
def run_jenkins_job(
self, creds, name, parameters, start_timeout, build_timeout):
"""Execute a Jenkins job with provided parameters
:param creds: dict, Jenkins url and user credentials
:param name: string, Jenkins job to execute
:param parameters: dict, parameters for Jenkins job
:parameter start_timeout: int, timeout to wait until build is started
:parameter build_timeout: int, timeout to wait until build is finished
:return: tuple, Jenkins job build execution status, high level
description of the build and verbose decription of executed job
stages
"""
jenkins_url, jenkins_user, jenkins_pass = (
creds['url'], creds['user'], creds['pass'])
build_status = run_jenkins_job.run_job(
host=jenkins_url,
username=jenkins_user,
password=jenkins_pass,
start_timeout=start_timeout,
build_timeout=build_timeout,
verbose=False,
job_name=name,
job_parameters=parameters)
description, stages = get_jenkins_job_stages.get_deployment_result(
host=jenkins_url,
username=jenkins_user,
password=jenkins_pass,
job_name=name,
build_number='lastBuild')
return build_status, description, stages
@pytest.mark.grab_versions
@pytest.mark.restart_osd_node
def test_restart_osd_node(
self,
salt_actions,
underlay_actions,
show_step):
"""Verify that Ceph OSD node is not affected by system restart
Scenario:
1. Find Ceph OSD nodes
2. Check Ceph cluster health before node restart (skipped until
PROD-31374 is fixed)
3. Restart 1 Ceph OSD node
4. Check Ceph cluster health after node restart (skipped until
PROD-31374 is fixed)
5. Run Tempest smoke test suite
6. Run test_ceph_status.py::test_ceph_osd and
test_services.py::test_check_services[osd] sanity tests
Duration: ~9 min
"""
salt = salt_actions
ssh = underlay_actions
# Find Ceph OSD nodes
show_step(1)
tgt = "I@ceph:osd"
osd_hosts = salt.local(tgt, "test.ping")['return'][0].keys()
# Select a node for the test
osd_host = osd_hosts[0]
# Check Ceph cluster health before node restart
show_step(2)
ceph_health = self.get_ceph_health(ssh, osd_hosts)
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Restart a Ceph OSD node
show_step(3)
LOG.info("Sending reboot command to '{}' node.".format(osd_host))
remote = ssh.remote(node_name=osd_host)
remote.execute_async("/sbin/shutdown -r now")
# Wait for restarted node to boot and become accessible
helpers.wait_pass(remote.reconnect, timeout=60 * 3, interval=5)
echo_request = "echo"
echo_response = salt.local(
osd_host, "test.echo", echo_request)['return'][0]
assert echo_request == echo_response[osd_host], (
"Minion on node '{}' node is not responding after node "
"reboot.".format(osd_host)
)
LOG.info("'{}' node is back after reboot.".format(osd_host))
# Check Ceph cluster health after node restart
show_step(4)
ceph_health = self.get_ceph_health(ssh, osd_hosts) # noqa
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Run Tempest smoke test suite
show_step(5)
jenkins_creds = salt.get_cluster_jenkins_creds()
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.TEMPEST_JOB_NAME,
self.TEMPEST_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing Tempest smoke "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.TEMPEST_JOB_NAME, status, description, '\n'.join(stages))
)
# Run Sanity test
show_step(6)
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.SANITY_JOB_NAME,
self.SANITY_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing selected sanity "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.SANITY_JOB_NAME, status, description, '\n'.join(stages))
)
@pytest.mark.grab_versions
@pytest.mark.restart_cmn_node
def test_restart_cmn_node(
self,
salt_actions,
underlay_actions,
show_step):
"""Verify that Ceph CMN node is not affected by system restart
Scenario:
1. Find Ceph CMN nodes
2. Check Ceph cluster health before node restart (skipped until
PROD-31374 is fixed)
3. Restart 1 Ceph CMN node
4. Check Ceph cluster health after node restart (skipped until
PROD-31374 is fixed)
5. Run Tempest smoke test suite
6. Run test_ceph_status.py::test_ceph_replicas and
test_services.py::test_check_services[cmn] sanity tests
Duration: ~9 min
"""
salt = salt_actions
ssh = underlay_actions
# Find Ceph CMN nodes
show_step(1)
tgt = "I@ceph:mon"
cmn_hosts = salt.local(tgt, "test.ping")['return'][0].keys()
# Select a node for the test
cmn_host = cmn_hosts[0]
# Check Ceph cluster health before node restart
show_step(2)
ceph_health = self.get_ceph_health(ssh, cmn_hosts)
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Restart a Ceph CMN node
show_step(3)
LOG.info("Sending reboot command to '{}' node.".format(cmn_host))
remote = ssh.remote(node_name=cmn_host)
remote.execute_async("/sbin/shutdown -r now")
# Wait for restarted node to boot and become accessible
helpers.wait_pass(remote.reconnect, timeout=60 * 3, interval=5)
echo_request = "echo"
echo_response = salt.local(
cmn_host, "test.echo", echo_request)['return'][0]
assert echo_request == echo_response[cmn_host], (
"Minion on node '{}' node is not responding after node "
"reboot.".format(cmn_host)
)
LOG.info("'{}' node is back after reboot.".format(cmn_host))
# Check Ceph cluster health after node restart
show_step(4)
ceph_health = self.get_ceph_health(ssh, cmn_hosts) # noqa
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Run Tempest smoke test suite
show_step(5)
jenkins_creds = salt.get_cluster_jenkins_creds()
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.TEMPEST_JOB_NAME,
self.TEMPEST_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing Tempest smoke "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.TEMPEST_JOB_NAME, status, description, '\n'.join(stages))
)
# Run Sanity test
show_step(6)
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.SANITY_JOB_NAME,
self.SANITY_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing selected sanity "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.SANITY_JOB_NAME, status, description, '\n'.join(stages))
)
@pytest.mark.grab_versions
@pytest.mark.restart_rgw_node
def test_restart_rgw_node(
self,
salt_actions,
underlay_actions,
show_step):
"""Verify that Ceph RGW node is not affected by system restart
Scenario:
1. Find Ceph RGW nodes
2. Check Ceph cluster health before node restart (skipped until
PROD-31374 is fixed)
3. Restart 1 Ceph RGW node
4. Check Ceph cluster health after node restart (skipped until
PROD-31374 is fixed)
5. Run Tempest smoke test suite
6. Run test_services.py::test_check_services[rgw] sanity test
Duration: ~9 min
"""
salt = salt_actions
ssh = underlay_actions
# Find Ceph RGW nodes
show_step(1)
tgt = "I@ceph:radosgw"
rgw_hosts = salt.local(tgt, "test.ping")['return'][0].keys()
# Select a node for the test
rgw_host = rgw_hosts[0]
# Check Ceph cluster health before node restart
show_step(2)
ceph_health = self.get_ceph_health(ssh, rgw_hosts)
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Restart a Ceph RGW node
show_step(3)
LOG.info("Sending reboot command to '{}' node.".format(rgw_host))
remote = ssh.remote(node_name=rgw_host)
remote.execute_async("/sbin/shutdown -r now")
# Wait for restarted node to boot and become accessible
helpers.wait_pass(remote.reconnect, timeout=60 * 3, interval=5)
echo_request = "echo"
echo_response = salt.local(
rgw_host, "test.echo", echo_request)['return'][0]
assert echo_request == echo_response[rgw_host], (
"Minion on node '{}' node is not responding after node "
"reboot.".format(rgw_host)
)
LOG.info("'{}' node is back after reboot.".format(rgw_host))
# Check Ceph cluster health after node restart
show_step(4)
ceph_health = self.get_ceph_health(ssh, rgw_hosts) # noqa
# FIXME: uncomment the check once PROD-31374 is fixed
# status = all(
# ["OK" in status for node, status in ceph_health.items()])
# assert status, "Ceph health is not OK: {0}".format(ceph_health)
# Run Tempest smoke test suite
show_step(5)
jenkins_creds = salt.get_cluster_jenkins_creds()
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.TEMPEST_JOB_NAME,
self.TEMPEST_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing Tempest smoke "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.TEMPEST_JOB_NAME, status, description, '\n'.join(stages))
)
# Run Sanity test
show_step(6)
status, description, stages = self.run_jenkins_job(
jenkins_creds,
self.SANITY_JOB_NAME,
self.SANITY_JOB_PARAMETERS,
self.JENKINS_START_TIMEOUT,
self.JENKINS_BUILD_TIMEOUT
)
assert status == 'SUCCESS', (
"'{0}' job run status is {1} after executing selected sanity "
"tests. Please check the build:\n{2}\n\nExecuted build "
"stages:\n{3}".format(
self.SANITY_JOB_NAME, status, description, '\n'.join(stages))
)
# #######################################################################
# ############# Tests for fuel-devops deployed environments #############
# #######################################################################
def show_failed_msg(self, failed):
return "There are failed tempest tests:\n\n {0}".format(
'\n\n '.join([(name + ': ' + detail)
for name, detail in failed.items()]))
@pytest.mark.grab_versions
@pytest.mark.fail_snapshot
def _test_restart_osd_node(self, func_name, underlay, config,
openstack_deployed, ceph_deployed,
openstack_actions, hardware,
rally, show_step):
"""Test restart ceph osd node
Scenario:
1. Find ceph osd nodes
2. Check ceph health before restart
3. Restart 1 ceph osd node
4. Check ceph health after restart
5. Run tempest smoke after failover
6. Check tempest report for failed tests
Requiremets:
- Salt cluster
- OpenStack cluster
- Ceph cluster
"""
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #1
show_step(1)
osd_node_names = underlay.get_target_node_names(
target='osd')
# STEP #2
show_step(2)
# Get the ceph health output before restart
health_before = self.get_ceph_health(underlay, osd_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_before))
# STEP #3
show_step(3)
hardware.warm_restart_nodes(underlay, 'osd01')
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #4
show_step(4)
# Get the ceph health output after restart
health_after = self.get_ceph_health(underlay, osd_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_after))
rally.run_container()
# STEP #5
show_step(5)
results = rally.run_tempest(pattern='set=smoke',
conf_name='/var/lib/ceph_mcp.conf',
report_prefix=func_name,
designate_plugin=False,
timeout=1800)
# Step #6
show_step(6)
assert not results['fail'], self.show_failed_msg(results['fail'])
LOG.info("*************** DONE **************")
@pytest.mark.grab_versions
@pytest.mark.fail_snapshot
def _test_restart_cmn_node(self, func_name, underlay, config,
openstack_deployed, ceph_deployed,
core_actions,
salt_actions, openstack_actions,
rally, show_step, hardware):
"""Test restart ceph cmn node
Scenario:
1. Find ceph cmn nodes
2. Check ceph health before restart
3. Restart 1 ceph cmn node
4. Check ceph health after restart
5. Run tempest smoke after failover
6. Check tempest report for failed tests
Requiremets:
- Salt cluster
- OpenStack cluster
- Ceph cluster
"""
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #1
show_step(1)
cmn_node_names = underlay.get_target_node_names(
target='cmn')
# STEP #2
show_step(2)
# Get the ceph health output before restart
health_before = self.get_ceph_health(underlay, cmn_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_before))
# STEP #3
show_step(3)
hardware.warm_restart_nodes(underlay, 'cmn01')
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #4
show_step(4)
# Get the ceph health output after restart
health_after = self.get_ceph_health(underlay, cmn_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_after))
rally.run_container()
# STEP #5
show_step(5)
results = rally.run_tempest(pattern='set=smoke',
conf_name='/var/lib/ceph_mcp.conf',
report_prefix=func_name,
designate_plugin=False,
timeout=1800)
# Step #6
show_step(6)
assert not results['fail'], self.show_failed_msg(results['fail'])
LOG.info("*************** DONE **************")
@pytest.mark.grab_versions
@pytest.mark.fail_snapshot
def _test_restart_rgw_node(self, func_name, underlay, config,
openstack_deployed, ceph_deployed,
core_actions, hardware,
salt_actions, openstack_actions,
rally, show_step):
"""Test restart ceph rgw node
Scenario:
1. Find ceph rgw nodes
2. Check ceph health before restart
3. Restart 1 ceph rgw node
4. Check ceph health after restart
5. Run tempest smoke after failover
6. Check tempest report for failed tests
Requiremets:
- Salt cluster
- OpenStack cluster
- Ceph cluster
"""
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #1
show_step(1)
rgw_node_names = underlay.get_target_node_names(
target='rgw')
if not rgw_node_names:
pytest.skip('Skip as there are not rgw nodes in deploy')
# STEP #2
show_step(2)
# Get the ceph health output before restart
health_before = self.get_ceph_health(underlay, rgw_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_before))
# STEP #3
show_step(3)
hardware.warm_restart_nodes(underlay, 'rgw01')
openstack_actions._salt.local(
tgt='*', fun='cmd.run',
args='service ntp stop; ntpd -gq; service ntp start')
# STEP #4
show_step(4)
# Get the ceph health output after restart
health_after = self.get_ceph_health(underlay, rgw_node_names)
assert all(["OK" in p for n, p in health_before.items()]), (
"'Ceph health is not ok from node: {0}".format(health_after))
rally.run_container()
# STEP #5
show_step(5)
results = rally.run_tempest(pattern='set=smoke',
conf_name='/var/lib/ceph_mcp.conf',
designate_plugin=False,
report_prefix=func_name,
timeout=1800)
# Step #6
show_step(6)
assert not results['fail'], self.show_failed_msg(results['fail'])
LOG.info("*************** DONE **************")