#!/usr/bin/env python # -*- coding: utf-8 -*- # import fstools import report import reqif import json import os import platform import getpass import sys import subprocess import imp import xml.dom.minidom try: import jinja2 except ImportError: jinja2 = None import shutil HEADER = '\033[95m' OKBLUE = '\033[94m' OKGREEN = '\033[92m' WARNING = '\033[93m' FAIL = '\033[91m' ENDC = '\033[0m' BOLD = '\033[1m' UNDERLINE = '\033[4m' ARG_CLEAN = 'clean' ARG_RUN = 'run' ARG_FINALISE = 'finalise' ARG_PDF = 'pdf' ARG_STATUS = 'status' ARG_COPY = 'copy' ARG_RELEASE = 'release' UNITTEST_KEY_SYSTEM_INFO = 'system_information' UNITTEST_KEY_UNITTEST_INFO = 'unittest_information' UNITTEST_KEY_TESTOBJECT_INFO = 'testobject_information' UNITTEST_KEY_TESTRUNS = 'testrun_list' UNITTEST_KEY_COVERAGE_INFO = 'coverage_information' UNITTEST_KEY_SPECIFICATION = 'specification' FILES = { 'data-collection': 'unittest.json', 'tex-report': 'unittest.tex', 'coverage-xml': 'coverage.xml' } REPORT_FILES = [FILES['data-collection'], FILES['coverage-xml'], 'unittest.pdf'] class coverage_info(list): KEY_NAME = 'name' KEY_FILEPATH = 'filepath' KEY_LINE_COVERAGE = 'line_coverage' KEY_BRANCH_COVERAGE = 'branch_coverage' KEY_FILES = 'files' KEY_FRAGMENTS = 'fragments' KEY_START_LINE = 'start' KEY_END_LINE = 'end' KEY_COVERAGE_STATE = 'coverage_state' COVERED = 'covered' UNCOVERED = 'uncovered' CLEAN = 'clean' PARTIALLY_COVERED = 'partially-covered' def __init__(self, xml_filename, module_basepath): list.__init__(self) xmldoc = xml.dom.minidom.parse(xml_filename) itemlist = xmldoc.getElementsByTagName('package') for p in itemlist: module = {} module[self.KEY_NAME] = p.attributes['name'].value[len(module_basepath) + 1:] module[self.KEY_FILEPATH] = p.attributes['name'].value.replace('.', os.path.sep) module[self.KEY_LINE_COVERAGE] = float(p.attributes['line-rate'].value) * 100. try: module[self.KEY_BRANCH_COVERAGE] = float(p.attributes['branch-rate'].value) * 100. except AttributeError: module[self.KEY_BRANCH_COVERAGE] = None module[self.KEY_FILES] = [] for c in p.getElementsByTagName('class'): f = {} f[self.KEY_NAME] = c.attributes['filename'].value[len(module_basepath) + 1:].replace(os.path.sep, '.') f[self.KEY_FILEPATH] = c.attributes['filename'].value f[self.KEY_LINE_COVERAGE] = float(c.attributes['line-rate'].value) * 100. try: f[self.KEY_BRANCH_COVERAGE] = float(p.attributes['branch-rate'].value) * 100. except: f[self.KEY_BRANCH_COVERAGE] = None f[self.KEY_FRAGMENTS] = [] last_hit = None start_line = 1 end_line = 1 for line in c.getElementsByTagName('line'): line_no = int(line.attributes['number'].value) hit = bool(int(line.attributes['hits'].value)) if hit: cc = line.attributes.get('condition-coverage') if cc is not None and not cc.value.startswith('100%'): hit = self.PARTIALLY_COVERED else: hit = self.COVERED else: hit = self.UNCOVERED if line_no == 1: last_hit = hit elif last_hit != hit or line_no > end_line + 1: if last_hit is not None: line = {} line[self.KEY_START_LINE] = start_line line[self.KEY_END_LINE] = end_line line[self.KEY_COVERAGE_STATE] = last_hit f[self.KEY_FRAGMENTS].append(line) if line_no > end_line + 1: line = {} if last_hit is not None: line[self.KEY_START_LINE] = end_line + 1 else: line[self.KEY_START_LINE] = start_line line[self.KEY_END_LINE] = line_no - 1 line[self.KEY_COVERAGE_STATE] = self.CLEAN f[self.KEY_FRAGMENTS].append(line) start_line = line_no end_line = line_no last_hit = hit elif line_no == end_line + 1: end_line = line_no if last_hit is not None: line = {} line[self.KEY_START_LINE] = start_line line[self.KEY_END_LINE] = end_line line[self.KEY_COVERAGE_STATE] = last_hit f[self.KEY_FRAGMENTS].append(line) line = {} if last_hit is not None: line[self.KEY_START_LINE] = end_line + 1 else: line[self.KEY_START_LINE] = start_line line[self.KEY_END_LINE] = None line[self.KEY_COVERAGE_STATE] = self.CLEAN f[self.KEY_FRAGMENTS].append(line) module[self.KEY_FILES].append(f) self.append(module) def __str__(self): rv = '' for module in self: rv += '%s (%.1f%% - %s)\n' % (module.get(self.KEY_NAME), module.get(self.KEY_LINE_COVERAGE), module.get(self.KEY_FILEPATH)) for py_file in module.get(self.KEY_FILES): rv += ' %s (%.1f%% - %s)\n' % (py_file.get(self.KEY_NAME), py_file.get(self.KEY_LINE_COVERAGE), py_file.get(self.KEY_FILEPATH)) for fragment in py_file.get(self.KEY_FRAGMENTS): if fragment.get(self.KEY_END_LINE) is not None: rv += ' %d - %d: %s\n' % (fragment.get(self.KEY_START_LINE), fragment.get(self.KEY_END_LINE), repr(fragment.get(self.KEY_COVERAGE_STATE))) else: rv += ' %d - : %s\n' % (fragment.get(self.KEY_START_LINE), repr(fragment.get(self.KEY_COVERAGE_STATE))) return rv def unittest_filename(base_folder, filename): return os.path.join(base_folder, 'testresults', filename) def print_header(txt, color=BOLD + WARNING): print(color + txt + ENDC) def print_action(txt, color=BOLD): print(color + ' * ' + txt + ENDC) def print_info(txt, color=ENDC): print(' ' + color + txt + ENDC) def remove_file(filename): if os.path.exists(filename) and not filename.endswith('.gitkeep'): try: print_info('Removing %s' % filename) os.remove(filename) except OSError: pass def module_uid(path): return fstools.uid_filelist(path, '*.py', rekursive=True) def unittest(options, args, unittest_folder): if ARG_CLEAN in args: unittest_init(unittest_folder) elif ARG_RUN in args: unittest_run(unittest_folder, options) elif ARG_FINALISE in args: unittest_finalise(unittest_folder) elif ARG_PDF in args: unittest_pdf(unittest_folder) elif ARG_STATUS in args: unittest_status(unittest_folder) elif ARG_COPY in args: unittest_copy(unittest_folder) elif ARG_RELEASE in args: unittest_release(unittest_folder) def unittest_init(unittest_folder): config = imp.load_source('', os.path.join(unittest_folder, 'src', 'config.py')) # print_header("Initiating unittest for first testrun...") if not os.path.exists(unittest_filename(unittest_folder, '')): print_action('Creating outpout folder %s' % unittest_filename(unittest_folder, '')) fstools.mkdir(unittest_filename(unittest_folder, '')) # print_action('Cleaning up data from last testrun') for fn in os.listdir(unittest_filename(unittest_folder, '')): remove_file(unittest_filename(unittest_folder, fn)) remove_file(unittest_filename(unittest_folder, FILES['coverage-xml'])) # print_action('Creating unittest data-collection: %s' % unittest_filename(unittest_folder, FILES['data-collection'])) # system_info = {} system_info['Architecture'] = platform.architecture()[0] system_info['Machine'] = platform.machine() system_info['Hostname'] = platform.node() system_info['Distribution'] = ' '.join(platform.dist()) system_info['System'] = platform.system() system_info['Kernel'] = platform.release() + ' (%s)' % platform.version() system_info['Username'] = getpass.getuser() system_info['Path'] = unittest_folder # unittest_info = {} unittest_info['Version'] = module_uid(os.path.join(unittest_folder, 'src', 'tests')) # testobject_info = {} testobject_info['Name'] = config.lib.__name__ testobject_info['Version'] = module_uid(config.lib.__path__[0]) testobject_info['Description'] = config.lib.__DESCRIPTION__ testobject_info['Supported Interpreters'] = ', '.join(['python%d' % vers for vers in config.lib.__INTERPRETER__]) testobject_info['State'] = 'Released' if config.release_unittest_version == module_uid(os.path.join(unittest_folder, 'src', 'tests')) else 'In development' testobject_info['Dependencies'] = [] for dependency in config.lib.__DEPENDENCIES__: testobject_info['Dependencies'].append((dependency, module_uid(os.path.join(unittest_folder, 'src', dependency)))) # spec_filename = os.path.join(unittest_folder, '..', 'requirements', 'specification.reqif') print_action("Adding Requirement Specification from %s" % spec_filename) try: spec = reqif.reqif_dict(spec_filename, 'Heading', 'Software Specification') except FileNotFoundError: print_info('FAILED', FAIL) spec = {} else: print_info('SUCCESS', OKGREEN) # data_collection = { UNITTEST_KEY_SYSTEM_INFO: system_info, UNITTEST_KEY_UNITTEST_INFO: unittest_info, UNITTEST_KEY_TESTOBJECT_INFO: testobject_info, UNITTEST_KEY_SPECIFICATION: spec, UNITTEST_KEY_TESTRUNS: [], } with open(unittest_filename(unittest_folder, FILES['data-collection']), 'w') as fh: fh.write(json.dumps(data_collection, indent=4, sort_keys=True)) def unittest_run(unittest_folder, options): tests = imp.load_source('', os.path.join(unittest_folder, 'src', 'tests', '__init__.py')) config = imp.load_source('', os.path.join(unittest_folder, 'src', 'config.py')) # interpreter_version = 'python ' + '.'.join(['%d' % n for n in sys.version_info[:3]]) + ' (%s)' % sys.version_info[3] # execution_level = report.TCEL_REVERSE_NAMED.get(options.execution_level, report.TCEL_FULL) # if sys.version_info.major in config.lib.__INTERPRETER__: print_header("Running \"%s\" Unittest with %s" % (options.execution_level, interpreter_version)) print_action('Loading Testrun data from %s' % unittest_filename(unittest_folder, FILES['data-collection'])) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'r') as fh: data_collection = json.loads(fh.read()) print_action('Executing Testcases') heading_dict = {} for key in data_collection[UNITTEST_KEY_SPECIFICATION].get('item_dict', {}): heading_dict[key] = data_collection[UNITTEST_KEY_SPECIFICATION]['item_dict'][key]['Heading'] test_session = report.testSession( ['__unittest__', config.lib.logger_name] + config.additional_loggers_to_catch, interpreter=interpreter_version, testcase_execution_level=execution_level, testrun_id='p%d' % sys.version_info[0], heading_dict=heading_dict ) tests.testrun(test_session) # print_action('Adding Testrun data to %s' % unittest_filename(unittest_folder, FILES['data-collection'])) data_collection[UNITTEST_KEY_TESTRUNS].append(test_session) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'w') as fh: fh.write(json.dumps(data_collection, indent=4, sort_keys=True)) else: print_header("Library does not support %s." % interpreter_version) def unittest_finalise(unittest_folder): config = imp.load_source('', os.path.join(unittest_folder, 'src', 'config.py')) # print_action('Adding Testrun data to %s' % unittest_filename(unittest_folder, FILES['data-collection'])) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'r') as fh: data_collection = json.loads(fh.read()) # print_header("Adding Requirement information") # data_collection['lost_souls'] = {} # print_action("Adding Lost Requirement Soul") data_collection['lost_souls']['item_list'] = [] for req_id in data_collection['specification'].get('item_dict', {}): item = data_collection['specification']['item_dict'][req_id] if item['system_type_uid'] == '_MR7eNHYYEem_kd-7nxt1sg': testcase_available = False for testrun in data_collection['testrun_list']: if req_id in testrun['testcases']: testcase_available = True break if not testcase_available: data_collection['lost_souls']['item_list'].append(req_id) print_info('%s - "%s" has no corresponding testcase' % (item['system_uid'], item['Heading']), FAIL) # print_action("Adding Lost Testcase Soul") data_collection['lost_souls']['testcase_list'] = [] for testrun in data_collection['testrun_list']: for tc_id in testrun.get('testcases', {}): if tc_id not in data_collection['specification'].get('item_dict', {}) and tc_id not in data_collection['lost_souls']['testcase_list']: data_collection['lost_souls']['testcase_list'].append(tc_id) print_info('"%s" has no corresponding testcase' % tc_id, FAIL) # print_header("Adding Coverage information") print_action('Adding Coverage Information to %s' % unittest_filename(unittest_folder, FILES['data-collection'])) data_collection[UNITTEST_KEY_COVERAGE_INFO] = coverage_info(unittest_filename(unittest_folder, 'coverage.xml'), os.path.dirname(config.lib_path)) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'w') as fh: fh.write(json.dumps(data_collection, indent=4, sort_keys=True)) def unittest_pdf(unittest_folder): print_header("Creating PDF-Report of Unittest") print_action('Loading Testrun data from %s' % unittest_filename(unittest_folder, FILES['data-collection'])) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'r') as fh: data_collection = json.loads(fh.read()) if jinja2 is None: print_action('You need to install jinja2 to create a PDF-Report!', FAIL) else: fn = unittest_filename(unittest_folder, FILES['tex-report']) print_action('Creating LaTeX-File %s' % fn) with open(fn, 'w') as fh: # template_path = os.path.join(os.path.dirname(__file__), 'templates') template_filename = 'unittest.tex' jenv = jinja2.Environment(loader=jinja2.FileSystemLoader(template_path)) template = jenv.get_template(template_filename) fh.write(template.render(data=data_collection)) print_action('Creating PDF %s' % unittest_filename(unittest_folder, 'unittest.pdf')) for i in range(3): sys.stdout.write(' Starting run %d/3 of pdflatex... ' % (i + 1)) sys.stdout.flush() exit_value = os.system("pdflatex -interaction nonstopmode --output-directory %(path)s %(path)s/unittest.tex 1> /dev/null" % {'path': unittest_filename(unittest_folder, '')}) if exit_value != 0: print(FAIL + 'FAILED' + ENDC) break else: print(OKGREEN + 'SUCCESS' + ENDC) def unittest_status(unittest_folder): config = imp.load_source('', os.path.join(unittest_folder, 'src', 'config.py')) # print_header('Checking status of all submodules') print_action('Updating all submodules (fetch)') process = subprocess.Popen("LANGUAGE='en_US.UTF-8 git' git submodule foreach git fetch", cwd=os.path.dirname(unittest_folder), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) stderroutput = process.communicate()[1] if stderroutput == b'': print_info('SUCCESS', color=OKGREEN) else: print_info('FAILED', color=FAIL) print_action('Checking status...') process = subprocess.Popen("LANGUAGE='en_US.UTF-8 git' git submodule foreach git status", cwd=os.path.dirname(unittest_folder), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) stdoutput, stderroutput = process.communicate() if stderroutput == b'': module = None data = {} for line in stdoutput.splitlines(): line = str(line) if 'Entering' in line: m = line[line.index("'") + 1:] m = str(m[:m.index("'")]) if m != module: data[m] = '' module = m else: data[m] += line for key in data: if "working tree clean" not in data[key] and "working directory clean" not in data[key]: data[key] = ("local changes", WARNING) elif "Your branch is behind" in data[key]: data[key] = ("no up to date (try git pull)", FAIL) elif "HEAD detached at" in data[key]: data[key] = ("no up to date (try git checkout master)", FAIL) elif "Your branch is ahead of" in data[key]: data[key] = ("push required", WARNING) elif "nothing to commit" in data[key]: data[key] = ("clean", OKGREEN) else: data[key] = ("unknown", FAIL) print_info('Submodule %s... %s' % (key, data[key][1] + data[key][0])) else: print_info('FAILED', color=FAIL) # print_header('Checking status of unittest and testresults in the library') print_action('Loading Testrun data from %s' % unittest_filename(unittest_folder, FILES['data-collection'])) with open(unittest_filename(unittest_folder, FILES['data-collection']), 'r') as fh: data_collection = json.loads(fh.read()) print_action('Checking release state of this testrun... ') if data_collection['testobject_information']['State'] != 'Released': print_info("FAILED", FAIL) else: print_info("SUCCESS", OKGREEN) # print_action('Checking up to dateness of testrults in library...') try: with open(os.path.join(unittest_folder, '..', 'pylibs', config.lib.__name__, '_testresults_', FILES['data-collection']), 'r') as fh: lib_result = json.loads(fh.read()) except FileNotFoundError: print_info("FAILED: Testresults not in library", FAIL) else: if data_collection['testobject_information'] != lib_result['testobject_information'] or data_collection['unittest_information'] != lib_result['unittest_information']: print_info("FAILED", FAIL) else: print_info("SUCCESS", OKGREEN) def unittest_copy(unittest_folder): config = imp.load_source('', os.path.join(unittest_folder, 'src', 'config.py')) # print_header('Copy unittest files to library') target_folder = os.path.join(config.lib_path, '_testresults_') print_action('Copying Unittest Files to %s' % target_folder) if not os.path.exists(target_folder): print_info('Creating folder %s' % target_folder) fstools.mkdir(target_folder) else: for fn in os.listdir(target_folder): remove_file(os.path.join(target_folder, fn)) for fn in REPORT_FILES: src = unittest_filename(unittest_folder, fn) dst = os.path.join(target_folder, fn) print_info('copying %s -> %s' % (src, dst)) shutil.copyfile(src, dst) def unittest_release(unittest_folder): unittest_uid = module_uid(os.path.join(unittest_folder, 'src', 'tests')) config_file = os.path.join(unittest_folder, 'src', 'config.py') print_header('Releasing unittest') with open(config_file, 'r') as fh: conf_file = fh.read() print_action('Setting release_unittest_version = %s in %s' % (unittest_uid, config_file)) with open(config_file, 'w') as fh: for line in conf_file.splitlines(): if line.startswith('release_unittest_version'): fh.write("release_unittest_version = '%s'\n" % unittest_uid) else: fh.write(line + '\n')