blob: f72275dfea643f122d592fda9631dc23616ee7de [file] [log] [blame]
import pytest
import json
import utils
import logging
def is_deb_in_exception(inconsistency_rule, package_name, error_node_list):
short_names_in_error_nodes = [n.split('.')[0] for n in error_node_list]
for node, excluded_packages in inconsistency_rule.items():
if package_name in excluded_packages and node in short_names_in_error_nodes:
return True
return False
@pytest.mark.full
def test_check_package_versions(local_salt_client, nodes_in_group):
"""Validate package has same versions on all the nodes.
Steps:
1) Collect packages for nodes_in_group
"salt -C '<group_of_nodes>' cmd.run 'lowpkg.list_pkgs'"
2) Exclude nodes without packages and exceptions
3) Go through each package and save it with version from each node to the
list. Mark 'No version' if package is not found.
If pachage name in the eception list or in inconsistency_rule, ignore it.
4) Compare items in that list - they should be equal and match the amout of nodes
"""
# defines packages specific to the concrete nodes
inconsistency_rule = {"kvm03": ["rsync", "sysstat", "xz-utils"], "log01": ["python-elasticsearch"], "ctl01": ["python-gnocchiclient", "python-ujson"]}
exclude_packages = utils.get_configuration().get("skipped_packages", [])
group, nodes = nodes_in_group
packages_versions = local_salt_client.cmd(tgt="L@"+','.join(nodes),
fun='lowpkg.list_pkgs',
expr_form='compound')
# Let's exclude cid01 and dbs01 nodes from this check
exclude_nodes = list(local_salt_client.test_ping(tgt="I@galera:master or I@gerrit:client",
expr_form='compound').keys())
# PROD-30833
gtw01 = local_salt_client.pillar_get(
param='_param:openstack_gateway_node01_hostname') or 'gtw01'
cluster_domain = local_salt_client.pillar_get(
param='_param:cluster_domain') or '.local'
gtw01 += '.' + cluster_domain
if gtw01 in nodes:
octavia = local_salt_client.cmd(tgt="L@" + ','.join(nodes),
fun='pillar.get',
param='octavia:manager:enabled',
expr_form='compound')
gtws = [gtw for gtw in list(octavia.values()) if gtw]
if len(gtws) == 1:
exclude_nodes.append(gtw01)
logging.info("gtw01 node is skipped in test_check_package_versions")
total_nodes = [i for i in nodes if i not in exclude_nodes]
if len(total_nodes) < 2:
pytest.skip("Nothing to compare - only 1 node")
nodes_with_packages = []
packages_with_different_versions = []
packages_names = set()
for node in total_nodes:
if not packages_versions[node]:
# TODO: do not skip node
logging.warning("Node {} is skipped".format(node))
continue
nodes_with_packages.append(node)
packages_names.update(list(packages_versions[node].keys()))
for deb in packages_names:
if deb in exclude_packages:
continue
diff = []
row = []
for node in nodes_with_packages:
if not packages_versions[node]:
continue
if deb in list(packages_versions[node].keys()):
diff.append(packages_versions[node][deb])
row.append("{}: {}".format(node, packages_versions[node][deb]))
else:
row.append("{}: No package".format(node))
if diff.count(diff[0]) < len(nodes_with_packages):
if not is_deb_in_exception(inconsistency_rule, deb, row):
row.sort()
row.insert(0, deb)
packages_with_different_versions.append(row)
assert len(packages_with_different_versions) == 0, (
"Non-uniform package versions are installed on '{}' group of nodes:\n"
"{}".format(
group, json.dumps(packages_with_different_versions, indent=4))
)
@pytest.mark.full
def test_packages_are_latest(local_salt_client, nodes_in_group):
config = utils.get_configuration()
skip = config.get("test_packages")["skip_test"]
if skip.lower() == 'true':
pytest.skip("Test for the latest packages is disabled")
skipped_pkg = config.get("test_packages")["skipped_packages"]
group, nodes = nodes_in_group
info_salt = local_salt_client.cmd(
tgt='L@' + ','.join(nodes),
param='apt list --upgradable 2>/dev/null | grep -v Listing',
expr_form='compound')
for node in nodes:
result = []
if info_salt[node]:
upg_list = info_salt[node].split('\n')
for i in upg_list:
if i.split('/')[0] not in skipped_pkg:
result.append(i)
assert not result, (
"Packages are not of latest version on '{}' node:\n{}".format(
node, "\n".join(result))
)
@pytest.mark.full
def test_check_module_versions(local_salt_client, nodes_in_group):
# defines modules specific to the concrete nodes
inconsistency_rule = {"ctl01": ["gnocchiclient", "ujson"], "log01": ["elasticsearch"]}
exclude_modules = utils.get_configuration().get("skipped_modules", [])
group, nodes = nodes_in_group
pre_check = local_salt_client.cmd(
tgt="L@"+','.join(nodes),
param='dpkg -l | grep "python-pip "',
expr_form='compound')
if list(pre_check.values()).count('') > 0:
pytest.skip("pip is not installed on one or more nodes")
exclude_nodes = list(local_salt_client.test_ping(tgt="I@galera:master or I@gerrit:client",
expr_form='compound').keys())
# PROD-30833
gtw01 = local_salt_client.pillar_get(
param='_param:openstack_gateway_node01_hostname') or 'gtw01'
cluster_domain = local_salt_client.pillar_get(
param='_param:cluster_domain') or '.local'
gtw01 += '.' + cluster_domain
if gtw01 in nodes:
octavia = local_salt_client.cmd(tgt="L@" + ','.join(nodes),
fun='pillar.get',
param='octavia:manager:enabled',
expr_form='compound')
gtws = [gtw for gtw in list(octavia.values()) if gtw]
if len(gtws) == 1:
exclude_nodes.append(gtw01)
logging.info("gtw01 node is skipped in test_check_module_versions")
total_nodes = [i for i in list(pre_check.keys()) if i not in exclude_nodes]
if len(total_nodes) < 2:
pytest.skip("Nothing to compare - only 1 node")
list_of_pip_packages = local_salt_client.cmd(
tgt="L@"+','.join(nodes),
fun='pip.freeze', expr_form='compound')
nodes_with_packages = []
modules_with_different_versions = []
packages_names = set()
for node in total_nodes:
nodes_with_packages.append(node)
packages_names.update([x.split("=")[0] for x in list_of_pip_packages[node]])
list_of_pip_packages[node] = dict([x.split("==") for x in list_of_pip_packages[node]])
for deb in packages_names:
if deb in exclude_modules:
continue
diff = []
row = []
for node in nodes_with_packages:
if deb in list(list_of_pip_packages[node].keys()):
diff.append(list_of_pip_packages[node][deb])
row.append("{}: {}".format(node, list_of_pip_packages[node][deb]))
else:
row.append("{}: No module".format(node))
if diff.count(diff[0]) < len(nodes_with_packages):
if not is_deb_in_exception(inconsistency_rule, deb, row):
row.sort()
row.insert(0, deb)
modules_with_different_versions.append(row)
assert len(modules_with_different_versions) == 0, (
"Non-uniform pip modules are installed on '{}' group of nodes:\n"
"{}".format(
group, json.dumps(modules_with_different_versions, indent=4))
)