minifileset.py
92 lines
| 3.5 KiB
| text/x-python
|
PythonLexer
/ mercurial / minifileset.py
Matt Harbison
|
r35634 | # minifileset.py - a simple language to select files | ||
# | ||||
# Copyright 2017 Facebook, Inc. | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
from .i18n import _ | ||||
from . import ( | ||||
error, | ||||
fileset, | ||||
Yuya Nishihara
|
r38841 | filesetlang, | ||
Augie Fackler
|
r37893 | pycompat, | ||
Matt Harbison
|
r35634 | ) | ||
Yuya Nishihara
|
r38709 | def _sizep(x): | ||
# i18n: "size" is a keyword | ||||
Yuya Nishihara
|
r38841 | expr = filesetlang.getstring(x, _("size requires an expression")) | ||
Yuya Nishihara
|
r38709 | return fileset.sizematcher(expr) | ||
Matt Harbison
|
r35634 | def _compile(tree): | ||
if not tree: | ||||
raise error.ParseError(_("missing argument")) | ||||
op = tree[0] | ||||
Yuya Nishihara
|
r38915 | if op == 'withstatus': | ||
return _compile(tree[1]) | ||||
elif op in {'symbol', 'string', 'kindpat'}: | ||||
Yuya Nishihara
|
r38841 | name = filesetlang.getpattern(tree, {'path'}, _('invalid file pattern')) | ||
Matt Harbison
|
r35634 | if name.startswith('**'): # file extension test, ex. "**.tar.gz" | ||
ext = name[2:] | ||||
Augie Fackler
|
r37893 | for c in pycompat.bytestr(ext): | ||
Matt Harbison
|
r35634 | if c in '*{}[]?/\\': | ||
raise error.ParseError(_('reserved character: %s') % c) | ||||
return lambda n, s: n.endswith(ext) | ||||
Yuya Nishihara
|
r35758 | elif name.startswith('path:'): # directory or full path test | ||
Matt Harbison
|
r35634 | p = name[5:] # prefix | ||
pl = len(p) | ||||
Augie Fackler
|
r37893 | f = lambda n, s: n.startswith(p) and (len(n) == pl | ||
or n[pl:pl + 1] == '/') | ||||
Matt Harbison
|
r35634 | return f | ||
Matt Harbison
|
r35818 | raise error.ParseError(_("unsupported file pattern: %s") % name, | ||
Matt Harbison
|
r35634 | hint=_('paths must be prefixed with "path:"')) | ||
Yuya Nishihara
|
r38901 | elif op in {'or', 'patterns'}: | ||
Yuya Nishihara
|
r38840 | funcs = [_compile(x) for x in tree[1:]] | ||
return lambda n, s: any(f(n, s) for f in funcs) | ||||
Matt Harbison
|
r35634 | elif op == 'and': | ||
func1 = _compile(tree[1]) | ||||
func2 = _compile(tree[2]) | ||||
return lambda n, s: func1(n, s) and func2(n, s) | ||||
elif op == 'not': | ||||
return lambda n, s: not _compile(tree[1])(n, s) | ||||
elif op == 'func': | ||||
symbols = { | ||||
'all': lambda n, s: True, | ||||
'none': lambda n, s: False, | ||||
Yuya Nishihara
|
r38709 | 'size': lambda n, s: _sizep(tree[2])(s), | ||
Matt Harbison
|
r35634 | } | ||
Yuya Nishihara
|
r38841 | name = filesetlang.getsymbol(tree[1]) | ||
Yuya Nishihara
|
r35709 | if name in symbols: | ||
Matt Harbison
|
r35634 | return symbols[name] | ||
raise error.UnknownIdentifier(name, symbols.keys()) | ||||
elif op == 'minus': # equivalent to 'x and not y' | ||||
func1 = _compile(tree[1]) | ||||
func2 = _compile(tree[2]) | ||||
return lambda n, s: func1(n, s) and not func2(n, s) | ||||
elif op == 'list': | ||||
raise error.ParseError(_("can't use a list in this context"), | ||||
Martin von Zweigbergk
|
r38846 | hint=_('see \'hg help "filesets.x or y"\'')) | ||
Matt Harbison
|
r35634 | raise error.ProgrammingError('illegal tree: %r' % (tree,)) | ||
def compile(text): | ||||
"""generate a function (path, size) -> bool from filter specification. | ||||
"text" could contain the operators defined by the fileset language for | ||||
common logic operations, and parenthesis for grouping. The supported path | ||||
tests are '**.extname' for file extension test, and '"path:dir/subdir"' | ||||
for prefix test. The ``size()`` predicate is borrowed from filesets to test | ||||
file size. The predicates ``all()`` and ``none()`` are also supported. | ||||
Yuya Nishihara
|
r35759 | '(**.php & size(">10MB")) | **.zip | (path:bin & !path:bin/README)' for | ||
Matt Harbison
|
r35634 | example, will catch all php files whose size is greater than 10 MB, all | ||
files whose name ends with ".zip", and all files under "bin" in the repo | ||||
root except for "bin/README". | ||||
""" | ||||
Yuya Nishihara
|
r38841 | tree = filesetlang.parse(text) | ||
Yuya Nishihara
|
r38862 | tree = filesetlang.analyze(tree) | ||
Yuya Nishihara
|
r38865 | tree = filesetlang.optimize(tree) | ||
Matt Harbison
|
r35634 | return _compile(tree) | ||