|  | #    Copyright 2016 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 tcp_tests.helpers import ext | 
|  | from tcp_tests.helpers import utils | 
|  | from tcp_tests import logger | 
|  | from tcp_tests import settings | 
|  | from tcp_tests.managers import underlay_ssh_manager | 
|  |  | 
|  | LOG = logger.logger | 
|  |  | 
|  |  | 
|  | @pytest.fixture(scope="session") | 
|  | def hardware(request, config): | 
|  | """Fixture for manage the hardware layer. | 
|  |  | 
|  | - start/stop/reboot libvirt/IPMI(/MaaS?) nodes | 
|  | - snapshot/revert libvirt nodes (fuel-devops only) | 
|  | - block/unblock libvirt  networks/interfaces (fuel-devops only) | 
|  |  | 
|  | This fixture should get a hardware configuration from | 
|  | 'config' object or create a virtual/baremetal underlay | 
|  | using EnvironmentManager. | 
|  |  | 
|  | Creates a snapshot 'hardware' with ready-to-use virtual environment | 
|  | (Only for config.hardware.env_manager='devops'): | 
|  | - just created virtual nodes in power-on state | 
|  | - node volumes filled with necessary content | 
|  | - node network interfaces connected to necessary devices | 
|  |  | 
|  | config.hardware.env_manager: one of ('devops', 'maas', None) | 
|  | config.hardware.config: path to the config file for the env_manager | 
|  | config.hardware.current_snapshot = Latest created or reverted snapshot | 
|  |  | 
|  | :rtype EnvironmentModel: if config.hardware.env_manager == 'devops' | 
|  | :rtype EnvironmentManagerEmpty: | 
|  | if config.hardware.env_manager == 'empty' | 
|  | """ | 
|  | env = None | 
|  |  | 
|  | env_manager = config.hardware.env_manager | 
|  |  | 
|  | if env_manager == 'empty': | 
|  | # No environment manager is used. | 
|  | # 'config' should contain config.underlay.ssh settings | 
|  | # 'config' should contain config.underlay.current_snapshot setting | 
|  | from tcp_tests.managers import envmanager_empty | 
|  | env = envmanager_empty.EnvironmentManagerEmpty(config=config) | 
|  |  | 
|  | elif env_manager == 'devops': | 
|  | # fuel-devops environment manager is used. | 
|  | # config.underlay.ssh settings can be empty or witn SSH to existing env | 
|  | # config.underlay.current_snapshot | 
|  | from tcp_tests.managers import envmanager_devops | 
|  | env = envmanager_devops.EnvironmentManager(config=config) | 
|  |  | 
|  | elif env_manager == 'heat': | 
|  | # heat environment manager is used. | 
|  | # config.underlay.ssh settings can be empty or witn SSH to existing env | 
|  | # config.underlay.current_snapshot | 
|  | from tcp_tests.managers import envmanager_heat | 
|  | env = envmanager_heat.EnvironmentManagerHeat(config=config) | 
|  | else: | 
|  | raise Exception("Unknown hardware manager: '{}'".format(env_manager)) | 
|  |  | 
|  | # for devops env_manager: power on nodes and wait for SSH | 
|  | # for empty env_manager: do nothing | 
|  | # for maas env_manager: provision nodes and wait for SSH | 
|  | if not env.has_snapshot(ext.SNAPSHOT.hardware): | 
|  | env.create_snapshot(ext.SNAPSHOT.hardware) | 
|  |  | 
|  | def fin(): | 
|  | if settings.SHUTDOWN_ENV_ON_TEARDOWN: | 
|  | LOG.info("Shutdown environment...") | 
|  | env.stop() | 
|  |  | 
|  | request.addfinalizer(fin) | 
|  | return env | 
|  |  | 
|  |  | 
|  | @pytest.fixture(scope='function') | 
|  | def revert_snapshot(request, hardware): | 
|  | """Revert snapshot for the test case | 
|  |  | 
|  | Usage: | 
|  | @pytest.mark.revert_snapshot(name='<required_snapshot_name>') | 
|  |  | 
|  | If the mark 'revert_snapshot' is absend, or <required_snapshot_name> | 
|  | not found, then an initial 'hardware' snapshot will be reverted. | 
|  |  | 
|  | :rtype string: name of the reverted snapshot or None | 
|  | """ | 
|  | top_fixtures_snapshots = utils.get_top_fixtures_marks( | 
|  | request, 'revert_snapshot') | 
|  |  | 
|  | # Try to revert the best matches snapshot for the test | 
|  | for snapshot_name in top_fixtures_snapshots: | 
|  | if hardware.has_snapshot(snapshot_name) and \ | 
|  | hardware.has_snapshot_config(snapshot_name): | 
|  | hardware.revert_snapshot(snapshot_name) | 
|  | return snapshot_name | 
|  |  | 
|  | # Fallback to the basic snapshot | 
|  | hardware.revert_snapshot(ext.SNAPSHOT.hardware) | 
|  | return None | 
|  |  | 
|  |  | 
|  | @pytest.fixture(scope='function') | 
|  | def snapshot(request, hardware): | 
|  | """Fixture for creating snapshot at the end of test if it's needed | 
|  |  | 
|  | Marks: | 
|  | snapshot_needed(name=None) - make snapshot if test is passed. If | 
|  | name argument provided, it will be used for creating snapshot, | 
|  | otherwise, test function name will be used | 
|  |  | 
|  | fail_snapshot - make snapshot if test failed | 
|  |  | 
|  | :param request: pytest.python.FixtureRequest | 
|  | :param env: envmanager.EnvironmentManager | 
|  | """ | 
|  | snapshot_needed = request.keywords.get('snapshot_needed', None) | 
|  | fail_snapshot = request.keywords.get('fail_snapshot', None) | 
|  |  | 
|  | def test_fin(): | 
|  | default_snapshot_name = getattr(request.node.function, | 
|  | '_snapshot_name', | 
|  | request.node.function.__name__) | 
|  | if hasattr(request.node, 'rep_call') and request.node.rep_call.passed \ | 
|  | and snapshot_needed: | 
|  | snapshot_name = utils.extract_name_from_mark(snapshot_needed) or \ | 
|  | "{}_passed".format(default_snapshot_name) | 
|  | hardware.create_snapshot(snapshot_name, force=True) | 
|  |  | 
|  | elif hasattr(request.node, 'rep_setup') and \ | 
|  | request.node.rep_setup.failed and fail_snapshot: | 
|  | snapshot_name = "{0}_prep_failed".format(default_snapshot_name) | 
|  | hardware.create_snapshot(snapshot_name, force=True) | 
|  |  | 
|  | elif hasattr(request.node, 'rep_call') and \ | 
|  | request.node.rep_call.failed and fail_snapshot: | 
|  | snapshot_name = "{0}_failed".format(default_snapshot_name) | 
|  | hardware.create_snapshot(snapshot_name, force=True) | 
|  |  | 
|  | request.addfinalizer(test_fin) | 
|  |  | 
|  |  | 
|  | @pytest.fixture(scope="function") | 
|  | def underlay_actions(config): | 
|  | """Fixture that provides SSH access to underlay objects. | 
|  |  | 
|  | :param config: oslo_config object that keeps various parameters | 
|  | across the fixtures, tests and test runs. | 
|  | All SSH data is taken from the provided config. | 
|  | :rtype UnderlaySSHManager: Object that encapsulate SSH credentials; | 
|  | - provide list of underlay nodes; | 
|  | - provide SSH access to underlay nodes using | 
|  | node names or node IPs. | 
|  | """ | 
|  | return underlay_ssh_manager.UnderlaySSHManager(config) | 
|  |  | 
|  |  | 
|  | @pytest.mark.revert_snapshot(ext.SNAPSHOT.underlay) | 
|  | @pytest.fixture(scope="function") | 
|  | def underlay(request, revert_snapshot, config, hardware, underlay_actions): | 
|  | """Fixture that bootstraps the environment underlay. | 
|  |  | 
|  | - Starts the 'hardware' environment and creates 'underlay' with required | 
|  | configuration. | 
|  | - Fills the following object using the 'hardware' fixture: | 
|  | config.underlay.ssh = JSONList of SSH access credentials for nodes. | 
|  | This list will be used for initialization the | 
|  | model UnderlaySSHManager, see it for details. | 
|  |  | 
|  | :rtype UnderlaySSHManager: Object that encapsulate SSH credentials; | 
|  | - provide list of underlay nodes; | 
|  | - provide SSH access to underlay nodes using | 
|  | node names or node IPs. | 
|  | """ | 
|  | # Create Underlay | 
|  |  | 
|  | def basic_underlay(): | 
|  | # If config.underlay.ssh wasn't provided from external config, then | 
|  | # try to get necessary data from hardware env_manager (fuel-devops) | 
|  |  | 
|  | # for devops env_manager: power on nodes and wait for SSH | 
|  | # for empty env_manager: do nothing | 
|  | # for maas env_manager: provision nodes and wait for SSH | 
|  | hardware.start(underlay_node_roles=config.underlay.roles, | 
|  | timeout=config.underlay.bootstrap_timeout) | 
|  |  | 
|  | config.underlay.ssh = hardware.get_ssh_data( | 
|  | roles=config.underlay.roles) | 
|  |  | 
|  | LOG.info("Config - {}".format(config)) | 
|  | underlay_actions.add_config_ssh(config.underlay.ssh) | 
|  |  | 
|  | hardware.create_snapshot(ext.SNAPSHOT.underlay) | 
|  |  | 
|  | return underlay_actions | 
|  |  | 
|  | def day1_underlay(): | 
|  | hardware.start( | 
|  | underlay_node_roles=['salt_master'], | 
|  | timeout=config.underlay.bootstrap_timeout) | 
|  |  | 
|  | config.underlay.ssh = hardware.get_ssh_data( | 
|  | roles=config.underlay.roles) | 
|  | underlay_actions.add_config_ssh(config.underlay.ssh) | 
|  |  | 
|  | LOG.info("Generate MACs for MaaS") | 
|  | macs = { | 
|  | n.name.split('.')[0]: { | 
|  | "interface": { | 
|  | "mac": n.get_interface_by_network_name('admin').mac_address},  # noqa | 
|  | "power_parameters": { | 
|  | "power_address": "{}:{}".format( | 
|  | n.get_interface_by_network_name('admin').l2_network_device.address_pool.get_ip('l2_network_device'),  # noqa | 
|  | n.bmc_port | 
|  | )}} for n in hardware.slave_nodes} | 
|  |  | 
|  | config.day1_cfg_config.maas_machines_macs = { | 
|  | "parameters": { | 
|  | "maas": { | 
|  | "region": { | 
|  | "machines": macs}}}} | 
|  |  | 
|  | for node in hardware.slave_nodes: | 
|  | # For correct comissioning by MaaS nodes should be powered off | 
|  | node.destroy() | 
|  |  | 
|  | hardware.create_snapshot(ext.SNAPSHOT.underlay) | 
|  |  | 
|  | return underlay_actions | 
|  |  | 
|  | if not config.underlay.ssh: | 
|  | if request.node.get_marker('day1_underlay'): | 
|  | return day1_underlay() | 
|  | else: | 
|  | return basic_underlay() | 
|  | else: | 
|  | # 1. hardware environment created and powered on | 
|  | # 2. config.underlay.ssh contains SSH access to provisioned nodes | 
|  | #    (can be passed from external config with TESTS_CONFIGS variable) | 
|  | return underlay_actions | 
|  |  | 
|  |  | 
|  | @pytest.fixture(scope='function') | 
|  | def grab_versions(request, func_name, underlay): | 
|  | """Fixture for grab package versions at the end of test | 
|  |  | 
|  | Marks: | 
|  | grab_versions(name=None) - make snapshot if test is passed. If | 
|  | name argument provided, it will be used for creating data, | 
|  | otherwise, test function name will be used | 
|  |  | 
|  | """ | 
|  | grab_version = request.keywords.get('grab_versions', None) | 
|  |  | 
|  | def test_fin(): | 
|  | fixture_failed = (hasattr(request.node, 'rep_setup') and | 
|  | request.node.rep_setup.failed) | 
|  | test_passed = (hasattr(request.node, 'rep_call') and | 
|  | request.node.rep_call.passed) | 
|  | test_failed = (hasattr(request.node, 'rep_call') and | 
|  | request.node.rep_call.failed) | 
|  |  | 
|  | if fixture_failed or test_passed or test_failed: | 
|  | artifact_name = utils.extract_name_from_mark(grab_version) or \ | 
|  | "{}".format(func_name) | 
|  | underlay.get_logs(artifact_name) | 
|  |  | 
|  | if grab_version: | 
|  | request.addfinalizer(test_fin) |