blob: c132294eaaa0dd5bdaab7b9aae8af7a0a187d788 [file] [log] [blame]
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +03001import pytest
2import json
3import utils
Hanna Arhipova1eef8312019-05-06 20:14:18 +03004import logging
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +03005
Ekaterina Chernova77594f72019-07-29 17:09:12 +03006def is_deb_in_exception(inconsistency_rule, package_name, error_node_list):
Ekaterina Chernova9c232f92019-07-26 16:02:18 +03007 short_names_in_error_nodes = [n.split('.')[0] for n in error_node_list]
8 for node, excluded_packages in inconsistency_rule.iteritems():
9 if package_name in excluded_packages and node in short_names_in_error_nodes:
10 return True
11 return False
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030012
Oleksii Zhurba23c18332019-05-09 18:53:40 -050013@pytest.mark.full
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030014def test_check_package_versions(local_salt_client, nodes_in_group):
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030015 """Validate package has same versions on all the nodes.
16 Steps:
17 1) Collect packages for nodes_in_group
18 "salt -C '<group_of_nodes>' cmd.run 'lowpkg.list_pkgs'"
19 2) Exclude nodes without packages and exceptions
Ekaterina Chernova77594f72019-07-29 17:09:12 +030020 3) Go through each package and save it with version from each node to the
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030021 list. Mark 'No version' if package is not found.
22 If pachage name in the eception list or in inconsistency_rule, ignore it.
23 4) Compare items in that list - they should be equal and match the amout of nodes
24
25 """
Ekaterina Chernova77594f72019-07-29 17:09:12 +030026 # defines packages specific to the concrete nodes
27 inconsistency_rule = {"kvm03": ["rsync", "sysstat", "xz-utils"], "log01": ["python-elasticsearch"], "ctl01": ["python-gnocchiclient", "python-ujson"]}
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030028 exclude_packages = utils.get_configuration().get("skipped_packages", [])
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020029 group, nodes = nodes_in_group
30 packages_versions = local_salt_client.cmd(tgt="L@"+','.join(nodes),
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030031 fun='lowpkg.list_pkgs',
32 expr_form='compound')
33 # Let's exclude cid01 and dbs01 nodes from this check
34 exclude_nodes = local_salt_client.test_ping(tgt="I@galera:master or I@gerrit:client",
35 expr_form='compound').keys()
Oleksii Zhurbaa3e79ce2019-06-04 17:26:51 -050036 # PROD-30833
37 gtw01 = local_salt_client.pillar_get(
38 param='_param:openstack_gateway_node01_hostname') or 'gtw01'
39 cluster_domain = local_salt_client.pillar_get(
40 param='_param:cluster_domain') or '.local'
41 gtw01 += '.' + cluster_domain
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020042 if gtw01 in nodes:
43 octavia = local_salt_client.cmd(tgt="L@" + ','.join(nodes),
Oleksii Zhurba29baebd2019-06-12 16:11:40 -050044 fun='pillar.get',
45 param='octavia:manager:enabled',
46 expr_form='compound')
47 gtws = [gtw for gtw in octavia.values() if gtw]
48 if len(gtws) == 1:
Oleksii Zhurbaa3e79ce2019-06-04 17:26:51 -050049 exclude_nodes.append(gtw01)
50 logging.info("gtw01 node is skipped in test_check_package_versions")
51
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020052 total_nodes = [i for i in nodes if i not in exclude_nodes]
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030053 if len(total_nodes) < 2:
54 pytest.skip("Nothing to compare - only 1 node")
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030055 nodes_with_packages = []
56 packages_with_different_versions = []
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030057 packages_names = set()
58
59 for node in total_nodes:
60 if not packages_versions[node]:
61 # TODO: do not skip node
Hanna Arhipova1eef8312019-05-06 20:14:18 +030062 logging.warning("Node {} is skipped".format(node))
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030063 continue
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030064 nodes_with_packages.append(node)
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030065 packages_names.update(packages_versions[node].keys())
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030066 for deb in packages_names:
67 if deb in exclude_packages:
68 continue
69 diff = []
70 row = []
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030071 for node in nodes_with_packages:
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030072 if not packages_versions[node]:
73 continue
74 if deb in packages_versions[node].keys():
75 diff.append(packages_versions[node][deb])
76 row.append("{}: {}".format(node, packages_versions[node][deb]))
77 else:
78 row.append("{}: No package".format(node))
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030079
80 if diff.count(diff[0]) < len(nodes_with_packages):
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020081 if not is_deb_in_exception(inconsistency_rule, deb, row):
Ekaterina Chernova9c232f92019-07-26 16:02:18 +030082 row.sort()
83 row.insert(0, deb)
84 packages_with_different_versions.append(row)
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020085 assert len(packages_with_different_versions) == 0, (
86 "Non-uniform package versions are installed on '{}' group of nodes:\n"
87 "{}".format(
88 group, json.dumps(packages_with_different_versions, indent=4))
89 )
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030090
91
Oleksii Zhurba23c18332019-05-09 18:53:40 -050092@pytest.mark.full
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +030093def test_packages_are_latest(local_salt_client, nodes_in_group):
94 config = utils.get_configuration()
95 skip = config.get("test_packages")["skip_test"]
96 if skip:
97 pytest.skip("Test for the latest packages is disabled")
98 skipped_pkg = config.get("test_packages")["skipped_packages"]
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +020099 group, nodes = nodes_in_group
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300100 info_salt = local_salt_client.cmd(
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200101 tgt='L@' + ','.join(nodes),
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300102 param='apt list --upgradable 2>/dev/null | grep -v Listing',
103 expr_form='compound')
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200104 for node in nodes:
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300105 result = []
106 if info_salt[node]:
107 upg_list = info_salt[node].split('\n')
108 for i in upg_list:
109 if i.split('/')[0] not in skipped_pkg:
110 result.append(i)
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200111 assert not result, (
112 "Packages are not of latest version on '{}' node:\n{}".format(
113 node, "\n".join(result))
114 )
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300115
116
Oleksii Zhurba23c18332019-05-09 18:53:40 -0500117@pytest.mark.full
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300118def test_check_module_versions(local_salt_client, nodes_in_group):
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300119 # defines modules specific to the concrete nodes
120 inconsistency_rule = {"ctl01": ["gnocchiclient", "ujson"], "log01": ["elasticsearch"]}
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300121 exclude_modules = utils.get_configuration().get("skipped_modules", [])
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200122 group, nodes = nodes_in_group
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300123 pre_check = local_salt_client.cmd(
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200124 tgt="L@"+','.join(nodes),
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300125 param='dpkg -l | grep "python-pip "',
126 expr_form='compound')
127 if pre_check.values().count('') > 0:
128 pytest.skip("pip is not installed on one or more nodes")
129
130 exclude_nodes = local_salt_client.test_ping(tgt="I@galera:master or I@gerrit:client",
131 expr_form='compound').keys()
Oleksii Zhurbaa3e79ce2019-06-04 17:26:51 -0500132
133 # PROD-30833
134 gtw01 = local_salt_client.pillar_get(
135 param='_param:openstack_gateway_node01_hostname') or 'gtw01'
136 cluster_domain = local_salt_client.pillar_get(
137 param='_param:cluster_domain') or '.local'
138 gtw01 += '.' + cluster_domain
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200139 if gtw01 in nodes:
140 octavia = local_salt_client.cmd(tgt="L@" + ','.join(nodes),
Oleksii Zhurba29baebd2019-06-12 16:11:40 -0500141 fun='pillar.get',
142 param='octavia:manager:enabled',
143 expr_form='compound')
144 gtws = [gtw for gtw in octavia.values() if gtw]
145 if len(gtws) == 1:
Oleksii Zhurbaa3e79ce2019-06-04 17:26:51 -0500146 exclude_nodes.append(gtw01)
147 logging.info("gtw01 node is skipped in test_check_module_versions")
148
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300149 total_nodes = [i for i in pre_check.keys() if i not in exclude_nodes]
150
151 if len(total_nodes) < 2:
152 pytest.skip("Nothing to compare - only 1 node")
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200153 list_of_pip_packages = local_salt_client.cmd(
154 tgt="L@"+','.join(nodes),
155 fun='pip.freeze', expr_form='compound')
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300156
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300157 nodes_with_packages = []
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300158
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300159 modules_with_different_versions = []
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300160 packages_names = set()
161
162 for node in total_nodes:
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300163 nodes_with_packages.append(node)
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300164 packages_names.update([x.split("=")[0] for x in list_of_pip_packages[node]])
165 list_of_pip_packages[node] = dict([x.split("==") for x in list_of_pip_packages[node]])
166
167 for deb in packages_names:
168 if deb in exclude_modules:
169 continue
170 diff = []
171 row = []
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300172 for node in nodes_with_packages:
Hanna Arhipovae6ed8e42019-05-15 16:27:08 +0300173 if deb in list_of_pip_packages[node].keys():
174 diff.append(list_of_pip_packages[node][deb])
175 row.append("{}: {}".format(node, list_of_pip_packages[node][deb]))
176 else:
177 row.append("{}: No module".format(node))
Ekaterina Chernova77594f72019-07-29 17:09:12 +0300178 if diff.count(diff[0]) < len(nodes_with_packages):
179 if not is_deb_in_exception(inconsistency_rule, deb, row):
180 row.sort()
181 row.insert(0, deb)
182 modules_with_different_versions.append(row)
Dmitriy Kruglovbc0a88b2019-08-20 11:45:35 +0200183 assert len(modules_with_different_versions) == 0, (
184 "Non-uniform pip modules are installed on '{}' group of nodes:\n"
185 "{}".format(
186 group, json.dumps(modules_with_different_versions, indent=4))
187 )