From 7803275a8aaeedf2124706f51b6a54cfcfb2d032 Mon Sep 17 00:00:00 2001 From: Victor Zhestkov Date: Thu, 1 Sep 2022 14:45:13 +0300 Subject: [PATCH] Fix the regression in schedule module releasded in 3004 (bsc#1202631) Co-authored-by: Gareth J. Greenaway --- changelog/61324.changed | 1 + salt/modules/schedule.py | 449 ++++++++++++++------ tests/pytests/unit/modules/test_schedule.py | 138 +++++- 3 files changed, 442 insertions(+), 146 deletions(-) create mode 100644 changelog/61324.changed diff --git a/changelog/61324.changed b/changelog/61324.changed new file mode 100644 index 0000000000..d67051a8da --- /dev/null +++ b/changelog/61324.changed @@ -0,0 +1 @@ +Adding the ability to add, delete, purge, and modify Salt scheduler jobs when the Salt minion is not running. diff --git a/salt/modules/schedule.py b/salt/modules/schedule.py index bcd64f2851..913a101ea6 100644 --- a/salt/modules/schedule.py +++ b/salt/modules/schedule.py @@ -15,6 +15,7 @@ import salt.utils.event import salt.utils.files import salt.utils.odict import salt.utils.yaml +import yaml try: import dateutil.parser as dateutil_parser @@ -64,7 +65,35 @@ SCHEDULE_CONF = [ ] -def list_(show_all=False, show_disabled=True, where=None, return_yaml=True): +def _get_schedule_config_file(): + """ + Return the minion schedule configuration file + """ + config_dir = __opts__.get("conf_dir", None) + if config_dir is None and "conf_file" in __opts__: + config_dir = os.path.dirname(__opts__["conf_file"]) + if config_dir is None: + config_dir = salt.syspaths.CONFIG_DIR + + minion_d_dir = os.path.join( + config_dir, + os.path.dirname( + __opts__.get( + "default_include", + salt.config.DEFAULT_MINION_OPTS["default_include"], + ) + ), + ) + + if not os.path.isdir(minion_d_dir): + os.makedirs(minion_d_dir) + + return os.path.join(minion_d_dir, "_schedule.conf") + + +def list_( + show_all=False, show_disabled=True, where=None, return_yaml=True, offline=False +): """ List the jobs currently scheduled on the minion @@ -83,24 +112,33 @@ def list_(show_all=False, show_disabled=True, where=None, return_yaml=True): """ schedule = {} - try: - with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: - res = __salt__["event.fire"]( - {"func": "list", "where": where}, "manage_schedule" - ) - if res: - event_ret = event_bus.get_event( - tag="/salt/minion/minion_schedule_list_complete", wait=30 + if offline: + schedule_config = _get_schedule_config_file() + if os.path.exists(schedule_config): + with salt.utils.files.fopen(schedule_config) as fp_: + schedule_yaml = fp_.read() + if schedule_yaml: + schedule_contents = yaml.safe_load(schedule_yaml) + schedule = schedule_contents.get("schedule", {}) + else: + try: + with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: + res = __salt__["event.fire"]( + {"func": "list", "where": where}, "manage_schedule" ) - if event_ret and event_ret["complete"]: - schedule = event_ret["schedule"] - except KeyError: - # Effectively a no-op, since we can't really return without an event system - ret = {} - ret["comment"] = "Event module not available. Schedule list failed." - ret["result"] = True - log.debug("Event module not available. Schedule list failed.") - return ret + if res: + event_ret = event_bus.get_event( + tag="/salt/minion/minion_schedule_list_complete", wait=30 + ) + if event_ret and event_ret["complete"]: + schedule = event_ret["schedule"] + except KeyError: + # Effectively a no-op, since we can't really return without an event system + ret = {} + ret["comment"] = "Event module not available. Schedule list failed." + ret["result"] = True + log.debug("Event module not available. Schedule list failed.") + return ret _hidden = ["enabled", "skip_function", "skip_during_range"] for job in list(schedule.keys()): # iterate over a copy since we will mutate it @@ -139,14 +177,11 @@ def list_(show_all=False, show_disabled=True, where=None, return_yaml=True): # remove _seconds from the listing del schedule[job]["_seconds"] - if schedule: - if return_yaml: - tmp = {"schedule": schedule} - return salt.utils.yaml.safe_dump(tmp, default_flow_style=False) - else: - return schedule + if return_yaml: + tmp = {"schedule": schedule} + return salt.utils.yaml.safe_dump(tmp, default_flow_style=False) else: - return {"schedule": {}} + return schedule def is_enabled(name=None): @@ -186,11 +221,18 @@ def purge(**kwargs): .. code-block:: bash salt '*' schedule.purge + + # Purge jobs on Salt minion + salt '*' schedule.purge + """ - ret = {"comment": [], "result": True} + ret = {"comment": [], "changes": {}, "result": True} - for name in list_(show_all=True, return_yaml=False): + current_schedule = list_( + show_all=True, return_yaml=False, offline=kwargs.get("offline") + ) + for name in pycopy.deepcopy(current_schedule): if name == "enabled": continue if name.startswith("__"): @@ -202,37 +244,65 @@ def purge(**kwargs): "Job: {} would be deleted from schedule.".format(name) ) else: - persist = kwargs.get("persist", True) + if kwargs.get("offline"): + del current_schedule[name] - try: - with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: - res = __salt__["event.fire"]( - {"name": name, "func": "delete", "persist": persist}, - "manage_schedule", - ) - if res: - event_ret = event_bus.get_event( - tag="/salt/minion/minion_schedule_delete_complete", wait=30 + ret["comment"].append("Deleted job: {} from schedule.".format(name)) + ret["changes"][name] = "removed" + + else: + persist = kwargs.get("persist", True) + try: + with salt.utils.event.get_event( + "minion", opts=__opts__ + ) as event_bus: + res = __salt__["event.fire"]( + {"name": name, "func": "delete", "persist": persist}, + "manage_schedule", ) - if event_ret and event_ret["complete"]: - _schedule_ret = event_ret["schedule"] - if name not in _schedule_ret: - ret["result"] = True - ret["comment"].append( - "Deleted job: {} from schedule.".format(name) - ) - else: - ret["comment"].append( - "Failed to delete job {} from schedule.".format( - name + if res: + event_ret = event_bus.get_event( + tag="/salt/minion/minion_schedule_delete_complete", + wait=30, + ) + if event_ret and event_ret["complete"]: + _schedule_ret = event_ret["schedule"] + if name not in _schedule_ret: + ret["result"] = True + ret["changes"][name] = "removed" + ret["comment"].append( + "Deleted job: {} from schedule.".format(name) ) - ) - ret["result"] = True + else: + ret["comment"].append( + "Failed to delete job {} from schedule.".format( + name + ) + ) + ret["result"] = True + + except KeyError: + # Effectively a no-op, since we can't really return without an event system + ret["comment"] = "Event module not available. Schedule add failed." + ret["result"] = True + + # wait until the end to write file in offline mode + if kwargs.get("offline"): + schedule_conf = _get_schedule_config_file() + + try: + with salt.utils.files.fopen(schedule_conf, "wb+") as fp_: + fp_.write( + salt.utils.stringutils.to_bytes( + salt.utils.yaml.safe_dump({"schedule": current_schedule}) + ) + ) + except OSError: + log.error( + "Failed to persist the updated schedule", + exc_info_on_loglevel=logging.DEBUG, + ) - except KeyError: - # Effectively a no-op, since we can't really return without an event system - ret["comment"] = "Event module not available. Schedule add failed." - ret["result"] = True return ret @@ -245,6 +315,10 @@ def delete(name, **kwargs): .. code-block:: bash salt '*' schedule.delete job1 + + # Delete job on Salt minion when the Salt minion is not running + salt '*' schedule.delete job1 + """ ret = { @@ -260,45 +334,86 @@ def delete(name, **kwargs): ret["comment"] = "Job: {} would be deleted from schedule.".format(name) ret["result"] = True else: - persist = kwargs.get("persist", True) + if kwargs.get("offline"): + current_schedule = list_( + show_all=True, + where="opts", + return_yaml=False, + offline=kwargs.get("offline"), + ) - if name in list_(show_all=True, where="opts", return_yaml=False): - event_data = {"name": name, "func": "delete", "persist": persist} - elif name in list_(show_all=True, where="pillar", return_yaml=False): - event_data = { - "name": name, - "where": "pillar", - "func": "delete", - "persist": False, - } - else: - ret["comment"] = "Job {} does not exist.".format(name) - return ret + del current_schedule[name] - try: - with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: - res = __salt__["event.fire"](event_data, "manage_schedule") - if res: - event_ret = event_bus.get_event( - tag="/salt/minion/minion_schedule_delete_complete", - wait=30, + schedule_conf = _get_schedule_config_file() + + try: + with salt.utils.files.fopen(schedule_conf, "wb+") as fp_: + fp_.write( + salt.utils.stringutils.to_bytes( + salt.utils.yaml.safe_dump({"schedule": current_schedule}) + ) ) - if event_ret and event_ret["complete"]: - schedule = event_ret["schedule"] - if name not in schedule: - ret["result"] = True - ret["comment"] = "Deleted Job {} from schedule.".format( - name - ) - ret["changes"][name] = "removed" - else: - ret[ - "comment" - ] = "Failed to delete job {} from schedule.".format(name) - return ret - except KeyError: - # Effectively a no-op, since we can't really return without an event system - ret["comment"] = "Event module not available. Schedule add failed." + except OSError: + log.error( + "Failed to persist the updated schedule", + exc_info_on_loglevel=logging.DEBUG, + ) + + ret["result"] = True + ret["comment"] = "Deleted Job {} from schedule.".format(name) + ret["changes"][name] = "removed" + else: + persist = kwargs.get("persist", True) + + if name in list_( + show_all=True, + where="opts", + return_yaml=False, + offline=kwargs.get("offline"), + ): + event_data = {"name": name, "func": "delete", "persist": persist} + elif name in list_( + show_all=True, + where="pillar", + return_yaml=False, + offline=kwargs.get("offline"), + ): + event_data = { + "name": name, + "where": "pillar", + "func": "delete", + "persist": False, + } + else: + ret["comment"] = "Job {} does not exist.".format(name) + return ret + + try: + with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: + res = __salt__["event.fire"](event_data, "manage_schedule") + if res: + event_ret = event_bus.get_event( + tag="/salt/minion/minion_schedule_delete_complete", + wait=30, + ) + if event_ret and event_ret["complete"]: + schedule = event_ret["schedule"] + if name not in schedule: + ret["result"] = True + ret["comment"] = "Deleted Job {} from schedule.".format( + name + ) + ret["changes"][name] = "removed" + else: + ret[ + "comment" + ] = "Failed to delete job {} from schedule.".format( + name + ) + return ret + except KeyError: + # Effectively a no-op, since we can't really return without an event system + ret["comment"] = "Event module not available. Schedule add failed." return ret @@ -438,6 +553,10 @@ def add(name, **kwargs): salt '*' schedule.add job1 function='test.ping' seconds=3600 # If function have some arguments, use job_args salt '*' schedule.add job2 function='cmd.run' job_args="['date >> /tmp/date.log']" seconds=60 + + # Add job to Salt minion when the Salt minion is not running + salt '*' schedule.add job1 function='test.ping' seconds=3600 offline=True + """ ret = { @@ -445,8 +564,11 @@ def add(name, **kwargs): "result": False, "changes": {}, } + current_schedule = list_( + show_all=True, return_yaml=False, offline=kwargs.get("offline") + ) - if name in list_(show_all=True, return_yaml=False): + if name in current_schedule: ret["comment"] = "Job {} already exists in schedule.".format(name) ret["result"] = False return ret @@ -486,32 +608,56 @@ def add(name, **kwargs): ret["comment"] = "Job: {} would be added to schedule.".format(name) ret["result"] = True else: - try: - with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: - res = __salt__["event.fire"]( - { - "name": name, - "schedule": schedule_data, - "func": "add", - "persist": persist, - }, - "manage_schedule", + if kwargs.get("offline"): + current_schedule.update(schedule_data) + + schedule_conf = _get_schedule_config_file() + + try: + with salt.utils.files.fopen(schedule_conf, "wb+") as fp_: + fp_.write( + salt.utils.stringutils.to_bytes( + salt.utils.yaml.safe_dump({"schedule": current_schedule}) + ) + ) + except OSError: + log.error( + "Failed to persist the updated schedule", + exc_info_on_loglevel=logging.DEBUG, ) - if res: - event_ret = event_bus.get_event( - tag="/salt/minion/minion_schedule_add_complete", - wait=30, + + ret["result"] = True + ret["comment"] = "Added job: {} to schedule.".format(name) + ret["changes"][name] = "added" + else: + try: + with salt.utils.event.get_event("minion", opts=__opts__) as event_bus: + res = __salt__["event.fire"]( + { + "name": name, + "schedule": schedule_data, + "func": "add", + "persist": persist, + }, + "manage_schedule", ) - if event_ret and event_ret["complete"]: - schedule = event_ret["schedule"] - if name in schedule: - ret["result"] = True - ret["comment"] = "Added job: {} to schedule.".format(name) - ret["changes"][name] = "added" - return ret - except KeyError: - # Effectively a no-op, since we can't really return without an event system - ret["comment"] = "Event module not available. Schedule add failed." + if res: + event_ret = event_bus.get_event( + tag="/salt/minion/minion_schedule_add_complete", + wait=30, + ) + if event_ret and event_ret["complete"]: + schedule = event_ret["schedule"] + if name in schedule: + ret["result"] = True + ret["comment"] = "Added job: {} to schedule.".format( + name + ) + ret["changes"][name] = "added" + return ret + except KeyError: + # Effectively a no-op, since we can't really return without an event system + ret["comment"] = "Event module not available. Schedule add failed." return ret @@ -524,6 +670,10 @@ def modify(name, **kwargs): .. code-block:: bash salt '*' schedule.modify job1 function='test.ping' seconds=3600 + + # Modify job on Salt minion when the Salt minion is not running + salt '*' schedule.modify job1 function='test.ping' seconds=3600 offline=True + """ ret = {"comment": "", "changes": {}, "result": True} @@ -549,7 +699,9 @@ def modify(name, **kwargs): ret["comment"] = 'Unable to use "when" and "cron" options together. Ignoring.' return ret - current_schedule = list_(show_all=True, return_yaml=False) + current_schedule = list_( + show_all=True, return_yaml=False, offline=kwargs.get("offline") + ) if name not in current_schedule: ret["comment"] = "Job {} does not exist in schedule.".format(name) @@ -566,8 +718,7 @@ def modify(name, **kwargs): _current["seconds"] = _current.pop("_seconds") # Copy _current _new, then update values from kwargs - _new = pycopy.deepcopy(_current) - _new.update(kwargs) + _new = build_schedule_item(name, **kwargs) # Remove test from kwargs, it's not a valid schedule option _new.pop("test", None) @@ -587,29 +738,51 @@ def modify(name, **kwargs): if "test" in kwargs and kwargs["test"]: ret["comment"] = "Job: {} would be modified in schedule.".format(name) else: - persist = kwargs.get("persist", True) - if name in list_(show_all=True, where="opts", return_yaml=False): - event_data = { - "name": name, - "schedule": _new, - "func": "modify", - "persist": persist, - } - elif name in list_(show_all=True, where="pillar", return_yaml=False): - event_data = { - "name": name, - "schedule": _new, - "where": "pillar", - "func": "modify", - "persist": False, - } + if kwargs.get("offline"): + current_schedule[name].update(_new) - out = __salt__["event.fire"](event_data, "manage_schedule") - if out: + schedule_conf = _get_schedule_config_file() + + try: + with salt.utils.files.fopen(schedule_conf, "wb+") as fp_: + fp_.write( + salt.utils.stringutils.to_bytes( + salt.utils.yaml.safe_dump({"schedule": current_schedule}) + ) + ) + except OSError: + log.error( + "Failed to persist the updated schedule", + exc_info_on_loglevel=logging.DEBUG, + ) + + ret["result"] = True ret["comment"] = "Modified job: {} in schedule.".format(name) + else: - ret["comment"] = "Failed to modify job {} in schedule.".format(name) - ret["result"] = False + persist = kwargs.get("persist", True) + if name in list_(show_all=True, where="opts", return_yaml=False): + event_data = { + "name": name, + "schedule": _new, + "func": "modify", + "persist": persist, + } + elif name in list_(show_all=True, where="pillar", return_yaml=False): + event_data = { + "name": name, + "schedule": _new, + "where": "pillar", + "func": "modify", + "persist": False, + } + + out = __salt__["event.fire"](event_data, "manage_schedule") + if out: + ret["comment"] = "Modified job: {} in schedule.".format(name) + else: + ret["comment"] = "Failed to modify job {} in schedule.".format(name) + ret["result"] = False return ret diff --git a/tests/pytests/unit/modules/test_schedule.py b/tests/pytests/unit/modules/test_schedule.py index e6cb134982..02914be82f 100644 --- a/tests/pytests/unit/modules/test_schedule.py +++ b/tests/pytests/unit/modules/test_schedule.py @@ -8,7 +8,8 @@ import pytest import salt.modules.schedule as schedule import salt.utils.odict from salt.utils.event import SaltEvent -from tests.support.mock import MagicMock, patch +from salt.utils.odict import OrderedDict +from tests.support.mock import MagicMock, call, mock_open, patch log = logging.getLogger(__name__) @@ -29,6 +30,11 @@ def sock_dir(tmp_path): return str(tmp_path / "test-socks") +@pytest.fixture +def schedule_config_file(tmp_path): + return "/etc/salt/minion.d/_schedule.conf" + + @pytest.fixture def configure_loader_modules(): return {schedule: {}} @@ -36,24 +42,56 @@ def configure_loader_modules(): # 'purge' function tests: 1 @pytest.mark.slow_test -def test_purge(sock_dir): +def test_purge(sock_dir, job1, schedule_config_file): """ Test if it purge all the jobs currently scheduled on the minion. """ + _schedule_data = {"job1": job1} with patch.dict(schedule.__opts__, {"schedule": {}, "sock_dir": sock_dir}): mock = MagicMock(return_value=True) with patch.dict(schedule.__salt__, {"event.fire": mock}): _ret_value = {"complete": True, "schedule": {}} with patch.object(SaltEvent, "get_event", return_value=_ret_value): - assert schedule.purge() == { - "comment": ["Deleted job: schedule from schedule."], + with patch.object( + schedule, "list_", MagicMock(return_value=_schedule_data) + ): + assert schedule.purge() == { + "comment": ["Deleted job: job1 from schedule."], + "changes": {"job1": "removed"}, + "result": True, + } + + _schedule_data = {"job1": job1, "job2": job1, "job3": job1} + comm = [ + "Deleted job: job1 from schedule.", + "Deleted job: job2 from schedule.", + "Deleted job: job3 from schedule.", + ] + + changes = {"job1": "removed", "job2": "removed", "job3": "removed"} + + with patch.dict( + schedule.__opts__, {"schedule": {"job1": "salt"}, "sock_dir": sock_dir} + ): + with patch("salt.utils.files.fopen", mock_open(read_data="")) as fopen_mock: + with patch.object( + schedule, "list_", MagicMock(return_value=_schedule_data) + ): + assert schedule.purge(offline=True) == { + "comment": comm, + "changes": changes, "result": True, } + _call = call(b"schedule: {}\n") + write_calls = fopen_mock.filehandles[schedule_config_file][ + 0 + ].write._mock_mock_calls + assert _call in write_calls # 'delete' function tests: 1 @pytest.mark.slow_test -def test_delete(sock_dir): +def test_delete(sock_dir, job1, schedule_config_file): """ Test if it delete a job from the minion's schedule. """ @@ -68,6 +106,28 @@ def test_delete(sock_dir): "result": False, } + _schedule_data = {"job1": job1} + comm = "Deleted Job job1 from schedule." + changes = {"job1": "removed"} + with patch.dict( + schedule.__opts__, {"schedule": {"job1": "salt"}, "sock_dir": sock_dir} + ): + with patch("salt.utils.files.fopen", mock_open(read_data="")) as fopen_mock: + with patch.object( + schedule, "list_", MagicMock(return_value=_schedule_data) + ): + assert schedule.delete("job1", offline="True") == { + "comment": comm, + "changes": changes, + "result": True, + } + + _call = call(b"schedule: {}\n") + write_calls = fopen_mock.filehandles[schedule_config_file][ + 0 + ].write._mock_mock_calls + assert _call in write_calls + # 'build_schedule_item' function tests: 1 def test_build_schedule_item(sock_dir): @@ -120,7 +180,7 @@ def test_build_schedule_item_invalid_when(sock_dir): @pytest.mark.slow_test -def test_add(sock_dir): +def test_add(sock_dir, schedule_config_file): """ Test if it add a job to the schedule. """ @@ -163,6 +223,24 @@ def test_add(sock_dir): "result": True, } + comm1 = "Added job: job3 to schedule." + changes1 = {"job3": "added"} + with patch.dict( + schedule.__opts__, {"schedule": {"job1": "salt"}, "sock_dir": sock_dir} + ): + with patch("salt.utils.files.fopen", mock_open(read_data="")) as fopen_mock: + assert schedule.add( + "job3", function="test.ping", seconds=3600, offline="True" + ) == {"comment": comm1, "changes": changes1, "result": True} + + _call = call( + b"schedule:\n job3: {function: test.ping, seconds: 3600, maxrunning: 1, name: job3, enabled: true,\n jid_include: true}\n" + ) + write_calls = fopen_mock.filehandles[schedule_config_file][ + 1 + ].write._mock_mock_calls + assert _call in write_calls + # 'run_job' function tests: 1 @@ -444,7 +522,7 @@ def test_copy(sock_dir, job1): @pytest.mark.slow_test -def test_modify(sock_dir): +def test_modify(sock_dir, job1, schedule_config_file): """ Test if modifying job to the schedule. """ @@ -564,7 +642,6 @@ def test_modify(sock_dir): for key in [ "maxrunning", "function", - "seconds", "jid_include", "name", "enabled", @@ -586,6 +663,51 @@ def test_modify(sock_dir): ret = schedule.modify("job2", function="test.version", test=True) assert ret == expected5 + _schedule_data = {"job1": job1} + comm = "Modified job: job1 in schedule." + changes = {"job1": "removed"} + + changes = { + "job1": { + "new": OrderedDict( + [ + ("function", "test.version"), + ("maxrunning", 1), + ("name", "job1"), + ("enabled", True), + ("jid_include", True), + ] + ), + "old": OrderedDict( + [ + ("function", "test.ping"), + ("maxrunning", 1), + ("name", "job1"), + ("jid_include", True), + ("enabled", True), + ] + ), + } + } + with patch.dict( + schedule.__opts__, {"schedule": {"job1": "salt"}, "sock_dir": sock_dir} + ): + with patch("salt.utils.files.fopen", mock_open(read_data="")) as fopen_mock: + with patch.object( + schedule, "list_", MagicMock(return_value=_schedule_data) + ): + assert schedule.modify( + "job1", function="test.version", offline="True" + ) == {"comment": comm, "changes": changes, "result": True} + + _call = call( + b"schedule:\n job1: {enabled: true, function: test.version, jid_include: true, maxrunning: 1,\n name: job1}\n" + ) + write_calls = fopen_mock.filehandles[schedule_config_file][ + 0 + ].write._mock_mock_calls + assert _call in write_calls + # 'is_enabled' function tests: 1 -- 2.37.2