Add tests for early rewrite bailout code and handle patterns with subdirectories

This commit is contained in:
Bruno Oliveira 2018-09-01 10:59:21 -03:00
parent d53e449296
commit 4675912d89
3 changed files with 118 additions and 22 deletions

View File

@ -67,7 +67,7 @@ class AssertionRewritingHook(object):
# flag to guard against trying to rewrite a pyc file while we are already writing another pyc file, # flag to guard against trying to rewrite a pyc file while we are already writing another pyc file,
# which might result in infinite recursion (#3506) # which might result in infinite recursion (#3506)
self._writing_pyc = False self._writing_pyc = False
self._basenames_to_check_rewrite = set('conftest',) self._basenames_to_check_rewrite = {"conftest"}
self._marked_for_rewrite_cache = {} self._marked_for_rewrite_cache = {}
self._session_paths_checked = False self._session_paths_checked = False
@ -75,6 +75,10 @@ class AssertionRewritingHook(object):
self.session = session self.session = session
self._session_paths_checked = False self._session_paths_checked = False
def _imp_find_module(self, name, path=None):
"""Indirection so we can mock calls to find_module originated from the hook during testing"""
return imp.find_module(name, path)
def find_module(self, name, path=None): def find_module(self, name, path=None):
if self._writing_pyc: if self._writing_pyc:
return None return None
@ -93,7 +97,7 @@ class AssertionRewritingHook(object):
pth = path[0] pth = path[0]
if pth is None: if pth is None:
try: try:
fd, fn, desc = imp.find_module(lastname, path) fd, fn, desc = self._imp_find_module(lastname, path)
except ImportError: except ImportError:
return None return None
if fd is not None: if fd is not None:
@ -179,8 +183,7 @@ class AssertionRewritingHook(object):
from this class) is a major slowdown, so, this method tries to from this class) is a major slowdown, so, this method tries to
filter what we're sure won't be rewritten before getting to it. filter what we're sure won't be rewritten before getting to it.
""" """
if not self._session_paths_checked and self.session is not None \ if self.session is not None and not self._session_paths_checked:
and hasattr(self.session, '_initialpaths'):
self._session_paths_checked = True self._session_paths_checked = True
for path in self.session._initialpaths: for path in self.session._initialpaths:
# Make something as c:/projects/my_project/path.py -> # Make something as c:/projects/my_project/path.py ->
@ -190,14 +193,18 @@ class AssertionRewritingHook(object):
self._basenames_to_check_rewrite.add(os.path.splitext(parts[-1])[0]) self._basenames_to_check_rewrite.add(os.path.splitext(parts[-1])[0])
# Note: conftest already by default in _basenames_to_check_rewrite. # Note: conftest already by default in _basenames_to_check_rewrite.
parts = name.split('.') parts = name.split(".")
if parts[-1] in self._basenames_to_check_rewrite: if parts[-1] in self._basenames_to_check_rewrite:
return False return False
# For matching the name it must be as if it was a filename. # For matching the name it must be as if it was a filename.
parts[-1] = parts[-1] + '.py' parts[-1] = parts[-1] + ".py"
fn_pypath = py.path.local(os.path.sep.join(parts)) fn_pypath = py.path.local(os.path.sep.join(parts))
for pat in self.fnpats: for pat in self.fnpats:
# if the pattern contains subdirectories ("tests/**.py" for example) we can't bail out based
# on the name alone because we need to match against the full path
if os.path.dirname(pat):
return False
if fn_pypath.fnmatch(pat): if fn_pypath.fnmatch(pat):
return False return False
@ -289,6 +296,16 @@ class AssertionRewritingHook(object):
raise raise
return sys.modules[name] return sys.modules[name]
def is_package(self, name):
try:
fd, fn, desc = imp.find_module(name)
except ImportError:
return False
if fd is not None:
fd.close()
tp = desc[2]
return tp == imp.PKG_DIRECTORY
@classmethod @classmethod
def _register_with_pkg_resources(cls): def _register_with_pkg_resources(cls):
""" """

View File

@ -383,6 +383,7 @@ class Session(nodes.FSCollector):
self.trace = config.trace.root.get("collection") self.trace = config.trace.root.get("collection")
self._norecursepatterns = config.getini("norecursedirs") self._norecursepatterns = config.getini("norecursedirs")
self.startdir = py.path.local() self.startdir = py.path.local()
self._initialpaths = frozenset()
# Keep track of any collected nodes in here, so we don't duplicate fixtures # Keep track of any collected nodes in here, so we don't duplicate fixtures
self._node_cache = {} self._node_cache = {}
@ -565,7 +566,6 @@ class Session(nodes.FSCollector):
"""Convert a dotted module name to path. """Convert a dotted module name to path.
""" """
try: try:
with _patched_find_module(): with _patched_find_module():
loader = pkgutil.find_loader(x) loader = pkgutil.find_loader(x)

View File

@ -1106,22 +1106,21 @@ class TestIssue925(object):
class TestIssue2121: class TestIssue2121:
def test_simple(self, testdir): def test_rewrite_python_files_contain_subdirs(self, testdir):
testdir.tmpdir.join("tests/file.py").ensure().write( testdir.makepyfile(
""" **{
"tests/file.py": """
def test_simple_failure(): def test_simple_failure():
assert 1 + 1 == 3 assert 1 + 1 == 3
""" """
}
) )
testdir.tmpdir.join("pytest.ini").write( testdir.makeini(
textwrap.dedent(
""" """
[pytest] [pytest]
python_files = tests/**.py python_files = tests/**.py
""" """
) )
)
result = testdir.runpytest() result = testdir.runpytest()
result.stdout.fnmatch_lines("*E*assert (1 + 1) == 3") result.stdout.fnmatch_lines("*E*assert (1 + 1) == 3")
@ -1153,3 +1152,83 @@ def test_rewrite_infinite_recursion(testdir, pytestconfig, monkeypatch):
hook = AssertionRewritingHook(pytestconfig) hook = AssertionRewritingHook(pytestconfig)
assert hook.find_module("test_foo") is not None assert hook.find_module("test_foo") is not None
assert len(write_pyc_called) == 1 assert len(write_pyc_called) == 1
class TestEarlyRewriteBailout(object):
@pytest.fixture
def hook(self, pytestconfig, monkeypatch, testdir):
"""Returns a patched AssertionRewritingHook instance so we can configure its initial paths and track
if imp.find_module has been called.
"""
import imp
self.find_module_calls = []
self.initial_paths = set()
class StubSession(object):
_initialpaths = self.initial_paths
def isinitpath(self, p):
return p in self._initialpaths
def spy_imp_find_module(name, path):
self.find_module_calls.append(name)
return imp.find_module(name, path)
hook = AssertionRewritingHook(pytestconfig)
# use default patterns, otherwise we inherit pytest's testing config
hook.fnpats[:] = ["test_*.py", "*_test.py"]
monkeypatch.setattr(hook, "_imp_find_module", spy_imp_find_module)
hook.set_session(StubSession())
testdir.syspathinsert()
return hook
def test_basic(self, testdir, hook):
"""
Ensure we avoid calling imp.find_module when we know for sure a certain module will not be rewritten
to optimize assertion rewriting (#3918).
"""
testdir.makeconftest(
"""
import pytest
@pytest.fixture
def fix(): return 1
"""
)
testdir.makepyfile(test_foo="def test_foo(): pass")
testdir.makepyfile(bar="def bar(): pass")
foobar_path = testdir.makepyfile(foobar="def foobar(): pass")
self.initial_paths.add(foobar_path)
# conftest files should always be rewritten
assert hook.find_module("conftest") is not None
assert self.find_module_calls == ["conftest"]
# files matching "python_files" mask should always be rewritten
assert hook.find_module("test_foo") is not None
assert self.find_module_calls == ["conftest", "test_foo"]
# file does not match "python_files": early bailout
assert hook.find_module("bar") is None
assert self.find_module_calls == ["conftest", "test_foo"]
# file is an initial path (passed on the command-line): should be rewritten
assert hook.find_module("foobar") is not None
assert self.find_module_calls == ["conftest", "test_foo", "foobar"]
def test_pattern_contains_subdirectories(self, testdir, hook):
"""If one of the python_files patterns contain subdirectories ("tests/**.py") we can't bailout early
because we need to match with the full path, which can only be found by calling imp.find_module.
"""
p = testdir.makepyfile(
**{
"tests/file.py": """
def test_simple_failure():
assert 1 + 1 == 3
"""
}
)
testdir.syspathinsert(p.dirpath())
hook.fnpats[:] = ["tests/**.py"]
assert hook.find_module("file") is not None
assert self.find_module_calls == ["file"]