blob: e8339cb185b3bc833aeca0fd4001b93790667c31 [file] [log] [blame]
# Author: Alex Savatieiev (osavatieiev@mirantis.com; a.savex@gmail.com)
# Copyright 2019-2022 Mirantis, Inc.
import json
from cfg_checker.common import const, logger_cli
from cfg_checker.common.exception import ConfigException
from cfg_checker.common.other import merge_dict
from cfg_checker.common.settings import ENV_TYPE_SALT
from cfg_checker.helpers.console_utils import Progress
from cfg_checker.modules.packages.repos import RepoManager
from cfg_checker.nodes import SaltNodes, KubeNodes
from cfg_checker.reports import reporter
from .versions import DebianVersion, PkgVersions, VersionCmpResult
class CloudPackageChecker(object):
def __init__(
self,
config,
force_tag=None,
exclude_keywords=[],
skip_list=None,
skip_list_file=None
):
# check that this env tag is present in Manager
self.env_config = config
self.rm = RepoManager()
self.force_tag = force_tag
self.exclude_keywords = exclude_keywords
# Init salt master info
if not self.master.nodes:
self.master.nodes = self.master.get_nodes(
skip_list=skip_list,
skip_list_file=skip_list_file
)
_tags = self.rm.get_available_tags(tag=self.master.mcp_release)
if not _tags:
logger_cli.warning(
"\n# WARNING: '{0}' is not listed in repo index. "
"Consider running:\n\t{1}\nto add info on this tag's "
"release package versions".format(
self.master.mcp_release,
"mcp-checker packages versions --tag <target_tag>"
)
)
@staticmethod
def presort_packages(all_packages, full=None):
logger_cli.info("-> Presorting packages")
# labels
_data = {}
_data = {
"cs": {
"ok": const.VERSION_OK,
"up": const.VERSION_UP,
"down": const.VERSION_DOWN,
"warn": const.VERSION_WARN,
"err": const.VERSION_ERR
},
"ca": {
"na": const.ACT_NA,
"up": const.ACT_UPGRADE,
"need_up": const.ACT_NEED_UP,
"need_down": const.ACT_NEED_DOWN,
"repo": const.ACT_REPO
}
}
_data['status_err'] = const.VERSION_ERR
_data['status_warn'] = const.VERSION_WARN
_data['status_down'] = const.VERSION_DOWN
_data['status_skip'] = const.VERSION_NA
# Presort packages
_data['critical'] = {}
_data['system'] = {}
_data['other'] = {}
_data['unlisted'] = {}
_l = len(all_packages)
_progress = Progress(_l)
_progress_index = 0
# counters
_ec = _es = _eo = _eu = 0
_wc = _ws = _wo = _wu = 0
_dc = _ds = _do = _du = 0
while _progress_index < _l:
# progress bar
_progress_index += 1
_progress.write_progress(_progress_index)
# sort packages
_pn, _val = all_packages.popitem()
_c = _val['desc']['section']
_rkeys = _val['results'].keys()
if not full:
# Check if this packet has errors
# if all is ok -> just skip it
_max_status = max(_val['results'].keys())
if _max_status <= const.VERSION_OK:
_max_action = max(_val['results'][_max_status].keys())
if _max_action == const.ACT_NA:
# this package does not have any comments
# ...just skip it from report
continue
_differ = len(set(_val['results'].keys())) > 1
if _differ:
# in case package has different status across nodes
# Warning becomes Error.
if const.VERSION_WARN in _val['results']:
if const.VERSION_ERR in _val['results']:
# add warns to err
# should never happen, though
merge_dict(
_val['results'].pop(const.VERSION_WARN),
_val['results'][const.VERSION_ERR]
)
else:
_val['results'][const.VERSION_ERR] = \
_val['results'].pop(const.VERSION_WARN)
else:
# in case package has same status on all nodes
# Error becomes Warning
if const.VERSION_ERR in _val['results']:
if const.VERSION_WARN in _val['results']:
# add warns to err
# should never happen, though
merge_dict(
_val['results'].pop(const.VERSION_ERR),
_val['results'][const.VERSION_WARN]
)
else:
_val['results'][const.VERSION_WARN] = \
_val['results'].pop(const.VERSION_ERR)
if len(_c) > 0 and _val['is_mirantis'] is None:
# not listed package in version lib
_data['unlisted'].update({
_pn: _val
})
_eu += sum(x == const.VERSION_ERR for x in _rkeys)
_wu += sum(x == const.VERSION_WARN for x in _rkeys)
_du += sum(x == const.VERSION_DOWN for x in _rkeys)
# mirantis/critical
# elif len(_c) > 0 and _c != 'System':
elif _val['is_mirantis']:
# not blank and not system
_data['critical'].update({
_pn: _val
})
_ec += sum(x == const.VERSION_ERR for x in _rkeys)
_wc += sum(x == const.VERSION_WARN for x in _rkeys)
_dc += sum(x == const.VERSION_DOWN for x in _rkeys)
# system
elif _c == 'System':
_data['system'].update({
_pn: _val
})
_es += sum(x == const.VERSION_ERR for x in _rkeys)
_ws += sum(x == const.VERSION_WARN for x in _rkeys)
_ds += sum(x == const.VERSION_DOWN for x in _rkeys)
# rest
else:
_data['other'].update({
_pn: _val
})
_eo += sum(x == const.VERSION_ERR for x in _rkeys)
_wo += sum(x == const.VERSION_WARN for x in _rkeys)
_do += sum(x == const.VERSION_DOWN for x in _rkeys)
_progress.end()
_data['errors'] = {
'mirantis': _ec,
'system': _es,
'other': _eo,
'unlisted': _eu
}
_data['warnings'] = {
'mirantis': _wc,
'system': _ws,
'other': _wo,
'unlisted': _wu
}
_data['downgrades'] = {
'mirantis': _dc,
'system': _ds,
'other': _do,
'unlisted': _du
}
return _data
def collect_packages(self):
"""
Check package versions in repos vs installed
:return: no return values, all date put to dict in place
"""
# Preload OpenStack release versions
_desc = PkgVersions(self.env_config)
logger_cli.info(
"# Cross-comparing: Installed vs Candidates vs Release"
)
# shortcuts for this cloud values
_os = self.master.openstack_release
_mcp = self.master.mcp_release
_t = [self.force_tag] if self.force_tag else []
_t.append(_mcp)
logger_cli.info("# Tag search list: {}".format(", ".join(_t)))
logger_cli.info("# Openstack version: {}".format(_os))
logger_cli.info(
"# Release versions repos keyword exclude list: {}".format(
", ".join(self.exclude_keywords)
)
)
# Progress class
_progress = Progress(len(self.master.nodes.keys()))
_progress_index = 0
_total_processed = 0
# Collect packages from all of the nodes in flat dict
_all_packages = {}
for node_name, node_value in self.master.nodes.items():
_uniq_len = len(_all_packages.keys())
_progress_index += 1
# progress updates shown before next node only
# it is costly operation to do it for each of the 150k packages
_progress.write_progress(
_progress_index,
note="/ {} uniq out of {} packages found".format(
_uniq_len,
_total_processed
)
)
for _name, _value in node_value['packages'].items():
_total_processed += 1
# Parse versions from nodes
_ver_ins = DebianVersion(_value['installed'])
_ver_can = DebianVersion(_value['candidate'])
# Process package description and release version
# at a first sight
if _name not in _all_packages:
# get node attributes
_linux = \
self.master.nodes[node_name]['linux_codename']
_arch = self.master.nodes[node_name]['linux_arch']
# get versions for tag, Openstack release and repo headers
# excluding 'nightly' repos by default
_r = {}
# if there is a forced tag = use it
if self.force_tag:
_r = self.rm.get_filtered_versions(
_name,
tag=self.force_tag,
include=[_os, _linux, _arch],
exclude=self.exclude_keywords
)
# if nothing found, look everywhere
# but with no word 'openstack'
if not _r:
_r = self.rm.get_filtered_versions(
_name,
tag=self.force_tag,
include=[_linux, _arch],
exclude=self.exclude_keywords + ['openstack']
)
# if nothing is found at this point,
# repeat search using normal tags
if not _r:
_r = self.rm.get_filtered_versions(
_name,
tag=_mcp,
include=[_os, _linux, _arch],
exclude=self.exclude_keywords
)
# Once again, if nothing found, look everywhere
if not _r:
_r = self.rm.get_filtered_versions(
_name,
tag=_mcp,
include=[_linux, _arch],
exclude=self.exclude_keywords + ['openstack']
)
# repack versions in flat format
_vs = {}
_sections = {}
_apps = {}
for s, apps in _r.items():
for a, versions in apps.items():
for v, repos in versions.items():
for repo in repos:
if v not in _vs:
_vs[v] = []
_vs[v].append(repo)
if v not in _sections:
_sections[v] = []
_sections[v].append(s)
if v not in _apps:
_apps[v] = []
_apps[v].append(a)
# search for the newest version among filtered
_r_desc = []
_vs_keys = iter(_vs.keys())
# get next version, if any
try:
_newest = DebianVersion(next(_vs_keys))
except StopIteration:
_newest = DebianVersion('')
# iterate others, if any
for v in _vs_keys:
_this = DebianVersion(v)
if _this > _newest:
_newest = _this
_release = _newest
# Get best description for the package
if _release.version != 'n/a':
_r_desc = _vs[_release.version]
# preload special description
if _desc[_name]:
_pkg_desc = _desc[_name]
else:
_pkg_desc = _desc.dummy_desc
# Save repos list and desc for this version
# Check if we can provide better from the package
if _release.version != 'n/a':
if not _pkg_desc['section']:
_pkg_desc['section'] = \
"/".join(_sections[_release.version])
if not _pkg_desc['app']:
_pkg_desc['app'] = \
"/".join(_apps[_release.version])
# Populate package info, once for package
_m = _r_desc[0]["maintainer"] if _r_desc else 'n/a'
_all_packages[_name] = {
"desc": _pkg_desc,
"repos": _r_desc,
"maintainer": _m,
"is_mirantis": self.rm.is_mirantis(
_name,
tag=_mcp
),
"results": {},
"r": _release,
}
# Cross-compare versions
_cmp = VersionCmpResult(
_ver_ins,
_ver_can,
_all_packages[_name]['r']
)
# Update results structure
# shortcut to results
_res = _all_packages[_name]['results']
# update status
if _cmp.status not in _res:
_res[_cmp.status] = {}
# update action
if _cmp.action not in _res[_cmp.status]:
_res[_cmp.status][_cmp.action] = {}
# update node
if node_name not in _res[_cmp.status][_cmp.action]:
_res[_cmp.status][_cmp.action][node_name] = {}
# put result
_res[_cmp.status][_cmp.action][node_name] = {
'i': _ver_ins,
'c': _ver_can,
'res': _cmp,
'raw': _value['raw']
}
self._packages = _all_packages
_progress.end()
def create_report(self, filename, rtype, full=None):
"""
Create static html showing packages diff per node
:return: buff with html
"""
logger_cli.info("# Generating report to '{}'".format(filename))
if rtype == 'html':
_type = reporter.HTMLPackageCandidates(self.master)
elif rtype == 'csv':
_type = reporter.CSVAllPackages(self.master)
else:
raise ConfigException("Report type not set")
_report = reporter.ReportToFile(
_type,
filename
)
payload = {
"nodes": self.master.nodes,
"mcp_release": self.master.mcp_release,
"openstack_release": self.master.openstack_release
}
payload.update(self.presort_packages(self._packages, full))
_report(payload)
logger_cli.info("-> Done")
def collect_installed_packages(self):
"""
Collect installed packages on each node
sets 'installed' dict property in the class
:return: none
"""
logger_cli.info("# Collecting installed packages")
if self.master.env_type == ENV_TYPE_SALT:
self.master.prepare_script_on_active_nodes("pkg_versions.py")
_result = self.master.execute_script_on_active_nodes(
"pkg_versions.py"
)
for key in self.master.nodes.keys():
# due to much data to be passed from salt, it is happening in order
if key in _result and _result[key]:
_text = _result[key]
try:
_dict = json.loads(_text[_text.find('{'):])
except ValueError:
logger_cli.info("... no JSON for '{}'".format(
key
))
logger_cli.debug(
"ERROR:\n{}\n".format(_text[:_text.find('{')])
)
_dict = {}
self.master.nodes[key]['packages'] = _dict
else:
self.master.nodes[key]['packages'] = {}
logger_cli.debug("... {} has {} packages installed".format(
key,
len(self.master.nodes[key]['packages'].keys())
))
logger_cli.info("-> Done")
class SaltCloudPackageChecker(CloudPackageChecker):
def __init__(
self,
config,
force_tag=None,
exclude_keywords=[],
skip_list=None,
skip_list_file=None
):
self.master = SaltNodes(config)
super(SaltCloudPackageChecker, self).__init__(
config,
force_tag=force_tag,
exclude_keywords=[],
skip_list=skip_list,
skip_list_file=skip_list_file
)
class KubeCloudPackageChecker(CloudPackageChecker):
def __init__(
self,
config,
force_tag=None,
exclude_keywords=[],
skip_list=None,
skip_list_file=None
):
self.master = KubeNodes(config)
super(KubeCloudPackageChecker, self).__init__(
config,
force_tag=force_tag,
exclude_keywords=[],
skip_list=skip_list,
skip_list_file=skip_list_file
)