2014-10-23 12:17:09 -05:00
|
|
|
# Authors:
|
|
|
|
# Petr Viktorin <pviktori@redhat.com>
|
|
|
|
#
|
|
|
|
# Copyright (C) 2011 Red Hat
|
|
|
|
# see file 'COPYING' for use and warranty information
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""Pytest plugin for IPA Integration tests"""
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
import os
|
|
|
|
import tempfile
|
|
|
|
import shutil
|
|
|
|
|
2014-10-23 12:17:09 -05:00
|
|
|
import pytest
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
from ipapython import ipautil
|
|
|
|
from ipapython.ipa_log_manager import log_mgr
|
2014-10-23 12:17:09 -05:00
|
|
|
from ipatests.test_integration.config import get_global_config
|
|
|
|
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
log = log_mgr.get_logger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_addoption(parser):
|
|
|
|
group = parser.getgroup("IPA integration tests")
|
|
|
|
|
|
|
|
group.addoption(
|
|
|
|
'--logfile-dir', dest="logfile_dir", default=None,
|
|
|
|
help="Directory to store integration test logs in.")
|
|
|
|
|
|
|
|
|
|
|
|
def collect_test_logs(node, logs_dict, test_config):
|
|
|
|
"""Collect logs from a test
|
|
|
|
|
|
|
|
Calls collect_logs
|
|
|
|
|
|
|
|
:param node: The pytest collection node (request.node)
|
|
|
|
:param logs_dict: Mapping of host to list of log filnames to collect
|
|
|
|
:param test_config: Pytest configuration
|
|
|
|
"""
|
|
|
|
collect_logs(
|
|
|
|
name=node.nodeid.replace('/', '-').replace('::', '-'),
|
|
|
|
logs_dict=logs_dict,
|
|
|
|
logfile_dir=test_config.getoption('logfile_dir'),
|
|
|
|
beakerlib_plugin=test_config.pluginmanager.getplugin('BeakerLibPlugin'),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def collect_logs(name, logs_dict, logfile_dir=None, beakerlib_plugin=None):
|
|
|
|
"""Collect logs from remote hosts
|
|
|
|
|
|
|
|
Calls collect_logs
|
|
|
|
|
|
|
|
:param name: Name under which logs arecollected, e.g. name of the test
|
|
|
|
:param logs_dict: Mapping of host to list of log filnames to collect
|
|
|
|
:param logfile_dir: Directory to log to
|
|
|
|
:param beakerlib_plugin:
|
|
|
|
BeakerLibProcess or BeakerLibPlugin used to collect tests for BeakerLib
|
|
|
|
|
|
|
|
If neither logfile_dir nor beakerlib_plugin is given, no tests are
|
|
|
|
collected.
|
|
|
|
"""
|
|
|
|
if logs_dict and (logfile_dir or beakerlib_plugin):
|
|
|
|
|
|
|
|
if logfile_dir:
|
|
|
|
remove_dir = False
|
|
|
|
else:
|
|
|
|
logfile_dir = tempfile.mkdtemp()
|
|
|
|
remove_dir = True
|
|
|
|
|
|
|
|
topdirname = os.path.join(logfile_dir, name)
|
|
|
|
|
|
|
|
for host, logs in logs_dict.items():
|
|
|
|
log.info('Collecting logs from: %s', host.hostname)
|
|
|
|
|
|
|
|
# Tar up the logs on the remote server
|
|
|
|
cmd = host.run_command(['tar', 'cJv'] + logs, log_stdout=False,
|
|
|
|
raiseonerr=False)
|
|
|
|
if cmd.returncode:
|
|
|
|
log.warn('Could not collect all requested logs')
|
|
|
|
|
|
|
|
# Unpack on the local side
|
|
|
|
dirname = os.path.join(topdirname, host.hostname)
|
|
|
|
try:
|
|
|
|
os.makedirs(dirname)
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
tarname = os.path.join(dirname, 'logs.tar.xz')
|
|
|
|
with open(tarname, 'w') as f:
|
|
|
|
f.write(cmd.stdout_text)
|
|
|
|
ipautil.run(['tar', 'xJvf', 'logs.tar.xz'], cwd=dirname,
|
|
|
|
raiseonerr=False)
|
|
|
|
os.unlink(tarname)
|
|
|
|
|
|
|
|
if beakerlib_plugin:
|
|
|
|
# Use BeakerLib's rlFileSubmit on the indifidual files
|
|
|
|
# The resulting submitted filename will be
|
|
|
|
# $HOSTNAME-$FILENAME (with '/' replaced by '-')
|
|
|
|
beakerlib_plugin.run_beakerlib_command(['pushd', topdirname])
|
|
|
|
try:
|
|
|
|
for dirpath, dirnames, filenames in os.walk(topdirname):
|
|
|
|
for filename in filenames:
|
|
|
|
fullname = os.path.relpath(
|
|
|
|
os.path.join(dirpath, filename), topdirname)
|
|
|
|
log.debug('Submitting file: %s', fullname)
|
|
|
|
beakerlib_plugin.run_beakerlib_command(
|
|
|
|
['rlFileSubmit', fullname])
|
|
|
|
finally:
|
|
|
|
beakerlib_plugin.run_beakerlib_command(['popd'])
|
|
|
|
|
|
|
|
if remove_dir:
|
|
|
|
if beakerlib_plugin:
|
|
|
|
# The BeakerLib process runs asynchronously, let it clean up
|
|
|
|
# after it's done with the directory
|
|
|
|
beakerlib_plugin.run_beakerlib_command(
|
|
|
|
['rm', '-rvf', topdirname])
|
|
|
|
else:
|
|
|
|
shutil.rmtree(topdirname)
|
|
|
|
|
|
|
|
logs_dict.clear()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope='class')
|
|
|
|
def class_integration_logs():
|
|
|
|
"""Internal fixture providing class-level logs_dict"""
|
|
|
|
return {}
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture
|
|
|
|
def integration_logs(class_integration_logs, request):
|
|
|
|
"""Provides access to test integration logs, and collects after each test
|
|
|
|
"""
|
|
|
|
yield class_integration_logs
|
|
|
|
collect_test_logs(request.node, class_integration_logs, request.config)
|
|
|
|
|
|
|
|
|
2014-10-23 12:17:09 -05:00
|
|
|
@pytest.yield_fixture(scope='class')
|
2014-10-23 13:56:15 -05:00
|
|
|
def integration_config(request, class_integration_logs):
|
|
|
|
"""Integration test Config object
|
|
|
|
"""
|
2014-10-23 12:17:09 -05:00
|
|
|
cls = request.cls
|
|
|
|
|
|
|
|
def get_resources(resource_container, resource_str, num_needed):
|
|
|
|
if len(resource_container) < num_needed:
|
|
|
|
raise pytest.skip(
|
|
|
|
'Not enough %s available (have %s, need %s)' %
|
|
|
|
(resource_str, len(resource_container), num_needed))
|
|
|
|
return resource_container[:num_needed]
|
|
|
|
|
|
|
|
config = get_global_config()
|
|
|
|
if not config.domains:
|
|
|
|
raise pytest.skip('Integration testing not configured')
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
cls.logs_to_collect = class_integration_logs
|
2014-10-23 12:17:09 -05:00
|
|
|
|
|
|
|
cls.domain = config.domains[0]
|
|
|
|
|
|
|
|
# Check that we have enough resources available
|
|
|
|
cls.master = cls.domain.master
|
|
|
|
cls.replicas = get_resources(cls.domain.replicas, 'replicas',
|
|
|
|
cls.num_replicas)
|
|
|
|
cls.clients = get_resources(cls.domain.clients, 'clients',
|
|
|
|
cls.num_clients)
|
|
|
|
cls.ad_domains = get_resources(config.ad_domains, 'AD domains',
|
|
|
|
cls.num_ad_domains)
|
|
|
|
|
|
|
|
# Check that we have all required extra hosts at our disposal
|
|
|
|
available_extra_roles = [role for domain in cls.get_domains()
|
|
|
|
for role in domain.extra_roles]
|
|
|
|
missing_extra_roles = list(set(cls.required_extra_roles) -
|
|
|
|
set(available_extra_roles))
|
|
|
|
|
|
|
|
if missing_extra_roles:
|
|
|
|
raise pytest.skip("Not all required extra hosts available, "
|
|
|
|
"missing: %s, available: %s"
|
|
|
|
% (missing_extra_roles,
|
|
|
|
available_extra_roles))
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
def collect_log(host, filename):
|
|
|
|
log.info('Adding %s:%s to list of logs to collect' %
|
|
|
|
(host.external_hostname, filename))
|
|
|
|
class_integration_logs.setdefault(host, []).append(filename)
|
|
|
|
|
2014-10-23 12:17:09 -05:00
|
|
|
for host in cls.get_all_hosts():
|
2014-10-23 13:56:15 -05:00
|
|
|
host.add_log_collector(collect_log)
|
2014-10-23 12:17:09 -05:00
|
|
|
cls.prepare_host(host)
|
|
|
|
|
|
|
|
try:
|
|
|
|
cls.install()
|
|
|
|
except:
|
|
|
|
cls.uninstall()
|
|
|
|
raise
|
|
|
|
|
|
|
|
yield config
|
|
|
|
|
|
|
|
for host in cls.get_all_hosts():
|
|
|
|
host.remove_log_collector(collect_log)
|
|
|
|
|
2014-10-23 13:56:15 -05:00
|
|
|
collect_test_logs(request.node, class_integration_logs, request.config)
|
|
|
|
|
2014-10-23 12:17:09 -05:00
|
|
|
try:
|
|
|
|
cls.uninstall()
|
|
|
|
finally:
|
|
|
|
del cls.master
|
|
|
|
del cls.replicas
|
|
|
|
del cls.clients
|
|
|
|
del cls.ad_domains
|
|
|
|
del cls.domain
|