##// END OF EJS Templates
ci: also offer tests with Python 3.13...
ci: also offer tests with Python 3.13 Python3.13 is du to be released soon. We better make sure we work with it.

File last commit:

r52819:70fe33bd default
r52887:baf9e3a8 default
Show More
match.py
1782 lines | 55.4 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 #
Raphaël Gomès
contributor: change mentions of mpm to olivia...
r47575 # Copyright 2008, 2009 Olivia Mackall <olivia@selenic.com> and others
Martin Geisler
match: add copyright and license header
r8231 #
# 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
Matt Harbison
typing: add `from __future__ import annotations` to most files...
r52756 from __future__ import annotations
Gregory Szorc
match: use absolute_import
r25958
Kyle Lippincott
match: convert O(n) to O(log n) in exactmatcher.visitchildrenset...
r47634 import bisect
Gregory Szorc
match: use absolute_import
r25958 import copy
spectral
match: add visitchildrenset complement to visitdir...
r38990 import itertools
Gregory Szorc
match: use absolute_import
r25958 import os
import re
Matt Harbison
typing: add type hints for the overloads of `matchmod.readpatternfile()`...
r52819 import typing
from typing import (
Any,
Callable,
List,
Tuple,
Union,
overload,
)
Gregory Szorc
match: use absolute_import
r25958
from .i18n import _
Gregory Szorc
py3: manually import pycompat.open into files that need it...
r43355 from .pycompat import open
Gregory Szorc
match: use absolute_import
r25958 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,
Georges Racinet
rust: using policy.importrust from Python callers...
r42652 policy,
Augie Fackler
match: some minimal pycompat fixes guided by test-hgignore.t...
r36590 pycompat,
Gregory Szorc
match: use absolute_import
r25958 util,
)
Augie Fackler
formatting: blacken the codebase...
r43346 from .utils import stringutil
Matt Mackall
walk: introduce match objects
r6576
Raphaël Gomès
rust-filepatterns: remove bridge code for filepatterns-related functions...
r44589 rustmod = policy.importrust('dirstate')
Raphaël Gomès
rust-filepatterns: call new Rust implementations from Python...
r42516
Augie Fackler
formatting: blacken the codebase...
r43346 allpatternkinds = (
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b're',
b'glob',
b'path',
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 b'filepath',
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'relglob',
b'relpath',
b'relre',
b'rootglob',
b'listfile',
b'listfile0',
b'set',
b'include',
b'subinclude',
b'rootfilesin',
Augie Fackler
formatting: blacken the codebase...
r43346 )
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 cwdrelativepatternkinds = (b'relpath', b'glob')
Kostia Balytskyi
match: expose some data and functionality to other modules...
r33647
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636 propertycache = util.propertycache
Augie Fackler
formatting: blacken the codebase...
r43346
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _rematcher(regex):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 def _expandsets(cwd, kindpats, ctx=None, listsubrepos=False, badfn=None):
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 '''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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind == b'set':
Yuya Nishihara
match: fix assertion for fileset with no context (issue6046)...
r41144 if ctx is None:
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.ProgrammingError(
Martin von Zweigbergk
cleanup: join string literals that are already on one line...
r43387 b"fileset expression with no context"
Augie Fackler
formatting: blacken the codebase...
r43346 )
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 matchers.append(ctx.matchfileset(cwd, pat, badfn=badfn))
Matt Harbison
match: resolve filesets in subrepos for commands given the '-S' argument...
r25122
if listsubrepos:
for subpath in ctx.substate:
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 sm = ctx.sub(subpath).matchfileset(cwd, pat, badfn=badfn)
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 pm = prefixdirmatcher(subpath, sm, badfn=badfn)
Yuya Nishihara
fileset: restrict getfileset() to not return a computed set (API)...
r38631 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
Augie Fackler
formatting: blacken the codebase...
r43346
Durham Goode
match: enable 'subinclude:' syntax...
r25283 def _expandsubinclude(kindpats, root):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Returns the list of subinclude matcher args and the kindpats without the
subincludes in it."""
Durham Goode
match: enable 'subinclude:' syntax...
r25283 relmatchers = []
other = []
for kind, pat, source in kindpats:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind == b'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)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 matcherargs = (newroot, b'', [], [b'include:%s' % path])
Durham Goode
match: enable 'subinclude:' syntax...
r25283
prefix = pathutil.canonpath(root, root, newroot)
if prefix:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 prefix += b'/'
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
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 def _kindpatsalwaysmatch(kindpats):
Matt Harbison
formatting: drop a few extra double quotes in docstrings...
r46558 """Checks whether the kindspats match everything, as e.g.
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 'relpath:.' does.
"""
Durham Goode
match: add source to kindpats list...
r25213 for kind, pat, source in kindpats:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if pat != b'' or kind not in [b'relpath', b'glob']:
Martin von Zweigbergk
matcher: make e.g. 'relpath:.' lead to fast paths...
r24447 return False
return True
Augie Fackler
formatting: blacken the codebase...
r43346
def _buildkindpatsmatcher(
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 matchercls,
root,
cwd,
kindpats,
ctx=None,
listsubrepos=False,
badfn=None,
Augie Fackler
formatting: blacken the codebase...
r43346 ):
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 matchers = []
Augie Fackler
formatting: blacken the codebase...
r43346 fms, kindpats = _expandsets(
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 cwd,
kindpats,
ctx=ctx,
listsubrepos=listsubrepos,
badfn=badfn,
Augie Fackler
formatting: blacken the codebase...
r43346 )
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 if kindpats:
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 m = matchercls(root, kindpats, badfn=badfn)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 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:
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 return nevermatcher(badfn=badfn)
Yuya Nishihara
match: compose 'set:' pattern as matcher...
r38599 if len(matchers) == 1:
return matchers[0]
return unionmatcher(matchers)
Augie Fackler
formatting: blacken the codebase...
r43346
def match(
root,
cwd,
patterns=None,
include=None,
exclude=None,
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 default=b'glob',
Augie Fackler
formatting: blacken the codebase...
r43346 auditor=None,
ctx=None,
listsubrepos=False,
warn=None,
badfn=None,
icasefs=False,
):
Denis Laxalde
match: add doctest examples in match()...
r42253 r"""build an object to match a set of file patterns
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
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
Denis Laxalde
match: complete documentation of match() parameters
r42252 auditor - optional path auditor
ctx - optional changecontext
listsubrepos - if True, recurse into subrepositories
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 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
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 'filepath:<path>' - an exact path to a single file, relative to the
repository root
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 '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
Denis Laxalde
match: add doctest examples in match()...
r42253
Matt Harbison
tests: convert the `root` arg of matchmod.match() to local path separators...
r44416 >>> def _match(root, *args, **kwargs):
... return match(util.localpath(root), *args, **kwargs)
Denis Laxalde
match: add doctest examples in match()...
r42253 Usually a patternmatcher is returned:
Mads Kiilerich
py3: fix for Python 3.12 emitting SyntaxWarning on invalid escape sequences...
r51229 >>> _match(b'/foo', b'.', [br're:.*\.c$', b'path:foo/a', b'*.py'])
match: sort patterns before compiling them into a regex...
r51285 <patternmatcher patterns='[^/]*\\.py$|foo/a(?:/|$)|.*\\.c$'>
Denis Laxalde
match: add doctest examples in match()...
r42253
Combining 'patterns' with 'include' (resp. 'exclude') gives an
intersectionmatcher (resp. a differencematcher):
Mads Kiilerich
py3: fix for Python 3.12 emitting SyntaxWarning on invalid escape sequences...
r51229 >>> type(_match(b'/foo', b'.', [br're:.*\.c$'], include=[b'path:lib']))
Denis Laxalde
match: add doctest examples in match()...
r42253 <class 'mercurial.match.intersectionmatcher'>
Mads Kiilerich
py3: fix for Python 3.12 emitting SyntaxWarning on invalid escape sequences...
r51229 >>> type(_match(b'/foo', b'.', [br're:.*\.c$'], exclude=[b'path:build']))
Denis Laxalde
match: add doctest examples in match()...
r42253 <class 'mercurial.match.differencematcher'>
Notice that, if 'patterns' is empty, an alwaysmatcher is returned:
Matt Harbison
tests: convert the `root` arg of matchmod.match() to local path separators...
r44416 >>> _match(b'/foo', b'.', [])
Denis Laxalde
match: add doctest examples in match()...
r42253 <alwaysmatcher>
The 'default' argument determines which kind of pattern is assumed if a
pattern has no prefix:
Mads Kiilerich
py3: fix for Python 3.12 emitting SyntaxWarning on invalid escape sequences...
r51229 >>> _match(b'/foo', b'.', [br'.*\.c$'], default=b're')
Denis Laxalde
match: add doctest examples in match()...
r42253 <patternmatcher patterns='.*\\.c$'>
Matt Harbison
tests: convert the `root` arg of matchmod.match() to local path separators...
r44416 >>> _match(b'/foo', b'.', [b'main.py'], default=b'relpath')
Denis Laxalde
match: add doctest examples in match()...
r42253 <patternmatcher patterns='main\\.py(?:/|$)'>
Matt Harbison
tests: convert the `root` arg of matchmod.match() to local path separators...
r44416 >>> _match(b'/foo', b'.', [b'main.py'], default=b're')
Denis Laxalde
match: add doctest examples in match()...
r42253 <patternmatcher patterns='main.py'>
The primary use of matchers is to check whether a value (usually a file
name) matches againset one of the patterns given at initialization. There
are two ways of doing this check.
Mads Kiilerich
py3: fix for Python 3.12 emitting SyntaxWarning on invalid escape sequences...
r51229 >>> m = _match(b'/foo', b'', [br're:.*\.c$', b'relpath:a'])
Denis Laxalde
match: add doctest examples in match()...
r42253
1. Calling the matcher with a file name returns True if any pattern
matches that file name:
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'a')
Denis Laxalde
match: add doctest examples in match()...
r42253 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'main.c')
Denis Laxalde
match: add doctest examples in match()...
r42253 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'test.py')
Denis Laxalde
match: add doctest examples in match()...
r42253 False
2. Using the exact() method only returns True if the file name matches one
of the exact patterns (i.e. not re: or glob: patterns):
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m.exact(b'a')
Denis Laxalde
match: add doctest examples in match()...
r42253 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m.exact(b'main.c')
Denis Laxalde
match: add doctest examples in match()...
r42253 False
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394 """
Martin von Zweigbergk
match: make sure `root` argument is always an absolute path (API)...
r44401 assert os.path.isabs(root)
Matt Harbison
match: don't util.normpath() cwd...
r44417 cwd = os.path.join(root, util.localpath(cwd))
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 normalize = _donormalize
if icasefs:
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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind not in (b're', b'relre'): # regex can't be normalized
Martin von Zweigbergk
match: replace icasefsmatch() function by flag to regular match()...
r32400 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: remove unused "exact" argument (API)...
r41771 if 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):
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 m = alwaysmatcher(badfn)
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 else:
Augie Fackler
formatting: blacken the codebase...
r43346 m = _buildkindpatsmatcher(
patternmatcher,
root,
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 cwd,
Augie Fackler
formatting: blacken the codebase...
r43346 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: delete unused root and cwd arguments to constructors (API)...
r41824 m = alwaysmatcher(badfn)
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 if include:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 kindpats = normalize(include, b'glob', root, cwd, auditor, warn)
Augie Fackler
formatting: blacken the codebase...
r43346 im = _buildkindpatsmatcher(
includematcher,
root,
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 cwd,
Augie Fackler
formatting: blacken the codebase...
r43346 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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 kindpats = normalize(exclude, b'glob', root, cwd, auditor, warn)
Augie Fackler
formatting: blacken the codebase...
r43346 em = _buildkindpatsmatcher(
includematcher,
root,
Matt Harbison
match: resolve filesets against the passed `cwd`, not the current one...
r44461 cwd,
Augie Fackler
formatting: blacken the codebase...
r43346 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
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: delete unused root and cwd arguments from {always,never,exact}() (API)...
r41825 def exact(files, badfn=None):
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 return exactmatcher(files, badfn=badfn)
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: delete unused root and cwd arguments from {always,never,exact}() (API)...
r41825 def always(badfn=None):
return alwaysmatcher(badfn)
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: delete unused root and cwd arguments from {always,never,exact}() (API)...
r41825 def never(badfn=None):
return nevermatcher(badfn)
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600
Augie Fackler
formatting: blacken the codebase...
r43346
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
Augie Fackler
formatting: blacken the codebase...
r43346
Denis Laxalde
match: make _donormalize's auditor and warn arguments optional...
r42254 def _donormalize(patterns, default, root, cwd, auditor=None, warn=None):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Convert 'kind:pat' from the patterns list to tuples with kind and
normalized and rooted patterns and with listfiles expanded."""
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 kindpats = []
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 kinds_to_normalize = (
b'relglob',
b'path',
b'filepath',
b'rootfilesin',
b'rootglob',
)
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 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:
Denis Laxalde
match: make _donormalize's auditor and warn arguments optional...
r42254 pat = pathutil.canonpath(root, cwd, pat, auditor=auditor)
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 elif kind in kinds_to_normalize:
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 pat = util.normpath(pat)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif kind in (b'listfile', b'listfile0'):
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 try:
files = util.readfile(pat)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind == b'listfile0':
files = files.split(b'\0')
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 else:
files = files.splitlines()
files = [f for f in files if f]
except EnvironmentError:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.Abort(_(b"unable to read file list (%s)") % pat)
Augie Fackler
formatting: blacken the codebase...
r43346 for k, p, source in _donormalize(
files, default, root, cwd, auditor, warn
):
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 kindpats.append((k, p, pat))
continue
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif kind == b'include':
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 try:
fullpath = os.path.join(root, util.localpath(pat))
includepats = readpatternfile(fullpath, warn)
Augie Fackler
formatting: blacken the codebase...
r43346 for k, p, source in _donormalize(
includepats, default, root, cwd, auditor, warn
):
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 kindpats.append((k, p, source or pat))
except error.Abort as inst:
Augie Fackler
match: suppress error about subscripting an exception...
r43786 raise error.Abort(
b'%s: %s'
Martin von Zweigbergk
errors: name arguments to Abort constructor...
r46274 % (
pat,
inst.message,
Matt Harbison
typing: drop an unnecessary warning disabling comment in match.py...
r50700 )
Augie Fackler
match: suppress error about subscripting an exception...
r43786 )
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 except IOError as inst:
if warn:
Augie Fackler
formatting: blacken the codebase...
r43346 warn(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"skipping unreadable pattern file '%s': %s\n")
Augie Fackler
formatting: blacken the codebase...
r43346 % (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
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 kindpats.append((kind, pat, b''))
Martin von Zweigbergk
match: move body of _normalize() to a static function...
r32396 return kindpats
Augie Fackler
formatting: blacken the codebase...
r43346
Gregory Szorc
py3: use class X: instead of class X(object):...
r49801 class basematcher:
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, badfn=None):
Arseniy Alekseyev
largefiles: track if a matcher was tampered with...
r52494 self._was_tampered_with = False
Martin von Zweigbergk
match: extract base class for matchers...
r32454 if badfn is not None:
self.bad = badfn
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with_nonrec(self) -> bool:
Arseniy Alekseyev
largefiles: track if a matcher was tampered with...
r52494 # [_was_tampered_with] is used to track if when extensions changed the matcher
# behavior (crazy stuff!), so we disable the rust fast path.
return self._was_tampered_with
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with(self) -> bool:
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 return self.was_tampered_with_nonrec()
Martin von Zweigbergk
match: extract base class for matchers...
r32454 def __call__(self, fn):
return self.matchfn(fn)
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: extract base class for matchers...
r32454 # 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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Callback from dirstate.walk for each explicit file that can't be
found/accessed, with an error message."""
Martin von Zweigbergk
match: extract base class for matchers...
r32454
# If an traversedir is set, it will be called when a directory discovered
# by recursive traversal is visited.
traversedir = None
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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Explicitly listed files or patterns or roots:
Martin von Zweigbergk
match: extract base class for matchers...
r32454 if no patterns or .always(): empty list,
if exact: list exact files,
if not .anypats(): list all files and dirs,
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 else: optimal roots"""
Martin von Zweigbergk
match: extract base class for matchers...
r32454 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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Decides whether a directory should be visited based on whether it
Martin von Zweigbergk
match: extract base class for matchers...
r32454 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.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Durham Goode
match: make base matcher return True for visitdir...
r33478 return True
Martin von Zweigbergk
match: extract base class for matchers...
r32454
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Decides whether a directory should be visited based on whether it
spectral
match: add visitchildrenset complement to visitdir...
r38990 has potential matches in it or one of its subdirectories, and
potentially lists which subdirectories of that directory should be
visited. This is based on the match's primary, included, and excluded
patterns.
This function is very similar to 'visitdir', and the following mapping
can be applied:
visitdir | visitchildrenlist
----------+-------------------
False | set()
'all' | 'all'
Kyle Lippincott
match: document that visitchildrenset might return files...
r39296 True | 'this' OR non-empty set of subdirs -or files- to visit
spectral
match: add visitchildrenset complement to visitdir...
r38990
Example:
Assume matchers ['path:foo/bar', 'rootfilesin:qux'], we would return
the following values (assuming the implementation of visitchildrenset
is capable of recognizing this; some implementations are not).
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 '' -> {'foo', 'qux'}
spectral
match: add visitchildrenset complement to visitdir...
r38990 'baz' -> set()
'foo' -> {'bar'}
# Ideally this would be 'all', but since the prefix nature of matchers
Kyle Lippincott
match: document that visitchildrenset might return files...
r39296 # is applied to the entire matcher, we have to downgrade this to
# 'this' due to the non-prefix 'rootfilesin'-kind matcher being mixed
# in.
spectral
match: add visitchildrenset complement to visitdir...
r38990 'foo/bar' -> 'this'
'qux' -> 'this'
Kyle Lippincott
match: document that visitchildrenset might return files...
r39296
Important:
Most matchers do not know if they're representing files or
directories. They see ['path:dir/f'] and don't know whether 'f' is a
file or a directory, so visitchildrenset('dir') for most matchers will
return {'f'}, but if the matcher knows it's a file (like exactmatcher
does), it may return 'this'. Do not rely on the return being a set
indicating that there are no files in this dir to investigate (or
equivalently that if there are files to investigate in 'dir' that it
will always return 'this').
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990
Martin von Zweigbergk
match: extract base class for matchers...
r32454 def always(self):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Matcher will match the paths in .files() recursively --
optimization might be possible."""
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 return False
def anypats(self):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """None of .always(), .isexact(), and .prefix() is true --
optimizations will be difficult."""
Martin von Zweigbergk
match: express anypats(), not prefix(), in terms of the others...
r33379 return not self.always() and not self.isexact() and not self.prefix()
Martin von Zweigbergk
match: extract base class for matchers...
r32454
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 class alwaysmatcher(basematcher):
'''Matches everything.'''
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, badfn=None):
super(alwaysmatcher, self).__init__(badfn)
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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'all'
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'all'
spectral
match: add visitchildrenset complement to visitdir...
r38990
Martin von Zweigbergk
match: handle everything-matching using new alwaysmatcher...
r32553 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
Augie Fackler
formatting: blacken the codebase...
r43346
Siddharth Agarwal
match: introduce nevermatcher for when no ignore files are present...
r32600 class nevermatcher(basematcher):
'''Matches nothing.'''
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, badfn=None):
super(nevermatcher, self).__init__(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
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
return set()
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
Augie Fackler
formatting: blacken the codebase...
r43346
Yuya Nishihara
match: add basic wrapper for boolean function...
r38596 class predicatematcher(basematcher):
"""A matcher adapter for a simple boolean function"""
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, predfn, predrepr=None, badfn=None):
super(predicatematcher, self).__init__(badfn)
Yuya Nishihara
match: add basic wrapper for boolean function...
r38596 self.matchfn = predfn
self._predrepr = predrepr
@encoding.strmethod
def __repr__(self):
Augie Fackler
formatting: blacken the codebase...
r43346 s = stringutil.buildrepr(self._predrepr) or pycompat.byterepr(
self.matchfn
)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<predicatenmatcher pred=%s>' % s
Yuya Nishihara
match: add basic wrapper for boolean function...
r38596
Augie Fackler
formatting: blacken the codebase...
r43346
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614 def path_or_parents_in_set(path, prefix_set):
"""Returns True if `path` (or any parent of `path`) is in `prefix_set`."""
l = len(prefix_set)
if l == 0:
return False
if path in prefix_set:
return True
# If there's more than 5 paths in prefix_set, it's *probably* quicker to
# "walk up" the directory hierarchy instead, with the assumption that most
# directory hierarchies are relatively shallow and hash lookup is cheap.
if l > 5:
return any(
Matt Harbison
formatting: re-blacken match.py...
r46681 parentdir in prefix_set for parentdir in pathutil.finddirs(path)
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614 )
# FIXME: Ideally we'd never get to this point if this is the case - we'd
# recognize ourselves as an 'always' matcher and skip this.
if b'' in prefix_set:
return True
Gregory Szorc
match: delete Python 2 conditional code...
r49739 sl = ord(b'/')
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614
# We already checked that path isn't in prefix_set exactly, so
# `path[len(pf)] should never raise IndexError.
return any(path.startswith(pf) and path[len(pf)] == sl for pf in prefix_set)
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501 class patternmatcher(basematcher):
Daniel Ploch
py3: make doc strings containing deprecated '\.' escape sequence raw strings...
r44125 r"""Matches a set of (kind, pat, source) against a 'root' directory.
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250
>>> kindpats = [
Gregory Szorc
match: use raw strings to avoid illegal baskslash escape...
r42367 ... (b're', br'.*\.c$', b''),
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 ... (b'path', b'foo/a', b''),
... (b'relpath', b'b', b''),
... (b'glob', b'*.h', b''),
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 ... ]
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m = patternmatcher(b'foo', kindpats)
>>> m(b'main.c') # matches re:.*\.c$
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'b.txt')
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 False
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'foo/a') # matches path:foo/a
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'a') # does not match path:b, since 'root' is 'foo'
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 False
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'b') # matches relpath:b, since 'root' is 'foo'
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'lib.h') # matches glob:*.h
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
>>> m.files()
match: sort patterns before compiling them into a regex...
r51285 [b'', b'foo/a', b'', b'b']
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m.exact(b'foo/a')
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m.exact(b'b')
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m.exact(b'lib.h') # exact matches are for (rel)path kinds
Denis Laxalde
match: add a docstring with doctest examples to patternmatcher...
r42250 False
"""
Martin von Zweigbergk
match: replace match class by match function (API)...
r32394
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, root, kindpats, badfn=None):
super(patternmatcher, self).__init__(badfn)
match: sort patterns before compiling them into a regex...
r51285 kindpats.sort()
Matt Mackall
match: fold _matcher into match.__init__
r8581
Arseniy Alekseyev
matchers: support patternmatcher in rust
r52496 if rustmod is not None:
# We need to pass the patterns to Rust because they can contain
# patterns from the user interface
self._kindpats = kindpats
Arseniy Alekseyev
match: fix the "visitdir" method on "rootfilesin" matchers...
r52462 roots, dirs, parents = _rootsdirsandparents(kindpats)
Martin von Zweigbergk
match: remove special-casing of always-matching patterns in patternmatcher...
r32557 self._files = _explicitfiles(kindpats)
Arseniy Alekseyev
match: fix the "visitdir" method on "rootfilesin" matchers...
r52462 self._dirs_explicit = set(dirs)
self._dirs = parents
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 self._prefix = _prefix(kindpats)
match: match explicit file using a set...
r51286 self._pats, self._matchfn = _buildmatch(kindpats, b'$', root)
def matchfn(self, fn):
if fn in self._fileset:
return True
return self._matchfn(fn)
Matt Mackall
match: fold match into _match base class
r8587
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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'all'
Arseniy Alekseyev
match: fix the "visitdir" method on "rootfilesin" matchers...
r52462 return (
dir in self._dirs
or path_or_parents_in_set(dir, self._fileset)
or path_or_parents_in_set(dir, self._dirs_explicit)
)
Drew Gottlieb
treemanifest: further optimize treemanifest.matches()...
r24636
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
ret = self.visitdir(dir)
if ret is True:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990 elif not ret:
return set()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 assert ret == b'all'
return b'all'
spectral
match: add visitchildrenset complement to visitdir...
r38990
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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<patternmatcher patterns=%r>' % pycompat.bytestr(self._pats)
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
utils: move the `dirs` definition in pathutil (API)...
r43923 # This is basically a reimplementation of pathutil.dirs that stores the
# children instead of just a count of them, plus a small optional optimization
# to avoid some directories we don't need.
Gregory Szorc
py3: use class X: instead of class X(object):...
r49801 class _dirchildren:
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 def __init__(self, paths, onlyinclude=None):
self._dirs = {}
self._onlyinclude = onlyinclude or []
addpath = self.addpath
for f in paths:
addpath(f)
def addpath(self, path):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if path == b'':
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 return
dirs = self._dirs
findsplitdirs = _dirchildren._findsplitdirs
for d, b in findsplitdirs(path):
if d not in self._onlyinclude:
continue
dirs.setdefault(d, set()).add(b)
@staticmethod
def _findsplitdirs(path):
# yields (dirname, basename) tuples, walking back to the root. This is
Martin von Zweigbergk
utils: move finddirs() to pathutil...
r44032 # very similar to pathutil.finddirs, except:
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 # - produces a (dirname, basename) tuple, not just 'dirname'
# Unlike manifest._splittopdir, this does not suffix `dirname` with a
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 # slash.
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 oldpos = len(path)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 pos = path.rfind(b'/')
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 while pos != -1:
Augie Fackler
formatting: blacken the codebase...
r43346 yield path[:pos], path[pos + 1 : oldpos]
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 oldpos = pos
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 pos = path.rfind(b'/', 0, pos)
yield b'', path[:oldpos]
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494
def get(self, path):
return self._dirs.get(path, set())
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501 class includematcher(basematcher):
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, root, kindpats, badfn=None):
super(includematcher, self).__init__(badfn)
Raphaël Gomès
rust-status: use bare hg status fastpath from Python...
r45017 if rustmod is not None:
# We need to pass the patterns to Rust because they can contain
# patterns from the user interface
self._kindpats = kindpats
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 self._pats, self.matchfn = _buildmatch(kindpats, b'(?:/|$)', root)
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 self._prefix = _prefix(kindpats)
spectral
includematcher: separate "parents" from "dirs"...
r38989 roots, dirs, parents = _rootsdirsandparents(kindpats)
Martin von Zweigbergk
match: remove support for non-include patterns from includematcher...
r32502 # 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)
spectral
includematcher: separate "parents" from "dirs"...
r38989 # parents are directories which are non-recursively included because
# they are needed to get to items in _dirs or _roots.
Martin von Zweigbergk
match: de-flake test-doctest.py by not depending on util.dirs() order...
r42553 self._parents = parents
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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'all'
Augie Fackler
formatting: blacken the codebase...
r43346 return (
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614 dir in self._dirs
Augie Fackler
formatting: blacken the codebase...
r43346 or dir in self._parents
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614 or path_or_parents_in_set(dir, self._roots)
Augie Fackler
formatting: blacken the codebase...
r43346 )
Martin von Zweigbergk
match: split up main matcher into patternmatcher and includematcher...
r32501
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 @propertycache
def _allparentschildren(self):
# It may seem odd that we add dirs, roots, and parents, and then
# restrict to only parents. This is to catch the case of:
# dirs = ['foo/bar']
# parents = ['foo']
# if we asked for the children of 'foo', but had only added
# self._parents, we wouldn't be able to respond ['bar'].
return _dirchildren(
Augie Fackler
formatting: blacken the codebase...
r43346 itertools.chain(self._dirs, self._roots, self._parents),
onlyinclude=self._parents,
)
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
if self._prefix and dir in self._roots:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'all'
spectral
match: add visitchildrenset complement to visitdir...
r38990 # Note: this does *not* include the 'dir in self._parents' case from
# visitdir, that's handled below.
Augie Fackler
formatting: blacken the codebase...
r43346 if (
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b'' in self._roots
Augie Fackler
formatting: blacken the codebase...
r43346 or dir in self._dirs
Kyle Lippincott
match: skip walking up the directory hierarchy if the number of pats are small...
r46614 or path_or_parents_in_set(dir, self._roots)
Augie Fackler
formatting: blacken the codebase...
r43346 ):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990
if dir in self._parents:
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 return self._allparentschildren.get(dir) or set()
return set()
spectral
match: add visitchildrenset complement to visitdir...
r38990
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
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<includematcher includes=%r>' % pycompat.bytestr(self._pats)
Augie Fackler
formatting: blacken the codebase...
r43346
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):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 r"""Matches the input files exactly. They are interpreted as paths, not
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 patterns (so no kind-prefixes).
Denis Laxalde
match: add doctest examples for exactmatcher...
r42249
Gregory Szorc
match: use raw strings to avoid illegal baskslash escape...
r42367 >>> m = exactmatcher([b'a.txt', br're:.*\.c$'])
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'a.txt')
Denis Laxalde
match: add doctest examples for exactmatcher...
r42249 True
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'b.txt')
Denis Laxalde
match: add doctest examples for exactmatcher...
r42249 False
Input files that would be matched are exactly those returned by .files()
>>> m.files()
['a.txt', 're:.*\\.c$']
So pattern 're:.*\.c$' is not considered as a regex, but as a file name
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> m(b'main.c')
Denis Laxalde
match: add doctest examples for exactmatcher...
r42249 False
Gregory Szorc
match: use raw strings to avoid illegal baskslash escape...
r42367 >>> m(br're:.*\.c$')
Denis Laxalde
match: add doctest examples for exactmatcher...
r42249 True
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, files, badfn=None):
super(exactmatcher, self).__init__(badfn)
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499
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):
utils: move the `dirs` definition in pathutil (API)...
r43923 return set(pathutil.dirs(self._fileset))
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499
def visitdir(self, dir):
return dir in self._dirs
Kyle Lippincott
match: convert O(n) to O(log n) in exactmatcher.visitchildrenset...
r47634 @propertycache
def _visitchildrenset_candidates(self):
"""A memoized set of candidates for visitchildrenset."""
return self._fileset | self._dirs - {b''}
@propertycache
def _sorted_visitchildrenset_candidates(self):
"""A memoized sorted list of candidates for visitchildrenset."""
return sorted(self._visitchildrenset_candidates)
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
Kyle Lippincott
match: make exactmatcher.visitchildrenset return file children as well...
r39297 if not self._fileset or dir not in self._dirs:
return set()
Kyle Lippincott
match: convert O(n) to O(log n) in exactmatcher.visitchildrenset...
r47634 if dir == b'':
candidates = self._visitchildrenset_candidates
else:
candidates = self._sorted_visitchildrenset_candidates
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 d = dir + b'/'
Kyle Lippincott
match: convert O(n) to O(log n) in exactmatcher.visitchildrenset...
r47634 # Use bisect to find the first element potentially starting with d
# (i.e. >= d). This should always find at least one element (we'll
# assert later if this is not the case).
first = bisect.bisect_left(candidates, d)
# We need a representation of the first element that is > d that
# does not start with d, so since we added a `/` on the end of dir,
# we'll add whatever comes after slash (we could probably assume
# that `0` is after `/`, but let's not) to the end of dir instead.
dnext = dir + encoding.strtolocal(chr(ord(b'/') + 1))
# Use bisect to find the first element >= d_next
last = bisect.bisect_left(candidates, dnext, lo=first)
dlen = len(d)
candidates = {c[dlen:] for c in candidates[first:last]}
Kyle Lippincott
match: make exactmatcher.visitchildrenset return file children as well...
r39297 # self._dirs includes all of the directories, recursively, so if
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 # we're attempting to match foo/bar/baz.txt, it'll have '', 'foo',
Kyle Lippincott
match: make exactmatcher.visitchildrenset return file children as well...
r39297 # 'foo/bar' in it. Thus we can safely ignore a candidate that has a
# '/' in it, indicating a it's for a subdir-of-a-subdir; the
# immediate subdir will be in there without a slash.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 ret = {c for c in candidates if b'/' not in c}
Kyle Lippincott
match: make exactmatcher.visitchildrenset return file children as well...
r39297 # We really do not expect ret to be empty, since that would imply that
# there's something in _dirs that didn't have a file in _fileset.
assert ret
return ret
spectral
match: add visitchildrenset complement to visitdir...
r38990
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499 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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<exactmatcher files=%r>' % self._files
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle exact matching using new exactmatcher
r32499
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 class differencematcher(basematcher):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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
Martin von Zweigbergk
match: remove explicitdir attribute...
r44114 The second matcher's non-matching-attributes (bad, traversedir) are ignored.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 def __init__(self, m1, m2):
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 super(differencematcher, self).__init__()
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 self._m1 = m1
self._m2 = m2
self.bad = m1.bad
self.traversedir = m1.traversedir
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with(self) -> bool:
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 return (
self.was_tampered_with_nonrec()
or self._m1.was_tampered_with()
or self._m2.was_tampered_with()
)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if self._m2.visitdir(dir) == b'all':
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 return False
Pulkit Goyal
match: teach diffmatcher.visitdir() to return 'all' if possible...
r41669 elif not self._m2.visitdir(dir):
# m2 does not match dir, we can return 'all' here if possible
return self._m1.visitdir(dir)
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 return bool(self._m1.visitdir(dir))
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
m2_set = self._m2.visitchildrenset(dir)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if m2_set == b'all':
spectral
match: add visitchildrenset complement to visitdir...
r38990 return set()
m1_set = self._m1.visitchildrenset(dir)
# Possible values for m1: 'all', 'this', set(...), set()
# Possible values for m2: 'this', set(...), set()
# If m2 has nothing under here that we care about, return m1, even if
# it's 'all'. This is a change in behavior from visitdir, which would
# return True, not 'all', for some reason.
if not m2_set:
return m1_set
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if m1_set in [b'all', b'this']:
spectral
match: add visitchildrenset complement to visitdir...
r38990 # Never return 'all' here if m2_set is any kind of non-empty (either
# 'this' or set(foo)), since m2 might return set() for a
# subdirectory.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990 # Possible values for m1: set(...), set()
# Possible values for m2: 'this', set(...)
# We ignore m2's set results. They're possibly incorrect:
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 # m1 = path:dir/subdir, m2=rootfilesin:dir, visitchildrenset(''):
spectral
match: add visitchildrenset complement to visitdir...
r38990 # m1 returns {'dir'}, m2 returns {'dir'}, if we subtracted we'd
# return set(), which is *not* correct, we still need to visit 'dir'!
return m1_set
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465 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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<differencematcher m1=%r, m2=%r>' % (self._m1, self._m2)
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle excludes using new differencematcher...
r32465
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 def intersectmatchers(m1, m2):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Composes two matchers by matching if both of them match.
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497
Martin von Zweigbergk
match: remove explicitdir attribute...
r44114 The second matcher's non-matching-attributes (bad, traversedir) are ignored.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 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.traversedir = m1.traversedir
return m
if m2.always():
m = copy.copy(m1)
return m
return intersectionmatcher(m1, m2)
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 class intersectionmatcher(basematcher):
def __init__(self, m1, m2):
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 super(intersectionmatcher, self).__init__()
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 self._m1 = m1
self._m2 = m2
self.bad = m1.bad
self.traversedir = m1.traversedir
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with(self) -> bool:
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 return (
self.was_tampered_with_nonrec()
or self._m1.was_tampered_with()
or self._m2.was_tampered_with()
)
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 @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)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if visit1 == b'all':
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 return self._m2.visitdir(dir)
# bool() because visit1=True + visit2='all' should not be 'all'
return bool(visit1 and self._m2.visitdir(dir))
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
m1_set = self._m1.visitchildrenset(dir)
if not m1_set:
return set()
m2_set = self._m2.visitchildrenset(dir)
if not m2_set:
return set()
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if m1_set == b'all':
spectral
match: add visitchildrenset complement to visitdir...
r38990 return m2_set
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif m2_set == b'all':
spectral
match: add visitchildrenset complement to visitdir...
r38990 return m1_set
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if m1_set == b'this' or m2_set == b'this':
return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990
assert isinstance(m1_set, set) and isinstance(m2_set, set)
return m1_set.intersection(m2_set)
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497 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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<intersectionmatcher m1=%r, m2=%r>' % (self._m1, self._m2)
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: handle includes using new intersectionmatcher
r32497
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
Martin von Zweigbergk
tests: fix failing doctest in match.py by adding dummy auditor...
r44454 >>> m1 = match(util.localpath(b'/root'), b'', [b'a.txt', b'sub/b.txt'], auditor=lambda name: None)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> m2 = subdirmatcher(b'sub', m1)
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2(b'a.txt')
Martin Geisler
match: add narrowmatcher class...
r12165 False
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2(b'b.txt')
Martin Geisler
match: add narrowmatcher class...
r12165 True
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2.matchfn(b'a.txt')
Martin Geisler
match: add narrowmatcher class...
r12165 False
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> 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
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
Martin Geisler
match: add narrowmatcher class...
r12165 """
Matt Harbison
typing: add type hints around the matcher for subrepo archiving...
r52650 def __init__(self, path: bytes, matcher: basematcher) -> None:
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 super(subdirmatcher, self).__init__()
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
Augie Fackler
formatting: blacken the codebase...
r43346 self._files = [
f[len(path) + 1 :]
for f in matcher._files
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if f.startswith(path + b"/")
Augie Fackler
formatting: blacken the codebase...
r43346 ]
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
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with(self) -> bool:
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 return (
self.was_tampered_with_nonrec() or self._matcher.was_tampered_with()
)
Martin von Zweigbergk
match: avoid accessing match._pathrestricted from subdirmatcher...
r32325 def bad(self, f, msg):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 self._matcher.bad(self._path + b"/" + f, msg)
Martin von Zweigbergk
match: avoid accessing match._pathrestricted from subdirmatcher...
r32325
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.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return self._matcher.matchfn(self._path + b"/" + f)
Martin von Zweigbergk
match: override matchfn() the usual way in subdirmatcher
r32464
Martin von Zweigbergk
match: override visitdir() the usual way in subdirmatcher...
r32324 def visitdir(self, dir):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if dir == b'':
Martin von Zweigbergk
match: override visitdir() the usual way in subdirmatcher...
r32324 dir = self._path
else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 dir = self._path + b"/" + dir
Martin von Zweigbergk
match: override visitdir() the usual way in subdirmatcher...
r32324 return self._matcher.visitdir(dir)
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if dir == b'':
spectral
match: add visitchildrenset complement to visitdir...
r38990 dir = self._path
else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 dir = self._path + b"/" + dir
spectral
match: add visitchildrenset complement to visitdir...
r38990 return self._matcher.visitchildrenset(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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<subdirmatcher path=%r, matcher=%r>' % (
Augie Fackler
formatting: blacken the codebase...
r43346 self._path,
self._matcher,
)
Martin von Zweigbergk
match: add __repr__ for subdirmatcher...
r32552
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 class prefixdirmatcher(basematcher):
"""Adapt a matcher to work on a parent directory.
Martin von Zweigbergk
match: remove explicitdir attribute...
r44114 The matcher's non-matching-attributes (bad, traversedir) are ignored.
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630
The prefix path should usually be the relative path from the root of
this matcher to the root of the wrapped matcher.
Martin von Zweigbergk
match: make sure `root` argument is always an absolute path (API)...
r44401 >>> m1 = match(util.localpath(b'/root/d/e'), b'f', [b'../a.txt', b'b.txt'], auditor=lambda name: None)
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 >>> m2 = prefixdirmatcher(b'd/e', m1)
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2(b'a.txt')
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 False
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2(b'd/e/a.txt')
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 True
Denis Laxalde
match: let regex match function return a boolean...
r42256 >>> m2(b'd/e/b.txt')
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 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
"""
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 def __init__(self, path, matcher, badfn=None):
super(prefixdirmatcher, self).__init__(badfn)
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 if not path:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.ProgrammingError(b'prefix path must not be empty')
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 self._path = path
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 self._pathprefix = path + b'/'
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 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
Augie Fackler
formatting: blacken the codebase...
r43346 return self._matcher.matchfn(f[len(self._pathprefix) :])
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630
@propertycache
def _pathdirs(self):
Martin von Zweigbergk
utils: move finddirs() to pathutil...
r44032 return set(pathutil.finddirs(self._path))
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630
def visitdir(self, dir):
if dir == self._path:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return self._matcher.visitdir(b'')
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 if dir.startswith(self._pathprefix):
Augie Fackler
formatting: blacken the codebase...
r43346 return self._matcher.visitdir(dir[len(self._pathprefix) :])
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 return dir in self._pathdirs
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
if dir == self._path:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return self._matcher.visitchildrenset(b'')
spectral
match: add visitchildrenset complement to visitdir...
r38990 if dir.startswith(self._pathprefix):
Augie Fackler
formatting: blacken the codebase...
r43346 return self._matcher.visitchildrenset(dir[len(self._pathprefix) :])
spectral
match: add visitchildrenset complement to visitdir...
r38990 if dir in self._pathdirs:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
Kyle Lippincott
match: add missing "return set()", add FIXME to test to doc a bug...
r38993 return set()
spectral
match: add visitchildrenset complement to visitdir...
r38990
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630 def isexact(self):
return self._matcher.isexact()
def prefix(self):
return self._matcher.prefix()
@encoding.strmethod
def __repr__(self):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<prefixdirmatcher path=%r, matcher=%r>' % (
Augie Fackler
formatting: blacken the codebase...
r43346 pycompat.bytestr(self._path),
self._matcher,
)
Yuya Nishihara
match: add prefixdirmatcher to adapt subrepo matcher back...
r38630
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.
Martin von Zweigbergk
match: remove explicitdir attribute...
r44114 The non-matching-attributes (bad, traversedir) are taken from the first
matcher.
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 """
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]
Martin von Zweigbergk
match: delete unused root and cwd arguments to constructors (API)...
r41824 super(unionmatcher, self).__init__()
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 self.traversedir = m1.traversedir
Gregory Szorc
match: move matchers from sparse into core...
r33319 self._matchers = matchers
Matt Harbison
typing: add some trivial type hints to `mercurial/match.py`...
r52568 def was_tampered_with(self) -> bool:
Arseniy Alekseyev
match: make `was_tampered_with` work recursively...
r52518 return self.was_tampered_with_nonrec() or any(
map(lambda m: m.was_tampered_with(), self._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)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if v == b'all':
Martin von Zweigbergk
match: make unionmatcher a proper matcher...
r33448 return v
r |= v
return r
spectral
match: add visitchildrenset complement to visitdir...
r38990 def visitchildrenset(self, dir):
r = set()
this = False
for m in self._matchers:
v = m.visitchildrenset(dir)
if not v:
continue
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if v == b'all':
spectral
match: add visitchildrenset complement to visitdir...
r38990 return v
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if this or v == b'this':
spectral
match: add visitchildrenset complement to visitdir...
r38990 this = True
# don't break, we might have an 'all' in here.
continue
assert isinstance(v, set)
r = r.union(v)
if this:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'this'
spectral
match: add visitchildrenset complement to visitdir...
r38990 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):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'<unionmatcher matchers=%r>' % self._matchers
Augie Fackler
formatting: blacken the codebase...
r43346
Gregory Szorc
match: move matchers from sparse into core...
r33319
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def patkind(pattern, default=None):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 r"""If pattern is 'kind:pat' with a known kind, return kind.
Denis Laxalde
match: add doctest examples for patkind()
r42251
Gregory Szorc
match: use raw strings to avoid illegal baskslash escape...
r42367 >>> patkind(br're:.*\.c$')
Denis Laxalde
match: add doctest examples for patkind()
r42251 're'
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> patkind(b'glob:*.c')
Denis Laxalde
match: add doctest examples for patkind()
r42251 'glob'
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> patkind(b'relpath:test.py')
Denis Laxalde
match: add doctest examples for patkind()
r42251 'relpath'
Pulkit Goyal
py3: add b'' prefixes to new doctests in match.py...
r42267 >>> patkind(b'main.py')
>>> patkind(b'main.py', default=b're')
Denis Laxalde
match: add doctest examples for patkind()
r42251 're'
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 return _patsplit(pattern, default)[0]
Matt Mackall
match: move util match functions over
r8570
Augie Fackler
formatting: blacken the codebase...
r43346
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."""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b':' in pattern:
kind, pat = pattern.split(b':', 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
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
match: remove head and tail args from _globre
r8582 def _globre(pat):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 r"""Convert an extended glob string to a regexp string.
Mads Kiilerich
match: _globre doctests
r21112
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 \.\*\?
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Matt Mackall
match: move util match functions over
r8570 i, n = 0, len(pat)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res = b''
Matt Mackall
match: move util match functions over
r8570 group = 0
Boris Feld
match: provide and use a quick way to escape a single byte...
r40720 escape = util.stringutil.regexbytesescapemap.get
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Mackall
many, many trivial check-code fixups
r10282 def peek():
Augie Fackler
formatting: blacken the codebase...
r43346 return i < n and pat[i : i + 1]
Matt Mackall
match: move util match functions over
r8570 while i < n:
Augie Fackler
formatting: blacken the codebase...
r43346 c = pat[i : i + 1]
Matt Mackall
many, many trivial check-code fixups
r10282 i += 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if c not in b'*?[{},\\':
Boris Feld
match: provide and use a quick way to escape a single byte...
r40720 res += escape(c, c)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif c == b'*':
if peek() == b'*':
Matt Mackall
match: move util match functions over
r8570 i += 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if peek() == b'/':
Siddharth Agarwal
match: make glob '**/' match the empty string...
r21815 i += 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res += b'(?:.*/)?'
Siddharth Agarwal
match: make glob '**/' match the empty string...
r21815 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res += b'.*'
Matt Mackall
match: move util match functions over
r8570 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res += b'[^/]*'
elif c == b'?':
res += b'.'
elif c == b'[':
Matt Mackall
match: move util match functions over
r8570 j = i
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if j < n and pat[j : j + 1] in b'!]':
Matt Mackall
match: move util match functions over
r8570 j += 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 while j < n and pat[j : j + 1] != b']':
Matt Mackall
match: move util match functions over
r8570 j += 1
if j >= n:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res += b'\\['
Matt Mackall
match: move util match functions over
r8570 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 stuff = pat[i:j].replace(b'\\', b'\\\\')
Matt Mackall
match: move util match functions over
r8570 i = j + 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if stuff[0:1] == b'!':
stuff = b'^' + stuff[1:]
elif stuff[0:1] == b'^':
stuff = b'\\' + stuff
res = b'%s[%s]' % (res, stuff)
elif c == b'{':
Matt Mackall
match: move util match functions over
r8570 group += 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 res += b'(?:'
elif c == b'}' and group:
res += b')'
Matt Mackall
match: move util match functions over
r8570 group -= 1
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif c == b',' and group:
res += b'|'
elif c == b'\\':
Matt Mackall
match: move util match functions over
r8570 p = peek()
if p:
i += 1
Boris Feld
match: provide and use a quick way to escape a single byte...
r40720 res += escape(p, p)
Matt Mackall
match: move util match functions over
r8570 else:
Boris Feld
match: provide and use a quick way to escape a single byte...
r40720 res += escape(c, c)
Matt Mackall
match: move util match functions over
r8570 else:
Boris Feld
match: provide and use a quick way to escape a single byte...
r40720 res += escape(c, c)
Matt Mackall
match: remove head and tail args from _globre
r8582 return res
Matt Mackall
match: move util match functions over
r8570
Augie Fackler
formatting: blacken the codebase...
r43346
Matt Harbison
match: make the FLAG_RE pattern a raw string...
r50540 FLAG_RE = util.re.compile(br'^\(\?([aiLmsux]+)\)(.*)')
matcher: fix issues regex flag contained in pattern (issue6759)...
r50498
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 def _regex(kind, pat, globsuffix):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Convert a (normalized) pattern of any kind into a
Raphaël Gomès
rust-filepatterns: call new Rust implementations from Python...
r42516 regular expression.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 globsuffix is appended to the regexp of globs."""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if not pat and kind in (b'glob', b'relpath'):
return b''
if kind == b're':
Mads Kiilerich
match: improve documentation - docstrings and more descriptive variable naming...
r21111 return pat
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 if kind == b'filepath':
raise error.ProgrammingError(
"'filepath:' patterns should not be converted to a regex"
)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind in (b'path', b'relpath'):
if pat == b'.':
return b''
return util.stringutil.reescape(pat) + b'(?:/|$)'
if kind == b'rootfilesin':
if pat == b'.':
escaped = b''
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 else:
# Pattern is a directory name.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 escaped = util.stringutil.reescape(pat) + b'/'
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 # Anything after the pattern must be a non-directory.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return escaped + b'[^/]+$'
if kind == b'relglob':
Valentin Gatien-Baron
match: simplify the regexps created for glob patterns...
r43132 globre = _globre(pat)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if globre.startswith(b'[^/]*'):
Valentin Gatien-Baron
match: simplify the regexps created for glob patterns...
r43132 # When pat has the form *XYZ (common), make the returned regex more
# legible by returning the regex for **XYZ instead of **/*XYZ.
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'.*' + globre[len(b'[^/]*') :] + globsuffix
return b'(?:|.*/)' + globre + globsuffix
if kind == b'relre':
matcher: do not prepend '.*' to pattern using ^ after flags...
r50500 flag = None
m = FLAG_RE.match(pat)
if m:
flag, pat = m.groups()
if not pat.startswith(b'^'):
pat = b'.*' + pat
if flag is not None:
pat = br'(?%s:%s)' % (flag, pat)
return pat
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind in (b'glob', b'rootglob'):
Yuya Nishihara
match: explode if unsupported pattern passed down to _regex() builder
r38597 return _globre(pat) + globsuffix
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.ProgrammingError(b'not a regex pattern: %s:%s' % (kind, pat))
Matt Mackall
match: unnest functions in _matcher
r8574
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: delete unused argument "listsubrepos" from _buildmatch()...
r41818 def _buildmatch(kindpats, globsuffix, root):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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 = {}
Augie Fackler
formatting: blacken the codebase...
r43346
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
Augie Fackler
formatting: blacken the codebase...
r43346 if mf(f[len(prefix) :]):
Durham Goode
match: make subinclude construction lazy...
r32132 return True
Durham Goode
match: enable 'subinclude:' syntax...
r25283 return False
Augie Fackler
formatting: blacken the codebase...
r43346
Durham Goode
match: enable 'subinclude:' syntax...
r25283 matchfuncs.append(matchsubinclude)
Matt Mackall
match: introduce basic fileset support
r14675
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 regex = b''
Durham Goode
match: allow unioning arbitrary match functions...
r25239 if kindpats:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if all(k == b'rootfilesin' for k, p, s in kindpats):
Martin von Zweigbergk
match: optimize matcher when all patterns are of rootfilesin kind...
r40278 dirs = {p for k, p, s in kindpats}
Augie Fackler
formatting: blacken the codebase...
r43346
Martin von Zweigbergk
match: optimize matcher when all patterns are of rootfilesin kind...
r40278 def mf(f):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 i = f.rfind(b'/')
Martin von Zweigbergk
match: optimize matcher when all patterns are of rootfilesin kind...
r40278 if i >= 0:
dir = f[:i]
else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 dir = b'.'
Martin von Zweigbergk
match: optimize matcher when all patterns are of rootfilesin kind...
r40278 return dir in dirs
Augie Fackler
formatting: blacken the codebase...
r43346
Augie Fackler
match: fix up a repr to not crash on Python 3...
r40381 regex = b'rootfilesin: %s' % stringutil.pprint(list(sorted(dirs)))
Martin von Zweigbergk
match: optimize matcher when all patterns are of rootfilesin kind...
r40278 matchfuncs.append(mf)
else:
regex, mf = _buildregexmatch(kindpats, globsuffix)
matchfuncs.append(mf)
Durham Goode
match: allow unioning arbitrary match functions...
r25239
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
Augie Fackler
formatting: blacken the codebase...
r43346
Boris Feld
match: extract a literal constant into a symbolic one
r40810 MAX_RE_SIZE = 20000
Augie Fackler
formatting: blacken the codebase...
r43346
Boris Feld
match: extract function that group regexps
r40812 def _joinregexes(regexps):
"""gather multiple regular expressions into a single one"""
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 return b'|'.join(regexps)
Boris Feld
match: extract function that group regexps
r40812
Augie Fackler
formatting: blacken the codebase...
r43346
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,
Boris Feld
match: test for overflow error in pattern...
r40811 return regexp string and a matcher function.
Test too large input
>>> _buildregexmatch([
Augie Fackler
match: fix doctest to use bytes instead of str...
r40983 ... (b'relglob', b'?' * MAX_RE_SIZE, b'')
... ], b'$')
Boris Feld
match: test for overflow error in pattern...
r40811 Traceback (most recent call last):
...
Boris Feld
match: raise an Abort error instead of OverflowError...
r40814 Abort: matcher pattern is too long (20009 bytes)
Boris Feld
match: test for overflow error in pattern...
r40811 """
Matt Mackall
match: unnest functions in _matcher
r8574 try:
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 allgroups = []
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 regexps = []
exact = set()
Arseniy Alekseyev
match: fix the "visitdir" method on "rootfilesin" matchers...
r52462 for kind, pattern, _source in kindpats:
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 if kind == b'filepath':
exact.add(pattern)
continue
regexps.append(_regex(kind, pattern, globsuffix))
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 fullregexp = _joinregexes(regexps)
startidx = 0
Martin von Zweigbergk
match: drop unnecessary wrapping of regex in group...
r40818 groupsize = 0
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 for idx, r in enumerate(regexps):
piecesize = len(r)
Martin von Zweigbergk
match: drop unnecessary wrapping of regex in group...
r40818 if piecesize > MAX_RE_SIZE:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 msg = _(b"matcher pattern is too long (%d bytes)") % piecesize
Boris Feld
match: raise an Abort error instead of OverflowError...
r40814 raise error.Abort(msg)
Martin von Zweigbergk
match: make "groupsize" include the trailing "|"...
r40816 elif (groupsize + piecesize) > MAX_RE_SIZE:
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 group = regexps[startidx:idx]
allgroups.append(_joinregexes(group))
startidx = idx
Martin von Zweigbergk
match: drop unnecessary wrapping of regex in group...
r40818 groupsize = 0
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 groupsize += piecesize + 1
if startidx == 0:
Denis Laxalde
match: let regex match function return a boolean...
r42256 matcher = _rematcher(fullregexp)
match: fix re2 compability broken in 2e2699af5649...
r42268 func = lambda s: bool(matcher(s))
Boris Feld
match: avoid translating glob to matcher multiple times for large sets...
r40813 else:
group = regexps[startidx:]
allgroups.append(_joinregexes(group))
allmatchers = [_rematcher(g) for g in allgroups]
func = lambda s: any(m(s) for m in allmatchers)
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588
actualfunc = func
if exact:
# An empty regex will always match, so only call the regex if
# there were any actual patterns to match.
if not regexps:
actualfunc = lambda s: s in exact
else:
actualfunc = lambda s: s in exact or func(s)
return fullregexp, actualfunc
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:
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 if k == b'filepath':
continue
Matt Mackall
match: unnest functions in _matcher
r8574 try:
Martin von Zweigbergk
match: drop unnecessary wrapping of regex in group...
r40818 _rematcher(_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:
Augie Fackler
formatting: blacken the codebase...
r43346 raise error.Abort(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"%s: invalid pattern (%s): %s") % (s, k, p)
Augie Fackler
formatting: blacken the codebase...
r43346 )
Durham Goode
match: add source to kindpats list...
r25213 else:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 raise error.Abort(_(b"invalid pattern (%s): %s") % (k, p))
raise error.Abort(_(b"invalid pattern"))
Matt Mackall
match: unnest functions in _matcher
r8574
Augie Fackler
formatting: blacken the codebase...
r43346
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 def _patternrootsanddirs(kindpats):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """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.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind in (b'glob', b'rootglob'): # find the non-glob prefix
Matt Mackall
match: fold _globprefix into _roots
r8584 root = []
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 for p in pat.split(b'/'):
if b'[' in p or b'{' in p or b'*' in p or b'?' in p:
Matt Mackall
match: fold _globprefix into _roots
r8584 break
root.append(p)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 r.append(b'/'.join(root))
Raphaël Gomès
match: add `filepath:` pattern to match an exact filepath relative to the root...
r51588 elif kind in (b'relpath', b'path', b'filepath'):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if pat == b'.':
pat = b''
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 r.append(pat)
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif kind in (b'rootfilesin',):
if pat == b'.':
pat = b''
Martin von Zweigbergk
match: use '' instead of '.' for root directory (API)...
r42528 d.append(pat)
Augie Fackler
formatting: blacken the codebase...
r43346 else: # relglob, re, relre
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 r.append(b'')
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 return r, d
Augie Fackler
formatting: blacken the codebase...
r43346
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 def _roots(kindpats):
'''Returns root directories to match recursively from the given patterns.'''
roots, dirs = _patternrootsanddirs(kindpats)
return roots
Augie Fackler
formatting: blacken the codebase...
r43346
spectral
includematcher: separate "parents" from "dirs"...
r38989 def _rootsdirsandparents(kindpats):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Returns roots and exact directories from patterns.
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013
Kyle Lippincott
match: correct doc for _rootsdirsandparents after 5a7df82de142...
r38992 `roots` are directories to match recursively, `dirs` should
be matched non-recursively, and `parents` are the implicitly required
directories to walk to items in either roots or dirs.
Returns a tuple of (roots, dirs, parents).
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013
match: stabilize _rootsdirsandparents doctest...
r42559 >>> r = _rootsdirsandparents(
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 ... [(b'glob', b'g/h/*', b''), (b'glob', b'g/h', b''),
... (b'glob', b'g*', b'')])
match: stabilize _rootsdirsandparents doctest...
r42559 >>> print(r[0:2], sorted(r[2])) # the set has an unstable output
(['g/h', 'g/h', ''], []) ['', 'g']
>>> r = _rootsdirsandparents(
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 ... [(b'rootfilesin', b'g/h', b''), (b'rootfilesin', b'', b'')])
match: stabilize _rootsdirsandparents doctest...
r42559 >>> print(r[0:2], sorted(r[2])) # the set has an unstable output
([], ['g/h', '']) ['', 'g']
>>> r = _rootsdirsandparents(
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 ... [(b'relpath', b'r', b''), (b'path', b'p/p', b''),
... (b'path', b'', b'')])
match: stabilize _rootsdirsandparents doctest...
r42559 >>> print(r[0:2], sorted(r[2])) # the set has an unstable output
(['r', 'p/p', ''], []) ['', 'p']
>>> r = _rootsdirsandparents(
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 ... [(b'relglob', b'rg*', b''), (b're', b're/', b''),
... (b'relre', b'rr', b'')])
match: stabilize _rootsdirsandparents doctest...
r42559 >>> print(r[0:2], sorted(r[2])) # the set has an unstable output
(['', '', ''], []) ['']
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 r, d = _patternrootsanddirs(kindpats)
Martin von Zweigbergk
match: de-flake test-doctest.py by not depending on util.dirs() order...
r42553 p = set()
# Add the parents as non-recursive/exact directories, since they must be
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013 # scanned to get to either the roots or the other exact directories.
utils: move the `dirs` definition in pathutil (API)...
r43923 p.update(pathutil.dirs(d))
p.update(pathutil.dirs(r))
Rodrigo Damazio Bovendorp
match: making visitdir() deal with non-recursive entries...
r31013
Kyle Lippincott
match: improve includematcher.visitchildrenset to be much faster and cached...
r39494 # FIXME: all uses of this function convert these to sets, do so before
# returning.
# FIXME: all uses of this function do not need anything in 'roots' and
# 'dirs' to also be in 'parents', consider removing them before returning.
spectral
includematcher: separate "parents" from "dirs"...
r38989 return r, d, p
Matt Mackall
match: split up _normalizepats
r8576
Augie Fackler
formatting: blacken the codebase...
r43346
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 def _explicitfiles(kindpats):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """Returns the potential explicit filenames from the patterns.
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012
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 []
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
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).
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 filable = [kp for kp in kindpats if kp[0] not in (b'rootfilesin',)]
Rodrigo Damazio Bovendorp
match: adding support for matching files inside a directory...
r31012 return _roots(filable)
Augie Fackler
formatting: blacken the codebase...
r43346
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:
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if kind not in (b'path', b'relpath'):
Martin von Zweigbergk
match: inverse _anypats(), making it _prefix()
r33405 return False
return True
Durham Goode
ignore: move readpatternfile to match.py...
r25167
Augie Fackler
formatting: blacken the codebase...
r43346
Durham Goode
ignore: move readpatternfile to match.py...
r25167 _commentre = None
Matt Harbison
typing: add type hints for the overloads of `matchmod.readpatternfile()`...
r52819 if typing.TYPE_CHECKING:
from typing_extensions import (
Literal,
)
@overload
def readpatternfile(
filepath: bytes, warn: Callable[[bytes], Any], sourceinfo: Literal[True]
) -> List[Tuple[bytes, int, bytes]]:
...
@overload
def readpatternfile(
filepath: bytes,
warn: Callable[[bytes], Any],
sourceinfo: Literal[False],
) -> List[bytes]:
...
@overload
def readpatternfile(
filepath: bytes,
warn: Callable[[bytes], Any],
sourceinfo: bool = False,
) -> List[Union[Tuple[bytes, int, bytes], bytes]]:
...
Augie Fackler
formatting: blacken the codebase...
r43346
Laurent Charignon
match: add option to return line and lineno from readpattern...
r27595 def readpatternfile(filepath, warn, sourceinfo=False):
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """parse a pattern file, returning a list of
Durham Goode
ignore: move readpatternfile to match.py...
r25167 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
Valentin Gatien-Baron
match: support rooted globs in hgignore...
r41318 rootglob:pat # rooted glob (same root as ^ in regexps)
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:
Raphaël Gomès
rust-filepatterns: call new Rust implementations from Python...
r42516 (pattern, lineno, originalline).
This is useful to debug ignore patterns.
Augie Fackler
formating: upgrade to black 20.8b1...
r46554 """
Durham Goode
ignore: use 'include:' rules instead of custom syntax...
r25216
Boris Feld
match: reformat `syntaxes` dictionary for better maintainability
r40721 syntaxes = {
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 b're': b'relre:',
b'regexp': b'relre:',
b'glob': b'relglob:',
b'rootglob': b'rootglob:',
b'include': b'include',
b'subinclude': b'subinclude',
Boris Feld
match: reformat `syntaxes` dictionary for better maintainability
r40721 }
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 syntax = b'relre:'
Durham Goode
ignore: move readpatternfile to match.py...
r25167 patterns = []
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 fp = open(filepath, b'rb')
Gregory Szorc
py3: stop using util.iterfile()...
r49796 for lineno, line in enumerate(fp, start=1):
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if b"#" in line:
Durham Goode
ignore: move readpatternfile to match.py...
r25167 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:
Augie Fackler
formatting: blacken the codebase...
r43346 line = line[: m.end(1)]
Durham Goode
ignore: move readpatternfile to match.py...
r25167 # fixup properly escaped comments that survived the above
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 line = line.replace(b"\\#", b"#")
Durham Goode
ignore: move readpatternfile to match.py...
r25167 line = line.rstrip()
if not line:
continue
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 if line.startswith(b'syntax:'):
Durham Goode
ignore: move readpatternfile to match.py...
r25167 s = line[7:].strip()
try:
syntax = syntaxes[s]
except KeyError:
Durham Goode
match: add optional warn argument...
r25214 if warn:
Augie Fackler
formatting: blacken the codebase...
r43346 warn(
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 _(b"%s: ignoring invalid syntax '%s'\n") % (filepath, s)
Augie Fackler
formatting: blacken the codebase...
r43346 )
Durham Goode
ignore: move readpatternfile to match.py...
r25167 continue
linesyntax = syntax
Gregory Szorc
global: bulk replace simple pycompat.iteritems(x) with x.items()...
r49768 for s, rels in syntaxes.items():
Durham Goode
ignore: move readpatternfile to match.py...
r25167 if line.startswith(rels):
linesyntax = rels
Augie Fackler
formatting: blacken the codebase...
r43346 line = line[len(rels) :]
Durham Goode
ignore: move readpatternfile to match.py...
r25167 break
Augie Fackler
formatting: byteify all mercurial/ and hgext/ string literals...
r43347 elif line.startswith(s + b':'):
Durham Goode
ignore: move readpatternfile to match.py...
r25167 linesyntax = rels
Augie Fackler
formatting: blacken the codebase...
r43346 line = line[len(s) + 1 :]
Durham Goode
ignore: move readpatternfile to match.py...
r25167 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