mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 02:43:41 +00:00 
			
		
		
		
	 c7349ee2c6
			
		
	
	
		c7349ee2c6
		
	
	
	
	
		
			
			TemproraryFileWrapper wrapper anymore, and should be immune from the problem that a temp file inherited by a spawned process caused an attempt to close the temp file in the spawning process to blow up (the unlink in TemporaryFileWrapper.close() blew up with a "Permission denied" error because, despite that the temp file got closed in the spawning process, the spawned process still had it open by virtue of C-level file descriptor inheritance). In context, that bug took days to figure out <wink/sigh>.
		
			
				
	
	
		
			260 lines
		
	
	
	
		
			8.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
	
		
			8.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """Temporary files and filenames."""
 | |
| 
 | |
| # XXX This tries to be not UNIX specific, but I don't know beans about
 | |
| # how to choose a temp directory or filename on MS-DOS or other
 | |
| # systems so it may have to be changed...
 | |
| 
 | |
| import os
 | |
| 
 | |
| __all__ = ["mktemp", "TemporaryFile", "tempdir", "gettempprefix"]
 | |
| 
 | |
| # Parameters that the caller may set to override the defaults
 | |
| tempdir = None
 | |
| template = None
 | |
| 
 | |
| def gettempdir():
 | |
|     """Function to calculate the directory to use."""
 | |
|     global tempdir
 | |
|     if tempdir is not None:
 | |
|         return tempdir
 | |
| 
 | |
|     # _gettempdir_inner deduces whether a candidate temp dir is usable by
 | |
|     # trying to create a file in it, and write to it.  If that succeeds,
 | |
|     # great, it closes the file and unlinks it.  There's a race, though:
 | |
|     # the *name* of the test file it tries is the same across all threads
 | |
|     # under most OSes (Linux is an exception), and letting multiple threads
 | |
|     # all try to open, write to, close, and unlink a single file can cause
 | |
|     # a variety of bogus errors (e.g., you cannot unlink a file under
 | |
|     # Windows if anyone has it open, and two threads cannot create the
 | |
|     # same file in O_EXCL mode under Unix).  The simplest cure is to serialize
 | |
|     # calls to _gettempdir_inner.  This isn't a real expense, because the
 | |
|     # first thread to succeed sets the global tempdir, and all subsequent
 | |
|     # calls to gettempdir() reuse that without trying _gettempdir_inner.
 | |
|     _tempdir_lock.acquire()
 | |
|     try:
 | |
|         return _gettempdir_inner()
 | |
|     finally:
 | |
|         _tempdir_lock.release()
 | |
| 
 | |
| def _gettempdir_inner():
 | |
|     """Function to calculate the directory to use."""
 | |
|     global tempdir
 | |
|     if tempdir is not None:
 | |
|         return tempdir
 | |
|     try:
 | |
|         pwd = os.getcwd()
 | |
|     except (AttributeError, os.error):
 | |
|         pwd = os.curdir
 | |
|     attempdirs = ['/tmp', '/var/tmp', '/usr/tmp', pwd]
 | |
|     if os.name == 'nt':
 | |
|         attempdirs.insert(0, 'C:\\TEMP')
 | |
|         attempdirs.insert(0, '\\TEMP')
 | |
|     elif os.name == 'mac':
 | |
|         import macfs, MACFS
 | |
|         try:
 | |
|             refnum, dirid = macfs.FindFolder(MACFS.kOnSystemDisk,
 | |
|                                              MACFS.kTemporaryFolderType, 1)
 | |
|             dirname = macfs.FSSpec((refnum, dirid, '')).as_pathname()
 | |
|             attempdirs.insert(0, dirname)
 | |
|         except macfs.error:
 | |
|             pass
 | |
|     elif os.name == 'riscos':
 | |
|         scrapdir = os.getenv('Wimp$ScrapDir')
 | |
|         if scrapdir:
 | |
|             attempdirs.insert(0, scrapdir)
 | |
|     for envname in 'TMPDIR', 'TEMP', 'TMP':
 | |
|         if os.environ.has_key(envname):
 | |
|             attempdirs.insert(0, os.environ[envname])
 | |
|     testfile = gettempprefix() + 'test'
 | |
|     for dir in attempdirs:
 | |
|         try:
 | |
|             filename = os.path.join(dir, testfile)
 | |
|             if os.name == 'posix':
 | |
|                 try:
 | |
|                     fd = os.open(filename,
 | |
|                                  os.O_RDWR | os.O_CREAT | os.O_EXCL, 0700)
 | |
|                 except OSError:
 | |
|                     pass
 | |
|                 else:
 | |
|                     fp = os.fdopen(fd, 'w')
 | |
|                     fp.write('blat')
 | |
|                     fp.close()
 | |
|                     os.unlink(filename)
 | |
|                     del fp, fd
 | |
|                     tempdir = dir
 | |
|                     break
 | |
|             else:
 | |
|                 fp = open(filename, 'w')
 | |
|                 fp.write('blat')
 | |
|                 fp.close()
 | |
|                 os.unlink(filename)
 | |
|                 tempdir = dir
 | |
|                 break
 | |
|         except IOError:
 | |
|             pass
 | |
|     if tempdir is None:
 | |
|         msg = "Can't find a usable temporary directory amongst " + `attempdirs`
 | |
|         raise IOError, msg
 | |
|     return tempdir
 | |
| 
 | |
| 
 | |
| # template caches the result of gettempprefix, for speed, when possible.
 | |
| # XXX unclear why this isn't "_template"; left it "template" for backward
 | |
| # compatibility.
 | |
| if os.name == "posix":
 | |
|     # We don't try to cache the template on posix:  the pid may change on us
 | |
|     # between calls due to a fork, and on Linux the pid changes even for
 | |
|     # another thread in the same process.  Since any attempt to keep the
 | |
|     # cache in synch would have to call os.getpid() anyway in order to make
 | |
|     # sure the pid hasn't changed between calls, a cache wouldn't save any
 | |
|     # time.  In addition, a cache is difficult to keep correct with the pid
 | |
|     # changing willy-nilly, and earlier attempts proved buggy (races).
 | |
|     template = None
 | |
| 
 | |
| # Else the pid never changes, so gettempprefix always returns the same
 | |
| # string.
 | |
| elif os.name == "nt":
 | |
|     template = '~' + `os.getpid()` + '-'
 | |
| elif os.name in ('mac', 'riscos'):
 | |
|     template = 'Python-Tmp-'
 | |
| else:
 | |
|     template = 'tmp' # XXX might choose a better one
 | |
| 
 | |
| def gettempprefix():
 | |
|     """Function to calculate a prefix of the filename to use.
 | |
| 
 | |
|     This incorporates the current process id on systems that support such a
 | |
|     notion, so that concurrent processes don't generate the same prefix.
 | |
|     """
 | |
| 
 | |
|     global template
 | |
|     if template is None:
 | |
|         return '@' + `os.getpid()` + '.'
 | |
|     else:
 | |
|         return template
 | |
| 
 | |
| 
 | |
| def mktemp(suffix=""):
 | |
|     """User-callable function to return a unique temporary file name."""
 | |
|     dir = gettempdir()
 | |
|     pre = gettempprefix()
 | |
|     while 1:
 | |
|         i = _counter.get_next()
 | |
|         file = os.path.join(dir, pre + str(i) + suffix)
 | |
|         if not os.path.exists(file):
 | |
|             return file
 | |
| 
 | |
| 
 | |
| class TemporaryFileWrapper:
 | |
|     """Temporary file wrapper
 | |
| 
 | |
|     This class provides a wrapper around files opened for temporary use.
 | |
|     In particular, it seeks to automatically remove the file when it is
 | |
|     no longer needed.
 | |
|     """
 | |
| 
 | |
|     # Cache the unlinker so we don't get spurious errors at shutdown
 | |
|     # when the module-level "os" is None'd out.  Note that this must
 | |
|     # be referenced as self.unlink, because the name TemporaryFileWrapper
 | |
|     # may also get None'd out before __del__ is called.
 | |
|     unlink = os.unlink
 | |
| 
 | |
|     def __init__(self, file, path):
 | |
|         self.file = file
 | |
|         self.path = path
 | |
|         self.close_called = 0
 | |
| 
 | |
|     def close(self):
 | |
|         if not self.close_called:
 | |
|             self.close_called = 1
 | |
|             self.file.close()
 | |
|             self.unlink(self.path)
 | |
| 
 | |
|     def __del__(self):
 | |
|         self.close()
 | |
| 
 | |
|     def __getattr__(self, name):
 | |
|         file = self.__dict__['file']
 | |
|         a = getattr(file, name)
 | |
|         if type(a) != type(0):
 | |
|             setattr(self, name, a)
 | |
|         return a
 | |
| 
 | |
| 
 | |
| def TemporaryFile(mode='w+b', bufsize=-1, suffix=""):
 | |
|     """Create and return a temporary file (opened read-write by default)."""
 | |
|     name = mktemp(suffix)
 | |
|     if os.name == 'posix':
 | |
|         # Unix -- be very careful
 | |
|         fd = os.open(name, os.O_RDWR|os.O_CREAT|os.O_EXCL, 0700)
 | |
|         try:
 | |
|             os.unlink(name)
 | |
|             return os.fdopen(fd, mode, bufsize)
 | |
|         except:
 | |
|             os.close(fd)
 | |
|             raise
 | |
|     elif os.name == 'nt':
 | |
|         # Windows -- can't unlink an open file, but O_TEMPORARY creates a
 | |
|         # file that "deletes itself" when the last handle is closed.
 | |
|         # O_NOINHERIT ensures processes created via spawn() don't get a
 | |
|         # handle to this too.  That would be a security hole, and, on my
 | |
|         # Win98SE box, when an O_TEMPORARY file is inherited by a spawned
 | |
|         # process, the fd in the spawned process seems to lack the
 | |
|         # O_TEMPORARY flag, so the file doesn't go away by magic then if the
 | |
|         # spawning process closes it first.
 | |
|         flags = (os.O_RDWR | os.O_CREAT | os.O_EXCL |
 | |
|                  os.O_TEMPORARY | os.O_NOINHERIT)
 | |
|         if 'b' in mode:
 | |
|             flags |= os.O_BINARY
 | |
|         fd = os.open(name, flags, 0700)
 | |
|         return os.fdopen(fd, mode, bufsize)
 | |
|     else:
 | |
|         # Assume we can't unlink a file that's still open, or arrange for
 | |
|         # an automagically self-deleting file -- use wrapper.
 | |
|         file = open(name, mode, bufsize)
 | |
|         return TemporaryFileWrapper(file, name)
 | |
| 
 | |
| # In order to generate unique names, mktemp() uses _counter.get_next().
 | |
| # This returns a unique integer on each call, in a threadsafe way (i.e.,
 | |
| # multiple threads will never see the same integer).  The integer will
 | |
| # usually be a Python int, but if _counter.get_next() is called often
 | |
| # enough, it will become a Python long.
 | |
| # Note that the only names that survive this next block of code
 | |
| # are "_counter" and "_tempdir_lock".
 | |
| 
 | |
| class _ThreadSafeCounter:
 | |
|     def __init__(self, mutex, initialvalue=0):
 | |
|         self.mutex = mutex
 | |
|         self.i = initialvalue
 | |
| 
 | |
|     def get_next(self):
 | |
|         self.mutex.acquire()
 | |
|         result = self.i
 | |
|         try:
 | |
|             newi = result + 1
 | |
|         except OverflowError:
 | |
|             newi = long(result) + 1
 | |
|         self.i = newi
 | |
|         self.mutex.release()
 | |
|         return result
 | |
| 
 | |
| try:
 | |
|     import thread
 | |
| 
 | |
| except ImportError:
 | |
|     class _DummyMutex:
 | |
|         def acquire(self):
 | |
|             pass
 | |
| 
 | |
|         release = acquire
 | |
| 
 | |
|     _counter = _ThreadSafeCounter(_DummyMutex())
 | |
|     _tempdir_lock = _DummyMutes()
 | |
|     del _DummyMutex
 | |
| 
 | |
| else:
 | |
|     _counter = _ThreadSafeCounter(thread.allocate_lock())
 | |
|     _tempdir_lock = thread.allocate_lock()
 | |
|     del thread
 | |
| 
 | |
| del _ThreadSafeCounter
 |