| import pytest |
| import sys |
| import os |
| |
| from tcp_tests import logger |
| from tcp_tests import settings |
| |
| sys.path.append(os.getcwd()) |
| try: |
| from tcp_tests.fixtures import config_fixtures |
| from tcp_tests.managers import underlay_ssh_manager |
| from tcp_tests.managers import saltmanager as salt_manager |
| except ImportError: |
| print("ImportError: Run the application from the tcp-qa directory or " |
| "set the PYTHONPATH environment variable to directory which contains" |
| " ./tcp_tests") |
| sys.exit(1) |
| LOG = logger.logger |
| |
| |
| def has_only_similar(values_by_nodes): |
| """ |
| :param values_by_nodes: dict |
| :return: bool, True if all items in the dict have similar values |
| """ |
| values = list(values_by_nodes.values()) |
| return all(value == values[0] for value in values) |
| |
| |
| def get_control_plane_targets(): |
| config = config_fixtures.config() |
| underlay = underlay_ssh_manager.UnderlaySSHManager(config) |
| saltmanager = salt_manager.SaltManager(config, underlay) |
| targets = list() |
| telemetry_exists = False |
| barbican_exists = False |
| try: |
| targets += saltmanager.run_state( |
| "I@keystone:server", 'test.ping')[0]['return'][0].keys() |
| targets += saltmanager.run_state( |
| "I@nginx:server and not I@salt:master", |
| "test.ping")[0]['return'][0].keys() |
| telemetry_exists = saltmanager.get_single_pillar( |
| "I@salt:master", |
| "_param:openstack_telemetry_hostname") |
| barbican_exists = saltmanager.get_single_pillar( |
| "I@salt:master", |
| "_param:barbican_enabled") |
| except BaseException as err: |
| LOG.warning("Can't retrieve data from Salt. \ |
| Maybe cluster is not deployed completely.\ |
| Err: {}".format(err)) |
| |
| # check for Manila existence |
| # if saltmanager.get_single_pillar("I@salt:master", |
| # "_param:manila_service_protocol"): |
| # targets.append('share*') |
| |
| # check for Tenant Telemetry existence |
| if telemetry_exists: |
| targets.append('mdb*') |
| |
| # check for Barbican existence |
| if barbican_exists: |
| targets.append('kmn*') |
| return targets |
| |
| |
| @pytest.fixture(scope='class') |
| def switch_to_proposed_pipelines(reclass_actions, salt_actions): |
| reclass = reclass_actions |
| proposed_repo = "http://mirror.mirantis.com/update/proposed/" |
| repo_param = "parameters._param.linux_system_repo_update_url" |
| |
| proposed_pipeline_branch = "release/proposed/2019.2.0" |
| pipeline_branch_param = "parameters._param.jenkins_pipelines_branch" |
| infra_yml = "cluster/*/infra/init.yml" |
| |
| LOG.info("Check reclass has release/proposed/2019.2.0 branches") |
| if reclass.get_key(pipeline_branch_param, |
| infra_yml) == proposed_pipeline_branch \ |
| and reclass.get_key(repo_param, infra_yml) == proposed_repo: |
| return True |
| |
| LOG.info("Switch to release/proposed/2019.2.0 branches") |
| reclass.add_key(pipeline_branch_param, proposed_pipeline_branch, infra_yml) |
| |
| reclass.add_key(repo_param, proposed_repo, infra_yml) |
| reclass.add_key(repo_param, proposed_repo, "cluster/*/openstack/init.yml") |
| reclass.add_key(repo_param, proposed_repo, "cluster/*/stacklight/init.yml") |
| reclass.add_key(repo_param, proposed_repo, "cluster/*/ceph/init.yml") |
| |
| salt_actions.run_state("*", "saltutil.refresh_pillar") |
| salt_actions.enforce_state("*", "salt.minion") |
| salt_actions.enforce_state("I@jenkins:client", "jenkins.client") |
| |
| |
| @pytest.fixture |
| def wa_for_sphinx_prod34406(reclass_actions, salt_actions): |
| reclass_actions.add_key( |
| "parameters._param.sphinx_proxy_password_generated", |
| "a"*32, |
| "cluster/*/infra/secrets.yml") |
| salt_actions.run_state("I@nginx:server", |
| "saltutil.refresh_pillar") |
| |
| |
| @pytest.fixture(scope='class') |
| def enable_openstack_update(reclass_actions, salt_actions): |
| param = "parameters._param.openstack_upgrade_enabled" |
| context_file = "cluster/*/infra/init.yml" |
| |
| LOG.info("Enable openstack_upgrade_enabled in reclass") |
| reclass_actions.add_bool_key(param, "True", context_file) |
| salt_actions.run_state("*", "saltutil.refresh_pillar") |
| yield True |
| LOG.info("Disable openstack_upgrade_enabled in reclass") |
| reclass_actions.add_bool_key(param, "False", context_file) |
| salt_actions.run_state("*", "saltutil.refresh_pillar") |
| |
| |
| @pytest.mark.usefixtures("switch_to_proposed_pipelines", |
| "wa_for_sphinx_prod34406") |
| class TestUpdateMcpCluster(object): |
| """ |
| Following the steps in |
| https://docs.mirantis.com/mcp/master/mcp-operations-guide/update-upgrade/minor-update.html#minor-update |
| """ |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_drivetrain(self, salt_actions, drivetrain_actions, |
| show_step, _): |
| """Updating DriveTrain component to release/proposed/2019.2.0 version |
| |
| Scenario: |
| 1. Add workaround for PROD-32751 |
| 2. Run job git-mirror-downstream-mk-pipelines |
| 3. Run job git-mirror-downstream-pipeline-library |
| 4. If jobs are passed then start 'Deploy - upgrade MCP Drivetrain' |
| |
| Duration: ~70 min |
| """ |
| salt = salt_actions |
| dt = drivetrain_actions |
| |
| # #################### Add workaround for PROD-32751 ################# |
| show_step(1) |
| |
| # FIXME: workaround for PROD-32751 |
| salt.cmd_run("cfg01*", "cd /srv/salt/reclass; git add -u && \ |
| git commit --allow-empty -m 'Cluster model update'") |
| |
| # ################### Downstream mk-pipelines ######################### |
| show_step(2) |
| job_name = 'git-mirror-downstream-mk-pipelines' |
| job_parameters = { |
| 'BRANCHES': 'release/proposed/2019.2.0' |
| } |
| update_pipelines = dt.start_job_on_jenkins( |
| job_name=job_name, |
| job_parameters=job_parameters) |
| |
| assert update_pipelines == 'SUCCESS' |
| |
| # ################### Downstream pipeline-library #################### |
| show_step(3) |
| job_name = 'git-mirror-downstream-pipeline-library' |
| job_parameters = { |
| 'BRANCHES': 'release/proposed/2019.2.0' |
| } |
| update_pipeline_library = dt.start_job_on_jenkins( |
| job_name=job_name, |
| job_parameters=job_parameters) |
| |
| assert update_pipeline_library == 'SUCCESS' |
| |
| # ################### Start 'Deploy - upgrade MCP Drivetrain' job ##### |
| show_step(4) |
| |
| job_name = 'upgrade-mcp-release' |
| job_parameters = { |
| 'GIT_REFSPEC': 'release/proposed/2019.2.0', |
| 'MK_PIPELINES_REFSPEC': 'release/proposed/2019.2.0', |
| 'TARGET_MCP_VERSION': '2019.2.0' |
| } |
| update_drivetrain = dt.start_job_on_jenkins( |
| job_name=job_name, |
| job_parameters=job_parameters, |
| build_timeout=3 * 60 * 60) |
| |
| assert update_drivetrain == 'SUCCESS' |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_glusterfs(self, salt_actions, reclass_actions, |
| drivetrain_actions, show_step, _): |
| """ Upgrade GlusterFS |
| Scenario: |
| 1. In infra/init.yml in Reclass, add the glusterfs_version parameter |
| 2. Start linux.system.repo state |
| 3. Start "update-glusterfs" job |
| 4. Check version for GlusterFS servers |
| 5. Check version for GlusterFS clients |
| |
| """ |
| salt = salt_actions |
| reclass = reclass_actions |
| dt = drivetrain_actions |
| |
| # ############## Change reclass ###################################### |
| show_step(1) |
| reclass.add_key( |
| "parameters._param.linux_system_repo_mcp_glusterfs_version_number", |
| "5", |
| "cluster/*/infra/init.yml" |
| ) |
| # ################# Run linux.system state ########################### |
| show_step(2) |
| salt.enforce_state("*", "linux.system.repo") |
| |
| # ############## Start deploy-upgrade-galera job ##################### |
| show_step(3) |
| job_name = 'update-glusterfs' |
| |
| update_glusterfs = dt.start_job_on_jenkins( |
| job_name=job_name, |
| build_timeout=40 * 60) |
| |
| assert update_glusterfs == 'SUCCESS' |
| |
| # ################ Check GlusterFS version for servers ############## |
| show_step(4) |
| gluster_server_versions_by_nodes = salt.cmd_run( |
| "I@glusterfs:server", |
| "glusterd --version|head -n1")[0] |
| |
| assert has_only_similar(gluster_server_versions_by_nodes), \ |
| gluster_server_versions_by_nodes |
| |
| # ################ Check GlusterFS version for clients ############## |
| show_step(5) |
| gluster_client_versions_by_nodes = salt.cmd_run( |
| "I@glusterfs:client", |
| "glusterfs --version|head -n1")[0] |
| |
| assert has_only_similar(gluster_client_versions_by_nodes), \ |
| gluster_client_versions_by_nodes |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_galera(self, salt_actions, reclass_actions, |
| drivetrain_actions, show_step, _): |
| """ Upgrade Galera automatically |
| |
| Scenario: |
| 1. Include the Galera upgrade pipeline job to DriveTrain |
| 2. Apply the jenkins.client state on the Jenkins nodes |
| 3. set the openstack_upgrade_enabled parameter to true |
| 4. Refresh pillars |
| 5. Add repositories with new Galera packages |
| 6. Start job from Jenkins |
| """ |
| salt = salt_actions |
| reclass = reclass_actions |
| dt = drivetrain_actions |
| # ################### Enable pipeline ################################# |
| show_step(1) |
| reclass.add_class( |
| "system.jenkins.client.job.deploy.update.upgrade_galera", |
| "cluster/*/cicd/control/leader.yml") |
| show_step(2) |
| salt.enforce_state("I@jenkins:client", "jenkins.client") |
| |
| # ############### Enable automatic upgrade ############################ |
| show_step(3) |
| reclass.add_bool_key("parameters._param.openstack_upgrade_enabled", |
| "True", |
| "cluster/*/infra/init.yml") |
| |
| show_step(4) |
| salt.run_state("dbs*", "saltutil.refresh_pillar") |
| |
| # ############# Add repositories with new Galera packages ####### |
| show_step(5) |
| salt.enforce_state("dbs*", "linux.system.repo") |
| salt.enforce_state("cfg*", "salt.master") |
| |
| # #################### Login Jenkins on cid01 node ################### |
| show_step(6) |
| |
| job_name = 'deploy-upgrade-galera' |
| job_parameters = { |
| 'INTERACTIVE': 'false' |
| } |
| |
| update_galera = dt.start_job_on_jenkins( |
| job_name=job_name, |
| job_parameters=job_parameters, |
| build_timeout=40 * 60) |
| |
| assert update_galera == 'SUCCESS' |
| |
| @pytest.fixture |
| def disable_automatic_failover_neutron_for_test(self, salt_actions): |
| """ |
| On each OpenStack controller node, modify the neutron.conf file |
| Restart the neutron-server service |
| """ |
| |
| def comment_line(node, file_name, word): |
| """ |
| Adds '#' before the specific line in specific file |
| |
| :param node: string, salt target of node where the file locates |
| :param file_name: string, full path to the file |
| :param word: string, the begin of line which should be commented |
| :return: None |
| """ |
| salt_actions.cmd_run(node, |
| "sed -i 's/^{word}/#{word}/' {file}". |
| format(word=word, |
| file=file_name)) |
| |
| def add_line(node, file_name, line): |
| """ |
| Appends line to the end of file |
| |
| :param node: string, salt target of node where the file locates |
| :param file_name: string, full path to the file |
| :param line: string, line that should be added |
| :return: None |
| """ |
| salt_actions.cmd_run(node, "echo {line} >> {file}".format( |
| line=line, |
| file=file_name)) |
| |
| neutron_conf = '/etc/neutron/neutron.conf' |
| neutron_server = "I@neutron:server" |
| # ######## Create backup for config file ####################### |
| salt_actions.cmd_run( |
| neutron_server, |
| "cp -p {file} {file}.backup".format(file=neutron_conf)) |
| |
| # ## Change parameters in neutron.conf' |
| comment_line(neutron_server, neutron_conf, |
| "allow_automatic_l3agent_failover", ) |
| comment_line(neutron_server, neutron_conf, |
| "allow_automatic_dhcp_failover") |
| add_line(neutron_server, neutron_conf, |
| "allow_automatic_dhcp_failover = false") |
| add_line(neutron_server, neutron_conf, |
| "allow_automatic_l3agent_failover = false") |
| |
| # ## Apply changed config to the neutron-server service |
| result = salt_actions.cmd_run(neutron_server, |
| "service neutron-server restart") |
| # TODO: add check that neutron-server is up and running |
| yield result |
| # ## Revert file changes |
| salt_actions.cmd_run( |
| neutron_server, |
| "cp -p {file}.backup {file}".format(file=neutron_conf)) |
| salt_actions.cmd_run(neutron_server, |
| "service neutron-server restart") |
| |
| @pytest.fixture |
| def disable_neutron_agents_for_test(self, salt_actions): |
| """ |
| Disable the neutron services before the test and |
| enable it after test |
| """ |
| result = salt_actions.cmd_run("I@neutron:server", """ |
| service neutron-dhcp-agent stop && \ |
| service neutron-l3-agent stop && \ |
| service neutron-metadata-agent stop && \ |
| service neutron-openvswitch-agent stop |
| """) |
| yield result |
| # |
| salt_actions.cmd_run("I@neutron:server", """ |
| service neutron-dhcp-agent start && \ |
| service neutron-l3-agent start && \ |
| service neutron-metadata-agent start && \ |
| service neutron-openvswitch-agent start |
| """) |
| # TODO: add check that all services are UP and running |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_rabbit(self, salt_actions, reclass_actions, |
| drivetrain_actions, show_step, _, |
| disable_automatic_failover_neutron_for_test, |
| disable_neutron_agents_for_test): |
| """ Updates RabbitMQ |
| Scenario: |
| 1. Include the RabbitMQ upgrade pipeline job to DriveTrain |
| 2. Add repositories with new RabbitMQ packages |
| 3. Start Deploy - upgrade RabbitMQ pipeline |
| |
| Updating RabbitMq should be completed before the OpenStack updating |
| process starts |
| """ |
| salt = salt_actions |
| reclass = reclass_actions |
| dt = drivetrain_actions |
| |
| # ####### Include the RabbitMQ upgrade pipeline job to DriveTrain #### |
| show_step(1) |
| reclass.add_class( |
| "system.jenkins.client.job.deploy.update.upgrade_rabbitmq", |
| "cluster/*/cicd/control/leader.yml") |
| salt.enforce_state("I@jenkins:client", "jenkins.client") |
| |
| reclass.add_bool_key("parameters._param.openstack_upgrade_enabled", |
| "True", |
| "cluster/*/infra/init.yml") |
| salt.run_state("I@rabbitmq:server", "saltutil.refresh_pillar") |
| |
| # ########### Add repositories with new RabbitMQ packages ############ |
| show_step(2) |
| salt.enforce_state("I@rabbitmq:server", "linux.system.repo") |
| |
| # ########### Start Deploy - upgrade RabbitMQ pipeline ############ |
| show_step(3) |
| job_parameters = { |
| 'INTERACTIVE': 'false' |
| } |
| |
| update_rabbit = dt.start_job_on_jenkins( |
| job_name='deploy-upgrade-rabbitmq', |
| job_parameters=job_parameters, |
| build_timeout=40 * 60 |
| ) |
| assert update_rabbit == 'SUCCESS' |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_ceph(self, salt_actions, drivetrain_actions, show_step, _): |
| """ Updates Ceph to the latest minor version |
| |
| Scenario: |
| 1. Add workaround for unhealth Ceph |
| 2. Start ceph-upgrade job with default parameters |
| 3. Check Ceph version for all nodes |
| |
| https://docs.mirantis.com/mcp/master/mcp-operations-guide/update-upgrade/minor-update/ceph-update.html |
| """ |
| salt = salt_actions |
| dt = drivetrain_actions |
| |
| # ###################### Add workaround for unhealth Ceph ############ |
| show_step(1) |
| salt.cmd_run("I@ceph:radosgw", |
| "ceph config set 'mon pg warn max object skew' 20") |
| # ###################### Start ceph-upgrade pipeline ################# |
| show_step(2) |
| job_parameters = {} |
| |
| update_ceph = dt.start_job_on_jenkins( |
| job_name='ceph-update', |
| job_parameters=job_parameters) |
| |
| assert update_ceph == 'SUCCESS' |
| |
| # ########## Verify Ceph version ##################################### |
| show_step(3) |
| |
| ceph_version_by_nodes = salt.cmd_run( |
| "I@ceph:* and not I@ceph:monitoring and not I@ceph:backup:server", |
| "ceph version")[0] |
| |
| assert has_only_similar(ceph_version_by_nodes), ceph_version_by_nodes |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize("_", [settings.ENV_NAME]) |
| @pytest.mark.run_mcp_update |
| def test_update_stacklight(self, _, drivetrain_actions): |
| """ Update packages for Stacklight |
| Scenario: |
| 1. Start Deploy - upgrade Stacklight job |
| """ |
| drivetrain = drivetrain_actions |
| |
| job_parameters = { |
| "STAGE_UPGRADE_DOCKER_COMPONENTS": True, |
| "STAGE_UPGRADE_ES_KIBANA": True, |
| "STAGE_UPGRADE_SYSTEM_PART": True |
| } |
| upgrade_control_pipeline = drivetrain.start_job_on_jenkins( |
| job_name="stacklight-upgrade", |
| job_parameters=job_parameters) |
| |
| assert upgrade_control_pipeline == 'SUCCESS' |
| |
| |
| @pytest.mark.usefixtures("switch_to_proposed_pipelines", |
| "enable_openstack_update") |
| class TestOpenstackUpdate(object): |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.run_mcp_update |
| def test__pre_update__enable_pipeline_job(self, |
| reclass_actions, salt_actions, |
| show_step): |
| """ Enable pipeline in the Drivetrain |
| |
| Scenario: |
| 1. Add deploy.update.* classes to the reclass |
| 2. Start jenkins.client salt state |
| |
| """ |
| salt = salt_actions |
| reclass = reclass_actions |
| show_step(1) |
| reclass.add_class("system.jenkins.client.job.deploy.update.upgrade", |
| "cluster/*/cicd/control/leader.yml") |
| |
| reclass.add_class( |
| "system.jenkins.client.job.deploy.update.upgrade_ovs_gateway", |
| "cluster/*/cicd/control/leader.yml") |
| |
| reclass.add_class( |
| "system.jenkins.client.job.deploy.update.upgrade_compute", |
| "cluster/*/cicd/control/leader.yml") |
| |
| show_step(2) |
| r, errors = salt.enforce_state("I@jenkins:client", "jenkins.client") |
| assert errors is None |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.parametrize('target', get_control_plane_targets()) |
| @pytest.mark.run_mcp_update |
| def test__update__control_plane(self, drivetrain_actions, target): |
| """Start 'Deploy - upgrade control VMs' for specific node |
| """ |
| job_parameters = { |
| "TARGET_SERVERS": target, |
| "OS_DIST_UPGRADE": True, |
| "UPGRADE_SALTSTACK": False, |
| "OS_UPGRADE": True, |
| "INTERACTIVE": False} |
| upgrade_control_pipeline = drivetrain_actions.start_job_on_jenkins( |
| job_name="deploy-upgrade-control", |
| job_parameters=job_parameters) |
| |
| assert upgrade_control_pipeline == 'SUCCESS' |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.run_mcp_update |
| def test__update__data_plane(self, drivetrain_actions, salt_actions): |
| """Start 'Deploy - upgrade OVS gateway' |
| """ |
| if not salt_actions.cmd_run("gtw*", "test.ping")[0].keys(): |
| pytest.skip("This deployment doesn't have gtw* nodes") |
| job_parameters = { |
| "OS_DIST_UPGRADE": True, |
| "OS_UPGRADE": True, |
| "INTERACTIVE": False} |
| upgrade_data_pipeline = drivetrain_actions.start_job_on_jenkins( |
| job_name="deploy-upgrade-ovs-gateway", |
| job_parameters=job_parameters) |
| |
| assert upgrade_data_pipeline == 'SUCCESS' |
| |
| @pytest.mark.grab_versions |
| @pytest.mark.run_mcp_update |
| def test__update__computes(self, drivetrain_actions): |
| """Start 'Deploy - upgrade computes' |
| """ |
| job_parameters = { |
| "OS_DIST_UPGRADE": True, |
| "OS_UPGRADE": True, |
| "INTERACTIVE": False} |
| upgrade_compute_pipeline = drivetrain_actions.start_job_on_jenkins( |
| job_name="deploy-upgrade-compute", |
| job_parameters=job_parameters) |
| |
| assert upgrade_compute_pipeline == 'SUCCESS' |