# -*- coding: utf-8 -*- ''' :codeauthor: Pedro Algarvio (pedro@algarvio.me) tests.conftest ~~~~~~~~~~~~~~ Prepare py.test for our test suite ''' # Import python libs from __future__ import absolute_import import os import sys import stat import socket import logging from collections import namedtuple TESTS_DIR = os.path.dirname( os.path.normpath(os.path.abspath(__file__)) ) CODE_DIR = os.path.dirname(TESTS_DIR) os.chdir(CODE_DIR) try: # If we have a system-wide salt module imported, unload it import salt for module in list(sys.modules): if module.startswith(('salt',)): try: if not sys.modules[module].__file__.startswith(CODE_DIR): sys.modules.pop(module) except AttributeError: continue sys.path.insert(0, CODE_DIR) except ImportError: sys.path.insert(0, CODE_DIR) # Import test libs import tests.support.paths # pylint: disable=unused-import from tests.integration import TestDaemon # Import pytest libs import pytest from _pytest.terminal import TerminalReporter # Import 3rd-party libs import psutil from salt.ext import six # Import salt libs import salt.utils.files import salt.utils.path import salt.log.setup from salt.utils.odict import OrderedDict # Define the pytest plugins we rely on pytest_plugins = ['tempdir', 'helpers_namespace', 'salt-from-filenames'] # pylint: disable=invalid-name # Define where not to collect tests from collect_ignore = ['setup.py'] log = logging.getLogger('salt.testsuite') # Reset logging root handlers for handler in logging.root.handlers: logging.root.removeHandler(handler) def pytest_tempdir_basename(): ''' Return the temporary directory basename for the salt test suite. ''' return 'salt-tests-tmp' # ----- CLI Options Setup -------------------------------------------------------------------------------------------> def pytest_addoption(parser): ''' register argparse-style options and ini-style config values. ''' parser.addoption( '--sysinfo', default=False, action='store_true', help='Print some system information.' ) parser.addoption( '--transport', default='zeromq', choices=('zeromq', 'raet', 'tcp'), help=('Select which transport to run the integration tests with, ' 'zeromq, raet, or tcp. Default: %default') ) test_selection_group = parser.getgroup('Tests Selection') test_selection_group.addoption( '--ssh', '--ssh-tests', dest='ssh', action='store_true', default=False, help='Run salt-ssh tests. These tests will spin up a temporary ' 'SSH server on your machine. In certain environments, this ' 'may be insecure! Default: False' ) test_selection_group.addoption( '--proxy', '--proxy-tests', dest='proxy', action='store_true', default=False, help='Run proxy tests' ) test_selection_group.addoption( '--run-destructive', action='store_true', default=False, help='Run destructive tests. These tests can include adding ' 'or removing users from your system for example. ' 'Default: False' ) test_selection_group.addoption( '--run-expensive', action='store_true', default=False, help='Run expensive tests. These tests usually involve costs ' 'like for example bootstrapping a cloud VM. ' 'Default: False' ) output_options_group = parser.getgroup('Output Options') output_options_group.addoption( '--output-columns', default=80, type=int, help='Number of maximum columns to use on the output' ) output_options_group.addoption( '--no-colors', '--no-colours', default=False, action='store_true', help='Disable colour printing.' ) # <---- CLI Options Setup -------------------------------------------------------------------------------------------- # ----- CLI Terminal Reporter ---------------------------------------------------------------------------------------> class SaltTerminalReporter(TerminalReporter): def __init__(self, config): TerminalReporter.__init__(self, config) @pytest.hookimpl(trylast=True) def pytest_sessionstart(self, session): TerminalReporter.pytest_sessionstart(self, session) self._session = session def pytest_runtest_logreport(self, report): TerminalReporter.pytest_runtest_logreport(self, report) if self.verbosity <= 0: return if report.when != 'call': return if self.config.getoption('--sys-stats') is False: return test_daemon = getattr(self._session, 'test_daemon', None) if self.verbosity == 1: line = ' [CPU:{0}%|MEM:{1}%]'.format(psutil.cpu_percent(), psutil.virtual_memory().percent) self._tw.write(line) return else: self.ensure_newline() template = ' {} - CPU: {:6.2f} % MEM: {:6.2f} % SWAP: {:6.2f} %\n' self._tw.write( template.format( ' System', psutil.cpu_percent(), psutil.virtual_memory().percent, psutil.swap_memory().percent ) ) for name, psproc in self._session.stats_processes.items(): with psproc.oneshot(): cpu = psproc.cpu_percent() mem = psproc.memory_percent('vms') swap = psproc.memory_percent('swap') self._tw.write(template.format(name, cpu, mem, swap)) def pytest_sessionstart(session): session.stats_processes = OrderedDict(( #('Log Server', test_daemon.log_server), (' Test Suite Run', psutil.Process(os.getpid())), )) # <---- CLI Terminal Reporter ---------------------------------------------------------------------------------------- # ----- Register Markers --------------------------------------------------------------------------------------------> @pytest.mark.trylast def pytest_configure(config): ''' called after command line options have been parsed and all plugins and initial conftest files been loaded. ''' config.addinivalue_line('norecursedirs', os.path.join(CODE_DIR, 'templates')) config.addinivalue_line( 'markers', 'destructive_test: Run destructive tests. These tests can include adding ' 'or removing users from your system for example.' ) config.addinivalue_line( 'markers', 'skip_if_not_root: Skip if the current user is not `root`.' ) config.addinivalue_line( 'markers', 'skip_if_binaries_missing(*binaries, check_all=False, message=None): Skip if ' 'any of the passed binaries are not found in path. If \'check_all\' is ' '\'True\', then all binaries must be found.' ) config.addinivalue_line( 'markers', 'requires_network(only_local_network=False): Skip if no networking is set up. ' 'If \'only_local_network\' is \'True\', only the local network is checked.' ) # Register our terminal reporter if not getattr(config, 'slaveinput', None): standard_reporter = config.pluginmanager.getplugin('terminalreporter') salt_reporter = SaltTerminalReporter(standard_reporter.config) config.pluginmanager.unregister(standard_reporter) config.pluginmanager.register(salt_reporter, 'terminalreporter') # Transplant configuration TestDaemon.transplant_configs(transport=config.getoption('--transport')) # <---- Register Markers --------------------------------------------------------------------------------------------- # ----- Test Setup --------------------------------------------------------------------------------------------------> @pytest.hookimpl(tryfirst=True) def pytest_runtest_setup(item): ''' Fixtures injection based on markers or test skips based on CLI arguments ''' destructive_tests_marker = item.get_marker('destructive_test') if destructive_tests_marker is not None: if item.config.getoption('--run-destructive') is False: pytest.skip('Destructive tests are disabled') os.environ['DESTRUCTIVE_TESTS'] = six.text_type(item.config.getoption('--run-destructive')) expensive_tests_marker = item.get_marker('expensive_test') if expensive_tests_marker is not None: if item.config.getoption('--run-expensive') is False: pytest.skip('Expensive tests are disabled') os.environ['EXPENSIVE_TESTS'] = six.text_type(item.config.getoption('--run-expensive')) skip_if_not_root_marker = item.get_marker('skip_if_not_root') if skip_if_not_root_marker is not None: if os.getuid() != 0: pytest.skip('You must be logged in as root to run this test') skip_if_binaries_missing_marker = item.get_marker('skip_if_binaries_missing') if skip_if_binaries_missing_marker is not None: binaries = skip_if_binaries_missing_marker.args if len(binaries) == 1: if isinstance(binaries[0], (list, tuple, set, frozenset)): binaries = binaries[0] check_all = skip_if_binaries_missing_marker.kwargs.get('check_all', False) message = skip_if_binaries_missing_marker.kwargs.get('message', None) if check_all: for binary in binaries: if salt.utils.path.which(binary) is None: pytest.skip( '{0}The "{1}" binary was not found'.format( message and '{0}. '.format(message) or '', binary ) ) elif salt.utils.path.which_bin(binaries) is None: pytest.skip( '{0}None of the following binaries was found: {1}'.format( message and '{0}. '.format(message) or '', ', '.join(binaries) ) ) requires_network_marker = item.get_marker('requires_network') if requires_network_marker is not None: only_local_network = requires_network_marker.kwargs.get('only_local_network', False) has_local_network = False # First lets try if we have a local network. Inspired in verify_socket try: pubsock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) retsock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) pubsock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) pubsock.bind(('', 18000)) pubsock.close() retsock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) retsock.bind(('', 18001)) retsock.close() has_local_network = True except socket.error: # I wonder if we just have IPV6 support? try: pubsock = socket.socket(socket.AF_INET6, socket.SOCK_STREAM) retsock = socket.socket(socket.AF_INET6, socket.SOCK_STREAM) pubsock.setsockopt( socket.SOL_SOCKET, socket.SO_REUSEADDR, 1 ) pubsock.bind(('', 18000)) pubsock.close() retsock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) retsock.bind(('', 18001)) retsock.close() has_local_network = True except socket.error: # Let's continue pass if only_local_network is True: if has_local_network is False: # Since we're only supposed to check local network, and no # local network was detected, skip the test pytest.skip('No local network was detected') # We are using the google.com DNS records as numerical IPs to avoid # DNS lookups which could greatly slow down this check for addr in ('173.194.41.198', '173.194.41.199', '173.194.41.200', '173.194.41.201', '173.194.41.206', '173.194.41.192', '173.194.41.193', '173.194.41.194', '173.194.41.195', '173.194.41.196', '173.194.41.197'): try: sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.settimeout(0.25) sock.connect((addr, 80)) sock.close() # We connected? Stop the loop break except socket.error: # Let's check the next IP continue else: pytest.skip('No internet network connection was detected') # <---- Test Setup --------------------------------------------------------------------------------------------------- # ----- Automatic Markers Setup -------------------------------------------------------------------------------------> def pytest_collection_modifyitems(items): ''' Automatically add markers to tests based on directory layout ''' for item in items: fspath = str(item.fspath) if '/integration/' in fspath: if 'test_daemon' not in item.fixturenames: item.fixturenames.append('test_daemon') item.add_marker(pytest.mark.integration) for kind in ('cli', 'client', 'cloud', 'fileserver', 'loader', 'minion', 'modules', 'netapi', 'output', 'reactor', 'renderers', 'runners', 'sdb', 'shell', 'ssh', 'states', 'utils', 'wheel'): if '/{0}/'.format(kind) in fspath: item.add_marker(getattr(pytest.mark, kind)) break if '/unit/' in fspath: item.add_marker(pytest.mark.unit) for kind in ('acl', 'beacons', 'cli', 'cloud', 'config', 'grains', 'modules', 'netapi', 'output', 'pillar', 'renderers', 'runners', 'serializers', 'states', 'templates', 'transport', 'utils'): if '/{0}/'.format(kind) in fspath: item.add_marker(getattr(pytest.mark, kind)) break # <---- Automatic Markers Setup -------------------------------------------------------------------------------------- # ----- Pytest Helpers ----------------------------------------------------------------------------------------------> if six.PY2: # backport mock_open from the python 3 unittest.mock library so that we can # mock read, readline, readlines, and file iteration properly file_spec = None def _iterate_read_data(read_data): # Helper for mock_open: # Retrieve lines from read_data via a generator so that separate calls to # readline, read, and readlines are properly interleaved data_as_list = ['{0}\n'.format(l) for l in read_data.split('\n')] if data_as_list[-1] == '\n': # If the last line ended in a newline, the list comprehension will have an # extra entry that's just a newline. Remove this. data_as_list = data_as_list[:-1] else: # If there wasn't an extra newline by itself, then the file being # emulated doesn't have a newline to end the last line remove the # newline that our naive format() added data_as_list[-1] = data_as_list[-1][:-1] for line in data_as_list: yield line @pytest.helpers.mock.register def mock_open(mock=None, read_data=''): """ A helper function to create a mock to replace the use of `open`. It works for `open` called directly or used as a context manager. The `mock` argument is the mock object to configure. If `None` (the default) then a `MagicMock` will be created for you, with the API limited to methods or attributes available on standard file handles. `read_data` is a string for the `read` methoddline`, and `readlines` of the file handle to return. This is an empty string by default. """ _mock = pytest.importorskip('mock', minversion='2.0.0') def _readlines_side_effect(*args, **kwargs): if handle.readlines.return_value is not None: return handle.readlines.return_value return list(_data) def _read_side_effect(*args, **kwargs): if handle.read.return_value is not None: return handle.read.return_value return ''.join(_data) def _readline_side_effect(): if handle.readline.return_value is not None: while True: yield handle.readline.return_value for line in _data: yield line global file_spec if file_spec is None: file_spec = file # pylint: disable=undefined-variable if mock is None: mock = _mock.MagicMock(name='open', spec=open) handle = _mock.MagicMock(spec=file_spec) handle.__enter__.return_value = handle _data = _iterate_read_data(read_data) handle.write.return_value = None handle.read.return_value = None handle.readline.return_value = None handle.readlines.return_value = None handle.read.side_effect = _read_side_effect handle.readline.side_effect = _readline_side_effect() handle.readlines.side_effect = _readlines_side_effect mock.return_value = handle return mock else: @pytest.helpers.mock.register def mock_open(mock=None, read_data=''): _mock = pytest.importorskip('mock', minversion='2.0.0') return _mock.mock_open(mock=mock, read_data=read_data) # <---- Pytest Helpers ----------------------------------------------------------------------------------------------- # ----- Fixtures Overrides ------------------------------------------------------------------------------------------> # ----- Generate CLI Scripts ----------------------------------------------------------------------------------------> @pytest.fixture(scope='session') def cli_master_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_master' @pytest.fixture(scope='session') def cli_minion_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_minion' @pytest.fixture(scope='session') def cli_salt_script_name(): ''' Return the CLI script basename ''' return 'cli_salt' @pytest.fixture(scope='session') def cli_run_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_run' @pytest.fixture(scope='session') def cli_key_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_key' @pytest.fixture(scope='session') def cli_call_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_call' @pytest.fixture(scope='session') def cli_syndic_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_syndic' @pytest.fixture(scope='session') def cli_ssh_script_name(): ''' Return the CLI script basename ''' return 'cli_salt_ssh' @pytest.fixture(scope='session') def cli_bin_dir(tempdir, request, python_executable_path, cli_master_script_name, cli_minion_script_name, cli_salt_script_name, cli_call_script_name, cli_key_script_name, cli_run_script_name, cli_ssh_script_name): ''' Return the path to the CLI script directory to use ''' tmp_cli_scripts_dir = tempdir.join('cli-scrips-bin') tmp_cli_scripts_dir.ensure(dir=True) cli_bin_dir_path = tmp_cli_scripts_dir.strpath # Now that we have the CLI directory created, lets generate the required CLI scripts to run salt's test suite script_templates = { 'salt': [ 'from salt.scripts import salt_main\n', 'if __name__ == \'__main__\':\n' ' salt_main()' ], 'salt-api': [ 'import salt.cli\n', 'def main():\n', ' sapi = salt.cli.SaltAPI()', ' sapi.run()\n', 'if __name__ == \'__main__\':', ' main()' ], 'common': [ 'from salt.scripts import salt_{0}\n', 'if __name__ == \'__main__\':\n', ' salt_{0}()' ] } for script_name in (cli_master_script_name, cli_minion_script_name, cli_call_script_name, cli_key_script_name, cli_run_script_name, cli_salt_script_name, cli_ssh_script_name): original_script_name = script_name.split('cli_')[-1].replace('_', '-') script_path = os.path.join(cli_bin_dir_path, script_name) if not os.path.isfile(script_path): log.info('Generating {0}'.format(script_path)) with salt.utils.files.fopen(script_path, 'w') as sfh: script_template = script_templates.get(original_script_name, None) if script_template is None: script_template = script_templates.get('common', None) if script_template is None: raise RuntimeError( 'Salt\'s test suite does not know how to handle the "{0}" script'.format( original_script_name ) ) sfh.write( '#!{0}\n\n'.format(python_executable_path) + 'import sys\n' + 'CODE_DIR="{0}"\n'.format(request.config.startdir.realpath().strpath) + 'if CODE_DIR not in sys.path:\n' + ' sys.path.insert(0, CODE_DIR)\n\n' + '\n'.join(script_template).format(original_script_name.replace('salt-', '')) ) fst = os.stat(script_path) os.chmod(script_path, fst.st_mode | stat.S_IEXEC) # Return the CLI bin dir value return cli_bin_dir_path # <---- Generate CLI Scripts ----------------------------------------------------------------------------------------- # ----- Salt Configuration ------------------------------------------------------------------------------------------> @pytest.fixture(scope='session') def session_integration_files_dir(request): ''' Fixture which returns the salt integration files directory path. Creates the directory if it does not yet exist. ''' return request.config.startdir.join('tests').join('integration').join('files') @pytest.fixture(scope='session') def session_state_tree_root_dir(session_integration_files_dir): ''' Fixture which returns the salt state tree root directory path. Creates the directory if it does not yet exist. ''' return session_integration_files_dir.join('file') @pytest.fixture(scope='session') def session_pillar_tree_root_dir(session_integration_files_dir): ''' Fixture which returns the salt pillar tree root directory path. Creates the directory if it does not yet exist. ''' return session_integration_files_dir.join('pillar') # <---- Salt Configuration ------------------------------------------------------------------------------------------- # <---- Fixtures Overrides ------------------------------------------------------------------------------------------- # ----- Custom Fixtures Definitions ---------------------------------------------------------------------------------> @pytest.fixture(scope='session') def test_daemon(request): values = (('transport', request.config.getoption('--transport')), ('sysinfo', request.config.getoption('--sysinfo')), ('no_colors', request.config.getoption('--no-colors')), ('output_columns', request.config.getoption('--output-columns')), ('ssh', request.config.getoption('--ssh')), ('proxy', request.config.getoption('--proxy'))) options = namedtuple('options', [n for n, v in values])(*[v for n, v in values]) fake_parser = namedtuple('parser', 'options')(options) test_daemon = TestDaemon(fake_parser) with test_daemon as test_daemon_running: request.session.test_daemon = test_daemon_running request.session.stats_processes.update(OrderedDict(( (' Salt Master', psutil.Process(test_daemon.master_process.pid)), (' Salt Minion', psutil.Process(test_daemon.minion_process.pid)), (' Salt Sub Minion', psutil.Process(test_daemon.sub_minion_process.pid)), ('Salt Syndic Master', psutil.Process(test_daemon.smaster_process.pid)), (' Salt Syndic', psutil.Process(test_daemon.syndic_process.pid)), )).items()) yield TestDaemon.clean() # <---- Custom Fixtures Definitions ----------------------------------------------------------------------------------