mirror of
https://github.com/python/cpython.git
synced 2025-08-22 09:45:06 +00:00
Close #2501: Permission bits are once again correctly copied from the source file to the cached bytecode file. Test by Eric Snow.
This commit is contained in:
parent
36d188c7f7
commit
a508770e20
4 changed files with 4332 additions and 4220 deletions
|
@ -118,13 +118,14 @@ def _path_isdir(path):
|
||||||
return _path_is_mode_type(path, 0o040000)
|
return _path_is_mode_type(path, 0o040000)
|
||||||
|
|
||||||
|
|
||||||
def _write_atomic(path, data):
|
def _write_atomic(path, data, mode=0o666):
|
||||||
"""Best-effort function to write data to a path atomically.
|
"""Best-effort function to write data to a path atomically.
|
||||||
Be prepared to handle a FileExistsError if concurrent writing of the
|
Be prepared to handle a FileExistsError if concurrent writing of the
|
||||||
temporary file is attempted."""
|
temporary file is attempted."""
|
||||||
# id() is used to generate a pseudo-random filename.
|
# id() is used to generate a pseudo-random filename.
|
||||||
path_tmp = '{}.{}'.format(path, id(path))
|
path_tmp = '{}.{}'.format(path, id(path))
|
||||||
fd = _os.open(path_tmp, _os.O_EXCL | _os.O_CREAT | _os.O_WRONLY, 0o666)
|
fd = _os.open(path_tmp,
|
||||||
|
_os.O_EXCL | _os.O_CREAT | _os.O_WRONLY, mode & 0o666)
|
||||||
try:
|
try:
|
||||||
# We first write data to a temporary file, and then use os.replace() to
|
# We first write data to a temporary file, and then use os.replace() to
|
||||||
# perform an atomic rename.
|
# perform an atomic rename.
|
||||||
|
@ -887,6 +888,16 @@ class SourceLoader(_LoaderBasics):
|
||||||
"""
|
"""
|
||||||
return {'mtime': self.path_mtime(path)}
|
return {'mtime': self.path_mtime(path)}
|
||||||
|
|
||||||
|
def _cache_bytecode(self, source_path, cache_path, data):
|
||||||
|
"""Optional method which writes data (bytes) to a file path (a str).
|
||||||
|
|
||||||
|
Implementing this method allows for the writing of bytecode files.
|
||||||
|
|
||||||
|
The source path is needed in order to correctly transfer permissions
|
||||||
|
"""
|
||||||
|
# For backwards compatibility, we delegate to set_data()
|
||||||
|
return self.set_data(cache_path, data)
|
||||||
|
|
||||||
def set_data(self, path, data):
|
def set_data(self, path, data):
|
||||||
"""Optional method which writes data (bytes) to a file path (a str).
|
"""Optional method which writes data (bytes) to a file path (a str).
|
||||||
|
|
||||||
|
@ -974,7 +985,7 @@ class SourceLoader(_LoaderBasics):
|
||||||
data.extend(_w_long(len(source_bytes)))
|
data.extend(_w_long(len(source_bytes)))
|
||||||
data.extend(marshal.dumps(code_object))
|
data.extend(marshal.dumps(code_object))
|
||||||
try:
|
try:
|
||||||
self.set_data(bytecode_path, data)
|
self._cache_bytecode(source_path, bytecode_path, data)
|
||||||
_verbose_message('wrote {!r}', bytecode_path)
|
_verbose_message('wrote {!r}', bytecode_path)
|
||||||
except NotImplementedError:
|
except NotImplementedError:
|
||||||
pass
|
pass
|
||||||
|
@ -1029,7 +1040,11 @@ class SourceFileLoader(FileLoader, SourceLoader):
|
||||||
st = _os.stat(path)
|
st = _os.stat(path)
|
||||||
return {'mtime': st.st_mtime, 'size': st.st_size}
|
return {'mtime': st.st_mtime, 'size': st.st_size}
|
||||||
|
|
||||||
def set_data(self, path, data):
|
def _cache_bytecode(self, source_path, bytecode_path, data):
|
||||||
|
# Adapt between the two APIs
|
||||||
|
return self.set_data(bytecode_path, data, source_path=source_path)
|
||||||
|
|
||||||
|
def set_data(self, path, data, *, source_path=None):
|
||||||
"""Write bytes data to a file."""
|
"""Write bytes data to a file."""
|
||||||
parent, filename = _path_split(path)
|
parent, filename = _path_split(path)
|
||||||
path_parts = []
|
path_parts = []
|
||||||
|
@ -1049,8 +1064,14 @@ class SourceFileLoader(FileLoader, SourceLoader):
|
||||||
# If can't get proper access, then just forget about writing
|
# If can't get proper access, then just forget about writing
|
||||||
# the data.
|
# the data.
|
||||||
return
|
return
|
||||||
|
mode = 0o666
|
||||||
|
if source_path is not None:
|
||||||
try:
|
try:
|
||||||
_write_atomic(path, data)
|
mode = _os.stat(source_path).st_mode
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
_write_atomic(path, data, mode)
|
||||||
_verbose_message('created {!r}', path)
|
_verbose_message('created {!r}', path)
|
||||||
except (PermissionError, FileExistsError):
|
except (PermissionError, FileExistsError):
|
||||||
# Don't worry if you can't write bytecode or someone is writing
|
# Don't worry if you can't write bytecode or someone is writing
|
||||||
|
|
|
@ -120,12 +120,35 @@ class ImportTests(unittest.TestCase):
|
||||||
s = os.stat(fn)
|
s = os.stat(fn)
|
||||||
# Check that the umask is respected, and the executable bits
|
# Check that the umask is respected, and the executable bits
|
||||||
# aren't set.
|
# aren't set.
|
||||||
self.assertEqual(stat.S_IMODE(s.st_mode), 0o666 & ~mask)
|
self.assertEqual(oct(stat.S_IMODE(s.st_mode)), oct(0o666 & ~mask))
|
||||||
finally:
|
finally:
|
||||||
del sys.path[0]
|
del sys.path[0]
|
||||||
remove_files(TESTFN)
|
remove_files(TESTFN)
|
||||||
unload(TESTFN)
|
unload(TESTFN)
|
||||||
|
|
||||||
|
@unittest.skipUnless(os.name == 'posix',
|
||||||
|
"test meaningful only on posix systems")
|
||||||
|
def test_cached_mode_issue_2051(self):
|
||||||
|
mode = 0o600
|
||||||
|
source = TESTFN + ".py"
|
||||||
|
with script_helper.temp_dir() as tempdir:
|
||||||
|
path = script_helper.make_script(tempdir, TESTFN,
|
||||||
|
"key='top secret'")
|
||||||
|
os.chmod(path, mode)
|
||||||
|
compiled = imp.cache_from_source(path)
|
||||||
|
sys.path.insert(0, tempdir)
|
||||||
|
try:
|
||||||
|
__import__(TESTFN)
|
||||||
|
finally:
|
||||||
|
sys.path.remove(tempdir)
|
||||||
|
|
||||||
|
if not os.path.exists(compiled):
|
||||||
|
self.fail("__import__ did not result in creation of "
|
||||||
|
"either a .pyc or .pyo file")
|
||||||
|
stat_info = os.stat(compiled)
|
||||||
|
|
||||||
|
self.assertEqual(oct(stat.S_IMODE(stat_info.st_mode)), oct(mode))
|
||||||
|
|
||||||
def test_imp_module(self):
|
def test_imp_module(self):
|
||||||
# Verify that the imp module can correctly load and find .py files
|
# Verify that the imp module can correctly load and find .py files
|
||||||
# XXX (ncoghlan): It would be nice to use support.CleanImport
|
# XXX (ncoghlan): It would be nice to use support.CleanImport
|
||||||
|
|
|
@ -10,6 +10,11 @@ What's New in Python 3.3.0 Release Candidate 1?
|
||||||
Core and Builtins
|
Core and Builtins
|
||||||
-----------------
|
-----------------
|
||||||
|
|
||||||
|
- Issue #2501: Source file permission bits are once again correctly
|
||||||
|
copied to the cached bytecode file. (The migration to importlib
|
||||||
|
reintroduced this problem because these was no regression test. A test
|
||||||
|
has been added as part of this patch)
|
||||||
|
|
||||||
- Issue #15761: Fix crash when PYTHONEXECUTABLE is set on Mac OS X.
|
- Issue #15761: Fix crash when PYTHONEXECUTABLE is set on Mac OS X.
|
||||||
|
|
||||||
- Issue #15726: Fix incorrect bounds checking in PyState_FindModule.
|
- Issue #15726: Fix incorrect bounds checking in PyState_FindModule.
|
||||||
|
|
8491
Python/importlib.h
8491
Python/importlib.h
File diff suppressed because it is too large
Load diff
Loading…
Add table
Add a link
Reference in a new issue