]> granicus.if.org Git - python/commitdiff
Issue #12856: Ensure child processes do not inherit the parent's random seed for...
authorAntoine Pitrou <solipsis@pitrou.net>
Fri, 25 Nov 2011 20:28:15 +0000 (21:28 +0100)
committerAntoine Pitrou <solipsis@pitrou.net>
Fri, 25 Nov 2011 20:28:15 +0000 (21:28 +0100)
Patch by Brian Harring.

Lib/tempfile.py
Lib/test/test_tempfile.py
Misc/NEWS

index 48b77a87e21620bdee9a12c0b8a402b050a2a863..34dff30468b4f490e3de17c37045c02a01832a9b 100644 (file)
@@ -112,8 +112,13 @@ class _RandomNameSequence:
 
     characters = "abcdefghijklmnopqrstuvwxyz0123456789_"
 
-    def __init__(self):
-        self.rng = _Random()
+    @property
+    def rng(self):
+        cur_pid = _os.getpid()
+        if cur_pid != getattr(self, '_rng_pid', None):
+            self._rng = _Random()
+            self._rng_pid = cur_pid
+        return self._rng
 
     def __iter__(self):
         return self
index f7f5bdadbd38e935ad2f80602d3dbf8de084c816..50cf3b49cb844acea3126041292836b1252fa405 100644 (file)
@@ -1,6 +1,7 @@
 # tempfile.py unit tests.
 import tempfile
 import os
+import signal
 import sys
 import re
 import warnings
@@ -135,6 +136,37 @@ class test__RandomNameSequence(TC):
         except:
             self.failOnException("iteration")
 
+    @unittest.skipUnless(hasattr(os, 'fork'),
+        "os.fork is required for this test")
+    def test_process_awareness(self):
+        # ensure that the random source differs between
+        # child and parent.
+        read_fd, write_fd = os.pipe()
+        pid = None
+        try:
+            pid = os.fork()
+            if not pid:
+                os.close(read_fd)
+                os.write(write_fd, next(self.r).encode("ascii"))
+                os.close(write_fd)
+                # bypass the normal exit handlers- leave those to
+                # the parent.
+                os._exit(0)
+            parent_value = next(self.r)
+            child_value = os.read(read_fd, len(parent_value)).decode("ascii")
+        finally:
+            if pid:
+                # best effort to ensure the process can't bleed out
+                # via any bugs above
+                try:
+                    os.kill(pid, signal.SIGKILL)
+                except EnvironmentError:
+                    pass
+            os.close(read_fd)
+            os.close(write_fd)
+        self.assertNotEqual(child_value, parent_value)
+
+
 test_classes.append(test__RandomNameSequence)
 
 
index f65fafe2eaffaccb320efed0f378dc2d0217a768..fba2c3d7d98401499fedc4f7c84c52761aaf3efa 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -83,6 +83,10 @@ Core and Builtins
 Library
 -------
 
+- Issue #12856: Ensure child processes do not inherit the parent's random
+  seed for filename generation in the tempfile module.  Patch by Brian
+  Harring.
+
 - Issue #13458: Fix a memory leak in the ssl module when decoding a
   certificate with a subjectAltName.  Patch by Robert Xiao.