D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
modules
/
Filename :
logrotate.py
back
Copy
""" Module for managing logrotate. """ import logging import os import salt.utils.files import salt.utils.platform import salt.utils.stringutils from salt.exceptions import SaltInvocationError _LOG = logging.getLogger(__name__) _DEFAULT_CONF = "/etc/logrotate.conf" # Define a function alias in order not to shadow built-in's __func_alias__ = {"set_": "set"} def __virtual__(): """ Only work on POSIX-like systems """ if salt.utils.platform.is_windows(): return ( False, "The logrotate execution module cannot be loaded: only available " "on non-Windows systems.", ) return True def _convert_if_int(value): """ Convert to an int if necessary. :param str value: The value to check/convert. :return: The converted or passed value. :rtype: bool|int|str """ try: value = int(str(value)) except ValueError: pass return value def _parse_conf(conf_file=_DEFAULT_CONF): """ Parse a logrotate configuration file. Includes will also be parsed, and their configuration will be stored in the return dict, as if they were part of the main config file. A dict of which configs came from which includes will be stored in the 'include files' dict inside the return dict, for later reference by the user or module. """ ret = {} mode = "single" multi_names = [] multi = {} prev_comps = None with salt.utils.files.fopen(conf_file, "r") as ifile: for line in ifile: line = salt.utils.stringutils.to_unicode(line).strip() if not line: continue if line.startswith("#"): continue comps = line.split() if "{" in line and "}" not in line: mode = "multi" if len(comps) == 1 and prev_comps: multi_names = prev_comps else: multi_names = comps multi_names.pop() continue if "}" in line: mode = "single" for multi_name in multi_names: ret[multi_name] = multi multi_names = [] multi = {} continue if mode == "single": key = ret else: key = multi if comps[0] == "include": if "include files" not in ret: ret["include files"] = {} for include in os.listdir(comps[1]): if include not in ret["include files"]: ret["include files"][include] = [] include_path = os.path.join(comps[1], include) include_conf = _parse_conf(include_path) for file_key in include_conf: ret[file_key] = include_conf[file_key] ret["include files"][include].append(file_key) prev_comps = comps if len(comps) > 2: key[comps[0]] = " ".join(comps[1:]) elif len(comps) > 1: key[comps[0]] = _convert_if_int(comps[1]) else: key[comps[0]] = True return ret def show_conf(conf_file=_DEFAULT_CONF): """ Show parsed configuration :param str conf_file: The logrotate configuration file. :return: The parsed configuration. :rtype: dict CLI Example: .. code-block:: bash salt '*' logrotate.show_conf """ return _parse_conf(conf_file) def get(key, value=None, conf_file=_DEFAULT_CONF): """ Get the value for a specific configuration line. :param str key: The command or stanza block to configure. :param str value: The command value or command of the block specified by the key parameter. :param str conf_file: The logrotate configuration file. :return: The value for a specific configuration line. :rtype: bool|int|str CLI Example: .. code-block:: bash salt '*' logrotate.get rotate salt '*' logrotate.get /var/log/wtmp rotate /etc/logrotate.conf """ current_conf = _parse_conf(conf_file) stanza = current_conf.get(key, False) if value: if stanza: return stanza.get(value, False) _LOG.debug("Block '%s' not present or empty.", key) return stanza def set_(key, value, setting=None, conf_file=_DEFAULT_CONF): """ Set a new value for a specific configuration line. :param str key: The command or block to configure. :param str value: The command value or command of the block specified by the key parameter. :param str setting: The command value for the command specified by the value parameter. :param str conf_file: The logrotate configuration file. :return: A boolean representing whether all changes succeeded. :rtype: bool CLI Example: .. code-block:: bash salt '*' logrotate.set rotate 2 Can also be used to set a single value inside a multiline configuration block. For instance, to change rotate in the following block: .. code-block:: text /var/log/wtmp { monthly create 0664 root root rotate 1 } Use the following command: .. code-block:: bash salt '*' logrotate.set /var/log/wtmp rotate 2 This module also has the ability to scan files inside an include directory, and make changes in the appropriate file. """ conf = _parse_conf(conf_file) for include in conf["include files"]: if key in conf["include files"][include]: conf_file = os.path.join(conf["include"], include) new_line = "" kwargs = { "flags": 8, "backup": False, "path": conf_file, "pattern": "^{}.*".format(key), "show_changes": False, } if setting is None: current_value = conf.get(key, False) if isinstance(current_value, dict): raise SaltInvocationError( "Error: {} includes a dict, and a specific setting inside the " "dict was not declared".format(key) ) if value == current_value: _LOG.debug("Command '%s' already has: %s", key, value) return True # This is the new config line that will be set if value is True: new_line = key elif value: new_line = "{} {}".format(key, value) kwargs.update({"prepend_if_not_found": True}) else: stanza = conf.get(key, dict()) if stanza and not isinstance(stanza, dict): error_msg = ( "Error: A setting for a dict was declared, but the " "configuration line given is not a dict" ) raise SaltInvocationError(error_msg) if setting == stanza.get(value, False): _LOG.debug("Command '%s' already has: %s", value, setting) return True # We're going to be rewriting an entire stanza if setting: stanza[value] = setting else: del stanza[value] new_line = _dict_to_stanza(key, stanza) kwargs.update( { "pattern": "^{0}.*?{{.*?}}".format(key), "flags": 24, "append_if_not_found": True, } ) kwargs.update({"repl": new_line}) _LOG.debug("Setting file '%s' line: %s", conf_file, new_line) return __salt__["file.replace"](**kwargs) def _dict_to_stanza(key, stanza): """ Convert a dict to a multi-line stanza """ ret = "" for skey in stanza: if stanza[skey] is True: stanza[skey] = "" ret += " {} {}\n".format(skey, stanza[skey]) return "{0} {{\n{1}}}".format(key, ret)