From b68d402e416d94b69cd5f091a8ac023844394180 Mon Sep 17 00:00:00 2001 From: Andrea Waltlova Date: Thu, 27 Jun 2024 11:10:03 +0200 Subject: [PATCH 1/4] Use codecov token Signed-off-by: Andrea Waltlova --- .github/workflows/tests.yml | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 4c806de..945bb20 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -15,10 +15,23 @@ jobs: - uses: actions/checkout@v3 - name: Install requirements run: pip install --upgrade -r requirements.txt + - name: Test run: python -m pytest --cov --cov-report=xml + - name: Upload coverage reports to Codecov uses: codecov/codecov-action@v3 with: fail_ci_if_error: true verbose: true # optional (default = false) + + - name: Upload coverage to Codecov + id: UploadFirstAttempt + continue-on-error: true + uses: codecov/codecov-action@v4 + env: + CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} + with: + fail_ci_if_error: true + files: ./coverage.xml + verbose: true # optional (default = false) From 870d7390a135c6e08a1f771ab44179b89653004d Mon Sep 17 00:00:00 2001 From: Andrea Waltlova Date: Thu, 27 Jun 2024 14:28:11 +0200 Subject: [PATCH 2/4] Replace prints by logging and add sso-report logs Signed-off-by: Andrea Waltlova --- pytest.ini | 4 + scripts/leapp_script.py | 172 +++++++++++++++++++++++++++++++++------- tests/test_logging.py | 91 +++++++++++++++++++++ tests/test_main.py | 131 +++++++++++++++++++++++------- 4 files changed, 340 insertions(+), 58 deletions(-) create mode 100644 pytest.ini create mode 100644 tests/test_logging.py diff --git a/pytest.ini b/pytest.ini new file mode 100644 index 0000000..2d0fdf9 --- /dev/null +++ b/pytest.ini @@ -0,0 +1,4 @@ +[pytest] +testpaths = "tests" +log_cli = true +log_cli_level = 10 \ No newline at end of file diff --git a/scripts/leapp_script.py b/scripts/leapp_script.py index 6da9b73..6db3912 100644 --- a/scripts/leapp_script.py +++ b/scripts/leapp_script.py @@ -1,7 +1,13 @@ import json +import logging import os +import shutil +import sys import subprocess +from time import gmtime, strftime + + # SCRIPT_TYPE is either 'PREUPGRADE' or 'UPGRADE' # Value is set in signed yaml envelope in content_vars (RHC_WORKER_LEAPP_SCRIPT_TYPE) SCRIPT_TYPE = os.environ.get("RHC_WORKER_LEAPP_SCRIPT_TYPE", "None") @@ -30,8 +36,24 @@ } -# Both classes taken from: -# https://github.com/oamg/convert2rhel-worker-scripts/blob/main/scripts/preconversion_assessment_script.py +# Path to store the script logs +LOG_DIR = "/var/log/leapp-insights-tasks" +# Log filename for the script. It will be created based on the script type of +# execution. +LOG_FILENAME = "leapp-insights-tasks-%s.log" % ( + "upgrade" if IS_UPGRADE else "preupgrade" +) + +# Path to the sos extras folder +SOS_REPORT_FOLDER = "/etc/sos.extras.d" +# Name of the file based on the task type for sos report +SOS_REPORT_FILE = "leapp-insights-tasks-%s-logs" % ( + "upgrade" if IS_UPGRADE else "preupgrade" +) + +logger = logging.getLogger(__name__) + + class ProcessError(Exception): """Custom exception to report errors during setup and run of leapp""" @@ -87,9 +109,100 @@ def to_dict(self): } +def setup_sos_report(): + """Setup sos report log collection.""" + if not os.path.exists(SOS_REPORT_FOLDER): + os.makedirs(SOS_REPORT_FOLDER) + + script_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + sosreport_link_file = os.path.join(SOS_REPORT_FOLDER, SOS_REPORT_FILE) + # In case the file for sos report does not exist, lets create one and add + # the log file path to it. + if not os.path.exists(sosreport_link_file): + with open(sosreport_link_file, mode="w") as handler: + handler.write(":%s\n" % script_log_file) + + +def setup_logger_handler(): + """ + Setup custom logging levels, handlers, and so on. Call this method from + your application's main start point. + """ + # Receive the log level from the worker and try to parse it. If the log + # level is not compatible with what the logging library expects, set the + # log level to INFO automatically. + log_level = os.getenv("RHC_WORKER_LOG_LEVEL", "INFO").upper() + log_level = logging.getLevelName(log_level) + if isinstance(log_level, str): + log_level = logging.INFO + + # enable raising exceptions + logging.raiseExceptions = True + logger.setLevel(log_level) + + # create sys.stdout handler for info/debug + stdout_handler = logging.StreamHandler(sys.stdout) + formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s") + stdout_handler.setFormatter(formatter) + + # Create the directory if it don't exist + if not os.path.exists(LOG_DIR): + os.makedirs(LOG_DIR) + + log_filepath = os.path.join(LOG_DIR, LOG_FILENAME) + file_handler = logging.FileHandler(log_filepath) + file_handler.setFormatter(formatter) + + # can flush logs to the file that were logged before initializing the file handler + logger.addHandler(stdout_handler) + logger.addHandler(file_handler) + + +def archive_old_logger_files(): + """ + Archive the old log files to not mess with multiple runs outputs. Every + time a new run begins, this method will be called to archive the previous + logs if there is a `convert2rhel.log` file there, it will be archived using + the same name for the log file, but having an appended timestamp to it. + For example: + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162445070567607.log + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162478219820043.log + This way, the user can track the logs for each run individually based on + the timestamp. + """ + + current_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + archive_log_dir = os.path.join(LOG_DIR, "archive") + + # No log file found, that means it's a first run or it was manually deleted + if not os.path.exists(current_log_file): + return + + stat = os.stat(current_log_file) + + # Get the last modified time in UTC + last_modified_at = gmtime(stat.st_mtime) + + # Format time to a human-readable format + formatted_time = strftime("%Y%m%dT%H%M%SZ", last_modified_at) + + # Create the directory if it don't exist + if not os.path.exists(archive_log_dir): + os.makedirs(archive_log_dir) + + file_name, suffix = tuple(LOG_FILENAME.rsplit(".", 1)) + archive_log_file = "%s/%s-%s.%s" % ( + archive_log_dir, + file_name, + formatted_time, + suffix, + ) + shutil.move(current_log_file, archive_log_file) + + def get_rhel_version(): """Currently we execute the task only for RHEL 7 or 8""" - print("Checking OS distribution and version ID ...") + logger.info("Checking OS distribution and version ID ...") try: distribution_id = None version_id = None @@ -100,13 +213,13 @@ def get_rhel_version(): elif line.startswith("VERSION_ID="): version_id = line.split("=")[1].strip().strip('"') except IOError: - print("Couldn't read /etc/os-release") + logger.warn("Couldn't read /etc/os-release") return distribution_id, version_id def is_non_eligible_releases(release): """Check if the release is eligible for upgrade or preupgrade.""" - print("Exit if not RHEL 7 or RHEL 8 ...") + logger.info("Exit if not RHEL 7 or RHEL 8 ...") major_version, _ = release.split(".") if release is not None else (None, None) return release is None or major_version not in ALLOWED_RHEL_RELEASES @@ -129,7 +242,7 @@ def run_subprocess(cmd, print_cmd=True, env=None, wait=True): raise TypeError("cmd should be a list, not a str") if print_cmd: - print("Calling command '%s'" % " ".join(cmd)) + logger.info("Calling command '%s'", " ".join(cmd)) process = subprocess.Popen( cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, bufsize=1, env=env @@ -225,9 +338,9 @@ def _get_leapp_command_and_packages(version): def setup_leapp(version): leapp_install_command, rhel_rhui_packages = _get_leapp_command_and_packages(version) if _check_if_package_installed("leapp-upgrade"): - print("'leapp-upgrade' already installed, skipping ...") + logger.info("'leapp-upgrade' already installed, skipping ...") else: - print("Installing leapp ...") + logger.info("Installing leapp ...") output, returncode = run_subprocess(leapp_install_command) if returncode: raise ProcessError( @@ -236,7 +349,7 @@ def setup_leapp(version): % (returncode, output.rstrip("\n")), ) - print("Check installed rhui packages ...") + logger.info("Check installed rhui packages ...") for pkg in rhel_rhui_packages: if _check_if_package_installed(pkg["src_pkg"]): pkg["installed"] = True @@ -244,7 +357,7 @@ def setup_leapp(version): def should_use_no_rhsm_check(rhui_installed, command): - print("Checking if subscription manager and repositories are available ...") + logger.info("Checking if subscription manager and repositories are available ...") rhsm_repo_check_fail = True rhsm_installed_check = _check_if_package_installed("subscription-manager") if rhsm_installed_check: @@ -258,10 +371,9 @@ def should_use_no_rhsm_check(rhui_installed, command): ) if rhui_installed and not rhsm_repo_check_fail: - print( - "RHUI packages detected, adding --no-rhsm flag to {} command".format( - SCRIPT_TYPE.title() - ) + logger.info( + "RHUI packages detected, adding --no-rhsm flag to % command", + SCRIPT_TYPE.title() ) command.append("--no-rhsm") return True @@ -269,7 +381,7 @@ def should_use_no_rhsm_check(rhui_installed, command): def install_leapp_pkg_corresponding_to_installed_rhui(rhui_pkgs): - print("Installing leapp package corresponding to installed rhui packages") + logger.info("Installing leapp package corresponding to installed rhui packages") for pkg in rhui_pkgs: install_pkg = pkg["leapp_pkg"] install_output, returncode = run_subprocess( @@ -284,7 +396,7 @@ def install_leapp_pkg_corresponding_to_installed_rhui(rhui_pkgs): def remove_previous_reports(): - print("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") + logger.info("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") if os.path.exists(JSON_REPORT_PATH): os.remove(JSON_REPORT_PATH) @@ -303,7 +415,7 @@ def execute_operation(command): else: new_env[key] = value - print("Executing {} ...".format(SCRIPT_TYPE.title())) + logger.info("Executing %s ...", SCRIPT_TYPE.title()) output, _ = run_subprocess(command, env=new_env) return output @@ -313,7 +425,7 @@ def _find_highest_report_level(entries): """ Gather status codes from entries. """ - print("Collecting and combining report status.") + logger.info("Collecting and combining report status.") action_level_combined = [value["severity"] for value in entries] valid_action_levels = [ @@ -324,14 +436,14 @@ def _find_highest_report_level(entries): def parse_results(output, reboot_required=False): - print("Processing {} results ...".format(SCRIPT_TYPE.title())) + logger.info("Processing %s results ...", SCRIPT_TYPE.title()) report_json = "Not found" message = "Can't open json report at " + JSON_REPORT_PATH alert = True status = "ERROR" - print("Reading JSON report") + logger.info("Reading JSON report") if os.path.exists(JSON_REPORT_PATH): with open(JSON_REPORT_PATH, mode="r") as handler: report_json = json.load(handler) @@ -388,7 +500,7 @@ def parse_results(output, reboot_required=False): output.alert = alert output.message = message - print("Reading TXT report") + logger.info("Reading TXT report") report_txt = "Not found" if os.path.exists(TXT_REPORT_PATH): with open(TXT_REPORT_PATH, mode="r") as handler: @@ -399,24 +511,28 @@ def parse_results(output, reboot_required=False): def update_insights_inventory(output): """Call insights-client to update insights inventory.""" - print("Updating system status in Red Hat Insights.") + logger.info("Updating system status in Red Hat Insights.") _, returncode = run_subprocess(cmd=["/usr/bin/insights-client"]) if returncode: - print("System registration failed with exit code %s." % returncode) + logger.info("System registration failed with exit code %s.", returncode) output.message += " Failed to update Insights Inventory." output.alert = True return - print("System registered with insights-client successfully.") + logger.info("System registered with insights-client successfully.") def reboot_system(): - print("Rebooting system in 1 minute.") + logger.info("Rebooting system in 1 minute.") run_subprocess(["/usr/sbin/shutdown", "-r", "1"], wait=False) def main(): + """Main entrypoint for the script.""" + setup_sos_report() + archive_old_logger_files() + setup_logger_handler() try: # Exit if invalid value for SCRIPT_TYPE if SCRIPT_TYPE not in ["PREUPGRADE", "UPGRADE"]: @@ -454,11 +570,11 @@ def main(): upgrade_reboot_required = REBOOT_GUIDANCE_MESSAGE in leapp_output parse_results(output, upgrade_reboot_required) update_insights_inventory(output) - print("Operation {} finished successfully.".format(SCRIPT_TYPE.title())) + logger.info("Operation %s finished successfully.", SCRIPT_TYPE.title()) if upgrade_reboot_required: reboot_system() except ProcessError as exception: - print(exception.report) + logger.error(exception.report) output = OutputCollector( status="ERROR", alert=True, @@ -467,7 +583,7 @@ def main(): report=exception.report, ) except Exception as exception: - print(str(exception)) + logger.critical(str(exception)) output = OutputCollector( status="ERROR", alert=True, diff --git a/tests/test_logging.py b/tests/test_logging.py new file mode 100644 index 0000000..6e3c573 --- /dev/null +++ b/tests/test_logging.py @@ -0,0 +1,91 @@ +import os +import logging +from mock import patch +import pytest +import scripts + +from scripts.leapp_script import ( + setup_sos_report, + setup_logger_handler, + archive_old_logger_files, +) + +def test_setup_sos_report(monkeypatch, tmpdir): + sos_report_folder = str(tmpdir) + monkeypatch.setattr(scripts.leapp_script, "SOS_REPORT_FOLDER", sos_report_folder) + + + setup_sos_report() + + sos_report_file = os.path.join( + sos_report_folder, "leapp-insights-tasks-preupgrade-logs" + ) + assert os.path.exists(sos_report_folder) + assert os.path.exists(sos_report_file) + + with open(sos_report_file) as handler: + assert ( + ":/var/log/leapp-insights-tasks/leapp-insights-tasks-preupgrade.log" + == handler.read().strip() + ) + + +@patch("scripts.leapp_script.os.makedirs") +@patch("scripts.leapp_script.os.path.exists", side_effect=[False, True]) +def test_setup_sos_report_no_sos_report_folder( + patch_exists, patch_makedirs, monkeypatch, tmpdir +): + sos_report_folder = str(tmpdir) + monkeypatch.setattr(scripts.leapp_script, "SOS_REPORT_FOLDER", sos_report_folder) + + setup_sos_report() + + # Folder created + assert patch_exists.call_count == 2 + patch_makedirs.assert_called_once_with(sos_report_folder) + + +@patch("scripts.leapp_script.os.makedirs") +@patch("scripts.leapp_script.os.path.exists", side_effect=[False, True]) +@patch("scripts.leapp_script.os.getenv", return_value="unknown") +def test_setup_logger_handler(mock_getenv, mock_exist, mock_makedirs, monkeypatch, tmpdir): + log_dir = str(tmpdir) + monkeypatch.setattr(scripts.leapp_script, "LOG_DIR", log_dir) + monkeypatch.setattr(scripts.leapp_script, "LOG_FILENAME", "filelog.log") + + logger = logging.getLogger(__name__) + setup_logger_handler() + + # emitting some log entries + logger.info("Test info: %s", "data") + logger.debug("Test debug: %s", "other data") + + mock_getenv.assert_called_once_with("RHC_WORKER_LOG_LEVEL", "INFO") + mock_exist.assert_called_once_with(log_dir) + mock_makedirs.assert_called_once_with(log_dir) + + +def test_archive_old_logger_files(monkeypatch, tmpdir): + log_dir = str(tmpdir) + archive_dir = os.path.join(log_dir, "archive") + monkeypatch.setattr(scripts.leapp_script, "LOG_DIR", log_dir) + monkeypatch.setattr(scripts.leapp_script, "LOG_FILENAME", "test.log") + + original_log_file = tmpdir.join("test.log") + original_log_file.write("test") + + archive_old_logger_files() + + assert os.path.exists(log_dir) + assert os.path.exists(archive_dir) + assert len(os.listdir(archive_dir)) == 1 + + +def test_archive_old_logger_files_no_log_file(monkeypatch, tmpdir): + log_dir = str(tmpdir.join("something-else")) + monkeypatch.setattr(scripts.leapp_script, "LOG_DIR", log_dir) + monkeypatch.setattr(scripts.leapp_script, "LOG_FILENAME", "test.log") + + archive_old_logger_files() + + assert not os.path.exists(log_dir) \ No newline at end of file diff --git a/tests/test_main.py b/tests/test_main.py index b36d495..560fcc2 100644 --- a/tests/test_main.py +++ b/tests/test_main.py @@ -1,4 +1,4 @@ -from mock import patch +from mock import patch, Mock from scripts.leapp_script import ( main, OutputCollector, @@ -7,15 +7,23 @@ @patch("scripts.leapp_script.SCRIPT_TYPE", "TEST") -def test_main_invalid_script_type(capsys): +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) +def test_main_invalid_script_type( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, + caplog +): main() - captured = capsys.readouterr() + log = caplog.text - assert ( - "Allowed values for RHC_WORKER_LEAPP_SCRIPT_TYPE are 'PREUPGRADE' and 'UPGRADE'." - in captured.out - ) - assert "Exiting because RHC_WORKER_LEAPP_SCRIPT_TYPE='TEST'" in captured.out + assert "Exiting because RHC_WORKER_LEAPP_SCRIPT_TYPE='TEST'" in log + + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "PREUPGRADE") @@ -25,13 +33,19 @@ def test_main_invalid_script_type(capsys): @patch("scripts.leapp_script.setup_leapp") @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_non_eligible_release_preupgrade( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_output_collector, mock_update_insights_inventory, mock_setup_leapp, mock_is_non_eligible_releases, mock_get_rhel_version, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "6.9") mock_is_non_eligible_releases.return_value = True @@ -39,8 +53,7 @@ def test_main_non_eligible_release_preupgrade( main() - captured = capsys.readouterr() - assert 'Exiting because distribution="rhel" and version="6.9"' in captured.out + assert 'Exiting because distribution="rhel" and version="6.9"' in caplog.text mock_get_rhel_version.assert_called_once() mock_is_non_eligible_releases.assert_called_once() @@ -48,6 +61,10 @@ def test_main_non_eligible_release_preupgrade( mock_setup_leapp.assert_not_called() mock_update_insights_inventory.assert_not_called() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() + @patch("scripts.leapp_script.SCRIPT_TYPE", "PREUPGRADE") @patch("scripts.leapp_script.IS_PREUPGRADE", True) @@ -61,7 +78,13 @@ def test_main_non_eligible_release_preupgrade( @patch("scripts.leapp_script.execute_operation") @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_eligible_release_preupgrade( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_output_collector, mock_update_insights_inventory, mock_execute_operation, @@ -72,7 +95,7 @@ def test_main_eligible_release_preupgrade( mock_is_non_eligible_releases, mock_get_rhel_version, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -81,8 +104,8 @@ def test_main_eligible_release_preupgrade( mock_output_collector.return_value = OutputCollector(entries=["non-empty"]) main() - captured = capsys.readouterr() - assert "Operation Preupgrade finished successfully." in captured.out + + assert "Operation Preupgrade finished successfully." in caplog.text mock_setup_leapp.assert_called_once() mock_should_use_no_rhsm_check.assert_called_once() @@ -91,6 +114,9 @@ def test_main_eligible_release_preupgrade( mock_execute_operation.assert_called_once() mock_parse_results.assert_called_once() mock_update_insights_inventory.assert_called_once() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "UPGRADE") @@ -106,7 +132,13 @@ def test_main_eligible_release_preupgrade( @patch("scripts.leapp_script.execute_operation") @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_eligible_release_upgrade( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_output_collector, mock_update_insights_inventory, mock_execute_operation, @@ -118,7 +150,7 @@ def test_main_eligible_release_upgrade( mock_get_rhel_version, mock_reboot_system, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -130,8 +162,8 @@ def test_main_eligible_release_upgrade( ) main() - captured = capsys.readouterr() - assert "Operation Upgrade finished successfully." in captured.out + + assert "Operation Upgrade finished successfully." in caplog.text mock_setup_leapp.assert_called_once() mock_should_use_no_rhsm_check.assert_called_once() @@ -141,6 +173,9 @@ def test_main_eligible_release_upgrade( mock_parse_results.assert_called_once() mock_update_insights_inventory.assert_called_once() mock_reboot_system.assert_called_once() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "UPGRADE") @@ -156,7 +191,13 @@ def test_main_eligible_release_upgrade( @patch("scripts.leapp_script.execute_operation") @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_upgrade_not_sucessfull( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_output_collector, mock_update_insights_inventory, mock_execute_operation, @@ -168,7 +209,7 @@ def test_main_upgrade_not_sucessfull( mock_get_rhel_version, mock_reboot_system, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -178,8 +219,8 @@ def test_main_upgrade_not_sucessfull( mock_execute_operation.return_value = "LOREM IPSUM\n" + "\nDOLOR SIT AMET" main() - captured = capsys.readouterr() - assert "Operation Upgrade finished successfully." in captured.out + + assert "Operation Upgrade finished successfully." in caplog.text mock_setup_leapp.assert_called_once() mock_should_use_no_rhsm_check.assert_called_once() @@ -189,6 +230,9 @@ def test_main_upgrade_not_sucessfull( mock_parse_results.assert_called_once() mock_update_insights_inventory.assert_called_once() mock_reboot_system.assert_not_called() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "UPGRADE") @@ -204,7 +248,13 @@ def test_main_upgrade_not_sucessfull( @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") @patch("scripts.leapp_script.run_subprocess") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_setup_leapp_not_sucessfull( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_run_subprocess, mock_output_collector, mock_update_insights_inventory, @@ -216,7 +266,7 @@ def test_main_setup_leapp_not_sucessfull( mock_get_rhel_version, mock_reboot_system, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -224,8 +274,8 @@ def test_main_setup_leapp_not_sucessfull( mock_output_collector.return_value = OutputCollector(entries=["non-empty"]) main() - captured = capsys.readouterr() - assert "Installation of leapp failed with code '1'" in captured.out + + assert "Installation of leapp failed with code '1'" in caplog.text mock_should_use_no_rhsm_check.assert_not_called() mock_install_rhui.assert_not_called() @@ -234,6 +284,9 @@ def test_main_setup_leapp_not_sucessfull( mock_parse_results.assert_not_called() mock_update_insights_inventory.assert_not_called() mock_reboot_system.assert_not_called() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "UPGRADE") @@ -249,7 +302,13 @@ def test_main_setup_leapp_not_sucessfull( @patch("scripts.leapp_script.update_insights_inventory") @patch("scripts.leapp_script.OutputCollector") @patch("scripts.leapp_script.run_subprocess") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_install_corresponding_pkgs_not_sucessfull( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_run_subprocess, mock_output_collector, mock_update_insights_inventory, @@ -261,7 +320,7 @@ def test_main_install_corresponding_pkgs_not_sucessfull( mock_get_rhel_version, mock_reboot_system, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -270,10 +329,10 @@ def test_main_install_corresponding_pkgs_not_sucessfull( mock_output_collector.return_value = OutputCollector(entries=["non-empty"]) main() - captured = capsys.readouterr() + assert ( "Installation of to_install (coresponding pkg to '{'leapp_pkg': 'to_install'}') failed with exit code 1 and output: Installation failed." - in captured.out + in caplog.text ) mock_setup_leapp.assert_called_once() @@ -283,6 +342,9 @@ def test_main_install_corresponding_pkgs_not_sucessfull( mock_parse_results.assert_not_called() mock_update_insights_inventory.assert_not_called() mock_reboot_system.assert_not_called() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() @patch("scripts.leapp_script.SCRIPT_TYPE", "UPGRADE") @@ -298,7 +360,13 @@ def test_main_install_corresponding_pkgs_not_sucessfull( @patch("scripts.leapp_script.execute_operation") @patch("scripts.leapp_script.OutputCollector") @patch("scripts.leapp_script.run_subprocess") +@patch("scripts.leapp_script.setup_sos_report", side_effect=Mock()) +@patch("scripts.leapp_script.archive_old_logger_files", side_effect=Mock()) +@patch("scripts.leapp_script.setup_logger_handler", side_effect=Mock()) def test_main_update_inventory_not_sucessfull( + mock_setup_logger_handler, + mock_setup_sos_report, + mock_archive_old_logger_files, mock_run_subprocess, mock_output_collector, mock_execute_operation, @@ -310,7 +378,7 @@ def test_main_update_inventory_not_sucessfull( mock_get_rhel_version, mock_reboot_system, mock_parse_results, - capsys, + caplog, ): mock_get_rhel_version.return_value = ("rhel", "7.9") mock_is_non_eligible_releases.return_value = False @@ -321,9 +389,9 @@ def test_main_update_inventory_not_sucessfull( mock_run_subprocess.return_value = ("Installation failed", 1) main() - captured = capsys.readouterr() - assert "Updating system status in Red Hat Insights." in captured.out - assert "System registration failed with exit code 1." in captured.out + log = caplog.text + assert "Updating system status in Red Hat Insights." in log + assert "System registration failed with exit code 1." in log mock_setup_leapp.assert_called_once() mock_should_use_no_rhsm_check.assert_called_once() @@ -332,3 +400,6 @@ def test_main_update_inventory_not_sucessfull( mock_execute_operation.assert_called_once() mock_parse_results.assert_called_once() mock_reboot_system.assert_not_called() + mock_setup_logger_handler.assert_called_once() + mock_setup_sos_report.assert_called_once() + mock_archive_old_logger_files.assert_called_once() From e6dfb7b8d9cd4497cb0ae2ad8b58df12128c1f41 Mon Sep 17 00:00:00 2001 From: Andrea Waltlova Date: Thu, 27 Jun 2024 15:07:29 +0200 Subject: [PATCH 3/4] Update sync script Signed-off-by: Andrea Waltlova --- Makefile | 6 + misc/sync_scripts.py | 111 ++++++++--------- playbooks/leapp_preupgrade_script.yml | 172 +++++++++++++++++++++----- playbooks/leapp_upgrade_script.yml | 172 +++++++++++++++++++++----- 4 files changed, 344 insertions(+), 117 deletions(-) diff --git a/Makefile b/Makefile index 841085a..880a58f 100644 --- a/Makefile +++ b/Makefile @@ -30,3 +30,9 @@ install: install-deps pre-commit tests: install-deps . $(PYTHON_VENV)/bin/activate; \ $(PYTEST_CALL) + +sync: install-deps + python misc/sync_scripts.py worker + +sync-advisor: install-deps + python misc/sync_scripts.py advisor diff --git a/misc/sync_scripts.py b/misc/sync_scripts.py index 1b64a0a..514af83 100644 --- a/misc/sync_scripts.py +++ b/misc/sync_scripts.py @@ -1,49 +1,50 @@ +import sys import os -import argparse import ruamel.yaml +# Get the last argument used in the commandline if available, otherwise, use +# "worker" as the default value. +SYNC_PROJECT = sys.argv[1:][-1] if sys.argv[1:] else "worker" + # Scripts located in this project SCRIPT_PATH = "scripts/leapp_script.py" -REPO_PRE_UPGRADE_YAML_PATH = os.path.join(".", "playbooks/leapp_preupgrade_script.yml") -REPO_UPGRADE_YAML_PATH = os.path.join(".", "playbooks/leapp_upgrade_script.yml") - -WORKER_PRE_UPGRADE_YAML_PATH = os.path.join( - "..", "rhc-worker-script/development/nginx/data/leapp_preupgrade.yml" -) -WORKER_UPGRADE_YAML_PATH = os.path.join( - "..", "rhc-worker-script/development/nginx/data/leapp_upgrade.yml" -) - -DEFAULT_YAML_ENVELOPE = """ -- name: LEAPP - vars: - insights_signature: | - ascii_armored gpg signature - insights_signature_exclude: /vars/insights_signature - interpreter: /usr/bin/python - content: | - placeholder - content_vars: - # variables that will be handed to the script as environment vars - # will be prefixed with RHC_WORKER_* - LEAPP_SCRIPT_TYPE: type -""" +SCRIPTS_YAML_PATH = { + # TODO(r0x0d): Deprecate this in the future + "worker": ( + os.path.join( + "..", "rhc-worker-script/development/nginx/data/leapp_preupgrade_script.yml" + ), + os.path.join( + "..", "rhc-worker-script/development/nginx/data/leapp_upgrade_script.yml" + ), + ), + "tasks": ( + "playbooks/leapp_preupgrade_script.yml", + "playbooks/leapp_upgrade_script.yml", + ), + "advisor": ( + os.path.join( + "..", + "advisor-backend/api/advisor/tasks/playbooks/leapp_preupgrade_script.yml", + ), + os.path.join( + "..", + "advisor-backend/api/advisor/tasks/playbooks/leapp_upgrade_script.yml", + ), + ), +} def _get_updated_yaml_content(yaml_path, script_path): if not os.path.exists(yaml_path): - yaml = ruamel.yaml.YAML() - config = yaml.load(DEFAULT_YAML_ENVELOPE) - mapping = 2 - offset = 0 - else: - config, mapping, offset = ruamel.yaml.util.load_yaml_guess_indent( - open(yaml_path) - ) - print(mapping, offset) + raise SystemExit(f"Couldn't find yaml file: {yaml_path}") + + config, mapping, offset = ruamel.yaml.util.load_yaml_guess_indent( + open(yaml_path, encoding="utf-8") + ) - with open(script_path) as script: + with open(script_path, encoding="utf-8") as script: content = script.read() script_type = "PREUPGRADE" if "preupgrade" in yaml_path else "UPGRADE" @@ -57,36 +58,24 @@ def _write_content(config, path, mapping=None, offset=None): yaml = ruamel.yaml.YAML() if mapping and offset: yaml.indent(mapping=mapping, sequence=mapping, offset=offset) - with open(path, "w") as handler: + with open(path, "w", encoding="utf-8") as handler: yaml.dump(config, handler) def main(): - parser = argparse.ArgumentParser() - parser.add_argument( - "--target", - choices=["repo", "worker"], - help="Target to sync scripts to", - default="worker", - ) - args = parser.parse_args() - - if args.target == "repo": - print("Syncing scripts to repo") - pre_upgrade_path = REPO_PRE_UPGRADE_YAML_PATH - upgrade_path = REPO_UPGRADE_YAML_PATH - - elif args.target == "worker": - print("Syncing scripts to worker") - pre_upgrade_path = WORKER_PRE_UPGRADE_YAML_PATH - upgrade_path = WORKER_UPGRADE_YAML_PATH - - config, mapping, offset = _get_updated_yaml_content(pre_upgrade_path, SCRIPT_PATH) - print("Writing new content to %s" % pre_upgrade_path) - _write_content(config, pre_upgrade_path, mapping, offset) - config, mapping, offset = _get_updated_yaml_content(upgrade_path, SCRIPT_PATH) - print("Writing new content to %s" % upgrade_path) - _write_content(config, upgrade_path, mapping, offset) + if SYNC_PROJECT not in ("worker", "advisor", "tasks"): + raise SystemExit( + f"'{SYNC_PROJECT}' not recognized. Valid values are 'worker' or 'advisor'" + ) + + analysis_script, conversion_script = SCRIPTS_YAML_PATH[SYNC_PROJECT] + + config, mapping, offset = _get_updated_yaml_content(analysis_script, SCRIPT_PATH) + print(f"Writing new content to {analysis_script}") + _write_content(config, analysis_script, mapping, offset) + config, mapping, offset = _get_updated_yaml_content(conversion_script, SCRIPT_PATH) + print(f"Writing new content to {conversion_script}") + _write_content(config, conversion_script, mapping, offset) if __name__ == "__main__": diff --git a/playbooks/leapp_preupgrade_script.yml b/playbooks/leapp_preupgrade_script.yml index 6bce4b0..d6dd0ca 100644 --- a/playbooks/leapp_preupgrade_script.yml +++ b/playbooks/leapp_preupgrade_script.yml @@ -6,9 +6,15 @@ interpreter: /usr/bin/python content: | import json + import logging import os + import shutil + import sys import subprocess + from time import gmtime, strftime + + # SCRIPT_TYPE is either 'PREUPGRADE' or 'UPGRADE' # Value is set in signed yaml envelope in content_vars (RHC_WORKER_LEAPP_SCRIPT_TYPE) SCRIPT_TYPE = os.environ.get("RHC_WORKER_LEAPP_SCRIPT_TYPE", "None") @@ -37,8 +43,24 @@ } - # Both classes taken from: - # https://github.com/oamg/convert2rhel-worker-scripts/blob/main/scripts/preconversion_assessment_script.py + # Path to store the script logs + LOG_DIR = "/var/log/leapp-insights-tasks" + # Log filename for the script. It will be created based on the script type of + # execution. + LOG_FILENAME = "leapp-insights-tasks-%s.log" % ( + "upgrade" if IS_UPGRADE else "preupgrade" + ) + + # Path to the sos extras folder + SOS_REPORT_FOLDER = "/etc/sos.extras.d" + # Name of the file based on the task type for sos report + SOS_REPORT_FILE = "leapp-insights-tasks-%s-logs" % ( + "upgrade" if IS_UPGRADE else "preupgrade" + ) + + logger = logging.getLogger(__name__) + + class ProcessError(Exception): """Custom exception to report errors during setup and run of leapp""" @@ -94,9 +116,100 @@ } + def setup_sos_report(): + """Setup sos report log collection.""" + if not os.path.exists(SOS_REPORT_FOLDER): + os.makedirs(SOS_REPORT_FOLDER) + + script_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + sosreport_link_file = os.path.join(SOS_REPORT_FOLDER, SOS_REPORT_FILE) + # In case the file for sos report does not exist, lets create one and add + # the log file path to it. + if not os.path.exists(sosreport_link_file): + with open(sosreport_link_file, mode="w") as handler: + handler.write(":%s\n" % script_log_file) + + + def setup_logger_handler(): + """ + Setup custom logging levels, handlers, and so on. Call this method from + your application's main start point. + """ + # Receive the log level from the worker and try to parse it. If the log + # level is not compatible with what the logging library expects, set the + # log level to INFO automatically. + log_level = os.getenv("RHC_WORKER_LOG_LEVEL", "INFO").upper() + log_level = logging.getLevelName(log_level) + if isinstance(log_level, str): + log_level = logging.INFO + + # enable raising exceptions + logging.raiseExceptions = True + logger.setLevel(log_level) + + # create sys.stdout handler for info/debug + stdout_handler = logging.StreamHandler(sys.stdout) + formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s") + stdout_handler.setFormatter(formatter) + + # Create the directory if it don't exist + if not os.path.exists(LOG_DIR): + os.makedirs(LOG_DIR) + + log_filepath = os.path.join(LOG_DIR, LOG_FILENAME) + file_handler = logging.FileHandler(log_filepath) + file_handler.setFormatter(formatter) + + # can flush logs to the file that were logged before initializing the file handler + logger.addHandler(stdout_handler) + logger.addHandler(file_handler) + + + def archive_old_logger_files(): + """ + Archive the old log files to not mess with multiple runs outputs. Every + time a new run begins, this method will be called to archive the previous + logs if there is a `convert2rhel.log` file there, it will be archived using + the same name for the log file, but having an appended timestamp to it. + For example: + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162445070567607.log + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162478219820043.log + This way, the user can track the logs for each run individually based on + the timestamp. + """ + + current_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + archive_log_dir = os.path.join(LOG_DIR, "archive") + + # No log file found, that means it's a first run or it was manually deleted + if not os.path.exists(current_log_file): + return + + stat = os.stat(current_log_file) + + # Get the last modified time in UTC + last_modified_at = gmtime(stat.st_mtime) + + # Format time to a human-readable format + formatted_time = strftime("%Y%m%dT%H%M%SZ", last_modified_at) + + # Create the directory if it don't exist + if not os.path.exists(archive_log_dir): + os.makedirs(archive_log_dir) + + file_name, suffix = tuple(LOG_FILENAME.rsplit(".", 1)) + archive_log_file = "%s/%s-%s.%s" % ( + archive_log_dir, + file_name, + formatted_time, + suffix, + ) + shutil.move(current_log_file, archive_log_file) + + def get_rhel_version(): """Currently we execute the task only for RHEL 7 or 8""" - print("Checking OS distribution and version ID ...") + logger.info("Checking OS distribution and version ID ...") try: distribution_id = None version_id = None @@ -107,13 +220,13 @@ elif line.startswith("VERSION_ID="): version_id = line.split("=")[1].strip().strip('"') except IOError: - print("Couldn't read /etc/os-release") + logger.warn("Couldn't read /etc/os-release") return distribution_id, version_id def is_non_eligible_releases(release): """Check if the release is eligible for upgrade or preupgrade.""" - print("Exit if not RHEL 7 or RHEL 8 ...") + logger.info("Exit if not RHEL 7 or RHEL 8 ...") major_version, _ = release.split(".") if release is not None else (None, None) return release is None or major_version not in ALLOWED_RHEL_RELEASES @@ -136,7 +249,7 @@ raise TypeError("cmd should be a list, not a str") if print_cmd: - print("Calling command '%s'" % " ".join(cmd)) + logger.info("Calling command '%s'", " ".join(cmd)) process = subprocess.Popen( cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, bufsize=1, env=env @@ -232,9 +345,9 @@ def setup_leapp(version): leapp_install_command, rhel_rhui_packages = _get_leapp_command_and_packages(version) if _check_if_package_installed("leapp-upgrade"): - print("'leapp-upgrade' already installed, skipping ...") + logger.info("'leapp-upgrade' already installed, skipping ...") else: - print("Installing leapp ...") + logger.info("Installing leapp ...") output, returncode = run_subprocess(leapp_install_command) if returncode: raise ProcessError( @@ -243,7 +356,7 @@ % (returncode, output.rstrip("\n")), ) - print("Check installed rhui packages ...") + logger.info("Check installed rhui packages ...") for pkg in rhel_rhui_packages: if _check_if_package_installed(pkg["src_pkg"]): pkg["installed"] = True @@ -251,7 +364,7 @@ def should_use_no_rhsm_check(rhui_installed, command): - print("Checking if subscription manager and repositories are available ...") + logger.info("Checking if subscription manager and repositories are available ...") rhsm_repo_check_fail = True rhsm_installed_check = _check_if_package_installed("subscription-manager") if rhsm_installed_check: @@ -265,10 +378,9 @@ ) if rhui_installed and not rhsm_repo_check_fail: - print( - "RHUI packages detected, adding --no-rhsm flag to {} command".format( - SCRIPT_TYPE.title() - ) + logger.info( + "RHUI packages detected, adding --no-rhsm flag to % command", + SCRIPT_TYPE.title() ) command.append("--no-rhsm") return True @@ -276,7 +388,7 @@ def install_leapp_pkg_corresponding_to_installed_rhui(rhui_pkgs): - print("Installing leapp package corresponding to installed rhui packages") + logger.info("Installing leapp package corresponding to installed rhui packages") for pkg in rhui_pkgs: install_pkg = pkg["leapp_pkg"] install_output, returncode = run_subprocess( @@ -291,7 +403,7 @@ def remove_previous_reports(): - print("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") + logger.info("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") if os.path.exists(JSON_REPORT_PATH): os.remove(JSON_REPORT_PATH) @@ -301,7 +413,7 @@ def execute_operation(command): - print("Executing {} ...".format(SCRIPT_TYPE.title())) + logger.info("Executing %s ...", SCRIPT_TYPE.title()) output, _ = run_subprocess(command) return output @@ -311,7 +423,7 @@ """ Gather status codes from entries. """ - print("Collecting and combining report status.") + logger.info("Collecting and combining report status.") action_level_combined = [value["severity"] for value in entries] valid_action_levels = [ @@ -322,14 +434,14 @@ def parse_results(output, reboot_required=False): - print("Processing {} results ...".format(SCRIPT_TYPE.title())) + logger.info("Processing %s results ...", SCRIPT_TYPE.title()) report_json = "Not found" message = "Can't open json report at " + JSON_REPORT_PATH alert = True status = "ERROR" - print("Reading JSON report") + logger.info("Reading JSON report") if os.path.exists(JSON_REPORT_PATH): with open(JSON_REPORT_PATH, mode="r") as handler: report_json = json.load(handler) @@ -386,7 +498,7 @@ output.alert = alert output.message = message - print("Reading TXT report") + logger.info("Reading TXT report") report_txt = "Not found" if os.path.exists(TXT_REPORT_PATH): with open(TXT_REPORT_PATH, mode="r") as handler: @@ -397,24 +509,28 @@ def update_insights_inventory(output): """Call insights-client to update insights inventory.""" - print("Updating system status in Red Hat Insights.") + logger.info("Updating system status in Red Hat Insights.") _, returncode = run_subprocess(cmd=["/usr/bin/insights-client"]) if returncode: - print("System registration failed with exit code %s." % returncode) + logger.info("System registration failed with exit code %s.", returncode) output.message += " Failed to update Insights Inventory." output.alert = True return - print("System registered with insights-client successfully.") + logger.info("System registered with insights-client successfully.") def reboot_system(): - print("Rebooting system in 1 minute.") + logger.info("Rebooting system in 1 minute.") run_subprocess(["/usr/sbin/shutdown", "-r", "1"], wait=False) def main(): + """Main entrypoint for the script.""" + setup_sos_report() + archive_old_logger_files() + setup_logger_handler() try: # Exit if invalid value for SCRIPT_TYPE if SCRIPT_TYPE not in ["PREUPGRADE", "UPGRADE"]: @@ -452,11 +568,11 @@ upgrade_reboot_required = REBOOT_GUIDANCE_MESSAGE in leapp_output parse_results(output, upgrade_reboot_required) update_insights_inventory(output) - print("Operation {} finished successfully.".format(SCRIPT_TYPE.title())) + logger.info("Operation %s finished successfully.", SCRIPT_TYPE.title()) if upgrade_reboot_required: reboot_system() except ProcessError as exception: - print(exception.report) + logger.error(exception.report) output = OutputCollector( status="ERROR", alert=True, @@ -465,7 +581,7 @@ report=exception.report, ) except Exception as exception: - print(str(exception)) + logger.critical(str(exception)) output = OutputCollector( status="ERROR", alert=True, diff --git a/playbooks/leapp_upgrade_script.yml b/playbooks/leapp_upgrade_script.yml index d134e4a..66807e4 100644 --- a/playbooks/leapp_upgrade_script.yml +++ b/playbooks/leapp_upgrade_script.yml @@ -6,9 +6,15 @@ interpreter: /usr/bin/python content: | import json + import logging import os + import shutil + import sys import subprocess + from time import gmtime, strftime + + # SCRIPT_TYPE is either 'PREUPGRADE' or 'UPGRADE' # Value is set in signed yaml envelope in content_vars (RHC_WORKER_LEAPP_SCRIPT_TYPE) SCRIPT_TYPE = os.environ.get("RHC_WORKER_LEAPP_SCRIPT_TYPE", "None") @@ -37,8 +43,24 @@ } - # Both classes taken from: - # https://github.com/oamg/convert2rhel-worker-scripts/blob/main/scripts/preconversion_assessment_script.py + # Path to store the script logs + LOG_DIR = "/var/log/leapp-insights-tasks" + # Log filename for the script. It will be created based on the script type of + # execution. + LOG_FILENAME = "leapp-insights-tasks-%s.log" % ( + "upgrade" if IS_UPGRADE else "preupgrade" + ) + + # Path to the sos extras folder + SOS_REPORT_FOLDER = "/etc/sos.extras.d" + # Name of the file based on the task type for sos report + SOS_REPORT_FILE = "leapp-insights-tasks-%s-logs" % ( + "upgrade" if IS_UPGRADE else "preupgrade" + ) + + logger = logging.getLogger(__name__) + + class ProcessError(Exception): """Custom exception to report errors during setup and run of leapp""" @@ -94,9 +116,100 @@ } + def setup_sos_report(): + """Setup sos report log collection.""" + if not os.path.exists(SOS_REPORT_FOLDER): + os.makedirs(SOS_REPORT_FOLDER) + + script_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + sosreport_link_file = os.path.join(SOS_REPORT_FOLDER, SOS_REPORT_FILE) + # In case the file for sos report does not exist, lets create one and add + # the log file path to it. + if not os.path.exists(sosreport_link_file): + with open(sosreport_link_file, mode="w") as handler: + handler.write(":%s\n" % script_log_file) + + + def setup_logger_handler(): + """ + Setup custom logging levels, handlers, and so on. Call this method from + your application's main start point. + """ + # Receive the log level from the worker and try to parse it. If the log + # level is not compatible with what the logging library expects, set the + # log level to INFO automatically. + log_level = os.getenv("RHC_WORKER_LOG_LEVEL", "INFO").upper() + log_level = logging.getLevelName(log_level) + if isinstance(log_level, str): + log_level = logging.INFO + + # enable raising exceptions + logging.raiseExceptions = True + logger.setLevel(log_level) + + # create sys.stdout handler for info/debug + stdout_handler = logging.StreamHandler(sys.stdout) + formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s") + stdout_handler.setFormatter(formatter) + + # Create the directory if it don't exist + if not os.path.exists(LOG_DIR): + os.makedirs(LOG_DIR) + + log_filepath = os.path.join(LOG_DIR, LOG_FILENAME) + file_handler = logging.FileHandler(log_filepath) + file_handler.setFormatter(formatter) + + # can flush logs to the file that were logged before initializing the file handler + logger.addHandler(stdout_handler) + logger.addHandler(file_handler) + + + def archive_old_logger_files(): + """ + Archive the old log files to not mess with multiple runs outputs. Every + time a new run begins, this method will be called to archive the previous + logs if there is a `convert2rhel.log` file there, it will be archived using + the same name for the log file, but having an appended timestamp to it. + For example: + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162445070567607.log + /var/log/leapp-insights-tasks/archive/leapp-insights-tasks-1635162478219820043.log + This way, the user can track the logs for each run individually based on + the timestamp. + """ + + current_log_file = os.path.join(LOG_DIR, LOG_FILENAME) + archive_log_dir = os.path.join(LOG_DIR, "archive") + + # No log file found, that means it's a first run or it was manually deleted + if not os.path.exists(current_log_file): + return + + stat = os.stat(current_log_file) + + # Get the last modified time in UTC + last_modified_at = gmtime(stat.st_mtime) + + # Format time to a human-readable format + formatted_time = strftime("%Y%m%dT%H%M%SZ", last_modified_at) + + # Create the directory if it don't exist + if not os.path.exists(archive_log_dir): + os.makedirs(archive_log_dir) + + file_name, suffix = tuple(LOG_FILENAME.rsplit(".", 1)) + archive_log_file = "%s/%s-%s.%s" % ( + archive_log_dir, + file_name, + formatted_time, + suffix, + ) + shutil.move(current_log_file, archive_log_file) + + def get_rhel_version(): """Currently we execute the task only for RHEL 7 or 8""" - print("Checking OS distribution and version ID ...") + logger.info("Checking OS distribution and version ID ...") try: distribution_id = None version_id = None @@ -107,13 +220,13 @@ elif line.startswith("VERSION_ID="): version_id = line.split("=")[1].strip().strip('"') except IOError: - print("Couldn't read /etc/os-release") + logger.warn("Couldn't read /etc/os-release") return distribution_id, version_id def is_non_eligible_releases(release): """Check if the release is eligible for upgrade or preupgrade.""" - print("Exit if not RHEL 7 or RHEL 8 ...") + logger.info("Exit if not RHEL 7 or RHEL 8 ...") major_version, _ = release.split(".") if release is not None else (None, None) return release is None or major_version not in ALLOWED_RHEL_RELEASES @@ -136,7 +249,7 @@ raise TypeError("cmd should be a list, not a str") if print_cmd: - print("Calling command '%s'" % " ".join(cmd)) + logger.info("Calling command '%s'", " ".join(cmd)) process = subprocess.Popen( cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, bufsize=1, env=env @@ -232,9 +345,9 @@ def setup_leapp(version): leapp_install_command, rhel_rhui_packages = _get_leapp_command_and_packages(version) if _check_if_package_installed("leapp-upgrade"): - print("'leapp-upgrade' already installed, skipping ...") + logger.info("'leapp-upgrade' already installed, skipping ...") else: - print("Installing leapp ...") + logger.info("Installing leapp ...") output, returncode = run_subprocess(leapp_install_command) if returncode: raise ProcessError( @@ -243,7 +356,7 @@ % (returncode, output.rstrip("\n")), ) - print("Check installed rhui packages ...") + logger.info("Check installed rhui packages ...") for pkg in rhel_rhui_packages: if _check_if_package_installed(pkg["src_pkg"]): pkg["installed"] = True @@ -251,7 +364,7 @@ def should_use_no_rhsm_check(rhui_installed, command): - print("Checking if subscription manager and repositories are available ...") + logger.info("Checking if subscription manager and repositories are available ...") rhsm_repo_check_fail = True rhsm_installed_check = _check_if_package_installed("subscription-manager") if rhsm_installed_check: @@ -265,10 +378,9 @@ ) if rhui_installed and not rhsm_repo_check_fail: - print( - "RHUI packages detected, adding --no-rhsm flag to {} command".format( - SCRIPT_TYPE.title() - ) + logger.info( + "RHUI packages detected, adding --no-rhsm flag to % command", + SCRIPT_TYPE.title() ) command.append("--no-rhsm") return True @@ -276,7 +388,7 @@ def install_leapp_pkg_corresponding_to_installed_rhui(rhui_pkgs): - print("Installing leapp package corresponding to installed rhui packages") + logger.info("Installing leapp package corresponding to installed rhui packages") for pkg in rhui_pkgs: install_pkg = pkg["leapp_pkg"] install_output, returncode = run_subprocess( @@ -291,7 +403,7 @@ def remove_previous_reports(): - print("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") + logger.info("Removing previous leapp reports at /var/log/leapp/leapp-report.* ...") if os.path.exists(JSON_REPORT_PATH): os.remove(JSON_REPORT_PATH) @@ -301,7 +413,7 @@ def execute_operation(command): - print("Executing {} ...".format(SCRIPT_TYPE.title())) + logger.info("Executing %s ...", SCRIPT_TYPE.title()) output, _ = run_subprocess(command) return output @@ -311,7 +423,7 @@ """ Gather status codes from entries. """ - print("Collecting and combining report status.") + logger.info("Collecting and combining report status.") action_level_combined = [value["severity"] for value in entries] valid_action_levels = [ @@ -322,14 +434,14 @@ def parse_results(output, reboot_required=False): - print("Processing {} results ...".format(SCRIPT_TYPE.title())) + logger.info("Processing %s results ...", SCRIPT_TYPE.title()) report_json = "Not found" message = "Can't open json report at " + JSON_REPORT_PATH alert = True status = "ERROR" - print("Reading JSON report") + logger.info("Reading JSON report") if os.path.exists(JSON_REPORT_PATH): with open(JSON_REPORT_PATH, mode="r") as handler: report_json = json.load(handler) @@ -386,7 +498,7 @@ output.alert = alert output.message = message - print("Reading TXT report") + logger.info("Reading TXT report") report_txt = "Not found" if os.path.exists(TXT_REPORT_PATH): with open(TXT_REPORT_PATH, mode="r") as handler: @@ -397,24 +509,28 @@ def update_insights_inventory(output): """Call insights-client to update insights inventory.""" - print("Updating system status in Red Hat Insights.") + logger.info("Updating system status in Red Hat Insights.") _, returncode = run_subprocess(cmd=["/usr/bin/insights-client"]) if returncode: - print("System registration failed with exit code %s." % returncode) + logger.info("System registration failed with exit code %s.", returncode) output.message += " Failed to update Insights Inventory." output.alert = True return - print("System registered with insights-client successfully.") + logger.info("System registered with insights-client successfully.") def reboot_system(): - print("Rebooting system in 1 minute.") + logger.info("Rebooting system in 1 minute.") run_subprocess(["/usr/sbin/shutdown", "-r", "1"], wait=False) def main(): + """Main entrypoint for the script.""" + setup_sos_report() + archive_old_logger_files() + setup_logger_handler() try: # Exit if invalid value for SCRIPT_TYPE if SCRIPT_TYPE not in ["PREUPGRADE", "UPGRADE"]: @@ -452,11 +568,11 @@ upgrade_reboot_required = REBOOT_GUIDANCE_MESSAGE in leapp_output parse_results(output, upgrade_reboot_required) update_insights_inventory(output) - print("Operation {} finished successfully.".format(SCRIPT_TYPE.title())) + logger.info("Operation %s finished successfully.", SCRIPT_TYPE.title()) if upgrade_reboot_required: reboot_system() except ProcessError as exception: - print(exception.report) + logger.error(exception.report) output = OutputCollector( status="ERROR", alert=True, @@ -465,7 +581,7 @@ report=exception.report, ) except Exception as exception: - print(str(exception)) + logger.critical(str(exception)) output = OutputCollector( status="ERROR", alert=True, From b56dc26855fdb189dab206047ecd2c73a93f23eb Mon Sep 17 00:00:00 2001 From: Andrea Waltlova Date: Fri, 28 Jun 2024 09:45:56 +0200 Subject: [PATCH 4/4] Remove duplicate codecov action Signed-off-by: Andrea Waltlova --- .github/workflows/tests.yml | 6 ------ 1 file changed, 6 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 945bb20..e97c0a8 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -19,12 +19,6 @@ jobs: - name: Test run: python -m pytest --cov --cov-report=xml - - name: Upload coverage reports to Codecov - uses: codecov/codecov-action@v3 - with: - fail_ci_if_error: true - verbose: true # optional (default = false) - - name: Upload coverage to Codecov id: UploadFirstAttempt continue-on-error: true