##// END OF EJS Templates
debug: allow specifying a manifest node rather than a revision
debug: allow specifying a manifest node rather than a revision

File last commit:

r38773:a8bfaf59 stable
r38802:ddb15a83 default
Show More
match.py
1132 lines | 37.3 KiB | text/x-python | PythonLexer
timeless
Generally replace "file name" with "filename" in help and comments.
r8761 # match.py - filename matching
Martin Geisler
match: add copyright and license header
r8231 #
# Copyright 2008, 2009 Matt Mackall <mpm@selenic.com> and others
#
# This software may be used and distributed according to the terms of the
Matt Mackall
Update license to GPLv2+
r10263 # GNU General Public License version 2 or any later version.
Martin Geisler
match: add copyright and license header
r8231
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 from __future__ import absolute_import, print_function
Gregory Szorc
match: use absolute_import
r25958
import copy
import os
import re
from .i18n import _
from . import (
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 encoding,
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 error,
Gregory Szorc
match: use absolute_import
r25958 pathutil,
Augie Fackler
match: some minimal pycompat fixes guided by test-hgignore.t...
r36590 pycompat,
Gregory Szorc
match: use absolute_import
r25958 util,
)
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 from .utils import (
stringutil,
)
Matt Mackall
walk: introduce match objects
r6576
Kostia Balytskyi
match: expose some data and functionality to other modules...
r33647 allpatternkinds = ('re', 'glob', 'path', 'relglob', 'relpath', 'relre',
'listfile', 'listfile0', 'set', 'include', 'subinclude',
'rootfilesin')
cwdrelativepatternkinds = ('relpath', 'glob')
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636 propertycache = util.propertycache
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _rematcher(regex):
'''compile the regexp with the best available regexp engine and return a
matcher function'''
Siddharth Agarwal
match: use util.re.compile instead of util.compilere
r21909 m = util.re.compile(regex)
Bryan O'Sullivan
matcher: use re2 bindings if available...
r16943 try:
# slightly faster, provided by facebook's re2 bindings
return m.test_match
except AttributeError:
return m.match
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 def _expandsets(root, cwd, kindpats, ctx, listsubrepos, badfn):
'''Returns the kindpats list with the 'set' patterns expanded to matchers'''
matchers = []
Matt Mackall
match: introduce basic fileset support
r14675 other = []
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Matt Mackall
match: introduce basic fileset support
r14675 if kind == 'set':
if not ctx:
Martin von Zweigbergk
match: use ProgrammingError where appropriate
r32444 raise error.ProgrammingError("fileset expression with no "
"context")
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 matchers.append(ctx.matchfileset(pat, badfn=badfn))
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122
if listsubrepos:
for subpath in ctx.substate:
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 sm = ctx.sub(subpath).matchfileset(pat, badfn=badfn)
pm = prefixdirmatcher(root, cwd, subpath, sm, badfn=badfn)
matchers.append(pm)
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122
Matt Mackall
match: introduce basic fileset support
r14675 continue
Durham Goode
match: add source to kindpats list...
r25213 other.append((kind, pat, source))
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 return matchers, other
Matt Mackall
match: introduce basic fileset support
r14675
Durham Goode
match: enable 'subinclude:' syntax...
r25283 def _expandsubinclude(kindpats, root):
Durham Goode
match: make subinclude construction lazy...
r32132 '''Returns the list of subinclude matcher args and the kindpats without the
Durham Goode
match: enable 'subinclude:' syntax...
r25283 subincludes in it.'''
relmatchers = []
other = []
for kind, pat, source in kindpats:
if kind == 'subinclude':
Matt Harbison
match: normpath the ignore source when expanding the 'subinclude' kind...
r25301 sourceroot = pathutil.dirname(util.normpath(source))
Durham Goode
match: enable 'subinclude:' syntax...
r25283 pat = util.pconvert(pat)
path = pathutil.join(sourceroot, pat)
newroot = pathutil.dirname(path)
Durham Goode
match: make subinclude construction lazy...
r32132 matcherargs = (newroot, '', [], ['include:%s' % path])
Durham Goode
match: enable 'subinclude:' syntax...
r25283
prefix = pathutil.canonpath(root, root, newroot)
if prefix:
prefix += '/'
Durham Goode
match: make subinclude construction lazy...
r32132 relmatchers.append((prefix, matcherargs))
Durham Goode
match: enable 'subinclude:' syntax...
r25283 else:
other.append((kind, pat, source))
return relmatchers, other
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 def _kindpatsalwaysmatch(kindpats):
""""Checks whether the kindspats match everything, as e.g.
'relpath:.' does.
"""
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 if pat != '' or kind not in ['relpath', 'glob']:
return False
return True
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 def _buildkindpatsmatcher(matchercls, root, cwd, kindpats, ctx=None,
listsubrepos=False, badfn=None):
matchers = []
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 fms, kindpats = _expandsets(root, cwd, kindpats, ctx=ctx,
listsubrepos=listsubrepos, badfn=badfn)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 if kindpats:
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 m = matchercls(root, cwd, kindpats, listsubrepos=listsubrepos,
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 badfn=badfn)
matchers.append(m)
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 if fms:
matchers.extend(fms)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 if not matchers:
return nevermatcher(root, cwd, badfn=badfn)
if len(matchers) == 1:
return matchers[0]
return unionmatcher(matchers)
Martin von Zweigbergk
match: allow pats to be None...
r32728 def match(root, cwd, patterns=None, include=None, exclude=None, default='glob',
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 exact=False, auditor=None, ctx=None, listsubrepos=False, warn=None,
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 badfn=None, icasefs=False):
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 """build an object to match a set of file patterns
arguments:
root - the canonical root of the tree you're matching against
cwd - the current working directory, if relevant
patterns - patterns to find
include - patterns to include (unless they are excluded)
exclude - patterns to exclude (even if they are included)
default - if a pattern in patterns has no explicit type, assume this one
exact - patterns are actually filenames (include/exclude still apply)
warn - optional function used for printing warnings
badfn - optional bad() callback for this matcher instead of the default
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 icasefs - make a matcher for wdir on case insensitive filesystems, which
normalizes the given patterns to the case in the filesystem
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
a pattern is one of:
'glob:<glob>' - a glob relative to cwd
're:<regexp>' - a regular expression
'path:<path>' - a path relative to repository root, which is matched
recursively
'rootfilesin:<path>' - a path relative to repository root, which is
matched non-recursively (will not match subdirectories)
'relglob:<glob>' - an unrooted glob (*.c matches C files in all dirs)
'relpath:<path>' - a path relative to cwd
'relre:<regexp>' - a regexp that needn't match the start of a name
'set:<fileset>' - a fileset expression
'include:<path>' - a file of patterns to read and include
'subinclude:<path>' - a file of patterns to match against files under
the same directory
'<something>' - a pattern of the specified default type
"""
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 normalize = _donormalize
if icasefs:
Martin von Zweigbergk
match: catch attempts to create case-insenstive exact matchers...
r32415 if exact:
Martin von Zweigbergk
match: use ProgrammingError where appropriate
r32444 raise error.ProgrammingError("a case-insensitive exact matcher "
"doesn't make sense")
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 dirstate = ctx.repo().dirstate
dsnormalize = dirstate.normalize
def normalize(patterns, default, root, cwd, auditor, warn):
kp = _donormalize(patterns, default, root, cwd, auditor, warn)
kindpats = []
for kind, pats, source in kp:
if kind not in ('re', 'relre'): # regex can't be normalized
p = pats
pats = dsnormalize(pats)
# Preserve the original to handle a case only rename.
if p != pats and p in dirstate:
kindpats.append((kind, p, source))
kindpats.append((kind, pats, source))
return kindpats
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 if exact:
m = exactmatcher(root, cwd, patterns, badfn)
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 elif patterns:
Martin von Zweigbergk
match: move normalize() call out of matcher constructors...
r32556 kindpats = normalize(patterns, default, root, cwd, auditor, warn)
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 if _kindpatsalwaysmatch(kindpats):
m = alwaysmatcher(root, cwd, badfn, relativeuipath=True)
else:
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 m = _buildkindpatsmatcher(patternmatcher, root, cwd, kindpats,
ctx=ctx, listsubrepos=listsubrepos,
badfn=badfn)
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 else:
# It's a little strange that no patterns means to match everything.
Martin von Zweigbergk
match: simplify nevermatcher...
r32650 # Consider changing this to match nothing (probably using nevermatcher).
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 m = alwaysmatcher(root, cwd, badfn)
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 if include:
Martin von Zweigbergk
match: move normalize() call out of matcher constructors...
r32556 kindpats = normalize(include, 'glob', root, cwd, auditor, warn)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 im = _buildkindpatsmatcher(includematcher, root, cwd, kindpats, ctx=ctx,
listsubrepos=listsubrepos, badfn=None)
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 m = intersectmatchers(m, im)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 if exclude:
Martin von Zweigbergk
match: move normalize() call out of matcher constructors...
r32556 kindpats = normalize(exclude, 'glob', root, cwd, auditor, warn)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 em = _buildkindpatsmatcher(includematcher, root, cwd, kindpats, ctx=ctx,
listsubrepos=listsubrepos, badfn=None)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 m = differencematcher(m, em)
return m
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
def exact(root, cwd, files, badfn=None):
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 return exactmatcher(root, cwd, files, badfn=badfn)
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
def always(root, cwd):
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 return alwaysmatcher(root, cwd)
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600 def never(root, cwd):
return nevermatcher(root, cwd)
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 def badmatch(match, badfn):
"""Make a copy of the given matcher, replacing its bad method with the given
one.
"""
m = copy.copy(match)
m.bad = badfn
return m
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 def _donormalize(patterns, default, root, cwd, auditor, warn):
'''Convert 'kind:pat' from the patterns list to tuples with kind and
normalized and rooted patterns and with listfiles expanded.'''
kindpats = []
for kind, pat in [_patsplit(p, default) for p in patterns]:
Kostia Balytskyi
match: expose some data and functionality to other modules...
r33647 if kind in cwdrelativepatternkinds:
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 pat = pathutil.canonpath(root, cwd, pat, auditor)
elif kind in ('relglob', 'path', 'rootfilesin'):
pat = util.normpath(pat)
elif kind in ('listfile', 'listfile0'):
try:
files = util.readfile(pat)
if kind == 'listfile0':
files = files.split('\0')
else:
files = files.splitlines()
files = [f for f in files if f]
except EnvironmentError:
raise error.Abort(_("unable to read file list (%s)") % pat)
for k, p, source in _donormalize(files, default, root, cwd,
auditor, warn):
kindpats.append((k, p, pat))
continue
elif kind == 'include':
try:
fullpath = os.path.join(root, util.localpath(pat))
includepats = readpatternfile(fullpath, warn)
for k, p, source in _donormalize(includepats, default,
root, cwd, auditor, warn):
kindpats.append((k, p, source or pat))
except error.Abort as inst:
raise error.Abort('%s: %s' % (pat, inst[0]))
except IOError as inst:
if warn:
warn(_("skipping unreadable pattern file '%s': %s\n") %
Yuya Nishihara
stringutil: bulk-replace call sites to point to new module...
r37102 (pat, stringutil.forcebytestr(inst.strerror)))
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 continue
# else: re or relre - which cannot be normalized
kindpats.append((kind, pat, ''))
return kindpats
Martin von Zweigbergk
match: extract base class for matchers...
r32454 class basematcher(object):
Martin von Zweigbergk
match: move entire uipath() implementation to basematcher...
r32496 def __init__(self, root, cwd, badfn=None, relativeuipath=True):
Martin von Zweigbergk
match: extract base class for matchers...
r32454 self._root = root
self._cwd = cwd
if badfn is not None:
self.bad = badfn
Martin von Zweigbergk
match: move entire uipath() implementation to basematcher...
r32496 self._relativeuipath = relativeuipath
Martin von Zweigbergk
match: extract base class for matchers...
r32454
def __call__(self, fn):
return self.matchfn(fn)
def __iter__(self):
for f in self._files:
yield f
# Callbacks related to how the matcher is used by dirstate.walk.
# Subscribers to these events must monkeypatch the matcher object.
def bad(self, f, msg):
'''Callback from dirstate.walk for each explicit file that can't be
found/accessed, with an error message.'''
# If an explicitdir is set, it will be called when an explicitly listed
# directory is visited.
explicitdir = None
# If an traversedir is set, it will be called when a directory discovered
# by recursive traversal is visited.
traversedir = None
def abs(self, f):
'''Convert a repo path back to path that is relative to the root of the
matcher.'''
return f
def rel(self, f):
'''Convert repo path back to path that is relative to cwd of matcher.'''
return util.pathto(self._root, self._cwd, f)
def uipath(self, f):
'''Convert repo path to a display path. If patterns or -I/-X were used
to create this matcher, the display path will be relative to cwd.
Otherwise it is relative to the root of the repo.'''
Martin von Zweigbergk
match: move entire uipath() implementation to basematcher...
r32496 return (self._relativeuipath and self.rel(f)) or self.abs(f)
Martin von Zweigbergk
match: extract base class for matchers...
r32454
Martin von Zweigbergk
match: make basematcher._files a @propertycache...
r32455 @propertycache
def _files(self):
return []
Martin von Zweigbergk
match: extract base class for matchers...
r32454 def files(self):
'''Explicitly listed files or patterns or roots:
if no patterns or .always(): empty list,
if exact: list exact files,
if not .anypats(): list all files and dirs,
else: optimal roots'''
return self._files
@propertycache
def _fileset(self):
return set(self._files)
def exact(self, f):
'''Returns True if f is in .files().'''
return f in self._fileset
Martin von Zweigbergk
match: make matchfn a method on the class...
r32463 def matchfn(self, f):
return False
Martin von Zweigbergk
match: extract base class for matchers...
r32454 def visitdir(self, dir):
'''Decides whether a directory should be visited based on whether it
has potential matches in it or one of its subdirectories. This is
based on the match's primary, included, and excluded patterns.
Returns the string 'all' if the given directory and all subdirectories
should be visited. Otherwise returns True or False indicating whether
the given directory should be visited.
'''
Durham Goode
match: make base matcher return True for visitdir...
r33478 return True
Martin von Zweigbergk
match: extract base class for matchers...
r32454
def always(self):
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 '''Matcher will match everything and .files() will be empty --
optimization might be possible.'''
Martin von Zweigbergk
match: extract base class for matchers...
r32454 return False
def isexact(self):
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 '''Matcher will match exactly the list of files in .files() --
optimization might be possible.'''
Martin von Zweigbergk
match: extract base class for matchers...
r32454 return False
def prefix(self):
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 '''Matcher will match the paths in .files() recursively --
optimization might be possible.'''
return False
def anypats(self):
'''None of .always(), .isexact(), and .prefix() is true --
optimizations will be difficult.'''
return not self.always() and not self.isexact() and not self.prefix()
Martin von Zweigbergk
match: extract base class for matchers...
r32454
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 class alwaysmatcher(basematcher):
'''Matches everything.'''
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 def __init__(self, root, cwd, badfn=None, relativeuipath=False):
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 super(alwaysmatcher, self).__init__(root, cwd, badfn,
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 relativeuipath=relativeuipath)
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553
def always(self):
return True
def matchfn(self, f):
return True
def visitdir(self, dir):
return 'all'
def __repr__(self):
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 return r'<alwaysmatcher>'
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600 class nevermatcher(basematcher):
'''Matches nothing.'''
Martin von Zweigbergk
match: simplify nevermatcher...
r32650 def __init__(self, root, cwd, badfn=None):
super(nevermatcher, self).__init__(root, cwd, badfn)
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600
Martin von Zweigbergk
match: make nevermatcher an exact matcher and a prefix matcher...
r33378 # It's a little weird to say that the nevermatcher is an exact matcher
# or a prefix matcher, but it seems to make sense to let callers take
# fast paths based on either. There will be no exact matches, nor any
# prefixes (files() returns []), so fast paths iterating over them should
# be efficient (and correct).
def isexact(self):
return True
def prefix(self):
return True
Martin von Zweigbergk
match: override visitdir() in nevermatcher to return False...
r33583 def visitdir(self, dir):
return False
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600 def __repr__(self):
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 return r'<nevermatcher>'
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600
Yuya Nishihara
match: add basic wrapper for boolean function...
r38596 class predicatematcher(basematcher):
"""A matcher adapter for a simple boolean function"""
def __init__(self, root, cwd, predfn, predrepr=None, badfn=None):
super(predicatematcher, self).__init__(root, cwd, badfn)
self.matchfn = predfn
self._predrepr = predrepr
@encoding.strmethod
def __repr__(self):
s = (stringutil.buildrepr(self._predrepr)
or pycompat.byterepr(self.matchfn))
return '<predicatenmatcher pred=%s>' % s
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501 class patternmatcher(basematcher):
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 def __init__(self, root, cwd, kindpats, listsubrepos=False, badfn=None):
Martin von Zweigbergk
match: drop support for empty pattern list in patternmatcher...
r32555 super(patternmatcher, self).__init__(root, cwd, badfn)
Matt Mackall
match: fold _matcher into match.__init__
r8581
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 self._files = _explicitfiles(kindpats)
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 self._prefix = _prefix(kindpats)
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 self._pats, self.matchfn = _buildmatch(kindpats, '$', listsubrepos,
Martin von Zweigbergk
match: minor cleanups to patternmatcher and includematcher...
r33306 root)
Matt Mackall
match: fold match into _match base class
r8587
Martin von Zweigbergk
match: make _fileroots a @propertycache and rename it to _fileset...
r32323 @propertycache
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636 def _dirs(self):
Martin von Zweigbergk
match: make _fileroots a @propertycache and rename it to _fileset...
r32323 return set(util.dirs(self._fileset)) | {'.'}
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636
def visitdir(self, dir):
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 if self._prefix and dir in self._fileset:
Martin von Zweigbergk
treemanifest: don't iterate entire matching submanifests on match()...
r27343 return 'all'
Martin von Zweigbergk
match: optimize visitdir() for when no explicit files are listed...
r32554 return ('.' in self._fileset or
Martin von Zweigbergk
match: make _fileroots a @propertycache and rename it to _fileset...
r32323 dir in self._fileset or
Martin von Zweigbergk
match: break boolean expressions into one operand per line...
r25576 dir in self._dirs or
Martin von Zweigbergk
match: make _fileroots a @propertycache and rename it to _fileset...
r32323 any(parentdir in self._fileset
Martin von Zweigbergk
match: drop optimization (?) of 'parentdirs' calculation...
r25577 for parentdir in util.finddirs(dir)))
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 def prefix(self):
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 return self._prefix
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: implement __repr__() and update users (API)...
r32406 def __repr__(self):
Pulkit Goyal
py3: use pycompat.bytestr() on bytes before %r-ing it...
r38039 return ('<patternmatcher patterns=%r>' % pycompat.bytestr(self._pats))
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
class includematcher(basematcher):
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 def __init__(self, root, cwd, kindpats, listsubrepos=False, badfn=None):
Martin von Zweigbergk
match: remove support for non-include patterns from includematcher...
r32502 super(includematcher, self).__init__(root, cwd, badfn)
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 self._pats, self.matchfn = _buildmatch(kindpats, '(?:/|$)',
Martin von Zweigbergk
match: minor cleanups to patternmatcher and includematcher...
r33306 listsubrepos, root)
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 self._prefix = _prefix(kindpats)
Martin von Zweigbergk
match: remove support for non-include patterns from includematcher...
r32502 roots, dirs = _rootsanddirs(kindpats)
# roots are directories which are recursively included.
Martin von Zweigbergk
match: simplify includematcher a bit...
r32503 self._roots = set(roots)
Martin von Zweigbergk
match: remove support for non-include patterns from includematcher...
r32502 # dirs are directories which are non-recursively included.
Martin von Zweigbergk
match: simplify includematcher a bit...
r32503 self._dirs = set(dirs)
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
def visitdir(self, dir):
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 if self._prefix and dir in self._roots:
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501 return 'all'
Martin von Zweigbergk
match: simplify includematcher a bit...
r32503 return ('.' in self._roots or
dir in self._roots or
dir in self._dirs or
any(parentdir in self._roots
for parentdir in util.finddirs(dir)))
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501 def __repr__(self):
Augie Fackler
match: some minimal pycompat fixes guided by test-hgignore.t...
r36590 return ('<includematcher includes=%r>' % pycompat.bytestr(self._pats))
Martin von Zweigbergk
match: implement __repr__() and update users (API)...
r32406
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 class exactmatcher(basematcher):
'''Matches the input files exactly. They are interpreted as paths, not
patterns (so no kind-prefixes).
'''
def __init__(self, root, cwd, files, badfn=None):
super(exactmatcher, self).__init__(root, cwd, badfn)
if isinstance(files, list):
self._files = files
else:
self._files = list(files)
Yuya Nishihara
match: define exactmatcher.matchfn statically...
r32543
matchfn = basematcher.exact
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499
@propertycache
def _dirs(self):
return set(util.dirs(self._fileset)) | {'.'}
def visitdir(self, dir):
return dir in self._dirs
def isexact(self):
return True
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 def __repr__(self):
return ('<exactmatcher files=%r>' % self._files)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 class differencematcher(basematcher):
'''Composes two matchers by matching if the first matches and the second
Yuya Nishihara
match: do not weirdly include explicit files excluded by -X option...
r35677 does not.
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465
The second matcher's non-matching-attributes (root, cwd, bad, explicitdir,
traversedir) are ignored.
'''
def __init__(self, m1, m2):
super(differencematcher, self).__init__(m1._root, m1._cwd)
self._m1 = m1
self._m2 = m2
self.bad = m1.bad
self.explicitdir = m1.explicitdir
self.traversedir = m1.traversedir
def matchfn(self, f):
Yuya Nishihara
match: do not weirdly include explicit files excluded by -X option...
r35677 return self._m1(f) and not self._m2(f)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465
@propertycache
def _files(self):
if self.isexact():
return [f for f in self._m1.files() if self(f)]
# If m1 is not an exact matcher, we can't easily figure out the set of
# files, because its files() are not always files. For example, if
# m1 is "path:dir" and m2 is "rootfileins:.", we don't
# want to remove "dir" from the set even though it would match m2,
# because the "dir" in m1 may not be a file.
return self._m1.files()
def visitdir(self, dir):
if self._m2.visitdir(dir) == 'all':
return False
return bool(self._m1.visitdir(dir))
def isexact(self):
return self._m1.isexact()
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 def __repr__(self):
return ('<differencematcher m1=%r, m2=%r>' % (self._m1, self._m2))
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 def intersectmatchers(m1, m2):
'''Composes two matchers by matching if both of them match.
The second matcher's non-matching-attributes (root, cwd, bad, explicitdir,
traversedir) are ignored.
'''
if m1 is None or m2 is None:
return m1 or m2
if m1.always():
m = copy.copy(m2)
# TODO: Consider encapsulating these things in a class so there's only
# one thing to copy from m1.
m.bad = m1.bad
m.explicitdir = m1.explicitdir
m.traversedir = m1.traversedir
m.abs = m1.abs
m.rel = m1.rel
m._relativeuipath |= m1._relativeuipath
return m
if m2.always():
m = copy.copy(m1)
m._relativeuipath |= m2._relativeuipath
return m
return intersectionmatcher(m1, m2)
class intersectionmatcher(basematcher):
def __init__(self, m1, m2):
super(intersectionmatcher, self).__init__(m1._root, m1._cwd)
self._m1 = m1
self._m2 = m2
self.bad = m1.bad
self.explicitdir = m1.explicitdir
self.traversedir = m1.traversedir
@propertycache
def _files(self):
if self.isexact():
m1, m2 = self._m1, self._m2
if not m1.isexact():
m1, m2 = m2, m1
return [f for f in m1.files() if m2(f)]
# It neither m1 nor m2 is an exact matcher, we can't easily intersect
# the set of files, because their files() are not always files. For
# example, if intersecting a matcher "-I glob:foo.txt" with matcher of
# "path:dir2", we don't want to remove "dir2" from the set.
return self._m1.files() + self._m2.files()
def matchfn(self, f):
return self._m1(f) and self._m2(f)
def visitdir(self, dir):
visit1 = self._m1.visitdir(dir)
if visit1 == 'all':
return self._m2.visitdir(dir)
# bool() because visit1=True + visit2='all' should not be 'all'
return bool(visit1 and self._m2.visitdir(dir))
def always(self):
return self._m1.always() and self._m2.always()
def isexact(self):
return self._m1.isexact() or self._m2.isexact()
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 def __repr__(self):
return ('<intersectionmatcher m1=%r, m2=%r>' % (self._m1, self._m2))
Martin von Zweigbergk
match: make subdirmatcher extend basematcher...
r32456 class subdirmatcher(basematcher):
Martin Geisler
match: add narrowmatcher class...
r12165 """Adapt a matcher to work on a subdirectory only.
The paths are remapped to remove/insert the path as needed:
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> from . import pycompat
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> m1 = match(b'root', b'', [b'a.txt', b'sub/b.txt'])
>>> m2 = subdirmatcher(b'sub', m1)
>>> bool(m2(b'a.txt'))
Martin Geisler
match: add narrowmatcher class...
r12165 False
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> bool(m2(b'b.txt'))
Martin Geisler
match: add narrowmatcher class...
r12165 True
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> bool(m2.matchfn(b'a.txt'))
Martin Geisler
match: add narrowmatcher class...
r12165 False
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> bool(m2.matchfn(b'b.txt'))
Martin Geisler
match: add narrowmatcher class...
r12165 True
>>> m2.files()
['b.txt']
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> m2.exact(b'b.txt')
Martin Geisler
match: add narrowmatcher class...
r12165 True
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> util.pconvert(m2.rel(b'b.txt'))
Matt Harbison
narrowmatcher: propagate the rel() method...
r23686 'sub/b.txt'
Martin Geisler
narrowmatcher: propagate bad method...
r12268 >>> def bad(f, msg):
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 ... print(pycompat.sysstr(b"%s: %s" % (f, msg)))
Martin Geisler
narrowmatcher: propagate bad method...
r12268 >>> m1.bad = bad
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> m2.bad(b'x.txt', b'No such file')
Martin Geisler
narrowmatcher: propagate bad method...
r12268 sub/x.txt: No such file
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> m2.abs(b'c.txt')
Matt Harbison
match: add the abs() method...
r23685 'sub/c.txt'
Martin Geisler
match: add narrowmatcher class...
r12165 """
def __init__(self, path, matcher):
Martin von Zweigbergk
match: make subdirmatcher extend basematcher...
r32456 super(subdirmatcher, self).__init__(matcher._root, matcher._cwd)
Martin Geisler
match: add narrowmatcher class...
r12165 self._path = path
self._matcher = matcher
Martin von Zweigbergk
match: make subdirmatcher extend basematcher...
r32456 self._always = matcher.always()
Martin Geisler
match: add narrowmatcher class...
r12165
self._files = [f[len(path) + 1:] for f in matcher._files
if f.startswith(path + "/")]
Matt Harbison
match: explicitly naming a subrepo implies always() for the submatcher...
r25194
Martin von Zweigbergk
match: use match.prefix() in subdirmatcher...
r32326 # If the parent repo had a path to this subrepo and the matcher is
# a prefix matcher, this submatcher always matches.
if matcher.prefix():
Matt Mackall
merge with stable
r25195 self._always = any(f == path for f in matcher._files)
Matt Harbison
match: explicitly naming a subrepo implies always() for the submatcher...
r25194
Martin von Zweigbergk
match: avoid accessing match._pathrestricted from subdirmatcher...
r32325 def bad(self, f, msg):
self._matcher.bad(self._path + "/" + f, msg)
Matt Harbison
match: add the abs() method...
r23685 def abs(self, f):
return self._matcher.abs(self._path + "/" + f)
Matt Harbison
narrowmatcher: propagate the rel() method...
r23686 def rel(self, f):
return self._matcher.rel(self._path + "/" + f)
Martin von Zweigbergk
match: avoid accessing match._pathrestricted from subdirmatcher...
r32325 def uipath(self, f):
return self._matcher.uipath(self._path + "/" + f)
Martin von Zweigbergk
match: override matchfn() the usual way in subdirmatcher
r32464 def matchfn(self, f):
# Some information is lost in the superclass's constructor, so we
# can not accurately create the matching function for the subdirectory
# from the inputs. Instead, we override matchfn() and visitdir() to
# call the original matcher with the subdirectory path prepended.
return self._matcher.matchfn(self._path + "/" + f)
Martin von Zweigbergk
match: override visitdir() the usual way in subdirmatcher...
r32324 def visitdir(self, dir):
if dir == '.':
dir = self._path
else:
dir = self._path + "/" + dir
return self._matcher.visitdir(dir)
Martin von Zweigbergk
match: make subdirmatcher extend basematcher...
r32456 def always(self):
return self._always
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 def prefix(self):
return self._matcher.prefix() and not self._always
Martin von Zweigbergk
match: make subdirmatcher extend basematcher...
r32456
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Martin von Zweigbergk
match: add __repr__ for subdirmatcher...
r32552 def __repr__(self):
return ('<subdirmatcher path=%r, matcher=%r>' %
(self._path, self._matcher))
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 class prefixdirmatcher(basematcher):
"""Adapt a matcher to work on a parent directory.
The matcher's non-matching-attributes (root, cwd, bad, explicitdir,
traversedir) are ignored.
The prefix path should usually be the relative path from the root of
this matcher to the root of the wrapped matcher.
Yuya Nishihara
doctest: convert matcher root to native path...
r38773 >>> m1 = match(util.localpath(b'root/d/e'), b'f', [b'../a.txt', b'b.txt'])
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 >>> m2 = prefixdirmatcher(b'root', b'd/e/f', b'd/e', m1)
>>> bool(m2(b'a.txt'),)
False
>>> bool(m2(b'd/e/a.txt'))
True
>>> bool(m2(b'd/e/b.txt'))
False
>>> m2.files()
['d/e/a.txt', 'd/e/f/b.txt']
>>> m2.exact(b'd/e/a.txt')
True
>>> m2.visitdir(b'd')
True
>>> m2.visitdir(b'd/e')
True
>>> m2.visitdir(b'd/e/f')
True
>>> m2.visitdir(b'd/e/g')
False
>>> m2.visitdir(b'd/ef')
False
"""
def __init__(self, root, cwd, path, matcher, badfn=None):
super(prefixdirmatcher, self).__init__(root, cwd, badfn)
if not path:
raise error.ProgrammingError('prefix path must not be empty')
self._path = path
self._pathprefix = path + '/'
self._matcher = matcher
@propertycache
def _files(self):
return [self._pathprefix + f for f in self._matcher._files]
def matchfn(self, f):
if not f.startswith(self._pathprefix):
return False
return self._matcher.matchfn(f[len(self._pathprefix):])
@propertycache
def _pathdirs(self):
return set(util.finddirs(self._path)) | {'.'}
def visitdir(self, dir):
if dir == self._path:
return self._matcher.visitdir('.')
if dir.startswith(self._pathprefix):
return self._matcher.visitdir(dir[len(self._pathprefix):])
return dir in self._pathdirs
def isexact(self):
return self._matcher.isexact()
def prefix(self):
return self._matcher.prefix()
@encoding.strmethod
def __repr__(self):
return ('<prefixdirmatcher path=%r, matcher=%r>'
% (pycompat.bytestr(self._path), self._matcher))
Gregory Szorc
match: move matchers from sparse into core...
r33319 class unionmatcher(basematcher):
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 """A matcher that is the union of several matchers.
The non-matching-attributes (root, cwd, bad, explicitdir, traversedir) are
taken from the first matcher.
"""
Gregory Szorc
match: move matchers from sparse into core...
r33319 def __init__(self, matchers):
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 m1 = matchers[0]
super(unionmatcher, self).__init__(m1._root, m1._cwd)
self.explicitdir = m1.explicitdir
self.traversedir = m1.traversedir
Gregory Szorc
match: move matchers from sparse into core...
r33319 self._matchers = matchers
Martin von Zweigbergk
match: override matchfn instead of __call__ for consistency...
r33380 def matchfn(self, f):
Gregory Szorc
match: move matchers from sparse into core...
r33319 for match in self._matchers:
Martin von Zweigbergk
match: override matchfn instead of __call__ for consistency...
r33380 if match(f):
Gregory Szorc
match: move matchers from sparse into core...
r33319 return True
return False
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 def visitdir(self, dir):
r = False
for m in self._matchers:
v = m.visitdir(dir)
if v == 'all':
return v
r |= v
return r
Pulkit Goyal
py3: make sure we return str from __repr__...
r36067 @encoding.strmethod
Gregory Szorc
match: move matchers from sparse into core...
r33319 def __repr__(self):
return ('<unionmatcher matchers=%r>' % self._matchers)
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def patkind(pattern, default=None):
'''If pattern is 'kind:pat' with a known kind, return kind.'''
return _patsplit(pattern, default)[0]
Matt Mackall
match: move util match functions over
r8570
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _patsplit(pattern, default):
"""Split a string into the optional pattern kind prefix and the actual
pattern."""
if ':' in pattern:
kind, pat = pattern.split(':', 1)
Kostia Balytskyi
match: expose some data and functionality to other modules...
r33647 if kind in allpatternkinds:
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 return kind, pat
return default, pattern
Matt Mackall
match: move util match functions over
r8570
Matt Mackall
match: remove head and tail args from _globre
r8582 def _globre(pat):
Mads Kiilerich
match: _globre doctests
r21112 r'''Convert an extended glob string to a regexp string.
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> from . import pycompat
>>> def bprint(s):
... print(pycompat.sysstr(s))
>>> bprint(_globre(br'?'))
Mads Kiilerich
match: _globre doctests
r21112 .
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'*'))
Mads Kiilerich
match: _globre doctests
r21112 [^/]*
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'**'))
Mads Kiilerich
match: _globre doctests
r21112 .*
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'**/a'))
Siddharth Agarwal
match: make glob '**/' match the empty string...
r21815 (?:.*/)?a
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'a/**/b'))
Augie Fackler
cleanup: migrate from re.escape to stringutil.reescape...
r38494 a/(?:.*/)?b
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'[a*?!^][^b][!c]'))
Mads Kiilerich
match: _globre doctests
r21112 [a*?!^][\^b][^c]
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'{a,b}'))
Mads Kiilerich
match: _globre doctests
r21112 (?:a|b)
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 >>> bprint(_globre(br'.\*\?'))
Mads Kiilerich
match: _globre doctests
r21112 \.\*\?
'''
Matt Mackall
match: move util match functions over
r8570 i, n = 0, len(pat)
res = ''
group = 0
Augie Fackler
cleanup: migrate from re.escape to stringutil.reescape...
r38494 escape = util.stringutil.reescape
Matt Mackall
many, many trivial check-code fixups
r10282 def peek():
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 return i < n and pat[i:i + 1]
Matt Mackall
match: move util match functions over
r8570 while i < n:
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 c = pat[i:i + 1]
Matt Mackall
many, many trivial check-code fixups
r10282 i += 1
Matt Mackall
match: optimize escaping in _globre...
r8583 if c not in '*?[{},\\':
res += escape(c)
elif c == '*':
Matt Mackall
match: move util match functions over
r8570 if peek() == '*':
i += 1
Siddharth Agarwal
match: make glob '**/' match the empty string...
r21815 if peek() == '/':
i += 1
res += '(?:.*/)?'
else:
res += '.*'
Matt Mackall
match: move util match functions over
r8570 else:
res += '[^/]*'
elif c == '?':
res += '.'
elif c == '[':
j = i
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 if j < n and pat[j:j + 1] in '!]':
Matt Mackall
match: move util match functions over
r8570 j += 1
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 while j < n and pat[j:j + 1] != ']':
Matt Mackall
match: move util match functions over
r8570 j += 1
if j >= n:
res += '\\['
else:
stuff = pat[i:j].replace('\\','\\\\')
i = j + 1
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 if stuff[0:1] == '!':
Matt Mackall
match: move util match functions over
r8570 stuff = '^' + stuff[1:]
Pulkit Goyal
match: slice over bytes to get the byteschr instead of ascii value
r31421 elif stuff[0:1] == '^':
Matt Mackall
match: move util match functions over
r8570 stuff = '\\' + stuff
res = '%s[%s]' % (res, stuff)
elif c == '{':
group += 1
res += '(?:'
elif c == '}' and group:
res += ')'
group -= 1
elif c == ',' and group:
res += '|'
elif c == '\\':
p = peek()
if p:
i += 1
Matt Mackall
match: optimize escaping in _globre...
r8583 res += escape(p)
Matt Mackall
match: move util match functions over
r8570 else:
Matt Mackall
match: optimize escaping in _globre...
r8583 res += escape(c)
Matt Mackall
match: move util match functions over
r8570 else:
Matt Mackall
match: optimize escaping in _globre...
r8583 res += escape(c)
Matt Mackall
match: remove head and tail args from _globre
r8582 return res
Matt Mackall
match: move util match functions over
r8570
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _regex(kind, pat, globsuffix):
'''Convert a (normalized) pattern of any kind into a regular expression.
globsuffix is appended to the regexp of globs.'''
if not pat:
Matt Mackall
match: unnest functions in _matcher
r8574 return ''
if kind == 're':
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 return pat
Martin von Zweigbergk
match: combine regex code for path: and relpath:...
r33358 if kind in ('path', 'relpath'):
Matt Harbison
match: let 'path:.' and 'path:' match everything (issue4687)...
r25636 if pat == '.':
return ''
Augie Fackler
cleanup: migrate from re.escape to stringutil.reescape...
r38494 return util.stringutil.reescape(pat) + '(?:/|$)'
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 if kind == 'rootfilesin':
if pat == '.':
escaped = ''
else:
# Pattern is a directory name.
Augie Fackler
cleanup: migrate from re.escape to stringutil.reescape...
r38494 escaped = util.stringutil.reescape(pat) + '/'
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 # Anything after the pattern must be a non-directory.
Martin von Zweigbergk
match: remove unnecessary '^' from regexes...
r33357 return escaped + '[^/]+$'
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 if kind == 'relglob':
return '(?:|.*/)' + _globre(pat) + globsuffix
if kind == 'relre':
if pat.startswith('^'):
return pat
return '.*' + pat
Yuya Nishihara
match: explode if unsupported pattern passed down to _regex() builder
r38597 if kind == 'glob':
return _globre(pat) + globsuffix
raise error.ProgrammingError('not a regex pattern: %s:%s' % (kind, pat))
Matt Mackall
match: unnest functions in _matcher
r8574
Yuya Nishihara
match: remove ctx argument from code path down to _buildmatch()...
r38600 def _buildmatch(kindpats, globsuffix, listsubrepos, root):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Return regexp string and a matcher function for kindpats.
globsuffix is appended to the regexp of globs.'''
Durham Goode
match: allow unioning arbitrary match functions...
r25239 matchfuncs = []
Durham Goode
match: enable 'subinclude:' syntax...
r25283 subincludes, kindpats = _expandsubinclude(kindpats, root)
if subincludes:
Durham Goode
match: make subinclude construction lazy...
r32132 submatchers = {}
Durham Goode
match: enable 'subinclude:' syntax...
r25283 def matchsubinclude(f):
Durham Goode
match: make subinclude construction lazy...
r32132 for prefix, matcherargs in subincludes:
if f.startswith(prefix):
mf = submatchers.get(prefix)
if mf is None:
mf = match(*matcherargs)
submatchers[prefix] = mf
if mf(f[len(prefix):]):
return True
Durham Goode
match: enable 'subinclude:' syntax...
r25283 return False
matchfuncs.append(matchsubinclude)
Matt Mackall
match: introduce basic fileset support
r14675
Durham Goode
match: allow unioning arbitrary match functions...
r25239 regex = ''
if kindpats:
regex, mf = _buildregexmatch(kindpats, globsuffix)
matchfuncs.append(mf)
if len(matchfuncs) == 1:
return regex, matchfuncs[0]
else:
return regex, lambda f: any(mf(f) for mf in matchfuncs)
Matt Mackall
match: introduce basic fileset support
r14675
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _buildregexmatch(kindpats, globsuffix):
"""Build a match function from a list of kinds and kindpats,
return regexp string and a matcher function."""
Matt Mackall
match: unnest functions in _matcher
r8574 try:
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 regex = '(?:%s)' % '|'.join([_regex(k, p, globsuffix)
Durham Goode
match: add source to kindpats list...
r25213 for (k, p, s) in kindpats])
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 if len(regex) > 20000:
Brodie Rao
cleanup: "raise SomeException()" -> "raise SomeException"
r16687 raise OverflowError
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 return regex, _rematcher(regex)
Matt Mackall
match: unnest functions in _matcher
r8574 except OverflowError:
# We're using a Python with a tiny regex engine and we
# made it explode, so we'll divide the pattern list in two
# until it works
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 l = len(kindpats)
Matt Mackall
match: unnest functions in _matcher
r8574 if l < 2:
raise
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 regexa, a = _buildregexmatch(kindpats[:l//2], globsuffix)
regexb, b = _buildregexmatch(kindpats[l//2:], globsuffix)
Yuya Nishihara
match: fix NameError 'pat' on overflow of regex pattern length...
r21191 return regex, lambda s: a(s) or b(s)
Matt Mackall
match: unnest functions in _matcher
r8574 except re.error:
Durham Goode
match: add source to kindpats list...
r25213 for k, p, s in kindpats:
Matt Mackall
match: unnest functions in _matcher
r8574 try:
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 _rematcher('(?:%s)' % _regex(k, p, globsuffix))
Matt Mackall
match: unnest functions in _matcher
r8574 except re.error:
Durham Goode
match: add source to kindpats list...
r25213 if s:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("%s: invalid pattern (%s): %s") %
Durham Goode
match: add source to kindpats list...
r25213 (s, k, p))
else:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("invalid pattern (%s): %s") % (k, p))
raise error.Abort(_("invalid pattern"))
Matt Mackall
match: unnest functions in _matcher
r8574
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 def _patternrootsanddirs(kindpats):
'''Returns roots and directories corresponding to each pattern.
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 This calculates the roots and directories exactly matching the patterns and
returns a tuple of (roots, dirs) for each. It does not return other
directories which may also need to be considered, like the parent
directories.
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079 '''
Matt Mackall
match: split up _normalizepats
r8576 r = []
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 d = []
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Matt Mackall
match: fold _globprefix into _roots
r8584 if kind == 'glob': # find the non-glob prefix
root = []
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 for p in pat.split('/'):
Matt Mackall
match: fold _globprefix into _roots
r8584 if '[' in p or '{' in p or '*' in p or '?' in p:
break
root.append(p)
r.append('/'.join(root) or '.')
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 elif kind in ('relpath', 'path'):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 r.append(pat or '.')
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 elif kind in ('rootfilesin',):
d.append(pat or '.')
Mads Kiilerich
match: fix root calculation for combining regexps with simple paths...
r19107 else: # relglob, re, relre
Matt Mackall
match: split up _normalizepats
r8576 r.append('.')
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 return r, d
def _roots(kindpats):
'''Returns root directories to match recursively from the given patterns.'''
roots, dirs = _patternrootsanddirs(kindpats)
return roots
def _rootsanddirs(kindpats):
'''Returns roots and exact directories from patterns.
roots are directories to match recursively, whereas exact directories should
be matched non-recursively. The returned (roots, dirs) tuple will also
include directories that need to be implicitly considered as either, such as
parent directories.
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _rootsanddirs(
... [(b'glob', b'g/h/*', b''), (b'glob', b'g/h', b''),
... (b'glob', b'g*', b'')])
Martin von Zweigbergk
match: optimize visitdir() for patterns matching only root directory...
r32176 (['g/h', 'g/h', '.'], ['g', '.'])
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _rootsanddirs(
... [(b'rootfilesin', b'g/h', b''), (b'rootfilesin', b'', b'')])
Martin von Zweigbergk
match: optimize visitdir() for patterns matching only root directory...
r32176 ([], ['g/h', '.', 'g', '.'])
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _rootsanddirs(
... [(b'relpath', b'r', b''), (b'path', b'p/p', b''),
... (b'path', b'', b'')])
Martin von Zweigbergk
match: optimize visitdir() for patterns matching only root directory...
r32176 (['r', 'p/p', '.'], ['p', '.'])
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _rootsanddirs(
... [(b'relglob', b'rg*', b''), (b're', b're/', b''),
... (b'relre', b'rr', b'')])
Martin von Zweigbergk
match: optimize visitdir() for patterns matching only root directory...
r32176 (['.', '.', '.'], ['.'])
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 '''
r, d = _patternrootsanddirs(kindpats)
# Append the parents as non-recursive/exact directories, since they must be
# scanned to get to either the roots or the other exact directories.
d.extend(util.dirs(d))
d.extend(util.dirs(r))
Martin von Zweigbergk
match: optimize visitdir() for patterns matching only root directory...
r32176 # util.dirs() does not include the root directory, so add it manually
d.append('.')
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013
return r, d
Matt Mackall
match: split up _normalizepats
r8576
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 def _explicitfiles(kindpats):
'''Returns the potential explicit filenames from the patterns.
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _explicitfiles([(b'path', b'foo/bar', b'')])
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 ['foo/bar']
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _explicitfiles([(b'rootfilesin', b'foo/bar', b'')])
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 []
'''
# Keep only the pattern kinds where one can specify filenames (vs only
# directory names).
filable = [kp for kp in kindpats if kp[0] not in ('rootfilesin',)]
return _roots(filable)
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 def _prefix(kindpats):
'''Whether all the patterns match a prefix (i.e. recursively)'''
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 if kind not in ('path', 'relpath'):
return False
return True
Durham Goode
ignore: move readpatternfile to match.py...
r25167
_commentre = None
Laurent Charignon
match: add option to return line and lineno from readpattern...
r27595 def readpatternfile(filepath, warn, sourceinfo=False):
Durham Goode
ignore: move readpatternfile to match.py...
r25167 '''parse a pattern file, returning a list of
patterns. These patterns should be given to compile()
Durham Goode
ignore: use 'include:' rules instead of custom syntax...
r25216 to be validated and converted into a match function.
trailing white space is dropped.
the escape character is backslash.
comments start with #.
empty lines are skipped.
lines can be of the following formats:
syntax: regexp # defaults following lines to non-rooted regexps
syntax: glob # defaults following lines to non-rooted globs
re:pattern # non-rooted regular expression
glob:pattern # non-rooted glob
Laurent Charignon
match: add option to return line and lineno from readpattern...
r27595 pattern # pattern of the current default type
if sourceinfo is set, returns a list of tuples:
(pattern, lineno, originalline). This is useful to debug ignore patterns.
'''
Durham Goode
ignore: use 'include:' rules instead of custom syntax...
r25216
Durham Goode
match: add 'include:' syntax...
r25215 syntaxes = {'re': 'relre:', 'regexp': 'relre:', 'glob': 'relglob:',
Durham Goode
match: enable 'subinclude:' syntax...
r25283 'include': 'include', 'subinclude': 'subinclude'}
Durham Goode
ignore: move readpatternfile to match.py...
r25167 syntax = 'relre:'
patterns = []
Rishabh Madan
py3: open file in rb mode
r31403 fp = open(filepath, 'rb')
Jun Wu
match: migrate to util.iterfile
r30399 for lineno, line in enumerate(util.iterfile(fp), start=1):
Durham Goode
ignore: move readpatternfile to match.py...
r25167 if "#" in line:
global _commentre
if not _commentre:
Pulkit Goyal
match: make regular expression bytes to prevent TypeError
r31420 _commentre = util.re.compile(br'((?:^|[^\\])(?:\\\\)*)#.*')
Durham Goode
ignore: move readpatternfile to match.py...
r25167 # remove comments prefixed by an even number of escapes
Bryan O'Sullivan
match: use re2 in readpatternfile if possible...
r27327 m = _commentre.search(line)
if m:
line = line[:m.end(1)]
Durham Goode
ignore: move readpatternfile to match.py...
r25167 # fixup properly escaped comments that survived the above
line = line.replace("\\#", "#")
line = line.rstrip()
if not line:
continue
if line.startswith('syntax:'):
s = line[7:].strip()
try:
syntax = syntaxes[s]
except KeyError:
Durham Goode
match: add optional warn argument...
r25214 if warn:
warn(_("%s: ignoring invalid syntax '%s'\n") %
(filepath, s))
Durham Goode
ignore: move readpatternfile to match.py...
r25167 continue
linesyntax = syntax
for s, rels in syntaxes.iteritems():
if line.startswith(rels):
linesyntax = rels
line = line[len(rels):]
break
elif line.startswith(s+':'):
linesyntax = rels
line = line[len(s) + 1:]
break
Laurent Charignon
match: add option to return line and lineno from readpattern...
r27595 if sourceinfo:
patterns.append((linesyntax + line, lineno, line))
else:
patterns.append(linesyntax + line)
Durham Goode
ignore: move readpatternfile to match.py...
r25167 fp.close()
return patterns