import logging import os import signal import subprocess # Create Cppcheck project file import sys def create_gui_project_file(project_file, root_path=None, import_project=None, paths=None, exclude_paths=None, suppressions=None, addon=None): cppcheck_xml = ('\n' '\n') if root_path: cppcheck_xml += ' \n' if import_project: cppcheck_xml += ' ' + import_project + '\n' if paths: cppcheck_xml += ' \n' for path in paths: cppcheck_xml += ' \n' cppcheck_xml += ' \n' if exclude_paths: cppcheck_xml += ' \n' for path in exclude_paths: cppcheck_xml += ' \n' cppcheck_xml += ' \n' if suppressions: cppcheck_xml += ' \n' for suppression in suppressions: cppcheck_xml += ' \n' cppcheck_xml += ' \n' if addon: cppcheck_xml += ' \n' cppcheck_xml += ' %s\n' % addon cppcheck_xml += ' \n' cppcheck_xml += '\n' f = open(project_file, 'wt') f.write(cppcheck_xml) f.close() def __lookup_cppcheck_exe(): # path the script is located in script_path = os.path.dirname(os.path.realpath(__file__)) exe_name = "cppcheck" if sys.platform == "win32": exe_name += ".exe" exe_path = None if 'TEST_CPPCHECK_EXE_LOOKUP_PATH' in os.environ: lookup_paths = [os.environ['TEST_CPPCHECK_EXE_LOOKUP_PATH']] else: lookup_paths = [os.path.join(script_path, '..', '..'), '.'] for base in lookup_paths: for path in ('', 'bin', os.path.join('bin', 'debug')): tmp_exe_path = os.path.join(base, path, exe_name) if os.path.isfile(tmp_exe_path): exe_path = tmp_exe_path break if exe_path: exe_path = os.path.abspath(exe_path) print("using '{}'".format(exe_path)) return exe_path # Run Cppcheck with args def cppcheck(args, env=None, remove_checkers_report=True, cwd=None, cppcheck_exe=None, timeout=None): exe = cppcheck_exe if cppcheck_exe else __lookup_cppcheck_exe() assert exe is not None, 'no cppcheck binary found' if 'TEST_CPPCHECK_INJECT_J' in os.environ: found_j = False for arg in args: if arg.startswith('-j'): found_j = True break if not found_j: arg_j = '-j' + str(os.environ['TEST_CPPCHECK_INJECT_J']) args.append(arg_j) if 'TEST_CPPCHECK_INJECT_CLANG' in os.environ: found_clang = False for arg in args: if arg.startswith('--clang'): found_clang = True break if not found_clang: arg_clang = '--clang=' + str(os.environ['TEST_CPPCHECK_INJECT_CLANG']) args.append(arg_clang) if 'TEST_CPPCHECK_INJECT_EXECUTOR' in os.environ: found_jn = False found_executor = False for arg in args: if arg.startswith('-j') and arg != '-j1': found_jn = True continue if arg.startswith('--executor'): found_executor = True continue # only add '--executor' if we are actually using multiple jobs if found_jn and not found_executor: arg_executor = '--executor=' + str(os.environ['TEST_CPPCHECK_INJECT_EXECUTOR']) args.append(arg_executor) logging.info(exe + ' ' + ' '.join(args)) p = subprocess.Popen([exe] + args, stdout=subprocess.PIPE, stderr=subprocess.PIPE, env=env, cwd=cwd) try: comm = p.communicate(timeout=timeout) return_code = p.returncode p = None except subprocess.TimeoutExpired: import psutil # terminate all the child processes child_procs = psutil.Process(p.pid).children(recursive=True) if len(child_procs) > 0: for child in child_procs: child.terminate() try: # call with timeout since it might be stuck p.communicate(timeout=5) p = None except subprocess.TimeoutExpired: pass raise finally: if p: # sending the signal to the process groups causes the parent Python process to terminate as well #os.killpg(os.getpgid(p.pid), signal.SIGTERM) # Send the signal to all the process groups p.terminate() comm = p.communicate() stdout = comm[0].decode(encoding='utf-8', errors='ignore').replace('\r\n', '\n') stderr = comm[1].decode(encoding='utf-8', errors='ignore').replace('\r\n', '\n') if remove_checkers_report: if stderr.find('[checkersReport]\n') > 0: start_id = stderr.find('[checkersReport]\n') start_line = stderr.rfind('\n', 0, start_id) if start_line <= 0: stderr = '' else: stderr = stderr[:start_line + 1] elif stderr.find(': (information) Active checkers: ') >= 0: pos = stderr.find(': (information) Active checkers: ') if pos == 0: stderr = '' elif stderr[pos - 1] == '\n': stderr = stderr[:pos] return return_code, stdout, stderr def assert_cppcheck(args, ec_exp=None, out_exp=None, err_exp=None, env=None): exitcode, stdout, stderr = cppcheck(args, env) if ec_exp is not None: assert exitcode == ec_exp, stdout if out_exp is not None: out_lines = stdout.splitlines() assert out_lines == out_exp, stdout if err_exp is not None: err_lines = stderr.splitlines() assert err_lines == err_exp, stderr