D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
proc
/
self
/
root
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
beacons
/
Filename :
log_beacon.py
back
Copy
""" Beacon to fire events at specific log messages. .. versionadded:: 2017.7.0 """ import logging import salt.utils.beacons import salt.utils.files import salt.utils.platform try: import re HAS_REGEX = True except ImportError: HAS_REGEX = False __virtualname__ = "log" LOC_KEY = "log.loc" SKEL = {} SKEL["tag"] = "" SKEL["match"] = "no" SKEL["raw"] = "" SKEL["error"] = "" log = logging.getLogger(__name__) def __virtual__(): if not salt.utils.platform.is_windows() and HAS_REGEX: return __virtualname__ err_msg = "Not available for Windows systems or when regex library is missing." log.error("Unable to load %s beacon: %s", __virtualname__, err_msg) return False, err_msg def _get_loc(): """ return the active file location """ if LOC_KEY in __context__: return __context__[LOC_KEY] def validate(config): """ Validate the beacon configuration """ # Configuration for log beacon should be a list of dicts if not isinstance(config, list): return False, "Configuration for log beacon must be a list." config = salt.utils.beacons.list_to_dict(config) if "file" not in config: return False, "Configuration for log beacon must contain file option." return True, "Valid beacon configuration" # TODO: match values should be returned in the event def beacon(config): """ Read the log file and return match whole string .. code-block:: yaml beacons: log: - file: <path> - tags: <tag>: regex: <pattern> .. note:: regex matching is based on the `re`_ module .. _re: https://docs.python.org/3.6/library/re.html#regular-expression-syntax The defined tag is added to the beacon event tag. This is not the tag in the log. .. code-block:: yaml beacons: log: - file: /var/log/messages #path to log. - tags: goodbye/world: # tag added to beacon event tag. regex: .*good-bye.* # match good-bye string anywhere in the log entry. """ config = salt.utils.beacons.list_to_dict(config) ret = [] if "file" not in config: event = SKEL.copy() event["tag"] = "global" event["error"] = "file not defined in config" ret.append(event) return ret with salt.utils.files.fopen(config["file"], "r") as fp_: loc = __context__.get(LOC_KEY, 0) if loc == 0: fp_.seek(0, 2) __context__[LOC_KEY] = fp_.tell() return ret fp_.seek(0, 2) __context__[LOC_KEY] = fp_.tell() fp_.seek(loc) txt = fp_.read() log.info("txt %s", txt) d = {} for tag in config.get("tags", {}): if "regex" not in config["tags"][tag]: continue if not config["tags"][tag]["regex"]: continue try: d[tag] = re.compile(r"{}".format(config["tags"][tag]["regex"])) except Exception as e: # pylint: disable=broad-except event = SKEL.copy() event["tag"] = tag event["error"] = "bad regex" ret.append(event) for line in txt.splitlines(): for tag, reg in d.items(): try: m = reg.match(line) if m: event = SKEL.copy() event["tag"] = tag event["raw"] = line event["match"] = "yes" ret.append(event) except Exception: # pylint: disable=broad-except event = SKEL.copy() event["tag"] = tag event["error"] = "bad match" ret.append(event) return ret