From 8a584a8546667ab5390e9a2003a8ce3cb3add25c Mon Sep 17 00:00:00 2001 From: Victor Zhestkov Date: Fri, 28 Oct 2022 13:20:13 +0300 Subject: [PATCH] Pass the context to pillar ext modules * Pass __context__ to ext pillar * Add test for passing the context to pillar ext module * Align the test and pillar to prevent failing test --- salt/master.py | 7 ++- salt/pillar/__init__.py | 16 +++++- tests/pytests/unit/test_master.py | 91 ++++++++++++++++++++++++++++++- 3 files changed, 108 insertions(+), 6 deletions(-) diff --git a/salt/master.py b/salt/master.py index 7f41ffe77b..0110082626 100644 --- a/salt/master.py +++ b/salt/master.py @@ -951,6 +951,7 @@ class MWorker(salt.utils.process.SignalHandlingProcess): self.k_mtime = 0 self.stats = collections.defaultdict(lambda: {"mean": 0, "runs": 0}) self.stat_clock = time.time() + self.context = {} # We need __setstate__ and __getstate__ to also pickle 'SMaster.secrets'. # Otherwise, 'SMaster.secrets' won't be copied over to the spawned process @@ -1138,7 +1139,7 @@ class MWorker(salt.utils.process.SignalHandlingProcess): self.key, ) self.clear_funcs.connect() - self.aes_funcs = AESFuncs(self.opts) + self.aes_funcs = AESFuncs(self.opts, context=self.context) salt.utils.crypt.reinit_crypto() self.__bind() @@ -1202,7 +1203,7 @@ class AESFuncs(TransportMethods): "_ext_nodes", # To be removed in 3006 (Sulfur) #60980 ) - def __init__(self, opts): + def __init__(self, opts, context=None): """ Create a new AESFuncs @@ -1212,6 +1213,7 @@ class AESFuncs(TransportMethods): :returns: Instance for handling AES operations """ self.opts = opts + self.context = context self.event = salt.utils.event.get_master_event( self.opts, self.opts["sock_dir"], listen=False ) @@ -1603,6 +1605,7 @@ class AESFuncs(TransportMethods): pillarenv=load.get("pillarenv"), extra_minion_data=load.get("extra_minion_data"), clean_cache=load.get("clean_cache"), + context=self.context, ) data = pillar.compile_pillar() self.fs_.update_opts() diff --git a/salt/pillar/__init__.py b/salt/pillar/__init__.py index 7be963566a..906bdfe55d 100644 --- a/salt/pillar/__init__.py +++ b/salt/pillar/__init__.py @@ -46,6 +46,7 @@ def get_pillar( pillarenv=None, extra_minion_data=None, clean_cache=False, + context=None, ): """ Return the correct pillar driver based on the file_client option @@ -81,6 +82,7 @@ def get_pillar( pillar_override=pillar_override, pillarenv=pillarenv, clean_cache=clean_cache, + context=context, ) return ptype( opts, @@ -92,6 +94,7 @@ def get_pillar( pillar_override=pillar_override, pillarenv=pillarenv, extra_minion_data=extra_minion_data, + context=context, ) @@ -280,7 +283,7 @@ class AsyncRemotePillar(RemotePillarMixin): raise salt.ext.tornado.gen.Return(ret_pillar) def destroy(self): - if self._closing: + if hasattr(self, "_closing") and self._closing: return self._closing = True @@ -309,6 +312,7 @@ class RemotePillar(RemotePillarMixin): pillar_override=None, pillarenv=None, extra_minion_data=None, + context=None, ): self.opts = opts self.opts["saltenv"] = saltenv @@ -333,6 +337,7 @@ class RemotePillar(RemotePillarMixin): merge_lists=True, ) self._closing = False + self.context = context def compile_pillar(self): """ @@ -406,6 +411,7 @@ class PillarCache: pillarenv=None, extra_minion_data=None, clean_cache=False, + context=None, ): # Yes, we need all of these because we need to route to the Pillar object # if we have no cache. This is another refactor target. @@ -432,6 +438,8 @@ class PillarCache: minion_cache_path=self._minion_cache_path(minion_id), ) + self.context = context + def _minion_cache_path(self, minion_id): """ Return the path to the cache file for the minion. @@ -455,6 +463,7 @@ class PillarCache: functions=self.functions, pillar_override=self.pillar_override, pillarenv=self.pillarenv, + context=self.context, ) return fresh_pillar.compile_pillar() @@ -530,6 +539,7 @@ class Pillar: pillar_override=None, pillarenv=None, extra_minion_data=None, + context=None, ): self.minion_id = minion_id self.ext = ext @@ -568,7 +578,7 @@ class Pillar: if opts.get("pillar_source_merging_strategy"): self.merge_strategy = opts["pillar_source_merging_strategy"] - self.ext_pillars = salt.loader.pillars(ext_pillar_opts, self.functions) + self.ext_pillars = salt.loader.pillars(ext_pillar_opts, self.functions, context=context) self.ignored_pillars = {} self.pillar_override = pillar_override or {} if not isinstance(self.pillar_override, dict): @@ -1335,7 +1345,7 @@ class Pillar: """ This method exist in order to be API compatible with RemotePillar """ - if self._closing: + if hasattr(self, "_closing") and self._closing: return self._closing = True diff --git a/tests/pytests/unit/test_master.py b/tests/pytests/unit/test_master.py index a49ecfec3b..ca02c7788d 100644 --- a/tests/pytests/unit/test_master.py +++ b/tests/pytests/unit/test_master.py @@ -1,7 +1,7 @@ import time import salt.master -from tests.support.mock import patch +from tests.support.mock import MagicMock, patch def test_fileserver_duration(): @@ -14,3 +14,92 @@ def test_fileserver_duration(): update.called_once() # Timeout is 1 second assert 2 > end - start > 1 + + +def test_mworker_pass_context(): + """ + Test of passing the __context__ to pillar ext module loader + """ + req_channel_mock = MagicMock() + local_client_mock = MagicMock() + + opts = { + "req_server_niceness": None, + "mworker_niceness": None, + "sock_dir": "/tmp", + "conf_file": "/tmp/fake_conf", + "transport": "zeromq", + "fileserver_backend": ["roots"], + "file_client": "local", + "pillar_cache": False, + "state_top": "top.sls", + "pillar_roots": {}, + } + + data = { + "id": "MINION_ID", + "grains": {}, + "saltenv": None, + "pillarenv": None, + "pillar_override": {}, + "extra_minion_data": {}, + "ver": "2", + "cmd": "_pillar", + } + + test_context = {"testing": 123} + + def mworker_bind_mock(): + mworker.aes_funcs.run_func(data["cmd"], data) + + with patch("salt.client.get_local_client", local_client_mock), patch( + "salt.master.ClearFuncs", MagicMock() + ), patch("salt.minion.MasterMinion", MagicMock()), patch( + "salt.utils.verify.valid_id", return_value=True + ), patch( + "salt.loader.matchers", MagicMock() + ), patch( + "salt.loader.render", MagicMock() + ), patch( + "salt.loader.utils", MagicMock() + ), patch( + "salt.loader.fileserver", MagicMock() + ), patch( + "salt.loader.minion_mods", MagicMock() + ), patch( + "salt.loader._module_dirs", MagicMock() + ), patch( + "salt.loader.LazyLoader", MagicMock() + ) as loadler_pillars_mock: + mworker = salt.master.MWorker(opts, {}, {}, [req_channel_mock]) + + with patch.object(mworker, "_MWorker__bind", mworker_bind_mock), patch.dict( + mworker.context, test_context + ): + mworker.run() + assert ( + loadler_pillars_mock.call_args_list[0][1].get("pack").get("__context__") + == test_context + ) + + loadler_pillars_mock.reset_mock() + + opts.update( + { + "pillar_cache": True, + "pillar_cache_backend": "file", + "pillar_cache_ttl": 1000, + "cachedir": "/tmp", + } + ) + + mworker = salt.master.MWorker(opts, {}, {}, [req_channel_mock]) + + with patch.object(mworker, "_MWorker__bind", mworker_bind_mock), patch.dict( + mworker.context, test_context + ), patch("salt.utils.cache.CacheFactory.factory", MagicMock()): + mworker.run() + assert ( + loadler_pillars_mock.call_args_list[0][1].get("pack").get("__context__") + == test_context + ) -- 2.37.3