summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMax Hirschhorn <max.hirschhorn@mongodb.com>2019-03-05 13:46:14 -0500
committerMax Hirschhorn <max.hirschhorn@mongodb.com>2019-03-05 13:46:14 -0500
commitf80f05151377eed5119a8c9cb52667d50f668610 (patch)
treeb32cedb43330fe1471fd46a9d1f971cfd1300a61
parent2706c9fdc71546897edb1a366240ca0062e79d3c (diff)
downloadmongo-f80f05151377eed5119a8c9cb52667d50f668610.tar.gz
SERVER-38697 Use process creation time to detect pid reuse.
Changes start_cmd() to use psutil.Popen() rather than subprocess.Popen() in order to cache the creation time of the child process. (cherry picked from commit d43369c671a596ee816f44038fca3423a0a33126)
-rwxr-xr-xpytests/powertest.py47
1 files changed, 19 insertions, 28 deletions
diff --git a/pytests/powertest.py b/pytests/powertest.py
index b553b35308d..efe0ff8b12c 100755
--- a/pytests/powertest.py
+++ b/pytests/powertest.py
@@ -66,6 +66,10 @@ if os.name == "posix" and sys.version_info[0] == 2:
else:
import subprocess
+# We replace the subprocess module imported by the psutil package so we can safely use
+# psutil.Popen() in addition to subprocess.Popen().
+psutil.subprocess = subprocess
+
# Get relative imports to work when the package is not installed on the PYTHONPATH.
if __name__ == "__main__" and __package__ is None:
sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
@@ -245,42 +249,27 @@ def dump_stacks_and_exit(signum, frame): # pylint: disable=unused-argument
sys.exit(1)
-def child_processes(parent_pid):
- """Return a list of all child processes for a pid."""
- # The child processes cannot be obtained from the parent on Windows from psutil. See
- # https://stackoverflow.com/questions/30220732/python-psutil-not-showing-all-child-processes
- child_procs = []
- while psutil.pid_exists(parent_pid):
- try:
- child_procs = [p for p in psutil.process_iter(attrs=["pid"]) if parent_pid == p.ppid()]
- break
- except psutil.NoSuchProcess:
- pass
- for proc in child_procs:
- proc_children = child_processes(proc.pid)
- if proc_children:
- child_procs += proc_children
- return list(set(child_procs))
-
-
-def kill_process(pid, kill_children=True):
+def kill_process(parent, kill_children=True):
"""Kill a process, and optionally it's children, by it's pid. Returns 0 if successful."""
try:
- parent = psutil.Process(pid)
+ # parent.children() implicitly calls parent.is_running(), which raises a
+ # psutil.NoSuchProcess exception if the creation time for the process with pid=parent.pid is
+ # different than parent.create_time(). We can reliably detect pid reuse this way because
+ # 'parent' is the same psutil.Process instance returned by start_cmd() and therefore has an
+ # accurate notion of the creation time.
+ procs = parent.children(recursive=True) if kill_children else []
except psutil.NoSuchProcess:
- LOGGER.warn("Could not kill process %d, as it no longer exists", pid)
+ LOGGER.warn("Could not kill process %d, as it no longer exists", parent.pid)
return 0
- procs = [parent]
- if kill_children:
- procs += child_processes(pid)
+ procs.append(parent)
for proc in procs:
try:
LOGGER.debug("Killing process '%s' pid %d", proc.name(), proc.pid)
proc.kill()
except psutil.NoSuchProcess:
- LOGGER.warn("Could not kill process %d, as it no longer exists", pid)
+ LOGGER.warn("Could not kill process %d, as it no longer exists", proc.pid)
_, alive = psutil.wait_procs(procs, timeout=30, callback=None)
if alive:
@@ -293,7 +282,7 @@ def kill_processes(procs, kill_children=True):
"""Kill a list of processes and optionally it's children."""
for proc in procs:
LOGGER.debug("Starting kill of parent process %d", proc.pid)
- kill_process(proc.pid, kill_children=kill_children)
+ kill_process(proc, kill_children=kill_children)
ret = proc.wait()
LOGGER.debug("Finished kill of parent process %d has return code of %d", proc.pid, ret)
@@ -371,8 +360,10 @@ def start_cmd(cmd, use_file=False):
else:
LOGGER.debug("Executing '%s'", cmd)
- proc = subprocess.Popen(cmd, close_fds=True)
- LOGGER.debug("Spawned process %s pid %d", psutil.Process(proc.pid).name(), proc.pid)
+ # We use psutil.Popen() rather than subprocess.Popen() in order to cache the creation time of
+ # the process. This enables us to reliably detect pid reuse in kill_process().
+ proc = psutil.Popen(cmd, close_fds=True)
+ LOGGER.debug("Spawned process %s pid %d", proc.name(), proc.pid)
return proc