blob: 987168f85d215f7abdbb4fac013843d78a7cfc44 [file] [log] [blame]
# Author: Alex Savatieiev (osavatieiev@mirantis.com; a.savex@gmail.com)
# Copyright 2019-2022 Mirantis, Inc.
import functools
import os
import re
import subprocess
from cfg_checker.common import logger_cli
from cfg_checker.common.const import all_salt_roles_map, uknown_code, \
truth
from cfg_checker.common.exception import ConfigException
pkg_dir = os.path.dirname(__file__)
pkg_dir = os.path.join(pkg_dir, os.pardir, os.pardir)
pkg_dir = os.path.normpath(pkg_dir)
pkg_dir = os.path.abspath(pkg_dir)
# 'Dirty' and simple way to execute piped cmds
def piped_shell(command, code=False):
logger_cli.debug("... cmd:'{}'".format(command))
_code, _out = subprocess.getstatusoutput(command)
if _code:
logger_cli.error("Non-zero return code: {}, '{}'".format(_code, _out))
if code:
return _code, _out
else:
return _out
# 'Proper way to execute shell
def shell(command):
logger_cli.debug("... cmd:'{}'".format(command))
_ps = subprocess.Popen(
command.split(),
stdout=subprocess.PIPE,
universal_newlines=False
).communicate()[0].decode()
return _ps
def merge_dict(source, destination):
"""
Dict merger, thanks to vincent
http://stackoverflow.com/questions/20656135/python-deep-merge-dictionary-data
>>> a = { 'first' : { 'all_rows' : { 'pass' : 'dog', 'number' : '1' } } }
>>> b = { 'first' : { 'all_rows' : { 'fail' : 'cat', 'number' : '5' } } }
>>> merge(b, a) == {
'first': {
'all_rows': {
'pass': 'dog',
'fail': 'cat',
'number': '5'
}
}
}
True
"""
for key, value in source.items():
if isinstance(value, dict):
# get node or create one
node = destination.setdefault(key, {})
merge_dict(value, node)
else:
destination[key] = value
return destination
class Utils(object):
@staticmethod
def validate_name(fqdn, message=False):
"""
Function that tries to validate node name.
Checks if code contains letters, has '.' in it,
roles map contains code's role
:param fqdn: node FQDN name to supply for the check
:param message: True if validate should return error check message
:return: False if checks failed, True if all checks passed
"""
_message = "# Validation passed"
def _result():
return (True, _message) if message else True
# node role code checks
_code = re.findall(r"[a-zA-Z]+", fqdn.split('.')[0])
if len(_code) > 0:
if _code[0] in all_salt_roles_map:
return _result()
else:
# log warning here
_message = "# Node code is unknown, '{}'. " \
"# Please, update map".format(_code)
else:
# log warning here
_message = "# Node name is invalid, '{}'".format(fqdn)
# put other checks here
# output result
return _result()
@staticmethod
def node_string_to_list(node_string):
# supplied var should contain node list
# if there is no ',' -> use space as a delimiter
if node_string is not None:
if node_string.find(',') < 0:
return node_string.split(' ')
else:
return node_string.split(',')
else:
return []
def get_node_code(self, fqdn):
# validate
_isvalid, _message = self.validate_name(fqdn, message=True)
_code = re.findall(
r"[a-zA-Z]+?(?=(?:[0-9]+$)|(?:\-+?)|(?:\_+?)|$)",
fqdn.split('.')[0]
)
# check if it is valid and raise if not
if _isvalid:
# try to match it with ones in map
_c = _code[0]
match = any([r in _c for r in all_salt_roles_map.keys()])
if match:
# no match, try to find it
match = False
for r in all_salt_roles_map.keys():
_idx = _c.find(r)
if _idx > -1:
_c = _c[_idx:]
match = True
break
if match:
return _c
else:
return uknown_code
else:
return uknown_code
else:
raise ConfigException(_message)
def get_nodes_list(self, nodes_list, env_sting=None):
_list = []
if env_sting is None:
# nothing supplied, use the one in repo
try:
if not nodes_list:
return []
with open(nodes_list) as _f:
_list.extend(_f.read().splitlines())
except IOError as e:
raise ConfigException("# Error while loading file, '{}': "
"{}".format(e.filename, e.strerror))
else:
_list.extend(self.node_string_to_list(env_sting))
# validate names
_invalid = []
_valid = []
for idx in range(len(_list)):
_name = _list[idx]
if not self.validate_name(_name):
_invalid.append(_name)
else:
_valid.append(_name)
return _valid, _invalid
@staticmethod
def to_bool(value):
if value.lower() in truth:
return True
else:
return False
# helper functions to get nested attrs
# https://stackoverflow.com/questions/31174295/getattr-and-setattr-on-nested-subobjects-chained-properties
# using wonder's beautiful simplification:
# https://stackoverflow.com/questions/31174295/getattr-and-setattr-on-nested-objects/31174427?noredirect=1#comment86638618_31174427
def rsetattr(self, obj, attr, val):
pre, _, post = attr.rpartition('.')
return setattr(self.rgetattr(obj, pre) if pre else obj, post, val)
def rgetattr(self, obj, attr, *args):
def _getattr(obj, attr):
return getattr(obj, attr, *args)
return functools.reduce(_getattr, [obj] + attr.split('.'))
@staticmethod
def split_option_type(size):
# I know, but it is faster then regex
_numbers = [48, 49, 50, 51, 52, 53, 54, 55, 56, 57]
_s_int = "0"
_s_type = ""
for ch in size:
if ord(ch) in _numbers:
_s_int += ch
else:
_s_type += ch
return int(_s_int), _s_type
utils = Utils()