fileset.py
630 lines
| 19.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / fileset.py
Matt Mackall
|
r14511 | # fileset.py - file set queries for mercurial | ||
# | ||||
# Copyright 2010 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Gregory Szorc
|
r25938 | from __future__ import absolute_import | ||
Augie Fackler
|
r20034 | import re | ||
Gregory Szorc
|
r25938 | |||
from .i18n import _ | ||||
from . import ( | ||||
error, | ||||
merge, | ||||
parser, | ||||
FUJIWARA Katsunori
|
r28448 | registrar, | ||
Pierre-Yves David
|
r31193 | scmutil, | ||
Gregory Szorc
|
r25938 | util, | ||
) | ||||
Matt Mackall
|
r14511 | |||
elements = { | ||||
Yuya Nishihara
|
r25815 | # token-type: binding-strength, primary, prefix, infix, suffix | ||
"(": (20, None, ("group", 1, ")"), ("func", 1, ")"), None), | ||||
"-": (5, None, ("negate", 19), ("minus", 5), None), | ||||
"not": (10, None, ("not", 10), None, None), | ||||
"!": (10, None, ("not", 10), None, None), | ||||
"and": (5, None, None, ("and", 5), None), | ||||
"&": (5, None, None, ("and", 5), None), | ||||
"or": (4, None, None, ("or", 4), None), | ||||
"|": (4, None, None, ("or", 4), None), | ||||
"+": (4, None, None, ("or", 4), None), | ||||
",": (2, None, None, ("list", 2), None), | ||||
")": (0, None, None, None, None), | ||||
"symbol": (0, "symbol", None, None, None), | ||||
"string": (0, "string", None, None, None), | ||||
"end": (0, None, None, None, None), | ||||
Matt Mackall
|
r14511 | } | ||
Martin von Zweigbergk
|
r32291 | keywords = {'and', 'or', 'not'} | ||
Matt Mackall
|
r14511 | |||
Matt Mackall
|
r19470 | globchars = ".*{}[]?/\\_" | ||
Matt Mackall
|
r14551 | |||
Matt Mackall
|
r14511 | def tokenize(program): | ||
pos, l = 0, len(program) | ||||
while pos < l: | ||||
c = program[pos] | ||||
if c.isspace(): # skip inter-token whitespace | ||||
pass | ||||
elif c in "(),-|&+!": # handle simple operators | ||||
yield (c, None, pos) | ||||
elif (c in '"\'' or c == 'r' and | ||||
program[pos:pos + 2] in ("r'", 'r"')): # handle quoted strings | ||||
if c == 'r': | ||||
pos += 1 | ||||
c = program[pos] | ||||
decode = lambda x: x | ||||
else: | ||||
Yuya Nishihara
|
r26233 | decode = parser.unescapestr | ||
Matt Mackall
|
r14511 | pos += 1 | ||
s = pos | ||||
while pos < l: # find closing quote | ||||
d = program[pos] | ||||
if d == '\\': # skip over escaped characters | ||||
pos += 2 | ||||
continue | ||||
if d == c: | ||||
yield ('string', decode(program[s:pos]), s) | ||||
break | ||||
pos += 1 | ||||
else: | ||||
raise error.ParseError(_("unterminated string"), s) | ||||
Matt Mackall
|
r14551 | elif c.isalnum() or c in globchars or ord(c) > 127: | ||
Matt Mackall
|
r14513 | # gather up a symbol/keyword | ||
Matt Mackall
|
r14511 | s = pos | ||
pos += 1 | ||||
while pos < l: # find end of symbol | ||||
d = program[pos] | ||||
Matt Mackall
|
r14551 | if not (d.isalnum() or d in globchars or ord(d) > 127): | ||
Matt Mackall
|
r14511 | break | ||
pos += 1 | ||||
sym = program[s:pos] | ||||
if sym in keywords: # operator keywords | ||||
yield (sym, None, s) | ||||
else: | ||||
yield ('symbol', sym, s) | ||||
pos -= 1 | ||||
else: | ||||
raise error.ParseError(_("syntax error"), pos) | ||||
pos += 1 | ||||
yield ('end', None, pos) | ||||
Yuya Nishihara
|
r20208 | def parse(expr): | ||
Yuya Nishihara
|
r25654 | p = parser.parser(elements) | ||
tree, pos = p.parse(tokenize(expr)) | ||||
Yuya Nishihara
|
r25252 | if pos != len(expr): | ||
raise error.ParseError(_("invalid token"), pos) | ||||
return tree | ||||
Matt Mackall
|
r14511 | |||
Matt Mackall
|
r14551 | def getstring(x, err): | ||
if x and (x[0] == 'string' or x[0] == 'symbol'): | ||||
return x[1] | ||||
raise error.ParseError(err) | ||||
def getset(mctx, x): | ||||
if not x: | ||||
raise error.ParseError(_("missing argument")) | ||||
return methods[x[0]](mctx, *x[1:]) | ||||
def stringset(mctx, x): | ||||
m = mctx.matcher([x]) | ||||
return [f for f in mctx.subset if m(f)] | ||||
def andset(mctx, x, y): | ||||
return getset(mctx.narrow(getset(mctx, x)), y) | ||||
def orset(mctx, x, y): | ||||
# needs optimizing | ||||
xl = getset(mctx, x) | ||||
yl = getset(mctx, y) | ||||
return xl + [f for f in yl if f not in xl] | ||||
def notset(mctx, x): | ||||
s = set(getset(mctx, x)) | ||||
return [r for r in mctx.subset if r not in s] | ||||
Patrick Mezard
|
r17363 | def minusset(mctx, x, y): | ||
xl = getset(mctx, x) | ||||
yl = set(getset(mctx, y)) | ||||
return [f for f in xl if f not in yl] | ||||
Matt Mackall
|
r14551 | def listset(mctx, a, b): | ||
timeless
|
r27518 | raise error.ParseError(_("can't use a list in this context"), | ||
hint=_('see hg help "filesets.x or y"')) | ||||
Matt Mackall
|
r14551 | |||
FUJIWARA Katsunori
|
r27460 | # symbols are callable like: | ||
# fun(mctx, x) | ||||
# with: | ||||
# mctx - current matchctx instance | ||||
# x - argument in tree form | ||||
symbols = {} | ||||
FUJIWARA Katsunori
|
r27461 | # filesets using matchctx.status() | ||
FUJIWARA Katsunori
|
r27463 | _statuscallers = set() | ||
FUJIWARA Katsunori
|
r27461 | |||
FUJIWARA Katsunori
|
r27462 | # filesets using matchctx.existing() | ||
FUJIWARA Katsunori
|
r27463 | _existingcallers = set() | ||
FUJIWARA Katsunori
|
r27462 | |||
FUJIWARA Katsunori
|
r28448 | predicate = registrar.filesetpredicate() | ||
FUJIWARA Katsunori
|
r27460 | |||
FUJIWARA Katsunori
|
r27461 | @predicate('modified()', callstatus=True) | ||
Matt Mackall
|
r14677 | def modified(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is modified according to :hg:`status`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "modified" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("modified takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().modified) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('added()', callstatus=True) | ||
Matt Mackall
|
r14677 | def added(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is added according to :hg:`status`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "added" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("added takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().added) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('removed()', callstatus=True) | ||
Matt Mackall
|
r14677 | def removed(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is removed according to :hg:`status`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "removed" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("removed takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().removed) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('deleted()', callstatus=True) | ||
Matt Mackall
|
r14677 | def deleted(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """Alias for ``missing()``. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "deleted" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("deleted takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().deleted) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('missing()', callstatus=True) | ||
liscju
|
r27024 | def missing(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is missing according to :hg:`status`. | ||
liscju
|
r27024 | """ | ||
# i18n: "missing" is a keyword | ||||
getargs(x, 0, 0, _("missing takes no arguments")) | ||||
Gregory Szorc
|
r31697 | s = set(mctx.status().deleted) | ||
liscju
|
r27024 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('unknown()', callstatus=True) | ||
Matt Mackall
|
r14677 | def unknown(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is unknown according to :hg:`status`. These files will only be | ||
Matt Mackall
|
r14681 | considered if this predicate is used. | ||
""" | ||||
Wagner Bruna
|
r14785 | # i18n: "unknown" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("unknown takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().unknown) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('ignored()', callstatus=True) | ||
Matt Mackall
|
r14677 | def ignored(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is ignored according to :hg:`status`. These files will only be | ||
Matt Mackall
|
r14681 | considered if this predicate is used. | ||
""" | ||||
Wagner Bruna
|
r14785 | # i18n: "ignored" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("ignored takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().ignored) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
FUJIWARA Katsunori
|
r27461 | @predicate('clean()', callstatus=True) | ||
Matt Mackall
|
r14677 | def clean(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is clean according to :hg:`status`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "clean" is a keyword | ||
Matt Mackall
|
r14677 | getargs(x, 0, 0, _("clean takes no arguments")) | ||
Gregory Szorc
|
r31697 | s = set(mctx.status().clean) | ||
Matt Mackall
|
r14677 | return [f for f in mctx.subset if f in s] | ||
Matt Mackall
|
r14676 | def func(mctx, a, b): | ||
if a[0] == 'symbol' and a[1] in symbols: | ||||
FUJIWARA Katsunori
|
r27464 | funcname = a[1] | ||
enabled = mctx._existingenabled | ||||
mctx._existingenabled = funcname in _existingcallers | ||||
try: | ||||
return symbols[funcname](mctx, b) | ||||
finally: | ||||
mctx._existingenabled = enabled | ||||
Matt Harbison
|
r25633 | |||
keep = lambda fn: getattr(fn, '__doc__', None) is not None | ||||
syms = [s for (s, fn) in symbols.items() if keep(fn)] | ||||
raise error.UnknownIdentifier(a[1], syms) | ||||
Matt Mackall
|
r14676 | |||
def getlist(x): | ||||
if not x: | ||||
return [] | ||||
if x[0] == 'list': | ||||
return getlist(x[1]) + [x[2]] | ||||
return [x] | ||||
def getargs(x, min, max, err): | ||||
l = getlist(x) | ||||
if len(l) < min or len(l) > max: | ||||
raise error.ParseError(err) | ||||
return l | ||||
FUJIWARA Katsunori
|
r27462 | @predicate('binary()', callexisting=True) | ||
Matt Mackall
|
r14676 | def binary(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that appears to be binary (contains NUL bytes). | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "binary" is a keyword | ||
Matt Mackall
|
r14676 | getargs(x, 0, 0, _("binary takes no arguments")) | ||
Jun Wu
|
r32134 | return [f for f in mctx.existing() if mctx.ctx[f].isbinary()] | ||
Matt Mackall
|
r14676 | |||
FUJIWARA Katsunori
|
r27462 | @predicate('exec()', callexisting=True) | ||
Matt Mackall
|
r14676 | def exec_(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is marked as executable. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "exec" is a keyword | ||
Matt Mackall
|
r14676 | getargs(x, 0, 0, _("exec takes no arguments")) | ||
Matt Mackall
|
r15963 | return [f for f in mctx.existing() if mctx.ctx.flags(f) == 'x'] | ||
Matt Mackall
|
r14676 | |||
FUJIWARA Katsunori
|
r27462 | @predicate('symlink()', callexisting=True) | ||
Matt Mackall
|
r14676 | def symlink(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is marked as a symlink. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "symlink" is a keyword | ||
Matt Mackall
|
r14676 | getargs(x, 0, 0, _("symlink takes no arguments")) | ||
Matt Mackall
|
r15963 | return [f for f in mctx.existing() if mctx.ctx.flags(f) == 'l'] | ||
Matt Mackall
|
r14676 | |||
FUJIWARA Katsunori
|
r27460 | @predicate('resolved()') | ||
Matt Mackall
|
r14679 | def resolved(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is marked resolved according to :hg:`resolve -l`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "resolved" is a keyword | ||
Matt Mackall
|
r14679 | getargs(x, 0, 0, _("resolved takes no arguments")) | ||
if mctx.ctx.rev() is not None: | ||||
return [] | ||||
Siddharth Agarwal
|
r26995 | ms = merge.mergestate.read(mctx.ctx.repo()) | ||
Matt Mackall
|
r14679 | return [f for f in mctx.subset if f in ms and ms[f] == 'r'] | ||
FUJIWARA Katsunori
|
r27460 | @predicate('unresolved()') | ||
Matt Mackall
|
r14679 | def unresolved(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is marked unresolved according to :hg:`resolve -l`. | ||
Matt Mackall
|
r14681 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "unresolved" is a keyword | ||
Matt Mackall
|
r14679 | getargs(x, 0, 0, _("unresolved takes no arguments")) | ||
if mctx.ctx.rev() is not None: | ||||
return [] | ||||
Siddharth Agarwal
|
r26995 | ms = merge.mergestate.read(mctx.ctx.repo()) | ||
Matt Mackall
|
r14679 | return [f for f in mctx.subset if f in ms and ms[f] == 'u'] | ||
FUJIWARA Katsunori
|
r27460 | @predicate('hgignore()') | ||
Matt Mackall
|
r14680 | def hgignore(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that matches the active .hgignore pattern. | ||
Matt Mackall
|
r14681 | """ | ||
FUJIWARA Katsunori
|
r23113 | # i18n: "hgignore" is a keyword | ||
Matt Mackall
|
r14680 | getargs(x, 0, 0, _("hgignore takes no arguments")) | ||
Matt Harbison
|
r24334 | ignore = mctx.ctx.repo().dirstate._ignore | ||
Matt Mackall
|
r14680 | return [f for f in mctx.subset if ignore(f)] | ||
FUJIWARA Katsunori
|
r27460 | @predicate('portable()') | ||
Siddharth Agarwal
|
r24408 | def portable(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that has a portable name. (This doesn't include filenames with case | ||
Siddharth Agarwal
|
r24408 | collisions.) | ||
""" | ||||
# i18n: "portable" is a keyword | ||||
getargs(x, 0, 0, _("portable takes no arguments")) | ||||
checkwinfilename = util.checkwinfilename | ||||
return [f for f in mctx.subset if checkwinfilename(f) is None] | ||||
FUJIWARA Katsunori
|
r27462 | @predicate('grep(regex)', callexisting=True) | ||
Matt Mackall
|
r14682 | def grep(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File contains the given regular expression. | ||
Matt Mackall
|
r14682 | """ | ||
Patrick Mezard
|
r17368 | try: | ||
# i18n: "grep" is a keyword | ||||
r = re.compile(getstring(x, _("grep requires a pattern"))) | ||||
Gregory Szorc
|
r25660 | except re.error as e: | ||
Patrick Mezard
|
r17368 | raise error.ParseError(_('invalid match pattern: %s') % e) | ||
Matt Mackall
|
r15963 | return [f for f in mctx.existing() if r.search(mctx.ctx[f].data())] | ||
Matt Mackall
|
r14682 | |||
Matt Mackall
|
r14683 | def _sizetomax(s): | ||
try: | ||||
r25925 | s = s.strip().lower() | |||
Bryan O'Sullivan
|
r19194 | for k, v in util._sizeunits: | ||
Matt Mackall
|
r14683 | if s.endswith(k): | ||
# max(4k) = 5k - 1, max(4.5k) = 4.6k - 1 | ||||
n = s[:-len(k)] | ||||
inc = 1.0 | ||||
if "." in n: | ||||
inc /= 10 ** len(n.split(".")[1]) | ||||
return int((float(n) + inc) * v) - 1 | ||||
# no extension, this is a precise value | ||||
return int(s) | ||||
except ValueError: | ||||
Mads Kiilerich
|
r14716 | raise error.ParseError(_("couldn't parse size: %s") % s) | ||
Matt Mackall
|
r14683 | |||
FUJIWARA Katsunori
|
r27462 | @predicate('size(expression)', callexisting=True) | ||
Matt Mackall
|
r14683 | def size(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File size matches the given expression. Examples: | ||
Matt Mackall
|
r14683 | |||
timeless
|
r29987 | - size('1k') - files from 1024 to 2047 bytes | ||
- size('< 20k') - files less than 20480 bytes | ||||
- size('>= .5MB') - files at least 524288 bytes | ||||
- size('4k - 1MB') - files from 4096 bytes to 1048576 bytes | ||||
Matt Mackall
|
r14683 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "size" is a keyword | ||
Mads Kiilerich
|
r14717 | expr = getstring(x, _("size requires an expression")).strip() | ||
Matt Mackall
|
r14683 | if '-' in expr: # do we have a range? | ||
a, b = expr.split('-', 1) | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(a) | ||
b = util.sizetoint(b) | ||||
Matt Mackall
|
r14683 | m = lambda x: x >= a and x <= b | ||
elif expr.startswith("<="): | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(expr[2:]) | ||
Matt Mackall
|
r14683 | m = lambda x: x <= a | ||
elif expr.startswith("<"): | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(expr[1:]) | ||
Matt Mackall
|
r14683 | m = lambda x: x < a | ||
elif expr.startswith(">="): | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(expr[2:]) | ||
Matt Mackall
|
r14683 | m = lambda x: x >= a | ||
elif expr.startswith(">"): | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(expr[1:]) | ||
Matt Mackall
|
r14683 | m = lambda x: x > a | ||
elif expr[0].isdigit or expr[0] == '.': | ||||
Bryan O'Sullivan
|
r19194 | a = util.sizetoint(expr) | ||
Matt Mackall
|
r14683 | b = _sizetomax(expr) | ||
Thomas Arendsen Hein
|
r14690 | m = lambda x: x >= a and x <= b | ||
Matt Mackall
|
r14683 | else: | ||
Mads Kiilerich
|
r14716 | raise error.ParseError(_("couldn't parse size: %s") % expr) | ||
Matt Mackall
|
r14683 | |||
Matt Mackall
|
r15963 | return [f for f in mctx.existing() if m(mctx.ctx[f].size())] | ||
Matt Mackall
|
r14683 | |||
FUJIWARA Katsunori
|
r27462 | @predicate('encoding(name)', callexisting=True) | ||
Matt Mackall
|
r14684 | def encoding(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File can be successfully decoded with the given character | ||
Matt Mackall
|
r14684 | encoding. May not be useful for encodings other than ASCII and | ||
UTF-8. | ||||
""" | ||||
Wagner Bruna
|
r14785 | # i18n: "encoding" is a keyword | ||
Matt Mackall
|
r14684 | enc = getstring(x, _("encoding requires an encoding name")) | ||
s = [] | ||||
Matt Mackall
|
r15963 | for f in mctx.existing(): | ||
Matt Mackall
|
r14684 | d = mctx.ctx[f].data() | ||
try: | ||||
d.decode(enc) | ||||
except LookupError: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_("unknown encoding '%s'") % enc) | ||
Matt Mackall
|
r14684 | except UnicodeDecodeError: | ||
continue | ||||
s.append(f) | ||||
return s | ||||
FUJIWARA Katsunori
|
r27462 | @predicate('eol(style)', callexisting=True) | ||
Matt Mackall
|
r18842 | def eol(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File contains newlines of the given style (dos, unix, mac). Binary | ||
Matt Mackall
|
r18842 | files are excluded, files with mixed line endings match multiple | ||
styles. | ||||
""" | ||||
Matt Harbison
|
r28056 | # i18n: "eol" is a keyword | ||
enc = getstring(x, _("eol requires a style name")) | ||||
Matt Mackall
|
r18842 | |||
s = [] | ||||
for f in mctx.existing(): | ||||
d = mctx.ctx[f].data() | ||||
if util.binary(d): | ||||
continue | ||||
if (enc == 'dos' or enc == 'win') and '\r\n' in d: | ||||
s.append(f) | ||||
elif enc == 'unix' and re.search('(?<!\r)\n', d): | ||||
s.append(f) | ||||
elif enc == 'mac' and re.search('\r(?!\n)', d): | ||||
s.append(f) | ||||
return s | ||||
FUJIWARA Katsunori
|
r27460 | @predicate('copied()') | ||
Matt Mackall
|
r14685 | def copied(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """File that is recorded as being copied. | ||
Matt Mackall
|
r14685 | """ | ||
Wagner Bruna
|
r14785 | # i18n: "copied" is a keyword | ||
Mads Kiilerich
|
r14718 | getargs(x, 0, 0, _("copied takes no arguments")) | ||
Matt Mackall
|
r14685 | s = [] | ||
for f in mctx.subset: | ||||
p = mctx.ctx[f].parents() | ||||
if p and p[0].path() != f: | ||||
s.append(f) | ||||
return s | ||||
Pierre-Yves David
|
r31193 | @predicate('revs(revs, pattern)') | ||
def revs(mctx, x): | ||||
Yuya Nishihara
|
r31254 | """Evaluate set in the specified revisions. If the revset match multiple | ||
revs, this will return file matching pattern in any of the revision. | ||||
Pierre-Yves David
|
r31193 | """ | ||
# i18n: "revs" is a keyword | ||||
r, x = getargs(x, 2, 2, _("revs takes two arguments")) | ||||
# i18n: "revs" is a keyword | ||||
revspec = getstring(r, _("first argument to revs must be a revision")) | ||||
repo = mctx.ctx.repo() | ||||
revs = scmutil.revrange(repo, [revspec]) | ||||
found = set() | ||||
result = [] | ||||
for r in revs: | ||||
ctx = repo[r] | ||||
for f in getset(mctx.switch(ctx, _buildstatus(ctx, x)), x): | ||||
if f not in found: | ||||
found.add(f) | ||||
result.append(f) | ||||
return result | ||||
Pierre-Yves David
|
r31195 | @predicate('status(base, rev, pattern)') | ||
def status(mctx, x): | ||||
Yuya Nishihara
|
r31254 | """Evaluate predicate using status change between ``base`` and | ||
Pierre-Yves David
|
r31195 | ``rev``. Examples: | ||
- ``status(3, 7, added())`` - matches files added from "3" to "7" | ||||
""" | ||||
repo = mctx.ctx.repo() | ||||
# i18n: "status" is a keyword | ||||
b, r, x = getargs(x, 3, 3, _("status takes three arguments")) | ||||
# i18n: "status" is a keyword | ||||
baseerr = _("first argument to status must be a revision") | ||||
baserevspec = getstring(b, baseerr) | ||||
if not baserevspec: | ||||
raise error.ParseError(baseerr) | ||||
reverr = _("second argument to status must be a revision") | ||||
revspec = getstring(r, reverr) | ||||
if not revspec: | ||||
raise error.ParseError(reverr) | ||||
basenode, node = scmutil.revpair(repo, [baserevspec, revspec]) | ||||
basectx = repo[basenode] | ||||
ctx = repo[node] | ||||
return getset(mctx.switch(ctx, _buildstatus(ctx, x, basectx=basectx)), x) | ||||
FUJIWARA Katsunori
|
r27460 | @predicate('subrepo([pattern])') | ||
Angel Ezquerra
|
r16443 | def subrepo(mctx, x): | ||
FUJIWARA Katsunori
|
r27460 | """Subrepositories whose paths match the given pattern. | ||
Angel Ezquerra
|
r16443 | """ | ||
# i18n: "subrepo" is a keyword | ||||
getargs(x, 0, 1, _("subrepo takes at most one argument")) | ||||
ctx = mctx.ctx | ||||
Mads Kiilerich
|
r18364 | sstate = sorted(ctx.substate) | ||
Angel Ezquerra
|
r16443 | if x: | ||
FUJIWARA Katsunori
|
r23113 | # i18n: "subrepo" is a keyword | ||
Angel Ezquerra
|
r16443 | pat = getstring(x, _("subrepo requires a pattern or no arguments")) | ||
Gregory Szorc
|
r25938 | from . import match as matchmod # avoid circular import issues | ||
Angel Ezquerra
|
r16443 | fast = not matchmod.patkind(pat) | ||
if fast: | ||||
def m(s): | ||||
return (s == pat) | ||||
else: | ||||
Matt Harbison
|
r24334 | m = matchmod.match(ctx.repo().root, '', [pat], ctx=ctx) | ||
Angel Ezquerra
|
r16443 | return [sub for sub in sstate if m(sub)] | ||
else: | ||||
return [sub for sub in sstate] | ||||
Matt Mackall
|
r14551 | methods = { | ||
'string': stringset, | ||||
'symbol': stringset, | ||||
'and': andset, | ||||
'or': orset, | ||||
Patrick Mezard
|
r17363 | 'minus': minusset, | ||
Matt Mackall
|
r14551 | 'list': listset, | ||
'group': getset, | ||||
Matt Mackall
|
r14676 | 'not': notset, | ||
'func': func, | ||||
Matt Mackall
|
r14551 | } | ||
class matchctx(object): | ||||
Yuya Nishihara
|
r31190 | def __init__(self, ctx, subset, status=None): | ||
Matt Mackall
|
r14551 | self.ctx = ctx | ||
self.subset = subset | ||||
Matt Mackall
|
r14677 | self._status = status | ||
FUJIWARA Katsunori
|
r27464 | self._existingenabled = False | ||
Matt Mackall
|
r14677 | def status(self): | ||
return self._status | ||||
Matt Mackall
|
r14673 | def matcher(self, patterns): | ||
return self.ctx.match(patterns) | ||||
Matt Mackall
|
r14551 | def filter(self, files): | ||
return [f for f in files if f in self.subset] | ||||
Matt Mackall
|
r15963 | def existing(self): | ||
FUJIWARA Katsunori
|
r27464 | assert self._existingenabled, 'unexpected existing() invocation' | ||
Patrick Mezard
|
r17365 | if self._status is not None: | ||
removed = set(self._status[3]) | ||||
Patrick Mezard
|
r17367 | unknown = set(self._status[4] + self._status[5]) | ||
Patrick Mezard
|
r17365 | else: | ||
removed = set() | ||||
Patrick Mezard
|
r17366 | unknown = set() | ||
Patrick Mezard
|
r17365 | return (f for f in self.subset | ||
Patrick Mezard
|
r17366 | if (f in self.ctx and f not in removed) or f in unknown) | ||
Matt Mackall
|
r14551 | def narrow(self, files): | ||
Matt Mackall
|
r14677 | return matchctx(self.ctx, self.filter(files), self._status) | ||
Yuya Nishihara
|
r31192 | def switch(self, ctx, status=None): | ||
subset = self.filter(_buildsubset(ctx, status)) | ||||
return matchctx(ctx, subset, status) | ||||
Matt Mackall
|
r14551 | |||
Yuya Nishihara
|
r31188 | class fullmatchctx(matchctx): | ||
"""A match context where any files in any revisions should be valid""" | ||||
Yuya Nishihara
|
r31190 | def __init__(self, ctx, status=None): | ||
subset = _buildsubset(ctx, status) | ||||
Yuya Nishihara
|
r31188 | super(fullmatchctx, self).__init__(ctx, subset, status) | ||
Yuya Nishihara
|
r31192 | def switch(self, ctx, status=None): | ||
return fullmatchctx(ctx, status) | ||||
# filesets using matchctx.switch() | ||||
_switchcallers = [ | ||||
Pierre-Yves David
|
r31193 | 'revs', | ||
Pierre-Yves David
|
r31195 | 'status', | ||
Yuya Nishihara
|
r31192 | ] | ||
Yuya Nishihara
|
r31188 | |||
Matt Mackall
|
r14678 | def _intree(funcs, tree): | ||
if isinstance(tree, tuple): | ||||
if tree[0] == 'func' and tree[1][0] == 'symbol': | ||||
if tree[1][1] in funcs: | ||||
return True | ||||
Yuya Nishihara
|
r31192 | if tree[1][1] in _switchcallers: | ||
# arguments won't be evaluated in the current context | ||||
return False | ||||
Matt Mackall
|
r14678 | for s in tree[1:]: | ||
if _intree(funcs, s): | ||||
return True | ||||
return False | ||||
Yuya Nishihara
|
r31189 | def _buildsubset(ctx, status): | ||
if status: | ||||
subset = [] | ||||
for c in status: | ||||
subset.extend(c) | ||||
return subset | ||||
else: | ||||
return list(ctx.walk(ctx.match([]))) | ||||
Matt Mackall
|
r14673 | def getfileset(ctx, expr): | ||
Yuya Nishihara
|
r25252 | tree = parse(expr) | ||
Yuya Nishihara
|
r31191 | return getset(fullmatchctx(ctx, _buildstatus(ctx, tree)), tree) | ||
Matt Mackall
|
r14678 | |||
Pierre-Yves David
|
r31194 | def _buildstatus(ctx, tree, basectx=None): | ||
Matt Mackall
|
r14678 | # do we need status info? | ||
Pierre-Yves David
|
r31194 | |||
# temporaty boolean to simplify the next conditional | ||||
purewdir = ctx.rev() is None and basectx is None | ||||
FUJIWARA Katsunori
|
r27461 | if (_intree(_statuscallers, tree) or | ||
Patrick Mezard
|
r17365 | # Using matchctx.existing() on a workingctx requires us to check | ||
# for deleted files. | ||||
Pierre-Yves David
|
r31194 | (purewdir and _intree(_existingcallers, tree))): | ||
Matt Mackall
|
r14678 | unknown = _intree(['unknown'], tree) | ||
ignored = _intree(['ignored'], tree) | ||||
Matt Harbison
|
r24334 | r = ctx.repo() | ||
Pierre-Yves David
|
r31194 | if basectx is None: | ||
basectx = ctx.p1() | ||||
return r.status(basectx, ctx, | ||||
Yuya Nishihara
|
r31191 | unknown=unknown, ignored=ignored, clean=True) | ||
Matt Mackall
|
r14678 | else: | ||
Yuya Nishihara
|
r31191 | return None | ||
Matt Mackall
|
r14681 | |||
Yuya Nishihara
|
r25255 | def prettyformat(tree): | ||
return parser.prettyformat(tree, ('string', 'symbol')) | ||||
FUJIWARA Katsunori
|
r28447 | def loadpredicate(ui, extname, registrarobj): | ||
"""Load fileset predicates from specified registrarobj | ||||
""" | ||||
for name, func in registrarobj._table.iteritems(): | ||||
symbols[name] = func | ||||
if func._callstatus: | ||||
_statuscallers.add(name) | ||||
if func._callexisting: | ||||
_existingcallers.add(name) | ||||
FUJIWARA Katsunori
|
r28448 | # load built-in predicates explicitly to setup _statuscallers/_existingcallers | ||
loadpredicate(None, None, predicate) | ||||
Matt Mackall
|
r14681 | # tell hggettext to extract docstrings from these functions: | ||
i18nfunctions = symbols.values() | ||||