fix issue305 - ignore any problems in writing a pyc file, but print out a trace.

This commit is contained in:
holger krekel 2013-08-01 15:43:42 +02:00
parent cbbbfcd101
commit 9ccd52d538
3 changed files with 30 additions and 12 deletions

View File

@ -6,6 +6,8 @@ Changes between 2.3.5 and 2.4.DEV
- remove implicit distribute_setup support from setup.py. - remove implicit distribute_setup support from setup.py.
- fix issue305: ignore any problems when writing pyc files.
- integrate tab-completion on options through use of "argcomplete". - integrate tab-completion on options through use of "argcomplete".
Thanks Anthon van der Neut for the PR. Thanks Anthon van der Neut for the PR.

View File

@ -156,7 +156,7 @@ class AssertionRewritingHook(object):
raise raise
return sys.modules[name] return sys.modules[name]
def _write_pyc(co, source_path, pyc): def _write_pyc(state, co, source_path, pyc):
# Technically, we don't have to have the same pyc format as # Technically, we don't have to have the same pyc format as
# (C)Python, since these "pycs" should never be seen by builtin # (C)Python, since these "pycs" should never be seen by builtin
# import. However, there's little reason deviate, and I hope # import. However, there's little reason deviate, and I hope
@ -167,15 +167,11 @@ def _write_pyc(co, source_path, pyc):
fp = open(pyc, "wb") fp = open(pyc, "wb")
except IOError: except IOError:
err = sys.exc_info()[1].errno err = sys.exc_info()[1].errno
if err in [errno.ENOENT, errno.ENOTDIR]: state.trace("error writing pyc file at %s: errno=%s" %(pyc, err))
# This happens when we get a EEXIST in find_module creating the # we ignore any failure to write the cache file
# __pycache__ directory and __pycache__ is by some non-dir node. # there are many reasons, permission-denied, __pycache__ being a
return False # file etc.
elif err == errno.EACCES: return False
# The directory is read-only; this can happen for example when
# running the tests in a package installed as root
return False
raise
try: try:
fp.write(imp.get_magic()) fp.write(imp.get_magic())
fp.write(struct.pack("<l", mtime)) fp.write(struct.pack("<l", mtime))
@ -249,12 +245,12 @@ def _make_rewritten_pyc(state, fn, pyc, co):
if sys.platform.startswith("win"): if sys.platform.startswith("win"):
# Windows grants exclusive access to open files and doesn't have atomic # Windows grants exclusive access to open files and doesn't have atomic
# rename, so just write into the final file. # rename, so just write into the final file.
_write_pyc(co, fn, pyc) _write_pyc(state, co, fn, pyc)
else: else:
# When not on windows, assume rename is atomic. Dump the code object # When not on windows, assume rename is atomic. Dump the code object
# into a file specific to this process and atomically replace it. # into a file specific to this process and atomically replace it.
proc_pyc = pyc + "." + str(os.getpid()) proc_pyc = pyc + "." + str(os.getpid())
if _write_pyc(co, fn, proc_pyc): if _write_pyc(state, co, fn, proc_pyc):
os.rename(proc_pyc, pyc) os.rename(proc_pyc, pyc)
def _read_pyc(source, pyc): def _read_pyc(source, pyc):

View File

@ -419,3 +419,23 @@ def test_rewritten():
res = testdir.runpytest() res = testdir.runpytest()
assert res.ret != 0 assert res.ret != 0
assert "SyntaxError: Non-ASCII character" in res.stdout.str() assert "SyntaxError: Non-ASCII character" in res.stdout.str()
def test_write_pyc(self, testdir, tmpdir, monkeypatch):
from _pytest.assertion.rewrite import _write_pyc
from _pytest.assertion import AssertionState
try:
import __builtin__ as b
except ImportError:
import builtins as b
config = testdir.parseconfig([])
state = AssertionState(config, "rewrite")
source_path = tmpdir.ensure("source.py")
pycpath = tmpdir.join("pyc").strpath
assert _write_pyc(state, [1], source_path, pycpath)
def open(*args):
e = IOError()
e.errno = 10
raise e
monkeypatch.setattr(b, "open", open)
assert not _write_pyc(state, [1], source_path, pycpath)