Commit 17c93260 authored by Antoine Pitrou's avatar Antoine Pitrou

Issue #18879: When a method is looked up on a temporary file, avoid closing...

Issue #18879: When a method is looked up on a temporary file, avoid closing the file before the method is possibly called.
parent bdce938a
...@@ -27,6 +27,7 @@ __all__ = [ ...@@ -27,6 +27,7 @@ __all__ = [
# Imports. # Imports.
import functools as _functools
import warnings as _warnings import warnings as _warnings
import sys as _sys import sys as _sys
import io as _io import io as _io
...@@ -349,6 +350,46 @@ def mktemp(suffix="", prefix=template, dir=None): ...@@ -349,6 +350,46 @@ def mktemp(suffix="", prefix=template, dir=None):
"No usable temporary filename found") "No usable temporary filename found")
class _TemporaryFileCloser:
"""A separate object allowing proper closing of a temporary file's
underlying file object, without adding a __del__ method to the
temporary file."""
def __init__(self, file, name, delete=True):
self.file = file
self.name = name
self.close_called = False
self.delete = delete
# NT provides delete-on-close as a primitive, so we don't need
# the wrapper to do anything special. We still use it so that
# file.name is useful (i.e. not "(fdopen)") with NamedTemporaryFile.
if _os.name != 'nt':
# 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 close(self):
if not self.close_called:
self.close_called = True
self.file.close()
if self.delete:
self.unlink(self.name)
# Need to ensure the file is deleted on __del__
def __del__(self):
self.close()
else:
def close(self):
if not self.close_called:
self.close_called = True
self.file.close()
class _TemporaryFileWrapper: class _TemporaryFileWrapper:
"""Temporary file wrapper """Temporary file wrapper
...@@ -360,8 +401,8 @@ class _TemporaryFileWrapper: ...@@ -360,8 +401,8 @@ class _TemporaryFileWrapper:
def __init__(self, file, name, delete=True): def __init__(self, file, name, delete=True):
self.file = file self.file = file
self.name = name self.name = name
self.close_called = False
self.delete = delete self.delete = delete
self._closer = _TemporaryFileCloser(file, name, delete)
def __getattr__(self, name): def __getattr__(self, name):
# Attribute lookups are delegated to the underlying file # Attribute lookups are delegated to the underlying file
...@@ -369,6 +410,15 @@ class _TemporaryFileWrapper: ...@@ -369,6 +410,15 @@ class _TemporaryFileWrapper:
# (i.e. methods are cached, closed and friends are not) # (i.e. methods are cached, closed and friends are not)
file = self.__dict__['file'] file = self.__dict__['file']
a = getattr(file, name) a = getattr(file, name)
if hasattr(a, '__call__'):
func = a
@_functools.wraps(func)
def func_wrapper(*args, **kwargs):
return func(*args, **kwargs)
# Avoid closing the file as long as the wrapper is alive,
# see issue #18879.
func_wrapper._closer = self._closer
a = func_wrapper
if not isinstance(a, int): if not isinstance(a, int):
setattr(self, name, a) setattr(self, name, a)
return a return a
...@@ -379,40 +429,22 @@ class _TemporaryFileWrapper: ...@@ -379,40 +429,22 @@ class _TemporaryFileWrapper:
self.file.__enter__() self.file.__enter__()
return self return self
# iter() doesn't use __getattr__ to find the __iter__ method
def __iter__(self):
return iter(self.file)
# NT provides delete-on-close as a primitive, so we don't need
# the wrapper to do anything special. We still use it so that
# file.name is useful (i.e. not "(fdopen)") with NamedTemporaryFile.
if _os.name != 'nt':
# 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 close(self):
if not self.close_called:
self.close_called = True
self.file.close()
if self.delete:
self.unlink(self.name)
def __del__(self):
self.close()
# Need to trap __exit__ as well to ensure the file gets # Need to trap __exit__ as well to ensure the file gets
# deleted when used in a with statement # deleted when used in a with statement
def __exit__(self, exc, value, tb): def __exit__(self, exc, value, tb):
result = self.file.__exit__(exc, value, tb) result = self.file.__exit__(exc, value, tb)
self.close() self.close()
return result return result
else:
def __exit__(self, exc, value, tb): def close(self):
self.file.__exit__(exc, value, tb) """
Close the temporary file, possibly deleting it.
"""
self._closer.close()
# iter() doesn't use __getattr__ to find the __iter__ method
def __iter__(self):
return iter(self.file)
def NamedTemporaryFile(mode='w+b', buffering=-1, encoding=None, def NamedTemporaryFile(mode='w+b', buffering=-1, encoding=None,
......
...@@ -8,6 +8,7 @@ import sys ...@@ -8,6 +8,7 @@ import sys
import re import re
import warnings import warnings
import contextlib import contextlib
import weakref
import unittest import unittest
from test import support from test import support
...@@ -674,6 +675,22 @@ class TestNamedTemporaryFile(BaseTestCase): ...@@ -674,6 +675,22 @@ class TestNamedTemporaryFile(BaseTestCase):
self.do_create(pre="a", suf="b") self.do_create(pre="a", suf="b")
self.do_create(pre="aa", suf=".txt") self.do_create(pre="aa", suf=".txt")
def test_method_lookup(self):
# Issue #18879: Looking up a temporary file method should keep it
# alive long enough.
f = self.do_create()
wr = weakref.ref(f)
write = f.write
write2 = f.write
del f
write(b'foo')
del write
write2(b'bar')
del write2
if support.check_impl_detail(cpython=True):
# No reference cycle was created.
self.assertIsNone(wr())
def test_creates_named(self): def test_creates_named(self):
# NamedTemporaryFile creates files with names # NamedTemporaryFile creates files with names
f = tempfile.NamedTemporaryFile() f = tempfile.NamedTemporaryFile()
......
...@@ -29,6 +29,9 @@ Core and Builtins ...@@ -29,6 +29,9 @@ Core and Builtins
Library Library
------- -------
- Issue #18879: When a method is looked up on a temporary file, avoid closing
the file before the method is possibly called.
- Issue #20034: Updated alias mapping to most recent locale.alias file - Issue #20034: Updated alias mapping to most recent locale.alias file
from X.org distribution using makelocalealias.py. from X.org distribution using makelocalealias.py.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment