1# Copyright (c) 2023 Nordic Semiconductor ASA 2# 3# SPDX-License-Identifier: Apache-2.0 4 5from __future__ import annotations 6 7import logging 8import os 9import platform 10import shlex 11import signal 12import subprocess 13import time 14 15import psutil 16 17_WINDOWS = platform.system() == 'Windows' 18 19 20def log_command(logger: logging.Logger, msg: str, args: list, level: int = logging.DEBUG) -> None: 21 """ 22 Platform-independent helper for logging subprocess invocations. 23 24 Will log a command string that can be copy/pasted into a POSIX 25 shell on POSIX platforms. This is not available on Windows, so 26 the entire args array is logged instead. 27 28 :param logger: logging.Logger to use 29 :param msg: message to associate with the command 30 :param args: argument list as passed to subprocess module 31 :param level: log level 32 """ 33 msg = f'{msg}: %s' 34 if _WINDOWS: 35 logger.log(level, msg, str(args)) 36 else: 37 logger.log(level, msg, shlex.join(args)) 38 39 40def terminate_process(proc: subprocess.Popen) -> None: 41 """ 42 Try to terminate provided process and all its subprocesses recursively. 43 """ 44 for child in psutil.Process(proc.pid).children(recursive=True): 45 try: 46 os.kill(child.pid, signal.SIGTERM) 47 except (ProcessLookupError, psutil.NoSuchProcess): 48 pass 49 proc.terminate() 50 # sleep for a while before attempting to kill 51 time.sleep(0.5) 52 proc.kill() 53