Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

process path aliases in run #1027

Draft
wants to merge 3 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 6 additions & 9 deletions coverage/control.py
Original file line number Diff line number Diff line change
Expand Up @@ -678,15 +678,12 @@ def combine(self, data_paths=None, strict=False):
self._post_init()
self.get_data()

aliases = None
if self.config.paths:
aliases = PathAliases()
for paths in self.config.paths.values():
result = paths[0]
for pattern in paths[1:]:
aliases.add(pattern, result)

combine_parallel_data(self._data, aliases=aliases, data_paths=data_paths, strict=strict)
combine_parallel_data(
self._data,
aliases=PathAliases.configure(self.config),
data_paths=data_paths,
strict=strict,
)

def get_data(self):
"""Get the collected data.
Expand Down
50 changes: 44 additions & 6 deletions coverage/files.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,12 +64,13 @@ def canonical_filename(filename):
for path in [os.curdir] + sys.path:
if path is None:
continue
f = os.path.join(path, filename)

try:
exists = os.path.exists(f)
f = os.path.join(str_filename(path), str_filename(filename))
except UnicodeError:
exists = False
if exists:
continue

if os.path.exists(f):
cf = f
break
cf = abs_file(cf)
Expand Down Expand Up @@ -143,6 +144,21 @@ def actual_path(filename):
return filename


if env.PY2:
def str_filename(filename):
return (
filename
if isinstance(filename, str)
else filename.encode(
sys.getfilesystemencoding() or sys.getdefaultencoding()
)
)
else:
@contract(filename='unicode', returns='unicode')
def str_filename(filename):
return filename


if env.PY2:
@contract(returns='unicode')
def unicode_filename(filename):
Expand Down Expand Up @@ -215,7 +231,8 @@ class TreeMatcher(object):
somewhere in a subtree rooted at one of the directories.

"""
def __init__(self, paths):
def __init__(self, paths, aliases=None):
self.aliases = aliases
self.paths = list(paths)

def __repr__(self):
Expand All @@ -227,6 +244,9 @@ def info(self):

def match(self, fpath):
"""Does `fpath` indicate a file in one of our trees?"""
if self.aliases is not None:
fpath = self.aliases.map(fpath)

for p in self.paths:
if fpath.startswith(p):
if fpath == p:
Expand Down Expand Up @@ -268,7 +288,8 @@ def match(self, module_name):

class FnmatchMatcher(object):
"""A matcher for files by file name pattern."""
def __init__(self, pats):
def __init__(self, pats, aliases=None):
self.aliases = aliases
self.pats = list(pats)
self.re = fnmatches_to_regex(self.pats, case_insensitive=env.WINDOWS)

Expand All @@ -281,6 +302,8 @@ def info(self):

def match(self, fpath):
"""Does `fpath` match one of our file name patterns?"""
if self.aliases is not None:
fpath = self.aliases.map(fpath)
return self.re.match(fpath) is not None


Expand Down Expand Up @@ -408,6 +431,21 @@ def map(self, path):
return path


@classmethod
def configure(cls, config):
paths = config.paths
if not paths:
return None

aliases = PathAliases()
for paths in paths.values():
result = paths[0]
for pattern in paths[1:]:
aliases.add(pattern, result)
return aliases



def find_python_files(dirname):
"""Yield all of the importable Python files in `dirname`, recursively.

Expand Down
13 changes: 7 additions & 6 deletions coverage/inorout.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
from coverage.backward import code_object
from coverage.disposition import FileDisposition, disposition_init
from coverage.files import TreeMatcher, FnmatchMatcher, ModuleMatcher
from coverage.files import prep_patterns, find_python_files, canonical_filename
from coverage.files import prep_patterns, find_python_files, canonical_filename, PathAliases
from coverage.misc import CoverageException
from coverage.python import source_for_file, source_for_morf

Expand Down Expand Up @@ -132,6 +132,7 @@ def __init__(self, warn, debug):

def configure(self, config):
"""Apply the configuration to get ready for decision-time."""
aliases = PathAliases.configure(config)
for src in config.source or []:
if os.path.isdir(src):
self.source.append(canonical_filename(src))
Expand Down Expand Up @@ -186,24 +187,24 @@ def debug(msg):
if self.source or self.source_pkgs:
against = []
if self.source:
self.source_match = TreeMatcher(self.source)
self.source_match = TreeMatcher(self.source, aliases=aliases)
against.append("trees {!r}".format(self.source_match))
if self.source_pkgs:
self.source_pkgs_match = ModuleMatcher(self.source_pkgs)
against.append("modules {!r}".format(self.source_pkgs_match))
debug("Source matching against " + " and ".join(against))
else:
if self.cover_paths:
self.cover_match = TreeMatcher(self.cover_paths)
self.cover_match = TreeMatcher(self.cover_paths, aliases=aliases)
debug("Coverage code matching: {!r}".format(self.cover_match))
if self.pylib_paths:
self.pylib_match = TreeMatcher(self.pylib_paths)
self.pylib_match = TreeMatcher(self.pylib_paths, aliases=aliases)
debug("Python stdlib matching: {!r}".format(self.pylib_match))
if self.include:
self.include_match = FnmatchMatcher(self.include)
self.include_match = FnmatchMatcher(self.include, aliases=aliases)
debug("Include matching: {!r}".format(self.include_match))
if self.omit:
self.omit_match = FnmatchMatcher(self.omit)
self.omit_match = FnmatchMatcher(self.omit, aliases=aliases)
debug("Omit matching: {!r}".format(self.omit_match))

def should_trace(self, filename, frame=None):
Expand Down
7 changes: 5 additions & 2 deletions tests/test_process.py
Original file line number Diff line number Diff line change
Expand Up @@ -1181,9 +1181,12 @@ def test_aliases_used_in_messages(self):
"coverage-%d.%d" % sys.version_info[:2],
]
for cmd in cmds:
out = self.run_command("%s foobar" % cmd)
out = self.run_command("{} foobar".format(cmd))
self.assertIn("Unknown command: 'foobar'", out)
self.assertIn("Use '%s help' for help" % cmd, out)
self.assertIn("Use '{} help' for help".format(cmd), out)

out = self.run_command("{} help version".format(cmd))
self.assertIn("ersion {coverage.__version__}".format(coverage=coverage), out)


class PydocTest(CoverageTest):
Expand Down