Merge pull request #4237 from boxed/master

Performance fixes
This commit is contained in:
Bruno Oliveira 2018-11-03 10:39:29 -03:00 committed by GitHub
commit 4cb838d978
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 26 additions and 14 deletions

View File

@ -18,6 +18,7 @@ from _pytest.config import directory_arg
from _pytest.config import hookimpl from _pytest.config import hookimpl
from _pytest.config import UsageError from _pytest.config import UsageError
from _pytest.outcomes import exit from _pytest.outcomes import exit
from _pytest.pathlib import parts
from _pytest.runner import collect_one_node from _pytest.runner import collect_one_node
@ -469,8 +470,8 @@ class Session(nodes.FSCollector):
return items return items
def collect(self): def collect(self):
for parts in self._initialparts: for initialpart in self._initialparts:
arg = "::".join(map(str, parts)) arg = "::".join(map(str, initialpart))
self.trace("processing argument", arg) self.trace("processing argument", arg)
self.trace.root.indent += 1 self.trace.root.indent += 1
try: try:
@ -488,7 +489,7 @@ class Session(nodes.FSCollector):
names = self._parsearg(arg) names = self._parsearg(arg)
argpath = names.pop(0).realpath() argpath = names.pop(0).realpath()
paths = [] paths = set()
root = self root = self
# Start with a Session root, and delve to argpath item (dir or file) # Start with a Session root, and delve to argpath item (dir or file)
@ -528,21 +529,26 @@ class Session(nodes.FSCollector):
def filter_(f): def filter_(f):
return f.check(file=1) return f.check(file=1)
seen_dirs = set()
for path in argpath.visit( for path in argpath.visit(
fil=filter_, rec=self._recurse, bf=True, sort=True fil=filter_, rec=self._recurse, bf=True, sort=True
): ):
pkginit = path.dirpath().join("__init__.py") dirpath = path.dirpath()
if pkginit.exists() and not any(x in pkginit.parts() for x in paths): if dirpath not in seen_dirs:
seen_dirs.add(dirpath)
pkginit = dirpath.join("__init__.py")
if pkginit.exists() and parts(pkginit.strpath).isdisjoint(paths):
for x in root._collectfile(pkginit): for x in root._collectfile(pkginit):
yield x yield x
paths.append(x.fspath.dirpath()) paths.add(x.fspath.dirpath())
if not any(x in path.parts() for x in paths): if parts(path.strpath).isdisjoint(paths):
for x in root._collectfile(path): for x in root._collectfile(path):
if (type(x), x.fspath) in self._node_cache: key = (type(x), x.fspath)
yield self._node_cache[(type(x), x.fspath)] if key in self._node_cache:
yield self._node_cache[key]
else: else:
self._node_cache[(type(x), x.fspath)] = x self._node_cache[key] = x
yield x yield x
else: else:
assert argpath.check(file=1) assert argpath.check(file=1)

View File

@ -303,3 +303,8 @@ def fnmatch_ex(pattern, path):
else: else:
name = six.text_type(path) name = six.text_type(path)
return fnmatch.fnmatch(name, pattern) return fnmatch.fnmatch(name, pattern)
def parts(s):
parts = s.split(sep)
return {sep.join(parts[: i + 1]) or sep for i in range(len(parts))}

View File

@ -41,6 +41,7 @@ from _pytest.mark.structures import get_unpacked_marks
from _pytest.mark.structures import normalize_mark_list from _pytest.mark.structures import normalize_mark_list
from _pytest.mark.structures import transfer_markers from _pytest.mark.structures import transfer_markers
from _pytest.outcomes import fail from _pytest.outcomes import fail
from _pytest.pathlib import parts
from _pytest.warning_types import PytestWarning from _pytest.warning_types import PytestWarning
from _pytest.warning_types import RemovedInPytest4Warning from _pytest.warning_types import RemovedInPytest4Warning
@ -567,9 +568,9 @@ class Package(Module):
if path.basename == "__init__.py" and path.dirpath() == this_path: if path.basename == "__init__.py" and path.dirpath() == this_path:
continue continue
parts = path.parts() parts_ = parts(path.strpath)
if any( if any(
pkg_prefix in parts and pkg_prefix.join("__init__.py") != path pkg_prefix in parts_ and pkg_prefix.join("__init__.py") != path
for pkg_prefix in pkg_prefixes for pkg_prefix in pkg_prefixes
): ):
continue continue