summaryrefslogtreecommitdiffstats
path: root/Lib/socketserver.py
diff options
context:
space:
mode:
authorCharles-François Natali <cf.natali@gmail.com>2014-06-20 21:37:35 (GMT)
committerCharles-François Natali <cf.natali@gmail.com>2014-06-20 21:37:35 (GMT)
commit504f5c36ef2419cb0c56450c3a733499c4576456 (patch)
treea1ceedcc70ed5716a7de11b8230e64b3a43e7565 /Lib/socketserver.py
parentaf9eb9628d833c8096e4438a48655ed07996137a (diff)
downloadcpython-504f5c36ef2419cb0c56450c3a733499c4576456.zip
cpython-504f5c36ef2419cb0c56450c3a733499c4576456.tar.gz
cpython-504f5c36ef2419cb0c56450c3a733499c4576456.tar.bz2
Issue #21491: socketserver: Fix a race condition in child processes reaping.
Diffstat (limited to 'Lib/socketserver.py')
-rw-r--r--Lib/socketserver.py56
1 files changed, 30 insertions, 26 deletions
diff --git a/Lib/socketserver.py b/Lib/socketserver.py
index 46ee7c5..2f395fa 100644
--- a/Lib/socketserver.py
+++ b/Lib/socketserver.py
@@ -523,35 +523,39 @@ class ForkingMixIn:
def collect_children(self):
"""Internal routine to wait for children that have exited."""
- if self.active_children is None: return
+ if self.active_children is None:
+ return
+
+ # If we're above the max number of children, wait and reap them until
+ # we go back below threshold. Note that we use waitpid(-1) below to be
+ # able to collect children in size(<defunct children>) syscalls instead
+ # of size(<children>): the downside is that this might reap children
+ # which we didn't spawn, which is why we only resort to this when we're
+ # above max_children.
while len(self.active_children) >= self.max_children:
- # XXX: This will wait for any child process, not just ones
- # spawned by this library. This could confuse other
- # libraries that expect to be able to wait for their own
- # children.
try:
- pid, status = os.waitpid(0, 0)
+ pid, _ = os.waitpid(-1, 0)
+ self.active_children.discard(pid)
+ except InterruptedError:
+ pass
+ except ChildProcessError:
+ # we don't have any children, we're done
+ self.active_children.clear()
except OSError:
- pid = None
- if pid not in self.active_children: continue
- self.active_children.remove(pid)
-
- # XXX: This loop runs more system calls than it ought
- # to. There should be a way to put the active_children into a
- # process group and then use os.waitpid(-pgid) to wait for any
- # of that set, but I couldn't find a way to allocate pgids
- # that couldn't collide.
- for child in self.active_children:
+ break
+
+ # Now reap all defunct children.
+ for pid in self.active_children.copy():
try:
- pid, status = os.waitpid(child, os.WNOHANG)
+ pid, _ = os.waitpid(pid, os.WNOHANG)
+ # if the child hasn't exited yet, pid will be 0 and ignored by
+ # discard() below
+ self.active_children.discard(pid)
+ except ChildProcessError:
+ # someone else reaped it
+ self.active_children.discard(pid)
except OSError:
- pid = None
- if not pid: continue
- try:
- self.active_children.remove(pid)
- except ValueError as e:
- raise ValueError('%s. x=%d and list=%r' % (e.message, pid,
- self.active_children))
+ pass
def handle_timeout(self):
"""Wait for zombies after self.timeout seconds of inactivity.
@@ -573,8 +577,8 @@ class ForkingMixIn:
if pid:
# Parent process
if self.active_children is None:
- self.active_children = []
- self.active_children.append(pid)
+ self.active_children = set()
+ self.active_children.add(pid)
self.close_request(request)
return
else: