##// END OF EJS Templates
run-tests: use a context manager for file I/O
run-tests: use a context manager for file I/O

File last commit:

r27595:9e2d0170 default
r27773:bf45edfa default
Show More
match.py
703 lines | 24.1 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
Gregory Szorc
match: use absolute_import
r25958 from __future__ import absolute_import
import copy
import os
import re
from .i18n import _
from . import (
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 error,
Gregory Szorc
match: use absolute_import
r25958 pathutil,
util,
)
Matt Mackall
walk: introduce match objects
r6576
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
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 def _expandsets(kindpats, ctx, listsubrepos):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Returns the kindpats list with the 'set' patterns expanded.'''
Matt Mackall
match: introduce basic fileset support
r14675 fset = set()
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:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort("fileset expression with no context")
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 s = ctx.getfileset(pat)
Matt Mackall
match: introduce basic fileset support
r14675 fset.update(s)
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122
if listsubrepos:
for subpath in ctx.substate:
s = ctx.sub(subpath).getfileset(pat)
fset.update(subpath + '/' + f for f in s)
Matt Mackall
match: introduce basic fileset support
r14675 continue
Durham Goode
match: add source to kindpats list...
r25213 other.append((kind, pat, source))
Matt Mackall
match: introduce basic fileset support
r14675 return fset, other
Durham Goode
match: enable 'subinclude:' syntax...
r25283 def _expandsubinclude(kindpats, root):
'''Returns the list of subinclude matchers and the kindpats without the
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)
relmatcher = match(newroot, '', [], ['include:%s' % path])
prefix = pathutil.canonpath(root, root, newroot)
if prefix:
prefix += '/'
relmatchers.append((prefix, relmatcher))
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
Matt Mackall
match: fold match into _match base class
r8587 class match(object):
Matt Mackall
match: add some default args
r8567 def __init__(self, root, cwd, patterns, include=[], exclude=[],
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 default='glob', exact=False, auditor=None, ctx=None,
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 listsubrepos=False, warn=None, badfn=None):
Matt Mackall
match: fold _matcher into match.__init__
r8581 """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
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 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)
Durham Goode
match: add optional warn argument...
r25214 warn - optional function used for printing warnings
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 badfn - optional bad() callback for this matcher instead of the default
Matt Mackall
match: fold _matcher into match.__init__
r8581
a pattern is one of:
'glob:<glob>' - a glob relative to cwd
're:<regexp>' - a regular expression
Mads Kiilerich
fix wording and not-completely-trivial spelling errors and bad docstrings
r17425 'path:<path>' - a path relative to repository root
Matt Mackall
match: fold _matcher into match.__init__
r8581 'relglob:<glob>' - an unrooted glob (*.c matches C files in all dirs)
'relpath:<path>' - a path relative to cwd
Matt Mackall
match: fold match into _match base class
r8587 'relre:<regexp>' - a regexp that needn't match the start of a name
Matt Mackall
match: introduce basic fileset support
r14675 'set:<fileset>' - a fileset expression
Durham Goode
match: add 'include:' syntax...
r25215 'include:<path>' - a file of patterns to read and include
Durham Goode
match: enable 'subinclude:' syntax...
r25283 'subinclude:<path>' - a file of patterns to match against files under
the same directory
Matt Mackall
match: fold match into _match base class
r8587 '<something>' - a pattern of the specified default type
Matt Mackall
match: fold _matcher into match.__init__
r8581 """
Matt Mackall
match: fold match into _match base class
r8587 self._root = root
self._cwd = cwd
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079 self._files = [] # exact files and roots of patterns
Matt Mackall
match: fold match into _match base class
r8587 self._anypats = bool(include or exclude)
Bryan O'Sullivan
match: more accurately report when we're always going to match...
r18713 self._always = False
Matt Harbison
match: introduce uipath() to properly style a file path...
r23480 self._pathrestricted = bool(include or exclude or patterns)
Durham Goode
match: add optional warn argument...
r25214 self._warn = warn
Drew Gottlieb
match: have visitdir() consider includes and excludes...
r25231 self._includeroots = set()
self._includedirs = set(['.'])
self._excluderoots = set()
Matt Mackall
match: fold _matcher into match.__init__
r8581
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 if badfn is not None:
self.bad = badfn
Matt Mackall
match: fold _matcher into match.__init__
r8581
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 matchfns = []
Matt Mackall
match: add exact flag to match() to unify all match forms
r8586 if include:
Matt Harbison
match: move _normalize() into the match class...
r24789 kindpats = self._normalize(include, 'glob', root, cwd, auditor)
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 self.includepat, im = _buildmatch(ctx, kindpats, '(?:/|$)',
Durham Goode
match: add root to _buildmatch...
r25238 listsubrepos, root)
Drew Gottlieb
match: have visitdir() consider includes and excludes...
r25231 self._includeroots.update(_roots(kindpats))
self._includedirs.update(util.dirs(self._includeroots))
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 matchfns.append(im)
Matt Mackall
match: add exact flag to match() to unify all match forms
r8586 if exclude:
Matt Harbison
match: move _normalize() into the match class...
r24789 kindpats = self._normalize(exclude, 'glob', root, cwd, auditor)
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 self.excludepat, em = _buildmatch(ctx, kindpats, '(?:/|$)',
Durham Goode
match: add root to _buildmatch...
r25238 listsubrepos, root)
Martin von Zweigbergk
treemanifest: visit directory 'foo' when given e.g. '-X foo/ba?'...
r25362 if not _anypats(kindpats):
self._excluderoots.update(_roots(kindpats))
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 matchfns.append(lambda f: not em(f))
Matt Mackall
match: add exact flag to match() to unify all match forms
r8586 if exact:
FUJIWARA Katsunori
match: make 'match.files()' return list object always...
r16789 if isinstance(patterns, list):
self._files = patterns
else:
self._files = list(patterns)
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 matchfns.append(self.exact)
Matt Mackall
match: add exact flag to match() to unify all match forms
r8586 elif patterns:
Matt Harbison
match: move _normalize() into the match class...
r24789 kindpats = self._normalize(patterns, default, root, cwd, auditor)
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 if not _kindpatsalwaysmatch(kindpats):
self._files = _roots(kindpats)
self._anypats = self._anypats or _anypats(kindpats)
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 self.patternspat, pm = _buildmatch(ctx, kindpats, '$',
Durham Goode
match: add root to _buildmatch...
r25238 listsubrepos, root)
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 matchfns.append(pm)
Matt Mackall
match: fold _matcher into match.__init__
r8581
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 if not matchfns:
m = util.always
self._always = True
elif len(matchfns) == 1:
m = matchfns[0]
Matt Mackall
match: fold _matcher into match.__init__
r8581 else:
Martin von Zweigbergk
match: simplify brittle predicate construction...
r22513 def m(f):
for matchfn in matchfns:
if not matchfn(f):
return False
return True
Matt Mackall
match: fold _matcher into match.__init__
r8581
Matt Mackall
match: fold match into _match base class
r8587 self.matchfn = m
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 self._fileroots = set(self._files)
Matt Mackall
match: fold match into _match base class
r8587
def __call__(self, fn):
return self.matchfn(fn)
def __iter__(self):
for f in self._files:
yield f
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
# Callbacks related to how the matcher is used by dirstate.walk.
# Subscribers to these events must monkeypatch the matcher object.
Matt Mackall
match: fold match into _match base class
r8587 def bad(self, f, msg):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Callback from dirstate.walk for each explicit file that can't be
found/accessed, with an error message.'''
Matt Mackall
match: ignore return of match.bad...
r8680 pass
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
# If an explicitdir is set, it will be called when an explicitly listed
# directory is visited.
Siddharth Agarwal
match: make explicitdir and traversedir None by default...
r19143 explicitdir = None
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
# If an traversedir is set, it will be called when a directory discovered
# by recursive traversal is visited.
Siddharth Agarwal
match: make explicitdir and traversedir None by default...
r19143 traversedir = None
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Matt Harbison
match: add the abs() method...
r23685 def abs(self, f):
'''Convert a repo path back to path that is relative to the root of the
matcher.'''
return f
Matt Mackall
match: fold match into _match base class
r8587 def rel(self, f):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Convert repo path back to path that is relative to cwd of matcher.'''
Matt Mackall
match: fold match into _match base class
r8587 return util.pathto(self._root, self._cwd, f)
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Matt Harbison
match: introduce uipath() to properly style a file path...
r23480 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.'''
Matt Harbison
narrowmatcher: propagate the rel() method...
r23686 return (self._pathrestricted and self.rel(f)) or self.abs(f)
Matt Harbison
match: introduce uipath() to properly style a file path...
r23480
Matt Mackall
match: fold match into _match base class
r8587 def files(self):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''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'''
Matt Mackall
match: fold match into _match base class
r8587 return self._files
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636 @propertycache
def _dirs(self):
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 return set(util.dirs(self._fileroots)) | set(['.'])
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636
def visitdir(self, dir):
Drew Gottlieb
match: have visitdir() consider includes and excludes...
r25231 '''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.
Martin von Zweigbergk
treemanifest: don't iterate entire matching submanifests on match()...
r27343 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.
Drew Gottlieb
match: have visitdir() consider includes and excludes...
r25231 This function's behavior is undefined if it has returned False for
one of the dir's parent directories.
'''
Martin von Zweigbergk
treemanifest: don't iterate entire matching submanifests on match()...
r27343 if self.prefix() and dir in self._fileroots:
return 'all'
Drew Gottlieb
match: have visitdir() consider includes and excludes...
r25231 if dir in self._excluderoots:
return False
Martin von Zweigbergk
match: break boolean expressions into one operand per line...
r25576 if (self._includeroots and
Martin von Zweigbergk
match: don't remove '.' from _includeroots...
r25579 '.' not in self._includeroots and
Martin von Zweigbergk
match: break boolean expressions into one operand per line...
r25576 dir not in self._includeroots and
Martin von Zweigbergk
match: join two nested if-blocks...
r25578 dir not in self._includedirs and
not any(parent in self._includeroots
for parent in util.finddirs(dir))):
return False
Martin von Zweigbergk
match: break boolean expressions into one operand per line...
r25576 return (not self._fileroots or
'.' in self._fileroots or
dir in self._fileroots or
dir in self._dirs or
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 any(parentdir in self._fileroots
Martin von Zweigbergk
match: drop optimization (?) of 'parentdirs' calculation...
r25577 for parentdir in util.finddirs(dir)))
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def exact(self, f):
'''Returns True if f is in .files().'''
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 return f in self._fileroots
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Matt Mackall
match: fold match into _match base class
r8587 def anypats(self):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Matcher uses patterns or include/exclude.'''
Matt Mackall
match: fold match into _match base class
r8587 return self._anypats
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111
Jesse Glick
localrepo: optimize internode status calls using match.always...
r16645 def always(self):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 '''Matcher will match everything and .files() will be empty
- optimization might be possible and necessary.'''
Bryan O'Sullivan
match: more accurately report when we're always going to match...
r18713 return self._always
Matt Mackall
match: refactor patkind...
r8568
Drew Gottlieb
match: add match.ispartial()...
r25114 def ispartial(self):
'''True if the matcher won't always match.
Although it's just the inverse of _always in this implementation,
Mads Kiilerich
spelling: trivial spell checking
r26781 an extension such as narrowhg might make it return something
Drew Gottlieb
match: add match.ispartial()...
r25114 slightly different.'''
return not self._always
Martin von Zweigbergk
match: add isexact() method to hide internals...
r24448 def isexact(self):
return self.matchfn == self.exact
Martin von Zweigbergk
match: introduce boolean prefix() method...
r25233 def prefix(self):
return not self.always() and not self.isexact() and not self.anypats()
Matt Harbison
match: move _normalize() into the match class...
r24789 def _normalize(self, patterns, default, root, cwd, auditor):
'''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]:
if kind in ('glob', 'relpath'):
pat = pathutil.canonpath(root, cwd, pat, auditor)
elif kind in ('relglob', 'path'):
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:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("unable to read file list (%s)") % pat)
Durham Goode
match: add source to kindpats list...
r25213 for k, p, source in self._normalize(files, default, root, cwd,
auditor):
kindpats.append((k, p, pat))
Matt Harbison
match: move _normalize() into the match class...
r24789 continue
Durham Goode
match: add 'include:' syntax...
r25215 elif kind == 'include':
try:
Yuya Nishihara
ignore: fix path concatenation of .hgignore on Windows...
r25875 fullpath = os.path.join(root, util.localpath(pat))
Durham Goode
ignore: fix include: rules depending on current directory (issue4759)...
r25870 includepats = readpatternfile(fullpath, self._warn)
Durham Goode
match: add 'include:' syntax...
r25215 for k, p, source in self._normalize(includepats, default,
root, cwd, auditor):
kindpats.append((k, p, source or pat))
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 except error.Abort as inst:
raise error.Abort('%s: %s' % (pat, inst[0]))
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except IOError as inst:
Durham Goode
match: add 'include:' syntax...
r25215 if self._warn:
self._warn(_("skipping unreadable pattern file "
"'%s': %s\n") % (pat, inst.strerror))
Matt Harbison
match: move _normalize() into the match class...
r24789 continue
# else: re or relre - which cannot be normalized
Durham Goode
match: add source to kindpats list...
r25213 kindpats.append((kind, pat, ''))
Matt Harbison
match: move _normalize() into the match class...
r24789 return kindpats
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 def exact(root, cwd, files, badfn=None):
return match(root, cwd, files, exact=True, badfn=badfn)
Matt Mackall
match: redefine always and never in terms of match and exact
r8585
Martin von Zweigbergk
match: make 'always' and 'exact' functions, not classes...
r23549 def always(root, cwd):
return match(root, cwd, [])
Matt Mackall
match: redefine always and never in terms of match and exact
r8585
Matt Harbison
match: introduce badmatch() to eliminate long callback chains with subrepos...
r25433 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 Geisler
match: add narrowmatcher class...
r12165 class narrowmatcher(match):
"""Adapt a matcher to work on a subdirectory only.
The paths are remapped to remove/insert the path as needed:
>>> m1 = match('root', '', ['a.txt', 'sub/b.txt'])
>>> m2 = narrowmatcher('sub', m1)
>>> bool(m2('a.txt'))
False
>>> bool(m2('b.txt'))
True
>>> bool(m2.matchfn('a.txt'))
False
>>> bool(m2.matchfn('b.txt'))
True
>>> m2.files()
['b.txt']
>>> m2.exact('b.txt')
True
Matt Harbison
narrowmatcher: propagate the rel() method...
r23686 >>> util.pconvert(m2.rel('b.txt'))
'sub/b.txt'
Martin Geisler
narrowmatcher: propagate bad method...
r12268 >>> def bad(f, msg):
... print "%s: %s" % (f, msg)
>>> m1.bad = bad
>>> m2.bad('x.txt', 'No such file')
sub/x.txt: No such file
Matt Harbison
match: add the abs() method...
r23685 >>> m2.abs('c.txt')
'sub/c.txt'
Martin Geisler
match: add narrowmatcher class...
r12165 """
def __init__(self, path, matcher):
Martin Geisler
narrowmatcher: fix broken rel method
r12267 self._root = matcher._root
self._cwd = matcher._cwd
Martin Geisler
match: add narrowmatcher class...
r12165 self._path = path
self._matcher = matcher
Bryan O'Sullivan
match: more accurately report when we're always going to match...
r18713 self._always = matcher._always
Matt Harbison
match: introduce uipath() to properly style a file path...
r23480 self._pathrestricted = matcher._pathrestricted
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
# If the parent repo had a path to this subrepo and no patterns are
# specified, this submatcher always matches.
if not self._always and not matcher._anypats:
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 Geisler
match: add narrowmatcher class...
r12165 self._anypats = matcher._anypats
self.matchfn = lambda fn: matcher.matchfn(self._path + "/" + fn)
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 self._fileroots = set(self._files)
Martin Geisler
match: add narrowmatcher class...
r12165
Matt Harbison
match: add the abs() method...
r23685 def abs(self, f):
return self._matcher.abs(self._path + "/" + f)
Martin Geisler
narrowmatcher: propagate bad method...
r12268 def bad(self, f, msg):
self._matcher.bad(self._path + "/" + f, msg)
Matt Harbison
narrowmatcher: propagate the rel() method...
r23686 def rel(self, f):
return self._matcher.rel(self._path + "/" + f)
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790 class icasefsmatcher(match):
"""A matcher for wdir on case insensitive filesystems, which normalizes the
given patterns to the case in the filesystem.
"""
def __init__(self, root, cwd, patterns, include, exclude, default, auditor,
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 ctx, listsubrepos=False, badfn=None):
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790 init = super(icasefsmatcher, self).__init__
Matt Harbison
match: fix a caseonly rename + explicit path commit on icasefs (issue4768)...
r26000 self._dirstate = ctx.repo().dirstate
self._dsnormalize = self._dirstate.normalize
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790
init(root, cwd, patterns, include, exclude, default, auditor=auditor,
Matt Harbison
match: add an optional constructor parameter for a bad() override...
r25464 ctx=ctx, listsubrepos=listsubrepos, badfn=badfn)
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790
# m.exact(file) must be based off of the actual user input, otherwise
# inexact case matches are treated as exact, and not noted without -v.
if self._files:
Drew Gottlieb
match: rename _fmap to _fileroots for clarity...
r25189 self._fileroots = set(_roots(self._kp))
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790
def _normalize(self, patterns, default, root, cwd, auditor):
self._kp = super(icasefsmatcher, self)._normalize(patterns, default,
root, cwd, auditor)
kindpats = []
Durham Goode
match: add source to kindpats list...
r25213 for kind, pats, source in self._kp:
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790 if kind not in ('re', 'relre'): # regex can't be normalized
Matt Harbison
match: fix a caseonly rename + explicit path commit on icasefs (issue4768)...
r26000 p = pats
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790 pats = self._dsnormalize(pats)
Matt Harbison
match: fix a caseonly rename + explicit path commit on icasefs (issue4768)...
r26000
# Preserve the original to handle a case only rename.
if p != pats and p in self._dirstate:
kindpats.append((kind, p, source))
Durham Goode
match: add source to kindpats list...
r25213 kindpats.append((kind, pats, source))
Matt Harbison
match: add a subclass for dirstate normalizing of the matched patterns...
r24790 return kindpats
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)
Steve Borho
match: support reading pattern lists from files
r13218 if kind in ('re', 'glob', 'path', 'relglob', 'relpath', 'relre',
Durham Goode
match: enable 'subinclude:' syntax...
r25283 'listfile', 'listfile0', 'set', 'include', 'subinclude'):
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.
>>> print _globre(r'?')
.
>>> print _globre(r'*')
[^/]*
>>> print _globre(r'**')
.*
Siddharth Agarwal
match: make glob '**/' match the empty string...
r21815 >>> print _globre(r'**/a')
(?:.*/)?a
>>> print _globre(r'a/**/b')
a\/(?:.*/)?b
Mads Kiilerich
match: _globre doctests
r21112 >>> print _globre(r'[a*?!^][^b][!c]')
[a*?!^][\^b][^c]
>>> print _globre(r'{a,b}')
(?:a|b)
>>> print _globre(r'.\*\?')
\.\*\?
'''
Matt Mackall
match: move util match functions over
r8570 i, n = 0, len(pat)
res = ''
group = 0
Siddharth Agarwal
match: use util.re.escape instead of re.escape...
r21915 escape = util.re.escape
Matt Mackall
many, many trivial check-code fixups
r10282 def peek():
return i < n and pat[i]
Matt Mackall
match: move util match functions over
r8570 while i < n:
c = pat[i]
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
if j < n and pat[j] in '!]':
j += 1
while j < n and pat[j] != ']':
j += 1
if j >= n:
res += '\\['
else:
stuff = pat[i:j].replace('\\','\\\\')
i = j + 1
if stuff[0] == '!':
stuff = '^' + stuff[1:]
elif stuff[0] == '^':
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
if kind == 'path':
Matt Harbison
match: let 'path:.' and 'path:' match everything (issue4687)...
r25636 if pat == '.':
return ''
Siddharth Agarwal
match: use util.re.escape instead of re.escape...
r21915 return '^' + util.re.escape(pat) + '(?:/|$)'
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 if kind == 'relglob':
return '(?:|.*/)' + _globre(pat) + globsuffix
if kind == 'relpath':
Siddharth Agarwal
match: use util.re.escape instead of re.escape...
r21915 return util.re.escape(pat) + '(?:/|$)'
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 if kind == 'relre':
if pat.startswith('^'):
return pat
return '.*' + pat
return _globre(pat) + globsuffix
Matt Mackall
match: unnest functions in _matcher
r8574
Durham Goode
match: add root to _buildmatch...
r25238 def _buildmatch(ctx, 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:
def matchsubinclude(f):
for prefix, mf in subincludes:
if f.startswith(prefix) and mf(f[len(prefix):]):
return True
return False
matchfuncs.append(matchsubinclude)
Matt Mackall
match: introduce basic fileset support
r14675
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122 fset, kindpats = _expandsets(kindpats, ctx, listsubrepos)
Matt Mackall
match: introduce basic fileset support
r14675 if fset:
Durham Goode
match: allow unioning arbitrary match functions...
r25239 matchfuncs.append(fset.__contains__)
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
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _roots(kindpats):
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079 '''return roots and exact explicitly listed files from patterns
Durham Goode
match: add source to kindpats list...
r25213 >>> _roots([('glob', 'g/*', ''), ('glob', 'g', ''), ('glob', 'g*', '')])
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079 ['g', 'g', '.']
Durham Goode
match: add source to kindpats list...
r25213 >>> _roots([('relpath', 'r', ''), ('path', 'p/p', ''), ('path', '', '')])
Mads Kiilerich
match: make it more clear what _roots do and that it ends up in match()._files
r21079 ['r', 'p/p', '.']
Durham Goode
match: add source to kindpats list...
r25213 >>> _roots([('relglob', 'rg*', ''), ('re', 're/', ''), ('relre', 'rr', '')])
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 = []
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 '.')
Matt Mackall
match: unnest functions in _matcher
r8574 elif kind in ('relpath', 'path'):
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 r.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('.')
return r
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _anypats(kindpats):
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Patrick Mezard
match: consider filesets as "anypats"...
r16182 if kind in ('glob', 're', 'relglob', 'relre', 'set'):
Matt Mackall
match: split up _normalizepats
r8576 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 = []
fp = open(filepath)
Laurent Charignon
match: add option to return line and lineno from readpattern...
r27595 for lineno, line in enumerate(fp, start=1):
Durham Goode
ignore: move readpatternfile to match.py...
r25167 if "#" in line:
global _commentre
if not _commentre:
Bryan O'Sullivan
match: use re2 in readpatternfile if possible...
r27327 _commentre = util.re.compile(r'((?:^|[^\\])(?:\\\\)*)#.*')
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