From ebd21a41fb1ef17370e58be4f05fd49c005efd90 Mon Sep 17 00:00:00 2001 From: SegFaulti4 Date: Fri, 22 Dec 2023 19:55:01 +0300 Subject: [PATCH 1/3] Added ansible package to requirements. Fixed port number in contributing inventory file, disabled StrictHostKeyChecking, deleted unnecessary host group. Extended .gitignore --- .gitignore | 4 +++- contributing/contr_inv | 7 ++----- setup.cfg | 4 +++- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/.gitignore b/.gitignore index fc73503..90b889c 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,6 @@ cotea.egg-info/ __pycache__/ dist/ - +build/ +venv/ +.venv/ diff --git a/contributing/contr_inv b/contributing/contr_inv index 876f10c..dd002d5 100644 --- a/contributing/contr_inv +++ b/contributing/contr_inv @@ -1,5 +1,2 @@ -[host1] -host1 ansible_host=localhost ansible_connection=ssh ansible_ssh_port=2223 ansible_user=root ansible_password=veronika - -[targets] -host1 +[host1] +localhost ansible_connection=ssh ansible_ssh_port=2222 ansible_user=root ansible_password=veronika ansible_ssh_common_args='-o StrictHostKeyChecking=no' diff --git a/setup.cfg b/setup.cfg index a9fbb3d..6406465 100644 --- a/setup.cfg +++ b/setup.cfg @@ -18,7 +18,9 @@ classifiers = package_dir = = src packages = find: -python_requires = >=3.6 +python_requires = >=3.8 +install_requires = + ansible >=2.9.4 [options.packages.find] where = src From e51704dee7f80c0bbe3507e346264b751d6f7be5 Mon Sep 17 00:00:00 2001 From: SegFaulti4 Date: Fri, 22 Dec 2023 19:55:52 +0300 Subject: [PATCH 2/3] Added exception passing from Ansible thread via threading.excepthook --- src/cotea/ansible_execution_sync.py | 6 +++ src/cotea/runner.py | 82 +++++++++++++++++------------ 2 files changed, 54 insertions(+), 34 deletions(-) diff --git a/src/cotea/ansible_execution_sync.py b/src/cotea/ansible_execution_sync.py index 6092202..e9e97b1 100644 --- a/src/cotea/ansible_execution_sync.py +++ b/src/cotea/ansible_execution_sync.py @@ -7,6 +7,8 @@ def __init__(self, logger): self.ansible_event = threading.Event() self.logger = logger self.curr_breakpoint_label = None + # Used to pass exceptions from Ansible thread + self.exception = None def status(self): self.logger.debug("Runner event status: %s", self.runner_event.is_set()) @@ -17,6 +19,8 @@ def runner_just_wait(self): #self.logger.debug("runner: waiting...") self.runner_event.wait() self.runner_event.clear() + if self.exception is not None: + raise self.exception def ansible_just_wait(self): #self.logger.debug("ansible: waiting...") @@ -36,6 +40,8 @@ def continue_ansible_with_stop(self): self.runner_event.wait() self.runner_event.clear() #self.logger.debug("runner: ANSIBLE WAKED ME UP") + if self.exception is not None: + raise self.exception def continue_runner(self): #self.logger.debug("ansible: resume runner work") diff --git a/src/cotea/runner.py b/src/cotea/runner.py index d374714..01afe12 100644 --- a/src/cotea/runner.py +++ b/src/cotea/runner.py @@ -39,9 +39,9 @@ def __init__(self, pb_path, arg_maker, debug_mod=None, show_progress_bar=False): logging_lvl = logging.INFO if debug_mod: logging_lvl= logging.DEBUG - + self.show_progress_bar = show_progress_bar - + logging.basicConfig(format="%(name)s %(asctime)s %(message)s", \ datefmt="%H:%M:%S", level=logging_lvl) @@ -49,7 +49,7 @@ def __init__(self, pb_path, arg_maker, debug_mod=None, show_progress_bar=False): self.arg_maker = arg_maker self.logger = logging.getLogger("RUNNER") - + log_sync = logging.getLogger("SYNC") self.sync_obj = ans_sync(log_sync) @@ -67,7 +67,7 @@ def __init__(self, pb_path, arg_maker, debug_mod=None, show_progress_bar=False): self._set_wrappers() start_ok = self._start_ansible() self.logger.debug("Ansible start ok: %s", start_ok) - + def _set_wrappers(self): wrp_lgr = logging.getLogger("WRPR") @@ -110,7 +110,7 @@ def _set_wrappers(self): self.execution_tree, self.progress_bar) PlayIterator.add_tasks = self.iterator_add_task_wrp - + def _set_wrappers_back(self): PlaybookCLI.run = self.pbcli_run_wrp.func @@ -121,7 +121,18 @@ def _set_wrappers_back(self): PlayIterator.add_tasks = self.iterator_add_task_wrp.func if self.show_progress_bar: PlaybookExecutor.__init__ = self.playbook_executor_wrp.func - + + def _except_hook(self, args, /): + exc_type, exc_value, exc_traceback, thread = \ + args.exc_type, args.exc_value, args.exc_traceback, args.thread + + if (exc_type == SystemExit or + # NOTE: this probably should never happen + thread != self.ansible_thread): + return self._old_except_hook(args) + + self.sync_obj.exception = exc_value + self.sync_obj.continue_runner() def _start_ansible(self): args = self.arg_maker.args @@ -131,19 +142,22 @@ def _start_ansible(self): self.pbCLI = PlaybookCLI(args) self.ansible_thread = threading.Thread(target=self.pbCLI.run) + self._old_except_hook = threading.excepthook + threading.excepthook = self._except_hook + self.ansible_thread.start() self.sync_obj.runner_just_wait() if self.sync_obj.curr_breakpoint_label == self.breakpoint_labeles["before_playbook"]: return True - + return False - + def has_next_play(self): if self.sync_obj.curr_breakpoint_label == self.breakpoint_labeles["after_playbook"]: return False - + self.sync_obj.continue_ansible_with_stop() current_bp_label = self.sync_obj.curr_breakpoint_label self.logger.debug("has_next_play: %s", current_bp_label) @@ -180,18 +194,18 @@ def run_next_task(self): if current_bp_label != self.breakpoint_labeles["after_task"]: self.logger.debug("run_next_task() has come not in to the 'after_task'") - + for task_result_ansible_obj in self.update_conn_wrapper.current_results: res.append(TaskResult(task_result_ansible_obj)) self.task_wrp.set_next_to_prev() return res - + def rerun_last_task(self): self.task_wrp.rerun_last_task = True - + # returns True and empty string if success # False and error msg otherwise @@ -202,7 +216,7 @@ def add_new_task(self, new_task_str, is_dict=False): has_attrs, error_msg = cotea_utils.obj_has_attrs(prev_task, ["_parent"]) if not has_attrs: return False, error_msg - + curr_block = prev_task._parent block_attrs = ["_loader", "_play", "_role", "_variable_manager", "_use_handlers"] has_attrs, error_msg = cotea_utils.obj_has_attrs(curr_block, block_attrs) @@ -227,16 +241,16 @@ def add_new_task(self, new_task_str, is_dict=False): error_msg += "(from str-aka-dict to python ds): {}" return False, error_msg.format(is_dict, str(e)) ds = [new_task_str_dict] - + #print("DS:\n", ds) - + has_attrs, _ = cotea_utils.obj_has_attrs(ds, ["__len__"]) if not has_attrs: error_msg = "Python repr of the input string should have " error_msg += "__len__ attr. Maybe something wrong with input: {}\n" error_msg += "Python repr without __len__ attr: {}" return False, error_msg.format(new_task_str, str(ds)) - + if len(ds) != 1: error_msg = "You must add 1 new task. Instead you add: {}" return False, error_msg.format(str(ds)) @@ -261,7 +275,7 @@ def add_new_task(self, new_task_str, is_dict=False): error_msg = "Exception during load_list_of_tasks call " error_msg += "(creats Ansible.Task objects): {}" return False, error_msg.format(str(e)) - + has_attrs, _ = cotea_utils.obj_has_attrs(new_ansible_task, ["__len__"]) if not has_attrs: error_msg = "Python repr of the input string should have " @@ -274,23 +288,23 @@ def add_new_task(self, new_task_str, is_dict=False): error_msg = "The input '{}' has been interpreted into {} tasks " error_msg += "instead of 1. Interpretation result: {}" return False, error_msg.format(new_task_str, new_tasks_count, str(ds)) - + #self.task_wrp.new_task_to_add = True self.task_wrp.new_task = new_ansible_task[0] - + adding_res, error_msg = self.task_wrp.add_tasks(new_ansible_task) return adding_res, error_msg - + def get_new_added_task(self): return self.task_wrp.new_task - + def ignore_errors_of_next_task(self): self.task_wrp.next_task_ignore_errors = True - + def dont_add_last_task_after_new(self): self.task_wrp.dont_add_last_task_after_new() @@ -306,15 +320,15 @@ def get_already_ignore_unrch(self): def finish_ansible(self): while self.sync_obj.curr_breakpoint_label != self.breakpoint_labeles["after_playbook"]: self.sync_obj.continue_ansible_with_stop() - + self.sync_obj.continue_ansible() self.ansible_thread.join(timeout=5) self._set_wrappers_back() - + def get_cur_play_name(self): return str(self.play_wrp.current_play_name) - + def get_next_task(self): return self.task_wrp.get_next_task() @@ -322,15 +336,15 @@ def get_next_task(self): def get_next_task_name(self): return str(self.task_wrp.get_next_task_name()) - + def get_prev_task(self): return self.task_wrp.get_prev_task() - + def get_prev_task_name(self): return str(self.task_wrp.get_prev_task_name()) - + def get_last_task_result(self): res = [] @@ -339,17 +353,17 @@ def get_last_task_result(self): res.append(TaskResult(task_result_ansible_obj)) return res - + # returns True if there was an non ignored error def was_error(self): return self.play_wrp.was_error - + # returns list with all errors, including the ignored ones def get_all_error_msgs(self): return self.update_conn_wrapper.error_msgs - + # returns last error msg that wasn't ignored def get_error_msg(self): @@ -361,9 +375,9 @@ def get_error_msg(self): if errors_count > 0: res = self.update_conn_wrapper.error_msgs[errors_count - 1] - + return res - + def get_all_vars(self): variable_manager = self.play_wrp.variable_manager @@ -419,7 +433,7 @@ def get_variable(self, var_name): self.logger.info("There is no variable with name %s", var_name) return None - + def add_var_as_extra_var(self, new_var_name, value): variable_manager = self.play_wrp.variable_manager From 92d5f1d6e55c2eba311bed69a5e5faeb79378569 Mon Sep 17 00:00:00 2001 From: SegFaulti4 Date: Tue, 16 Jan 2024 21:18:48 +0300 Subject: [PATCH 3/3] Added tests for internal errors, simplified inventory file for tests, minor changes in excepthook --- .github/workflows/cotea_testing_workflow.yaml | 1 + src/cotea/runner.py | 16 ++--- src/cotea_internal_error_case.py | 66 +++++++++++++++++++ src/cotea_run_files/incorrect.yaml | 6 ++ src/cotea_run_files/inv | 5 +- 5 files changed, 81 insertions(+), 13 deletions(-) create mode 100644 src/cotea_internal_error_case.py create mode 100644 src/cotea_run_files/incorrect.yaml diff --git a/.github/workflows/cotea_testing_workflow.yaml b/.github/workflows/cotea_testing_workflow.yaml index e2ec3f6..4bea0de 100644 --- a/.github/workflows/cotea_testing_workflow.yaml +++ b/.github/workflows/cotea_testing_workflow.yaml @@ -31,3 +31,4 @@ jobs: cd ./src python cotea_ok_case.py python cotea_ansible_error_case.py + python cotea_internal_error_case.py diff --git a/src/cotea/runner.py b/src/cotea/runner.py index 01afe12..12dc95f 100644 --- a/src/cotea/runner.py +++ b/src/cotea/runner.py @@ -123,15 +123,12 @@ def _set_wrappers_back(self): PlaybookExecutor.__init__ = self.playbook_executor_wrp.func def _except_hook(self, args, /): - exc_type, exc_value, exc_traceback, thread = \ - args.exc_type, args.exc_value, args.exc_traceback, args.thread - - if (exc_type == SystemExit or + if (args.exc_type == SystemExit or # NOTE: this probably should never happen - thread != self.ansible_thread): + args.thread != self.ansible_thread): return self._old_except_hook(args) - self.sync_obj.exception = exc_value + self.sync_obj.exception = args.exc_value self.sync_obj.continue_runner() def _start_ansible(self): @@ -318,10 +315,11 @@ def get_already_ignore_unrch(self): def finish_ansible(self): - while self.sync_obj.curr_breakpoint_label != self.breakpoint_labeles["after_playbook"]: - self.sync_obj.continue_ansible_with_stop() + if self.sync_obj.exception is None: + while self.sync_obj.curr_breakpoint_label != self.breakpoint_labeles["after_playbook"]: + self.sync_obj.continue_ansible_with_stop() + self.sync_obj.continue_ansible() - self.sync_obj.continue_ansible() self.ansible_thread.join(timeout=5) self._set_wrappers_back() diff --git a/src/cotea_internal_error_case.py b/src/cotea_internal_error_case.py new file mode 100644 index 0000000..e495dc4 --- /dev/null +++ b/src/cotea_internal_error_case.py @@ -0,0 +1,66 @@ +import unittest + + +class TestCotea(unittest.TestCase): + + def tearDown(self) -> None: + from cotea.utils import remove_modules_from_imported + + # Remove any Ansible-related objects from memory + # to clear previous execution context + remove_modules_from_imported(module_name_like="cotea") + + def test_incorrect_playbook_path_case(self): + from cotea.runner import runner + from cotea.arguments_maker import argument_maker + + pb_path = "cotea_run_files/#%|&" + inv_path = "cotea_run_files/inv" + + arg_maker = argument_maker() + arg_maker.add_arg("-i", inv_path) + r = runner(pb_path, arg_maker, show_progress_bar=True) + + try: + while r.has_next_play(): + while r.has_next_task(): + r.run_next_task() + r.finish_ansible() + except Exception as e: + r.finish_ansible() + self.assertTrue(hasattr(e, "message"), msg="Exception is expected to have 'message' attribute") + self.assertTrue(e.message.startswith(f"the playbook: {pb_path} could not be found"), + msg="Unexpected exception message") + else: + self.assertFalse(True, msg="Ansible is supposed to fail due to syntax error " + "and its' exception should be passed to main thread") + + def test_incorrect_syntax_case(self): + from cotea.runner import runner + from cotea.arguments_maker import argument_maker + + pb_path = "cotea_run_files/incorrect.yaml" + inv_path = "cotea_run_files/inv" + + arg_maker = argument_maker() + arg_maker.add_arg("-i", inv_path) + r = runner(pb_path, arg_maker, show_progress_bar=True) + + try: + while r.has_next_play(): + while r.has_next_task(): + r.run_next_task() + r.finish_ansible() + except Exception as e: + r.finish_ansible() + # NOTE: e should be AnsibleParserError, but "isinstance" returns False for some reason + self.assertTrue(hasattr(e, "message"), msg="Exception is expected to have 'message' attribute") + self.assertTrue(e.message.startswith("couldn't resolve module/action"), + msg="Unexpected exception message") + else: + self.assertFalse(True, msg="Ansible is supposed to fail due to syntax error " + "and its' exception should be passed to main thread") + + +if __name__ == '__main__': + unittest.main() diff --git a/src/cotea_run_files/incorrect.yaml b/src/cotea_run_files/incorrect.yaml new file mode 100644 index 0000000..1beadab --- /dev/null +++ b/src/cotea_run_files/incorrect.yaml @@ -0,0 +1,6 @@ +--- +- name: Play1 + hosts: all + tasks: + - name: Syntactically incorrect command + battle_star_engine: diff --git a/src/cotea_run_files/inv b/src/cotea_run_files/inv index 95fc38d..d0162e9 100644 --- a/src/cotea_run_files/inv +++ b/src/cotea_run_files/inv @@ -1,5 +1,2 @@ [host1] -host1 ansible_host=localhost ansible_connection=ssh ansible_ssh_port=2222 ansible_user=root ansible_password=amella - -[targets] -host1 \ No newline at end of file +localhost ansible_connection=ssh ansible_ssh_port=2222 ansible_user=root ansible_password=amella ansible_ssh_common_args='-o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null'