From 6c8fea496e23267b633d4da09a69d1a526e58eb0 Mon Sep 17 00:00:00 2001 From: mk180007 Date: Mon, 24 Aug 2020 15:45:39 -0700 Subject: [PATCH 1/3] FEATURE: ReportSys test for SLES pallet patch --- .../report/system/tests/test_sles_patch.py | 124 ++++++++++++++++++ 1 file changed, 124 insertions(+) create mode 100644 common/src/stack/report-system/command/report/system/tests/test_sles_patch.py diff --git a/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py b/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py new file mode 100644 index 000000000..5d5f6c92f --- /dev/null +++ b/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py @@ -0,0 +1,124 @@ +import pytest +import os +import stack.api +import hashlib +import json + + +# Returns hash of filename +def get_hash(filename, sles): + # SLES15 uses sha256 + if '15sp' in sles: + hasher = hashlib.sha256() + else: + hasher = hashlib.md5() + + with open(filename, 'rb') as f: + hasher.update(f.read()) + + return hasher.hexdigest() + + +# Verify SLES patches are installed +def test_sles_pallet_patched(host, report_output): + total_matched = [] + base_dir = '/export/stack/pallets/SLES/' + if not os.path.isdir(base_dir): + pytest.skip('No SLES pallet found - skipping patch check') + sles_flavors = os.listdir(base_dir) + assert sles_flavors + + # Find out where is stack-sles*images*.rpm file(s) + result = host.run('find /export/stack/pallets/stacki/ -name "*stack-sles-*.rpm"') + RPM = result.stdout.splitlines() + assert RPM + + # If the stack-sles*images*.rpm installed? + result = host.run('rpm -qa | grep stack | grep images') + assert result + + # Test every sles flavor found + for sles_flavor in sles_flavors: + # Make sure installed patches match what's under /opt/stack/pallet-patches + patch_dir = '/opt/stack/pallet-patches' + patch_dir_files = [] + found_source = False + for (dir_path, dir_names, filenames) in os.walk(patch_dir): + # Only want particular SLES version + if sles_flavor in dir_path: + file_to_check = None + if '15sp' in sles_flavor: + file_to_check = 'CHECKSUMS' + else: + file_to_check = 'content' + patch_dir_files += [os.path.join(dir_path, file) for file in filenames if file == file_to_check] + + # We should have non-empty list here + assert patch_dir_files + + # Find img file from patch directory in SLES pallet + result = host.run(f'grep .img {patch_dir_files[0]}') + + # Last item is the partial image path + part_img_file = result.stdout.split()[-1] + + # Find full path to /export/stack/pallets/SLES/? + result = host.run(f'probepal {base_dir}') + assert result.rc == 0 + palinfo = json.loads(result.stdout) + + # It shouldn't be empty + assert palinfo[base_dir] + + sles_pallet_root = None + for pallet in palinfo[base_dir]: + # Grab the right sles sp level + if pallet['version'] == sles_flavor: + assert host.file(f"{pallet['pallet_root']}").is_directory + sles_pallet_root = pallet['pallet_root'] + break + + # .img file should exist in sles pallet directory in relative path + assert os.path.exists(sles_pallet_root + '/' + part_img_file) + + # Verify all the stack-sles-* rpm packages + for rpm in RPM: + # Make sure we found pm file + assert '.rpm' in rpm + + result = host.run(f'rpm -qp --dump {rpm}') + # We don't want file under /opt/stack/images to be included in this list + patch_files = [line.split() for line in result.stdout.splitlines() if sles_flavor in line and '/images/' not in line] + + matched_list = [] + # Verify file(s) from images RPM actually exist + for this_list in patch_files: + file_to_check = this_list[0] + assert os.path.exists(file_to_check) + hash_value = this_list[3] + + # Grab hash for this file in patch directory and SLES pallet directory and compare against hash from img file. They should match + # Need to build the equivalent path first + temp_path_list = file_to_check.split('add-stacki-squashfs') + temp_path = sles_pallet_root + temp_path_list[1] + if get_hash(file_to_check,sles_flavor) == hash_value and get_hash(temp_path, sles_flavor) == hash_value: + matched_list.append(file_to_check) + + # Check if we found hash match as expected. Should be 4 files. + if matched_list and len(matched_list) == 4: + found_source = True + # trim the string + temp_path = rpm.split('stacki/')[1] + temp_path_list = temp_path.split('/') + version = temp_path_list[0] + os_type = temp_path_list[1] + version_string = 'stacki-' + version + '-' + os_type + '-sles-x86_64' + # Don't want duplicate entry + if version_string not in total_matched: + total_matched.append(version_string) + + # Didn't find stacki source for this os + assert found_source == True + + for match in total_matched: + report_output('SLES pallet patch source', match) From 7fa09c134aab7f613fa3993a01105d403518b6a5 Mon Sep 17 00:00:00 2001 From: Meg M Kido Date: Wed, 9 Sep 2020 02:38:08 +0000 Subject: [PATCH 2/3] update --- .../stack/argument_processors/pallet.py | 30 +++++++++++++------ 1 file changed, 21 insertions(+), 9 deletions(-) diff --git a/common/src/stack/command/stack/argument_processors/pallet.py b/common/src/stack/command/stack/argument_processors/pallet.py index b00b00a0f..e66d44e08 100644 --- a/common/src/stack/command/stack/argument_processors/pallet.py +++ b/common/src/stack/command/stack/argument_processors/pallet.py @@ -6,6 +6,7 @@ import subprocess from stack.commands import Log from stack.exception import ArgNotFound +import logging # Pallet info from Probepal and the database columns do not have the same names. # They do have a mapping that ends up being the same values as one another. So @@ -138,11 +139,22 @@ def _get_pallet_hooks(self, operation, pallet_info): except FileNotFoundError as exception: self.notify(f"{script_file} specified in {hook_file} not found:\n\n{exception}") + def log_routine(self, msg, log_methods): + [ _logger(msg) for _logger in log_methods ] + def run_pallet_hooks(self, operation, pallet_info): + # Setup the logger + logging.basicConfig( + filename="/var/log/run-pallet.log", + format="%(asctime)s: %(message)s", + datefmt="%Y-%m-%d %H:%M:%S", + level=logging.DEBUG + ) + logger = logging.getLogger(__name__) + """Run the hooks for the pallet operation in progress.""" pallet_info = self._normalize_pallet_info(pallet_info) - self.notify(f'checking for hooks in {PALLET_HOOK_ROOT} for {"-".join(pallet_info_getter(pallet_info))}') - + self.log_routine(f'checking for hooks in {PALLET_HOOK_ROOT} for {"-".join(pallet_info_getter(pallet_info))}', [self.notify, logger.info]) # Find all executable files within the script directory and sort them by name. hooks = sorted( self._get_pallet_hooks(operation=operation, pallet_info=pallet_info), @@ -150,22 +162,22 @@ def run_pallet_hooks(self, operation, pallet_info): ) # Execute the hooks in name sorted order. for hook in hooks: - self.notify(f'running hook: {hook}') - Log(f'Running {operation} pallet hook for pallet {"-".join(pallet_info_getter(pallet_info))}: {hook} ') + self.log_routine(f'running hook: {hook}', [self.notify]) + self.log_routine(f'Running {operation} pallet hook for pallet {"-".join(pallet_info_getter(pallet_info))}: {hook} ', [Log, logger.info]) try: # subprocess's env mapping must be strings to strings! environ = dict((str(k), str(v)) for k, v in asdict(pallet_info).items()) result = self._exec(['/usr/bin/env', str(hook)], cwd=hook.parent, check=True, env=environ) - Log(f'Result of running {hook} (rc=={result.returncode}):') - Log(f'Env:\n{environ}') - Log(f'Stdout:\n{result.stdout}') - Log(f'Stderr:\n{result.stderr}') + self.log_routine(f'Result of running {hook} (rc=={result.returncode}):', [Log, logger.info]) + self.log_routine(f'Env:\n{environ}', [Log, logger.info]) + self.log_routine(f'Stdout:\n{result.stdout}', [Log, logger.info]) + self.log_routine(f'Stderr:\n{result.stderr}', [Log, logger.info]) except (PermissionError, subprocess.CalledProcessError) as exception: msg = f'Unable to run hook {hook}:\n\n{exception}\n' # CalledProcessError has additional info... if hasattr(exception, 'stdout'): msg += f'\nstdout:\n{exception.stdout}\n\nstderr:\n{exception.stderr}' - self.notify(msg) + self.log_routine(msg, [self.notify, logger.warning]) def get_pallet_hook_directory(self, pallet_info): """Calculate the hook directory for a given pallet.""" From 8ff53a23e473def22b9ba1852075699eaab042e6 Mon Sep 17 00:00:00 2001 From: Meg M Kido Date: Tue, 6 Oct 2020 21:00:26 +0000 Subject: [PATCH 3/3] update --- .../report/system/tests/test_sles_patch.py | 115 ++++++++++++------ 1 file changed, 78 insertions(+), 37 deletions(-) diff --git a/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py b/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py index 5d5f6c92f..535aa69f3 100644 --- a/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py +++ b/common/src/stack/report-system/command/report/system/tests/test_sles_patch.py @@ -3,7 +3,9 @@ import stack.api import hashlib import json - +import pathlib +import pprint +import collections # Returns hash of filename def get_hash(filename, sles): @@ -18,10 +20,11 @@ def get_hash(filename, sles): return hasher.hexdigest() +def nested_dict(): + return collections.defaultdict(nested_dict) # Verify SLES patches are installed def test_sles_pallet_patched(host, report_output): - total_matched = [] base_dir = '/export/stack/pallets/SLES/' if not os.path.isdir(base_dir): pytest.skip('No SLES pallet found - skipping patch check') @@ -35,32 +38,35 @@ def test_sles_pallet_patched(host, report_output): # If the stack-sles*images*.rpm installed? result = host.run('rpm -qa | grep stack | grep images') - assert result + assert result.rc == 0 + + overall_dict = nested_dict() # Test every sles flavor found for sles_flavor in sles_flavors: # Make sure installed patches match what's under /opt/stack/pallet-patches patch_dir = '/opt/stack/pallet-patches' patch_dir_files = [] - found_source = False + not_matched_list = [] for (dir_path, dir_names, filenames) in os.walk(patch_dir): # Only want particular SLES version if sles_flavor in dir_path: - file_to_check = None if '15sp' in sles_flavor: - file_to_check = 'CHECKSUMS' + pattern = f'SLES-{sles_flavor}-*/**/CHECKSUMS' else: - file_to_check = 'content' - patch_dir_files += [os.path.join(dir_path, file) for file in filenames if file == file_to_check] + pattern = f'SLES-{sles_flavor}-*/**/content' - # We should have non-empty list here - assert patch_dir_files + # SLES-12sp3-sles12-sles-x86_64 + patch_dir_files = list(pathlib.Path(patch_dir).glob(pattern)) - # Find img file from patch directory in SLES pallet - result = host.run(f'grep .img {patch_dir_files[0]}') + # We should have 1 file here + assert len(patch_dir_files) == 1 # Last item is the partial image path - part_img_file = result.stdout.split()[-1] + part_img_file = [ + line for line in patch_dir_files[0].read_text().splitlines() + if line.endswith('.img') + ][-1].split()[-1] # Find full path to /export/stack/pallets/SLES/? result = host.run(f'probepal {base_dir}') @@ -70,7 +76,7 @@ def test_sles_pallet_patched(host, report_output): # It shouldn't be empty assert palinfo[base_dir] - sles_pallet_root = None + #sles_pallet_root = None for pallet in palinfo[base_dir]: # Grab the right sles sp level if pallet['version'] == sles_flavor: @@ -83,14 +89,16 @@ def test_sles_pallet_patched(host, report_output): # Verify all the stack-sles-* rpm packages for rpm in RPM: - # Make sure we found pm file - assert '.rpm' in rpm - result = host.run(f'rpm -qp --dump {rpm}') # We don't want file under /opt/stack/images to be included in this list - patch_files = [line.split() for line in result.stdout.splitlines() if sles_flavor in line and '/images/' not in line] + patch_files = [ + line.split() for line in result.stdout.splitlines() + if sles_flavor in line + if '/add-stacki-squashfs/' in line + ] matched_list = [] + not_matched_list = [] # Verify file(s) from images RPM actually exist for this_list in patch_files: file_to_check = this_list[0] @@ -103,22 +111,55 @@ def test_sles_pallet_patched(host, report_output): temp_path = sles_pallet_root + temp_path_list[1] if get_hash(file_to_check,sles_flavor) == hash_value and get_hash(temp_path, sles_flavor) == hash_value: matched_list.append(file_to_check) - - # Check if we found hash match as expected. Should be 4 files. - if matched_list and len(matched_list) == 4: - found_source = True - # trim the string - temp_path = rpm.split('stacki/')[1] - temp_path_list = temp_path.split('/') - version = temp_path_list[0] - os_type = temp_path_list[1] - version_string = 'stacki-' + version + '-' + os_type + '-sles-x86_64' - # Don't want duplicate entry - if version_string not in total_matched: - total_matched.append(version_string) - - # Didn't find stacki source for this os - assert found_source == True - - for match in total_matched: - report_output('SLES pallet patch source', match) + else: + not_matched_list.append(file_to_check) + + # Add info we have into nested dictionary + good_dict = {rpm: []} + bad_dict = {rpm: []} + # Check if we found all the hash matches + if matched_list and len(matched_list) == len(patch_files): + # this indicates if hash match was found for a sles flavor or not + good_dict[rpm].append(matched_list) + overall_dict[sles_flavor]['good'].update({rpm: matched_list}) + elif not_matched_list:# and len(not_matched_list) == len(patch_files): + bad_dict[rpm].append(not_matched_list) + overall_dict[sles_flavor]['bad'].update({rpm: not_matched_list}) + + # Print entire dictionary for debug + #pp = pprint.PrettyPrinter(indent=4) + #pp.pprint(overall_dict) + + assert overall_dict + # over all test result + final_status = True + for os_, results in overall_dict.items(): + success = False + if 'good' in results: + success = True + for result, rpms in results.items(): + for rpm, filelist in rpms.items(): + if result == 'good': + temp_path = rpm.split('stacki/')[1] + temp_path_list = temp_path.split('/') + version = temp_path_list[0] + os_type = temp_path_list[1] + version_string = 'stacki-' + version + '-' + os_type + '-sles-x86_64' + output = version_string + '\n' + report_output(f'SLES{os_} pallet patch source', output) + else: + # if we already found good case then skip all the bad cases + if success: + continue; + else: + # overall status is bad if we find one valid bad case + final_status = False + output = '' + for val in rpms.get(rpm): + if len(output) == 0: + output = val + else: + output = output + '\n' + val + output += '\n' + report_output(f'SLES{os_} pallet patch source NOT found', output) + assert final_status