D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
proc
/
self
/
root
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
thorium
/
Filename :
__init__.py
back
Copy
""" The thorium system allows for advanced event tracking and reactions """ # Needed: # Use a top file to load sls files locally # use the existing state system to compile a low state # Create a new state runtime to run the low state flow programming style # Create the thorium plugin system # Add dynamic recompile of thorium ruleset on given interval import logging import os import time import traceback import salt.cache import salt.loader import salt.payload import salt.state from salt.exceptions import SaltRenderError log = logging.getLogger(__name__) class ThorState(salt.state.HighState): """ Compile the thorium state and manage it in the thorium runtime """ def __init__( self, opts, grains=False, grain_keys=None, pillar=False, pillar_keys=None ): self.grains = grains self.grain_keys = grain_keys self.pillar = pillar self.pillar_keys = pillar_keys opts["file_roots"] = opts["thorium_roots"] opts["saltenv"] = opts["thoriumenv"] opts["state_top"] = opts["thorium_top"] opts["file_client"] = "local" self.opts = opts if opts.get("minion_data_cache"): self.cache = salt.cache.factory(opts) salt.state.HighState.__init__(self, self.opts, loader="thorium") self.returners = salt.loader.returners(self.opts, {}) self.reg_ret = self.opts.get("register_returner", None) regdata = {} if self.reg_ret is not None: try: regdata = self.returners["{}.load_reg".format(self.reg_ret)]() except Exception as exc: # pylint: disable=broad-except log.error(exc) self.state.inject_globals = {"__reg__": regdata} self.event = salt.utils.event.get_master_event(self.opts, self.opts["sock_dir"]) def gather_cache(self): """ Gather the specified data from the minion data cache """ cache = {"grains": {}, "pillar": {}} if self.grains or self.pillar: if self.opts.get("minion_data_cache"): minions = self.cache.list("minions") if not minions: return cache for minion in minions: total = self.cache.fetch("minions/{}".format(minion), "data") if "pillar" in total: if self.pillar_keys: for key in self.pillar_keys: if key in total["pillar"]: cache["pillar"][minion][key] = total["pillar"][key] else: cache["pillar"][minion] = total["pillar"] else: cache["pillar"][minion] = {} if "grains" in total: if self.grain_keys: for key in self.grain_keys: if key in total["grains"]: cache["grains"][minion][key] = total["grains"][key] else: cache["grains"][minion] = total["grains"] else: cache["grains"][minion] = {} return cache def start_runtime(self): """ Start the system! """ while True: try: self.call_runtime() except Exception: # pylint: disable=broad-except log.error("Exception in Thorium: ", exc_info=True) time.sleep(self.opts["thorium_interval"]) def get_chunks(self, exclude=None, whitelist=None): """ Compile the top file and return the lowstate for the thorium runtime to iterate over """ ret = {} err = [] try: top = self.get_top() except SaltRenderError as err: return ret except Exception: # pylint: disable=broad-except trb = traceback.format_exc() err.append(trb) return err err += self.verify_tops(top) matches = self.top_matches(top) if not matches: msg = "No Top file found!" raise SaltRenderError(msg) matches = self.matches_whitelist(matches, whitelist) high, errors = self.render_highstate(matches) if exclude: if isinstance(exclude, str): exclude = exclude.split(",") if "__exclude__" in high: high["__exclude__"].extend(exclude) else: high["__exclude__"] = exclude err += errors high, ext_errors = self.state.reconcile_extend(high) err += ext_errors err += self.state.verify_high(high) if err: raise SaltRenderError(err) return self.state.compile_high_data(high) def get_events(self): """ iterate over the available events and return a list of events """ ret = [] while True: event = self.event.get_event(wait=1, full=True) if event is None: return ret ret.append(event) def call_runtime(self): """ Execute the runtime """ cache = self.gather_cache() chunks = self.get_chunks() interval = self.opts["thorium_interval"] recompile = self.opts.get("thorium_recompile", 300) r_start = time.time() while True: events = self.get_events() if not events: time.sleep(interval) continue start = time.time() self.state.inject_globals["__events__"] = events self.state.call_chunks(chunks) elapsed = time.time() - start left = interval - elapsed if left > 0: time.sleep(left) self.state.reset_run_num() if (start - r_start) > recompile: cache = self.gather_cache() chunks = self.get_chunks() if self.reg_ret is not None: self.returners["{}.save_reg".format(self.reg_ret)](chunks) r_start = time.time()